2020-08-18 09:53:45 +05:30
// Copyright 2020 The Gitea Authors. All rights reserved.
2022-11-27 23:50:29 +05:30
// SPDX-License-Identifier: MIT
2020-08-18 09:53:45 +05:30
package storage
import (
2020-10-13 09:28:34 +05:30
"context"
2023-03-22 01:32:49 +05:30
"fmt"
2020-08-18 09:53:45 +05:30
"io"
"net/url"
"os"
"path/filepath"
2020-10-16 09:21:06 +05:30
"code.gitea.io/gitea/modules/log"
2020-08-18 09:53:45 +05:30
"code.gitea.io/gitea/modules/util"
)
2022-01-20 23:16:10 +05:30
var _ ObjectStorage = & LocalStorage { }
2020-08-18 09:53:45 +05:30
2020-10-13 09:28:34 +05:30
// 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 18:49:17 +05:30
Path string ` ini:"PATH" `
TemporaryPath string ` ini:"TEMPORARY_PATH" `
2020-10-13 09:28:34 +05:30
}
2020-08-18 09:53:45 +05:30
// LocalStorage represents a local files storage
type LocalStorage struct {
2021-03-05 18:49:17 +05:30
ctx context . Context
dir string
tmpdir string
2020-08-18 09:53:45 +05:30
}
// NewLocalStorage returns a local files
2020-10-13 09:28:34 +05:30
func NewLocalStorage ( ctx context . Context , cfg interface { } ) ( ObjectStorage , error ) {
configInterface , err := toConfig ( LocalStorageConfig { } , cfg )
if err != nil {
return nil , err
}
config := configInterface . ( LocalStorageConfig )
2023-03-22 01:32:49 +05:30
if ! filepath . IsAbs ( config . Path ) {
return nil , fmt . Errorf ( "LocalStorageConfig.Path should have been prepared by setting/storage.go and should be an absolute path, but not: %q" , config . Path )
}
2020-10-16 09:21:06 +05:30
log . Info ( "Creating new Local Storage at %s" , config . Path )
2020-10-13 09:28:34 +05:30
if err := os . MkdirAll ( config . Path , os . ModePerm ) ; err != nil {
2020-08-18 09:53:45 +05:30
return nil , err
}
2021-03-05 18:49:17 +05:30
if config . TemporaryPath == "" {
2023-03-22 01:32:49 +05:30
config . TemporaryPath = filepath . Join ( config . Path , "tmp" )
}
if ! filepath . IsAbs ( config . TemporaryPath ) {
return nil , fmt . Errorf ( "LocalStorageConfig.TemporaryPath should be an absolute path, but not: %q" , config . TemporaryPath )
2021-03-05 18:49:17 +05:30
}
2020-08-18 09:53:45 +05:30
return & LocalStorage {
2021-03-05 18:49:17 +05:30
ctx : ctx ,
dir : config . Path ,
tmpdir : config . TemporaryPath ,
2020-08-18 09:53:45 +05:30
} , nil
}
2022-03-23 02:32:26 +05:30
func ( l * LocalStorage ) buildLocalPath ( p string ) string {
2023-03-22 01:32:49 +05:30
return util . FilePathJoinAbs ( l . dir , p )
2022-03-23 02:32:26 +05:30
}
2020-08-18 09:53:45 +05:30
// Open a file
2020-09-08 21:15:10 +05:30
func ( l * LocalStorage ) Open ( path string ) ( Object , error ) {
2022-03-23 02:32:26 +05:30
return os . Open ( l . buildLocalPath ( path ) )
2020-08-18 09:53:45 +05:30
}
// Save a file
2021-04-03 21:49:59 +05:30
func ( l * LocalStorage ) Save ( path string , r io . Reader , size int64 ) ( int64 , error ) {
2022-03-23 02:32:26 +05:30
p := l . buildLocalPath ( path )
2020-08-18 09:53:45 +05:30
if err := os . MkdirAll ( filepath . Dir ( p ) , os . ModePerm ) ; err != nil {
return 0 , err
}
2021-03-05 18:49:17 +05:30
// Create a temporary file to save to
if err := os . MkdirAll ( l . tmpdir , os . ModePerm ) ; err != nil {
2020-08-18 09:53:45 +05:30
return 0 , err
}
2021-09-22 11:08:34 +05:30
tmp , err := os . CreateTemp ( l . tmpdir , "upload-*" )
2021-03-05 18:49:17 +05:30
if err != nil {
return 0 , err
}
tmpRemoved := false
defer func ( ) {
if ! tmpRemoved {
_ = util . Remove ( tmp . Name ( ) )
}
} ( )
2020-08-18 09:53:45 +05:30
2021-03-05 18:49:17 +05:30
n , err := io . Copy ( tmp , r )
2020-08-18 09:53:45 +05:30
if err != nil {
return 0 , err
}
2021-03-05 18:49:17 +05:30
if err := tmp . Close ( ) ; err != nil {
return 0 , err
}
2021-07-15 21:16:07 +05:30
if err := util . Rename ( tmp . Name ( ) , p ) ; err != nil {
2021-03-05 18:49:17 +05:30
return 0 , err
}
2022-09-24 18:34:14 +05:30
// 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 06:20:36 +05:30
// 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 & 0 o666 ) ; err != nil {
2022-09-24 18:34:14 +05:30
return 0 , err
}
2021-03-05 18:49:17 +05:30
tmpRemoved = true
return n , nil
2020-08-18 09:53:45 +05:30
}
2020-09-08 21:15:10 +05:30
// Stat returns the info of the file
2020-09-29 14:35:13 +05:30
func ( l * LocalStorage ) Stat ( path string ) ( os . FileInfo , error ) {
2022-03-23 02:32:26 +05:30
return os . Stat ( l . buildLocalPath ( path ) )
2022-03-14 20:48:27 +05:30
}
2020-08-18 09:53:45 +05:30
// Delete delete a file
func ( l * LocalStorage ) Delete ( path string ) error {
2022-03-23 02:32:26 +05:30
return util . Remove ( l . buildLocalPath ( path ) )
2020-08-18 09:53:45 +05:30
}
// URL gets the redirect URL to a file
func ( l * LocalStorage ) URL ( path , name string ) ( * url . URL , error ) {
return nil , ErrURLNotSupported
}
2020-09-29 14:35:13 +05:30
// IterateObjects iterates across the objects in the local storage
2023-05-14 04:03:25 +05:30
func ( l * LocalStorage ) IterateObjects ( dirName string , fn func ( path string , obj Object ) error ) error {
dir := l . buildLocalPath ( dirName )
2023-03-13 15:53:51 +05:30
return filepath . WalkDir ( dir , func ( path string , d os . DirEntry , err error ) error {
2020-09-29 14:35:13 +05:30
if err != nil {
return err
}
2020-10-13 09:28:34 +05:30
select {
case <- l . ctx . Done ( ) :
return l . ctx . Err ( )
default :
}
2020-09-29 14:35:13 +05:30
if path == l . dir {
return nil
}
2023-01-16 21:51:44 +05:30
if d . IsDir ( ) {
2020-09-29 14:35:13 +05:30
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 09:28:34 +05:30
func init ( ) {
RegisterStorageType ( LocalStorageType , NewLocalStorage )
}