mirror of
				https://github.com/go-gitea/gitea
				synced 2025-11-04 05:18:25 +00:00 
			
		
		
		
	* Use vendored go-swagger * vendor go-swagger * revert un wanteed change * remove un-needed GO111MODULE * Update Makefile Co-Authored-By: techknowlogick <matti@mdranta.net>
		
			
				
	
	
		
			366 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
			
		
		
	
	
			366 lines
		
	
	
		
			7.3 KiB
		
	
	
	
		
			Go
		
	
	
	
		
			Vendored
		
	
	
	
// Copyright © 2014 Steve Francia <spf@spf13.com>.
 | 
						|
//
 | 
						|
// Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
// you may not use this file except in compliance with the License.
 | 
						|
// You may obtain a copy of the License at
 | 
						|
// http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
//
 | 
						|
// Unless required by applicable law or agreed to in writing, software
 | 
						|
// distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
// See the License for the specific language governing permissions and
 | 
						|
// limitations under the License.
 | 
						|
 | 
						|
package afero
 | 
						|
 | 
						|
import (
 | 
						|
	"fmt"
 | 
						|
	"log"
 | 
						|
	"os"
 | 
						|
	"path/filepath"
 | 
						|
	"strings"
 | 
						|
	"sync"
 | 
						|
	"time"
 | 
						|
 | 
						|
	"github.com/spf13/afero/mem"
 | 
						|
)
 | 
						|
 | 
						|
type MemMapFs struct {
 | 
						|
	mu   sync.RWMutex
 | 
						|
	data map[string]*mem.FileData
 | 
						|
	init sync.Once
 | 
						|
}
 | 
						|
 | 
						|
func NewMemMapFs() Fs {
 | 
						|
	return &MemMapFs{}
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) getData() map[string]*mem.FileData {
 | 
						|
	m.init.Do(func() {
 | 
						|
		m.data = make(map[string]*mem.FileData)
 | 
						|
		// Root should always exist, right?
 | 
						|
		// TODO: what about windows?
 | 
						|
		m.data[FilePathSeparator] = mem.CreateDir(FilePathSeparator)
 | 
						|
	})
 | 
						|
	return m.data
 | 
						|
}
 | 
						|
 | 
						|
func (*MemMapFs) Name() string { return "MemMapFS" }
 | 
						|
 | 
						|
