2020-08-18 12:23:45 +08:00
|
|
|
// Copyright 2020 The Gitea Authors. All rights reserved.
|
2022-11-28 02:20:29 +08:00
|
|
|
// SPDX-License-Identifier: MIT
|
2020-08-18 12:23:45 +08:00
|
|
|
|
|
|
|
package storage
|
|
|
|
|
|
|
|
import (
|
2020-10-13 11:58:34 +08:00
|
|
|
"context"
|
2020-08-18 12:23:45 +08:00
|
|
|
"io"
|
|
|
|
"net/url"
|
|
|
|
"os"
|
|
|
|
"path/filepath"
|
2022-03-14 23:18:27 +08:00
|
|
|
"strings"
|
2020-08-18 12:23:45 +08:00
|
|
|
|
2020-10-16 11:51:06 +08:00
|
|
|
"code.gitea.io/gitea/modules/log"
|
2020-08-18 12:23:45 +08:00
|
|
|
"code.gitea.io/gitea/modules/util"
|
|
|
|
)
|
|
|
|
|
2022-01-21 01:46:10 +08:00
|
|
|
var _ ObjectStorage = &LocalStorage{}
|
2020-08-18 12:23:45 +08:00
|
|
|
|
2020-10-13 11:58:34 +08:00
|
|
|
// LocalStorageType is the type descriptor for local storage
|
|
|
|
const LocalStorageType Type = "local"
|
|
|
|
|
|
|
|
// LocalStorageConfig represents the configuration for a local storage
|
|
|
|
type LocalStorageConfig struct {
|
2021-03-05 21:19:17 +08:00
|
|
|
Path string `ini:"PATH"`
|
|
|
|
TemporaryPath string `ini:"TEMPORARY_PATH"`
|
2020-10-13 11:58:34 +08:00
|
|
|
}
|
|
|
|
|
2020-08-18 12:23:45 +08:00
|
|
|
// LocalStorage represents a local files storage
|
|
|
|
type LocalStorage struct {
|
2021-03-05 21:19:17 +08:00
|
|
|
ctx context.Context
|
|
|
|
dir string
|
|
|
|
tmpdir string
|
2020-08-18 12:23:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// NewLocalStorage returns a local files
|
2020-10-13 11:58:34 +08:00
|
|
|
func NewLocalStorage(ctx context.Context, cfg interface{}) (ObjectStorage, error) {
|
|
|
|
configInterface, err := toConfig(LocalStorageConfig{}, cfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
config := configInterface.(LocalStorageConfig)
|
|
|
|
|
2020-10-16 11:51:06 +08:00
|
|
|
log.Info("Creating new Local Storage at %s", config.Path)
|
2020-10-13 11:58:34 +08:00
|
|
|
if err := os.MkdirAll(config.Path, os.ModePerm); err != nil {
|
2020-08-18 12:23:45 +08:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-03-05 21:19:17 +08:00
|
|
|
if config.TemporaryPath == "" {
|
|
|
|
config.TemporaryPath = config.Path + "/tmp"
|
|
|
|
}
|
|
|
|
|
2020-08-18 12:23:45 +08:00
|
|
|
return &LocalStorage{
|
2021-03-05 21:19:17 +08:00
|
|
|
ctx: ctx,
|
|
|
|
dir: config.Path,
|
|
|
|
tmpdir: config.TemporaryPath,
|
2020-08-18 12:23:45 +08:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2022-03-23 05:02:26 +08:00
|
|
|
func (l *LocalStorage) buildLocalPath(p string) string {
|
2023-03-08 20:17:39 +08:00
|
|
|
return filepath.Join(l.dir, util.CleanPath(strings.ReplaceAll(p, "\\", "/")))
|
2022-03-23 05:02:26 +08:00
|
|
|
}
|
|
|
|
|
2020-08-18 12:23:45 +08:00
|
|
|
// Open a file
|
2020-09-08 23:45:10 +08:00
|
|
|
func (l *LocalStorage) Open(path string) (Object, error) {
|
2022-03-23 05:02:26 +08:00
|
|
|
return os.Open(l.buildLocalPath(path))
|
2020-08-18 12:23:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Save a file
|
2021-04-04 00:19:59 +08:00
|
|
|
func (l *LocalStorage) Save(path string, r io.Reader, size int64) (int64, error) {
|
2022-03-23 05:02:26 +08:00
|
|
|
p := l.buildLocalPath(path)
|
2020-08-18 12:23:45 +08:00
|
|
|
if err := os.MkdirAll(filepath.Dir(p), os.ModePerm); err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2021-03-05 21:19:17 +08:00
|
|
|
// Create a temporary file to save to
|
|
|
|
if err := os.MkdirAll(l.tmpdir, os.ModePerm); err != nil {
|
2020-08-18 12:23:45 +08:00
|
|
|
return 0, err
|
|
|
|
}
|
2021-09-22 13:38:34 +08:00
|
|
|
tmp, err := os.CreateTemp(l.tmpdir, "upload-*")
|
2021-03-05 21:19:17 +08:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
tmpRemoved := false
|
|
|
|
defer func() {
|
|
|
|
if !tmpRemoved {
|
|
|
|
_ = util.Remove(tmp.Name())
|
|
|
|
}
|
|
|
|
}()
|
2020-08-18 12:23:45 +08:00
|
|
|
|
2021-03-05 21:19:17 +08:00
|
|
|
n, err := io.Copy(tmp, r)
|
2020-08-18 12:23:45 +08:00
|
|
|
if err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
2021-03-05 21:19:17 +08:00
|
|
|
|
|
|
|
if err := tmp.Close(); err != nil {
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
2021-07-15 23:46:07 +08:00
|
|
|
if err := util.Rename(tmp.Name(), p); err != nil {
|
2021-03-05 21:19:17 +08:00
|
|
|
return 0, err
|
|
|
|
}
|
2022-09-24 21:04:14 +08:00
|
|
|
// Golang's tmp file (os.CreateTemp) always have 0o600 mode, so we need to change the file to follow the umask (as what Create/MkDir does)
|
2022-12-19 08:50:36 +08:00
|
|
|
// but we don't want to make these files executable - so ensure that we mask out the executable bits
|
|
|
|
if err := util.ApplyUmask(p, os.ModePerm&0o666); err != nil {
|
2022-09-24 21:04:14 +08:00
|
|
|
return 0, err
|
|
|
|
}
|
2021-03-05 21:19:17 +08:00
|
|
|
|
|
|
|
tmpRemoved = true
|
|
|
|
|
|
|
|
return n, nil
|
2020-08-18 12:23:45 +08:00
|
|
|
}
|
|
|
|
|
2020-09-08 23:45:10 +08:00
|
|
|
// Stat returns the info of the file
|
2020-09-29 17:05:13 +08:00
|
|
|
func (l *LocalStorage) Stat(path string) (os.FileInfo, error) {
|
2022-03-23 05:02:26 +08:00
|
|
|
return os.Stat(l.buildLocalPath(path))
|
2022-03-14 23:18:27 +08:00
|
|
|
}
|
|
|
|
|
2020-08-18 12:23:45 +08:00
|
|
|
// Delete delete a file
|
|
|
|
func (l *LocalStorage) Delete(path string) error {
|
2022-03-23 05:02:26 +08:00
|
|
|
return util.Remove(l.buildLocalPath(path))
|
2020-08-18 12:23:45 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// URL gets the redirect URL to a file
|
|
|
|
func (l *LocalStorage) URL(path, name string) (*url.URL, error) {
|
|
|
|
return nil, ErrURLNotSupported
|
|
|
|
}
|
2020-09-29 17:05:13 +08:00
|
|
|
|
|
|
|
// IterateObjects iterates across the objects in the local storage
|
2023-03-13 18:23:51 +08:00
|
|
|
func (l *LocalStorage) IterateObjects(prefix string, fn func(path string, obj Object) error) error {
|
|
|
|
dir := l.dir
|
|
|
|
if prefix != "" {
|
|
|
|
dir = filepath.Join(l.dir, util.CleanPath(prefix))
|
|
|
|
}
|
|
|
|
return filepath.WalkDir(dir, func(path string, d os.DirEntry, err error) error {
|
2020-09-29 17:05:13 +08:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2020-10-13 11:58:34 +08:00
|
|
|
select {
|
|
|
|
case <-l.ctx.Done():
|
|
|
|
return l.ctx.Err()
|
|
|
|
default:
|
|
|
|
}
|
2020-09-29 17:05:13 +08:00
|
|
|
if path == l.dir {
|
|
|
|
return nil
|
|
|
|
}
|
2023-01-17 00:21:44 +08:00
|
|
|
if d.IsDir() {
|
2020-09-29 17:05:13 +08:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
relPath, err := filepath.Rel(l.dir, path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
obj, err := os.Open(path)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer obj.Close()
|
|
|
|
return fn(relPath, obj)
|
|
|
|
})
|
|
|
|
}
|
2020-10-13 11:58:34 +08:00
|
|
|
|
|
|
|
func init() {
|
|
|
|
RegisterStorageType(LocalStorageType, NewLocalStorage)
|
|
|
|
}
|