* Retry rename on lock induced failures Due to external locking on Windows it is possible for an os.Rename to fail if the files or directories are being used elsewhere. This PR simply suggests retrying the rename again similar to how we handle the os.Remove problems. Fix #16427 Signed-off-by: Andrew Thornton <art27@cantab.net> * resolve CI fail Co-authored-by: techknowlogick <techknowlogick@gitea.io>
		
			
				
	
	
		
			157 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			157 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // 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"
 | |
| 	"io"
 | |
| 	"io/ioutil"
 | |
| 	"net/url"
 | |
| 	"os"
 | |
| 	"path/filepath"
 | |
| 
 | |
| 	"code.gitea.io/gitea/modules/log"
 | |
| 	"code.gitea.io/gitea/modules/util"
 | |
| )
 | |
| 
 | |
| var (
 | |
| 	_ ObjectStorage = &LocalStorage{}
 | |
| )
 | |
| 
 | |
| // LocalStorageType is the type descriptor for local storage
 | |
| 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
 | |
| func NewLocalStorage(ctx context.Context, cfg interface{}) (ObjectStorage, error) {
 | |
| 	configInterface, err := toConfig(LocalStorageConfig{}, cfg)
 | |
| 	if err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 	config := configInterface.(LocalStorageConfig)
 | |
| 
 | |
| 	log.Info("Creating new Local Storage at %s", config.Path)
 | |
| 	if err := os.MkdirAll(config.Path, os.ModePerm); err != nil {
 | |
| 		return nil, err
 | |
| 	}
 | |
| 
 | |
| 	if config.TemporaryPath == "" {
 | |
| 		config.TemporaryPath = config.Path + "/tmp"
 | |
| 	}
 | |
| 
 | |
| 	return &LocalStorage{
 | |
| 		ctx:    ctx,
 | |
| 		dir:    config.Path,
 | |
| 		tmpdir: config.TemporaryPath,
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| // Open a file
 | |
| func (l *LocalStorage) Open(path string) (Object, error) {
 | |
| 	return os.Open(filepath.Join(l.dir, path))
 | |
| }
 | |
| 
 | |
| // Save a file
 | |
| func (l *LocalStorage) Save(path string, r io.Reader, size int64) (int64, error) {
 | |
| 	p := filepath.Join(l.dir, path)
 | |
| 	if err := os.MkdirAll(filepath.Dir(p), os.ModePerm); err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 
 | |
| 	// Create a temporary file to save to
 | |
| 	if err := os.MkdirAll(l.tmpdir, os.ModePerm); err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	tmp, err := ioutil.TempFile(l.tmpdir, "upload-*")
 | |
| 	if err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 	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 := util.Rename(tmp.Name(), p); err != nil {
 | |
| 		return 0, err
 | |
| 	}
 | |
| 
 | |
| 	tmpRemoved = true
 | |
| 
 | |
| 	return n, nil
 | |
| }
 | |
| 
 | |
| // Stat returns the info of the file
 | |
| func (l *LocalStorage) Stat(path string) (os.FileInfo, error) {
 | |
| 	return os.Stat(filepath.Join(l.dir, path))
 | |
| }
 | |
| 
 | |
| // Delete delete a file
 | |
| func (l *LocalStorage) Delete(path string) error {
 | |
| 	p := filepath.Join(l.dir, path)
 | |
| 	return util.Remove(p)
 | |
| }
 | |
| 
 | |
| // URL gets the redirect URL to a file
 | |
| func (l *LocalStorage) URL(path, name string) (*url.URL, error) {
 | |
| 	return nil, ErrURLNotSupported
 | |
| }
 | |
| 
 | |
| // IterateObjects iterates across the objects in the local storage
 | |
| func (l *LocalStorage) IterateObjects(fn func(path string, obj Object) error) error {
 | |
| 	return filepath.Walk(l.dir, func(path string, info os.FileInfo, err error) error {
 | |
| 		if err != nil {
 | |
| 			return err
 | |
| 		}
 | |
| 		select {
 | |
| 		case <-l.ctx.Done():
 | |
| 			return l.ctx.Err()
 | |
| 		default:
 | |
| 		}
 | |
| 		if path == l.dir {
 | |
| 			return nil
 | |
| 		}
 | |
| 		if info.IsDir() {
 | |
| 			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)
 | |
| 	})
 | |
| }
 | |
| 
 | |
| func init() {
 | |
| 	RegisterStorageType(LocalStorageType, NewLocalStorage)
 | |
| }
 |