2020-08-18 07:23:45 +03:00
|
|
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a MIT-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package storage
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"errors"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/url"
|
2020-09-08 18:45:10 +03:00
|
|
|
"time"
|
2020-08-18 07:23:45 +03:00
|
|
|
|
|
|
|
"code.gitea.io/gitea/modules/setting"
|
|
|
|
)
|
|
|
|
|
|
|
|
var (
|
|
|
|
// ErrURLNotSupported represents url is not supported
|
|
|
|
ErrURLNotSupported = errors.New("url method not supported")
|
|
|
|
)
|
|
|
|
|
2020-09-08 18:45:10 +03:00
|
|
|
// Object represents the object on the storage
|
|
|
|
type Object interface {
|
|
|
|
io.ReadCloser
|
|
|
|
io.Seeker
|
|
|
|
}
|
|
|
|
|
|
|
|
// ObjectInfo represents the object info on the storage
|
|
|
|
type ObjectInfo interface {
|
|
|
|
Name() string
|
|
|
|
Size() int64
|
|
|
|
ModTime() time.Time
|
|
|
|
}
|
|
|
|
|
2020-08-18 07:23:45 +03:00
|
|
|
// ObjectStorage represents an object storage to handle a bucket and files
|
|
|
|
type ObjectStorage interface {
|
2020-09-08 18:45:10 +03:00
|
|
|
Open(path string) (Object, error)
|
2020-08-18 07:23:45 +03:00
|
|
|
Save(path string, r io.Reader) (int64, error)
|
2020-09-08 18:45:10 +03:00
|
|
|
Stat(path string) (ObjectInfo, error)
|
2020-08-18 07:23:45 +03:00
|
|
|
Delete(path string) error
|
|
|
|
URL(path, name string) (*url.URL, error)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Copy copys a file from source ObjectStorage to dest ObjectStorage
|
|
|
|
func Copy(dstStorage ObjectStorage, dstPath string, srcStorage ObjectStorage, srcPath string) (int64, error) {
|
|
|
|
f, err := srcStorage.Open(srcPath)
|
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
defer f.Close()
|
|
|
|
|
|
|
|
return dstStorage.Save(dstPath, f)
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
// Attachments represents attachments storage
|
|
|
|
Attachments ObjectStorage
|
2020-09-08 18:45:10 +03:00
|
|
|
|
|
|
|
// LFS represents lfs storage
|
|
|
|
LFS ObjectStorage
|
2020-08-18 07:23:45 +03:00
|
|
|
)
|
|
|
|
|
|
|
|
// Init init the stoarge
|
|
|
|
func Init() error {
|
2020-09-08 18:45:10 +03:00
|
|
|
if err := initAttachments(); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return initLFS()
|
|
|
|
}
|
|
|
|
|
|
|
|
func initAttachments() error {
|
2020-08-18 07:23:45 +03:00
|
|
|
var err error
|
|
|
|
switch setting.Attachment.StoreType {
|
|
|
|
case "local":
|
|
|
|
Attachments, err = NewLocalStorage(setting.Attachment.Path)
|
|
|
|
case "minio":
|
|
|
|
minio := setting.Attachment.Minio
|
|
|
|
Attachments, err = NewMinioStorage(
|
|
|
|
context.Background(),
|
|
|
|
minio.Endpoint,
|
|
|
|
minio.AccessKeyID,
|
|
|
|
minio.SecretAccessKey,
|
|
|
|
minio.Bucket,
|
|
|
|
minio.Location,
|
|
|
|
minio.BasePath,
|
|
|
|
minio.UseSSL,
|
|
|
|
)
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("Unsupported attachment store type: %s", setting.Attachment.StoreType)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
2020-09-08 18:45:10 +03:00
|
|
|
|
|
|
|
func initLFS() error {
|
|
|
|
var err error
|
|
|
|
switch setting.LFS.StoreType {
|
|
|
|
case "local":
|
|
|
|
LFS, err = NewLocalStorage(setting.LFS.ContentPath)
|
|
|
|
case "minio":
|
|
|
|
minio := setting.LFS.Minio
|
|
|
|
LFS, err = NewMinioStorage(
|
|
|
|
context.Background(),
|
|
|
|
minio.Endpoint,
|
|
|
|
minio.AccessKeyID,
|
|
|
|
minio.SecretAccessKey,
|
|
|
|
minio.Bucket,
|
|
|
|
minio.Location,
|
|
|
|
minio.BasePath,
|
|
|
|
minio.UseSSL,
|
|
|
|
)
|
|
|
|
default:
|
|
|
|
return fmt.Errorf("Unsupported LFS store type: %s", setting.LFS.StoreType)
|
|
|
|
}
|
|
|
|
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|