summaryrefslogtreecommitdiff
path: root/internal/storage/storage.go
blob: 498ea873adc688e16215e11651bf06cff87e286b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
/*
   GoToSocial
   Copyright (C) 2021-2022 GoToSocial Authors admin@gotosocial.org

   This program is free software: you can redistribute it and/or modify
   it under the terms of the GNU Affero General Public License as published by
   the Free Software Foundation, either version 3 of the License, or
   (at your option) any later version.

   This program is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU Affero General Public License for more details.

   You should have received a copy of the GNU Affero General Public License
   along with this program.  If not, see <http://www.gnu.org/licenses/>.
*/

package storage

import (
	"context"
	"fmt"
	"mime"
	"net/url"
	"path"
	"time"

	"codeberg.org/gruf/go-store/v2/kv"
	"codeberg.org/gruf/go-store/v2/storage"
	"github.com/minio/minio-go/v7"
	"github.com/minio/minio-go/v7/pkg/credentials"
	"github.com/superseriousbusiness/gotosocial/internal/config"
)

// ErrAlreadyExists is a ptr to underlying storage.ErrAlreadyExists,
// to put the related errors in the same package as our storage wrapper.
var ErrAlreadyExists = storage.ErrAlreadyExists

// Driver wraps a kv.KVStore to also provide S3 presigned GET URLs.
type Driver struct {
	// Underlying storage
	*kv.KVStore
	Storage storage.Storage

	// S3-only parameters
	Proxy  bool
	Bucket string
}

// URL will return a presigned GET object URL, but only if running on S3 storage with proxying disabled.
func (d *Driver) URL(ctx context.Context, key string) *url.URL {
	// Check whether S3 *without* proxying is enabled
	s3, ok := d.Storage.(*storage.S3Storage)
	if !ok || d.Proxy {
		return nil
	}

	// If URL request fails, fallback is to fetch the file. So ignore the error here
	url, _ := s3.Client().PresignedGetObject(ctx, d.Bucket, key, time.Hour, url.Values{
		"response-content-type": []string{mime.TypeByExtension(path.Ext(key))},
	})

	return url
}

func AutoConfig() (*Driver, error) {
	var st storage.Storage

	switch backend := config.GetStorageBackend(); backend {
	case "s3":
		// Load runtime configuration
		endpoint := config.GetStorageS3Endpoint()
		access := config.GetStorageS3AccessKey()
		secret := config.GetStorageS3SecretKey()
		secure := config.GetStorageS3UseSSL()
		bucket := config.GetStorageS3BucketName()

		// Open the s3 storage implementation
		s3, err := storage.OpenS3(endpoint, bucket, &storage.S3Config{
			CoreOpts: minio.Options{
				Creds:  credentials.NewStaticV4(access, secret, ""),
				Secure: secure,
			},
			GetOpts:      minio.GetObjectOptions{},
			PutOpts:      minio.PutObjectOptions{},
			PutChunkSize: 5 * 1024 * 1024, // 5MiB
			StatOpts:     minio.StatObjectOptions{},
			RemoveOpts:   minio.RemoveObjectOptions{},
			ListSize:     200,
		})
		if err != nil {
			return nil, fmt.Errorf("error opening s3 storage: %w", err)
		}

		// Set storage impl
		st = s3

	case "local":
		// Load runtime configuration
		basePath := config.GetStorageLocalBasePath()

		// Open the disk storage implementation
		disk, err := storage.OpenDisk(basePath, &storage.DiskConfig{
			// Put the store lockfile in the storage dir itself.
			// Normally this would not be safe, since we could end up
			// overwriting the lockfile if we store a file called 'store.lock'.
			// However, in this case it's OK because the keys are set by
			// GtS and not the user, so we know we're never going to overwrite it.
			LockFile: path.Join(basePath, "store.lock"),
		})
		if err != nil {
			return nil, fmt.Errorf("error opening disk storage: %w", err)
		}

		// Set storage impl
		st = disk

	default:
		return nil, fmt.Errorf("invalid storage backend: %s", backend)
	}

	return &Driver{
		KVStore: kv.New(st),
		Proxy:   config.GetStorageS3Proxy(),
		Bucket:  config.GetStorageS3BucketName(),
		Storage: st,
	}, nil
}