func (m *MemMapFs) Create(name string) (File, error) {
 | 
						|
	name = normalizePath(name)
 | 
						|
	m.mu.Lock()
 | 
						|
	file := mem.CreateFile(name)
 | 
						|
	m.getData()[name] = file
 | 
						|
	m.registerWithParent(file)
 | 
						|
	m.mu.Unlock()
 | 
						|
	return mem.NewFileHandle(file), nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) unRegisterWithParent(fileName string) error {
 | 
						|
	f, err := m.lockfreeOpen(fileName)
 | 
						|
	if err != nil {
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	parent := m.findParent(f)
 | 
						|
	if parent == nil {
 | 
						|
		log.Panic("parent of ", f.Name(), " is nil")
 | 
						|
	}
 | 
						|
 | 
						|
	parent.Lock()
 | 
						|
	mem.RemoveFromMemDir(parent, f)
 | 
						|
	parent.Unlock()
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) findParent(f *mem.FileData) *mem.FileData {
 | 
						|
	pdir, _ := filepath.Split(f.Name())
 | 
						|
	pdir = filepath.Clean(pdir)
 | 
						|
	pfile, err := m.lockfreeOpen(pdir)
 | 
						|
	if err != nil {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
	return pfile
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) registerWithParent(f *mem.FileData) {
 | 
						|
	if f == nil {
 | 
						|
		return
 | 
						|
	}
 | 
						|
	parent := m.findParent(f)
 | 
						|
	if parent == nil {
 | 
						|
		pdir := filepath.Dir(filepath.Clean(f.Name()))
 | 
						|
		err := m.lockfreeMkdir(pdir, 0777)
 | 
						|
		if err != nil {
 | 
						|
			//log.Println("Mkdir error:", err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
		parent, err = m.lockfreeOpen(pdir)
 | 
						|
		if err != nil {
 | 
						|
			//log.Println("Open after Mkdir error:", err)
 | 
						|
			return
 | 
						|
		}
 | 
						|
	}
 | 
						|
 | 
						|
	parent.Lock()
 | 
						|
	mem.InitializeDir(parent)
 | 
						|
	mem.AddToMemDir(parent, f)
 | 
						|
	parent.Unlock()
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) lockfreeMkdir(name string, perm os.FileMode) error {
 | 
						|
	name = normalizePath(name)
 | 
						|
	x, ok := m.getData()[name]
 | 
						|
	if ok {
 | 
						|
		// Only return ErrFileExists if it's a file, not a directory.
 | 
						|
		i := mem.FileInfo{FileData: x}
 | 
						|
		if !i.IsDir() {
 | 
						|
			return ErrFileExists
 | 
						|
		}
 | 
						|
	} else {
 | 
						|
		item := mem.CreateDir(name)
 | 
						|
		m.getData()[name] = item
 | 
						|
		m.registerWithParent(item)
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) Mkdir(name string, perm os.FileMode) error {
 | 
						|
	name = normalizePath(name)
 | 
						|
 | 
						|
	m.mu.RLock()
 | 
						|
	_, ok := m.getData()[name]
 | 
						|
	m.mu.RUnlock()
 | 
						|
	if ok {
 | 
						|
		return &os.PathError{Op: "mkdir", Path: name, Err: ErrFileExists}
 | 
						|
	}
 | 
						|
 | 
						|
	m.mu.Lock()
 | 
						|
	item := mem.CreateDir(name)
 | 
						|
	m.getData()[name] = item
 | 
						|
	m.registerWithParent(item)
 | 
						|
	m.mu.Unlock()
 | 
						|
 | 
						|
	m.Chmod(name, perm|os.ModeDir)
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) MkdirAll(path string, perm os.FileMode) error {
 | 
						|
	err := m.Mkdir(path, perm)
 | 
						|
	if err != nil {
 | 
						|
		if err.(*os.PathError).Err == ErrFileExists {
 | 
						|
			return nil
 | 
						|
		}
 | 
						|
		return err
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
// Handle some relative paths
 | 
						|
func normalizePath(path string) string {
 | 
						|
	path = filepath.Clean(path)
 | 
						|
 | 
						|
	switch path {
 | 
						|
	case ".":
 | 
						|
		return FilePathSeparator
 | 
						|
	case "..":
 | 
						|
		return FilePathSeparator
 | 
						|
	default:
 | 
						|
		return path
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) Open(name string) (File, error) {
 | 
						|
	f, err := m.open(name)
 | 
						|
	if f != nil {
 | 
						|
		return mem.NewReadOnlyFileHandle(f), err
 | 
						|
	}
 | 
						|
	return nil, err
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) openWrite(name string) (File, error) {
 | 
						|
	f, err := m.open(name)
 | 
						|
	if f != nil {
 | 
						|
		return mem.NewFileHandle(f), err
 | 
						|
	}
 | 
						|
	return nil, err
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) open(name string) (*mem.FileData, error) {
 | 
						|
	name = normalizePath(name)
 | 
						|
 | 
						|
	m.mu.RLock()
 | 
						|
	f, ok := m.getData()[name]
 | 
						|
	m.mu.RUnlock()
 | 
						|
	if !ok {
 | 
						|
		return nil, &os.PathError{Op: "open", Path: name, Err: ErrFileNotFound}
 | 
						|
	}
 | 
						|
	return f, nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) lockfreeOpen(name string) (*mem.FileData, error) {
 | 
						|
	name = normalizePath(name)
 | 
						|
	f, ok := m.getData()[name]
 | 
						|
	if ok {
 | 
						|
		return f, nil
 | 
						|
	} else {
 | 
						|
		return nil, ErrFileNotFound
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) OpenFile(name string, flag int, perm os.FileMode) (File, error) {
 | 
						|
	chmod := false
 | 
						|
	file, err := m.openWrite(name)
 | 
						|
	if os.IsNotExist(err) && (flag&os.O_CREATE > 0) {
 | 
						|
		file, err = m.Create(name)
 | 
						|
		chmod = true
 | 
						|
	}
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	if flag == os.O_RDONLY {
 | 
						|
		file = mem.NewReadOnlyFileHandle(file.(*mem.File).Data())
 | 
						|
	}
 | 
						|
	if flag&os.O_APPEND > 0 {
 | 
						|
		_, err = file.Seek(0, os.SEEK_END)
 | 
						|
		if err != nil {
 | 
						|
			file.Close()
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if flag&os.O_TRUNC > 0 && flag&(os.O_RDWR|os.O_WRONLY) > 0 {
 | 
						|
		err = file.Truncate(0)
 | 
						|
		if err != nil {
 | 
						|
			file.Close()
 | 
						|
			return nil, err
 | 
						|
		}
 | 
						|
	}
 | 
						|
	if chmod {
 | 
						|
		m.Chmod(name, perm)
 | 
						|
	}
 | 
						|
	return file, nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) Remove(name string) error {
 | 
						|
	name = normalizePath(name)
 | 
						|
 | 
						|
	m.mu.Lock()
 | 
						|
	defer m.mu.Unlock()
 | 
						|
 | 
						|
	if _, ok := m.getData()[name]; ok {
 | 
						|
		err := m.unRegisterWithParent(name)
 | 
						|
		if err != nil {
 | 
						|
			return &os.PathError{Op: "remove", Path: name, Err: err}
 | 
						|
		}
 | 
						|
		delete(m.getData(), name)
 | 
						|
	} else {
 | 
						|
		return &os.PathError{Op: "remove", Path: name, Err: os.ErrNotExist}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) RemoveAll(path string) error {
 | 
						|
	path = normalizePath(path)
 | 
						|
	m.mu.Lock()
 | 
						|
	m.unRegisterWithParent(path)
 | 
						|
	m.mu.Unlock()
 | 
						|
 | 
						|
	m.mu.RLock()
 | 
						|
	defer m.mu.RUnlock()
 | 
						|
 | 
						|
	for p, _ := range m.getData() {
 | 
						|
		if strings.HasPrefix(p, path) {
 | 
						|
			m.mu.RUnlock()
 | 
						|
			m.mu.Lock()
 | 
						|
			delete(m.getData(), p)
 | 
						|
			m.mu.Unlock()
 | 
						|
			m.mu.RLock()
 | 
						|
		}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) Rename(oldname, newname string) error {
 | 
						|
	oldname = normalizePath(oldname)
 | 
						|
	newname = normalizePath(newname)
 | 
						|
 | 
						|
	if oldname == newname {
 | 
						|
		return nil
 | 
						|
	}
 | 
						|
 | 
						|
	m.mu.RLock()
 | 
						|
	defer m.mu.RUnlock()
 | 
						|
	if _, ok := m.getData()[oldname]; ok {
 | 
						|
		m.mu.RUnlock()
 | 
						|
		m.mu.Lock()
 | 
						|
		m.unRegisterWithParent(oldname)
 | 
						|
		fileData := m.getData()[oldname]
 | 
						|
		delete(m.getData(), oldname)
 | 
						|
		mem.ChangeFileName(fileData, newname)
 | 
						|
		m.getData()[newname] = fileData
 | 
						|
		m.registerWithParent(fileData)
 | 
						|
		m.mu.Unlock()
 | 
						|
		m.mu.RLock()
 | 
						|
	} else {
 | 
						|
		return &os.PathError{Op: "rename", Path: oldname, Err: ErrFileNotFound}
 | 
						|
	}
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) Stat(name string) (os.FileInfo, error) {
 | 
						|
	f, err := m.Open(name)
 | 
						|
	if err != nil {
 | 
						|
		return nil, err
 | 
						|
	}
 | 
						|
	fi := mem.GetFileInfo(f.(*mem.File).Data())
 | 
						|
	return fi, nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) Chmod(name string, mode os.FileMode) error {
 | 
						|
	name = normalizePath(name)
 | 
						|
 | 
						|
	m.mu.RLock()
 | 
						|
	f, ok := m.getData()[name]
 | 
						|
	m.mu.RUnlock()
 | 
						|
	if !ok {
 | 
						|
		return &os.PathError{Op: "chmod", Path: name, Err: ErrFileNotFound}
 | 
						|
	}
 | 
						|
 | 
						|
	m.mu.Lock()
 | 
						|
	mem.SetMode(f, mode)
 | 
						|
	m.mu.Unlock()
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) Chtimes(name string, atime time.Time, mtime time.Time) error {
 | 
						|
	name = normalizePath(name)
 | 
						|
 | 
						|
	m.mu.RLock()
 | 
						|
	f, ok := m.getData()[name]
 | 
						|
	m.mu.RUnlock()
 | 
						|
	if !ok {
 | 
						|
		return &os.PathError{Op: "chtimes", Path: name, Err: ErrFileNotFound}
 | 
						|
	}
 | 
						|
 | 
						|
	m.mu.Lock()
 | 
						|
	mem.SetModTime(f, mtime)
 | 
						|
	m.mu.Unlock()
 | 
						|
 | 
						|
	return nil
 | 
						|
}
 | 
						|
 | 
						|
func (m *MemMapFs) List() {
 | 
						|
	for _, x := range m.data {
 | 
						|
		y := mem.FileInfo{FileData: x}
 | 
						|
		fmt.Println(x.Name(), y.Size())
 | 
						|
	}
 | 
						|
}
 | 
						|
 | 
						|
// func debugMemMapList(fs Fs) {
 | 
						|
// 	if x, ok := fs.(*MemMapFs); ok {
 | 
						|
// 		x.List()
 | 
						|
// 	}
 | 
						|
// }
 |