139 lines
3.6 KiB
Go
139 lines
3.6 KiB
Go
|
// Copyright 2020 The Go Authors. All rights reserved.
|
||
|
// Use of this source code is governed by a BSD-style
|
||
|
// license that can be found in the LICENSE file.
|
||
|
|
||
|
package fs
|
||
|
|
||
|
import (
|
||
|
"errors"
|
||
|
"path"
|
||
|
)
|
||
|
|
||
|
// A SubFS is a file system with a Sub method.
|
||
|
type SubFS interface {
|
||
|
FS
|
||
|
|
||
|
// Sub returns an FS corresponding to the subtree rooted at dir.
|
||
|
Sub(dir string) (FS, error)
|
||
|
}
|
||
|
|
||
|
// Sub returns an FS corresponding to the subtree rooted at fsys's dir.
|
||
|
//
|
||
|
// If dir is ".", Sub returns fsys unchanged.
|
||
|
// Otherwise, if fs implements SubFS, Sub returns fsys.Sub(dir).
|
||
|
// Otherwise, Sub returns a new FS implementation sub that,
|
||
|
// in effect, implements sub.Open(name) as fsys.Open(path.Join(dir, name)).
|
||
|
// The implementation also translates calls to ReadDir, ReadFile, and Glob appropriately.
|
||
|
//
|
||
|
// Note that Sub(os.DirFS("/"), "prefix") is equivalent to os.DirFS("/prefix")
|
||
|
// and that neither of them guarantees to avoid operating system
|
||
|
// accesses outside "/prefix", because the implementation of os.DirFS
|
||
|
// does not check for symbolic links inside "/prefix" that point to
|
||
|
// other directories. That is, os.DirFS is not a general substitute for a
|
||
|
// chroot-style security mechanism, and Sub does not change that fact.
|
||
|
func Sub(fsys FS, dir string) (FS, error) {
|
||
|
if !ValidPath(dir) {
|
||
|
return nil, &PathError{Op: "sub", Path: dir, Err: errors.New("invalid name")}
|
||
|
}
|
||
|
if dir == "." {
|
||
|
return fsys, nil
|
||
|
}
|
||
|
if fsys, ok := fsys.(SubFS); ok {
|
||
|
return fsys.Sub(dir)
|
||
|
}
|
||
|
return &subFS{fsys, dir}, nil
|
||
|
}
|
||
|
|
||
|
type subFS struct {
|
||
|
fsys FS
|
||
|
dir string
|
||
|
}
|
||
|
|
||
|
// fullName maps name to the fully-qualified name dir/name.
|
||
|
func (f *subFS) fullName(op string, name string) (string, error) {
|
||
|
if !ValidPath(name) {
|
||
|
return "", &PathError{Op: op, Path: name, Err: errors.New("invalid name")}
|
||
|
}
|
||
|
return path.Join(f.dir, name), nil
|
||
|
}
|
||
|
|
||
|
// shorten maps name, which should start with f.dir, back to the suffix after f.dir.
|
||
|
func (f *subFS) shorten(name string) (rel string, ok bool) {
|
||
|
if name == f.dir {
|
||
|
return ".", true
|
||
|
}
|
||
|
if len(name) >= len(f.dir)+2 && name[len(f.dir)] == '/' && name[:len(f.dir)] == f.dir {
|
||
|
return name[len(f.dir)+1:], true
|
||
|
}
|
||
|
return "", false
|
||
|
}
|
||
|
|
||
|
// fixErr shortens any reported names in PathErrors by stripping f.dir.
|
||
|
func (f *subFS) fixErr(err error) error {
|
||
|
if e, ok := err.(*PathError); ok {
|
||
|
if short, ok := f.shorten(e.Path); ok {
|
||
|
e.Path = short
|
||
|
}
|
||
|
}
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
func (f *subFS) Open(name string) (File, error) {
|
||
|
full, err := f.fullName("open", name)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
file, err := f.fsys.Open(full)
|
||
|
return file, f.fixErr(err)
|
||
|
}
|
||
|
|
||
|
func (f *subFS) ReadDir(name string) ([]DirEntry, error) {
|
||
|
full, err := f.fullName("read", name)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
dir, err := ReadDir(f.fsys, full)
|
||
|
return dir, f.fixErr(err)
|
||
|
}
|
||
|
|
||
|
func (f *subFS) ReadFile(name string) ([]byte, error) {
|
||
|
full, err := f.fullName("read", name)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
data, err := ReadFile(f.fsys, full)
|
||
|
return data, f.fixErr(err)
|
||
|
}
|
||
|
|
||
|
func (f *subFS) Glob(pattern string) ([]string, error) {
|
||
|
// Check pattern is well-formed.
|
||
|
if _, err := path.Match(pattern, ""); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
if pattern == "." {
|
||
|
return []string{"."}, nil
|
||
|
}
|
||
|
|
||
|
full := f.dir + "/" + pattern
|
||
|
list, err := Glob(f.fsys, full)
|
||
|
for i, name := range list {
|
||
|
name, ok := f.shorten(name)
|
||
|
if !ok {
|
||
|
return nil, errors.New("invalid result from inner fsys Glob: " + name + " not in " + f.dir) // can't use fmt in this package
|
||
|
}
|
||
|
list[i] = name
|
||
|
}
|
||
|
return list, f.fixErr(err)
|
||
|
}
|
||
|
|
||
|
func (f *subFS) Sub(dir string) (FS, error) {
|
||
|
if dir == "." {
|
||
|
return f, nil
|
||
|
}
|
||
|
full, err := f.fullName("sub", dir)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return &subFS{f.fsys, full}, nil
|
||
|
}
|