mirror of
				https://gitee.com/gitea/gitea
				synced 2025-11-04 08:30:25 +08:00 
			
		
		
		
	Fix race in local storage (#14888)
LocalStorage should only put completed files in position Signed-off-by: Andrew Thornton <art27@cantab.net>
This commit is contained in:
		@@ -7,6 +7,7 @@ package storage
 | 
			
		||||
import (
 | 
			
		||||
	"context"
 | 
			
		||||
	"io"
 | 
			
		||||
	"io/ioutil"
 | 
			
		||||
	"net/url"
 | 
			
		||||
	"os"
 | 
			
		||||
	"path/filepath"
 | 
			
		||||
@@ -25,12 +26,14 @@ const LocalStorageType Type = "local"
 | 
			
		||||
// LocalStorageConfig represents the configuration for a local storage
 | 
			
		||||
type LocalStorageConfig struct {
 | 
			
		||||
	Path          string `ini:"PATH"`
 | 
			
		||||
	TemporaryPath string `ini:"TEMPORARY_PATH"`
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// LocalStorage represents a local files storage
 | 
			
		||||
type LocalStorage struct {
 | 
			
		||||
	ctx    context.Context
 | 
			
		||||
	dir    string
 | 
			
		||||
	tmpdir string
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// NewLocalStorage returns a local files
 | 
			
		||||
@@ -46,9 +49,14 @@ func NewLocalStorage(ctx context.Context, cfg interface{}) (ObjectStorage, error
 | 
			
		||||
		return nil, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if config.TemporaryPath == "" {
 | 
			
		||||
		config.TemporaryPath = config.Path + "/tmp"
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	return &LocalStorage{
 | 
			
		||||
		ctx:    ctx,
 | 
			
		||||
		dir:    config.Path,
 | 
			
		||||
		tmpdir: config.TemporaryPath,
 | 
			
		||||
	}, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
@@ -64,17 +72,37 @@ func (l *LocalStorage) Save(path string, r io.Reader) (int64, error) {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	// always override
 | 
			
		||||
	if err := util.Remove(p); err != nil {
 | 
			
		||||
	// Create a temporary file to save to
 | 
			
		||||
	if err := os.MkdirAll(l.tmpdir, os.ModePerm); err != nil {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	f, err := os.Create(p)
 | 
			
		||||
	tmp, err := ioutil.TempFile(l.tmpdir, "upload-*")
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
	defer f.Close()
 | 
			
		||||
	return io.Copy(f, r)
 | 
			
		||||
	tmpRemoved := false
 | 
			
		||||
	defer func() {
 | 
			
		||||
		if !tmpRemoved {
 | 
			
		||||
			_ = util.Remove(tmp.Name())
 | 
			
		||||
		}
 | 
			
		||||
	}()
 | 
			
		||||
 | 
			
		||||
	n, err := io.Copy(tmp, r)
 | 
			
		||||
	if err != nil {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if err := tmp.Close(); err != nil {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	if err := os.Rename(tmp.Name(), p); err != nil {
 | 
			
		||||
		return 0, err
 | 
			
		||||
	}
 | 
			
		||||
 | 
			
		||||
	tmpRemoved = true
 | 
			
		||||
 | 
			
		||||
	return n, nil
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Stat returns the info of the file
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user