mirror of
https://github.com/grafana/grafana.git
synced 2025-01-09 15:43:23 -06:00
7c5210a915
* Fix files with two dots in the name not being returned by LocalFS.Files() * Renamed variable for consistency * Add test * Fix typo * Fix wrong upperLevelPrefix value * Removed unnecessary check in LocalFS.Files()
288 lines
8.8 KiB
Go
288 lines
8.8 KiB
Go
package plugins
|
|
|
|
import (
|
|
"errors"
|
|
"fmt"
|
|
"io/fs"
|
|
"os"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/grafana/grafana/pkg/util"
|
|
)
|
|
|
|
var (
|
|
_ fs.File = &LocalFile{}
|
|
|
|
_ FS = &LocalFS{}
|
|
_ FS = &StaticFS{}
|
|
)
|
|
|
|
// LocalFS is a plugins.FS that allows accessing files on the local file system.
|
|
type LocalFS struct {
|
|
// basePath is the basePath that will be prepended to all the files to get their absolute path.
|
|
basePath string
|
|
}
|
|
|
|
// NewLocalFS returns a new LocalFS that can access any file in the specified base path on the filesystem.
|
|
// basePath must use os-specific path separator for Open() to work properly.
|
|
func NewLocalFS(basePath string) LocalFS {
|
|
return LocalFS{basePath: basePath}
|
|
}
|
|
|
|
// fileIsAllowed takes an absolute path to a file and an os.FileInfo for that file, and it checks if access to that
|
|
// file is allowed or not. Access to a file is allowed if the file is in the FS's Base() directory, and if it's a
|
|
// symbolic link it should not end up outside the plugin's directory.
|
|
func (f LocalFS) fileIsAllowed(basePath string, absolutePath string, info os.FileInfo) (bool, error) {
|
|
upperLevelPrefix := ".." + string(filepath.Separator)
|
|
if info.Mode()&os.ModeSymlink == os.ModeSymlink {
|
|
symlinkPath, err := filepath.EvalSymlinks(absolutePath)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
symlink, err := os.Stat(symlinkPath)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
|
|
// verify that symlinked file is within plugin directory
|
|
p, err := filepath.Rel(basePath, symlinkPath)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if p == ".." || strings.HasPrefix(p, upperLevelPrefix) {
|
|
return false, fmt.Errorf("file '%s' not inside of plugin directory", p)
|
|
}
|
|
|
|
// skip adding symlinked directories
|
|
if symlink.IsDir() {
|
|
return false, nil
|
|
}
|
|
}
|
|
|
|
// skip directories
|
|
if info.IsDir() {
|
|
return false, nil
|
|
}
|
|
|
|
// verify that file is within plugin directory
|
|
file, err := filepath.Rel(f.Base(), absolutePath)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if strings.HasPrefix(file, upperLevelPrefix) {
|
|
return false, fmt.Errorf("file '%s' not inside of plugin directory", file)
|
|
}
|
|
return true, nil
|
|
}
|
|
|
|
// walkFunc returns a filepath.WalkFunc that accumulates absolute file paths into acc by walking over f.Base().
|
|
// f.fileIsAllowed is used as WalkFunc, see its documentation for more information on which files are collected.
|
|
func (f LocalFS) walkFunc(basePath string, acc map[string]struct{}) filepath.WalkFunc {
|
|
return func(path string, info os.FileInfo, err error) error {
|
|
if err != nil {
|
|
return err
|
|
}
|
|
ok, err := f.fileIsAllowed(basePath, path, info)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !ok {
|
|
return nil
|
|
}
|
|
acc[path] = struct{}{}
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// Open opens the specified file on the local filesystem.
|
|
// The provided name must be a relative file name that uses os-specific path separators.
|
|
// The function returns the corresponding fs.File for accessing the file on the local filesystem.
|
|
// If a nil error is returned, the caller should take care of calling Close() the returned fs.File.
|
|
// If the file does not exist, ErrFileNotExist is returned.
|
|
func (f LocalFS) Open(name string) (fs.File, error) {
|
|
cleanPath, err := util.CleanRelativePath(name)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
basePath := f.Base()
|
|
absFn := filepath.Join(basePath, cleanPath)
|
|
finfo, err := os.Stat(absFn)
|
|
if err != nil {
|
|
return nil, ErrFileNotExist
|
|
}
|
|
// Make sure access to the file is allowed (symlink check, etc)
|
|
ok, err := f.fileIsAllowed(basePath, absFn, finfo)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !ok {
|
|
return nil, ErrFileNotExist
|
|
}
|
|
return &LocalFile{path: absFn}, nil
|
|
}
|
|
|
|
// Base returns the base path for the LocalFS.
|
|
// The returned string uses os-specific path separator.
|
|
func (f LocalFS) Base() string {
|
|
return f.basePath
|
|
}
|
|
|
|
// Files returns a slice of all the relative file paths on the LocalFS.
|
|
// The returned strings can be passed to Open() to open those files.
|
|
// The returned strings use os-specific path separator.
|
|
func (f LocalFS) Files() ([]string, error) {
|
|
// Accumulate all files into filesMap by calling f.collectFilesFunc, which will write into the accumulator.
|
|
// Those are absolute because filepath.Walk uses absolute file paths.
|
|
absFilePaths := make(map[string]struct{})
|
|
if err := filepath.Walk(f.basePath, f.walkFunc(f.Base(), absFilePaths)); err != nil {
|
|
return nil, fmt.Errorf("walk: %w", err)
|
|
}
|
|
// Convert the accumulator into a slice of relative path strings
|
|
relFiles := make([]string, 0, len(absFilePaths))
|
|
base := f.Base()
|
|
for fn := range absFilePaths {
|
|
relPath, err := filepath.Rel(base, fn)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
clenRelPath, err := util.CleanRelativePath(relPath)
|
|
if err != nil {
|
|
continue
|
|
}
|
|
relFiles = append(relFiles, clenRelPath)
|
|
}
|
|
return relFiles, nil
|
|
}
|
|
|
|
// Remove removes a plugin from the local filesystem by deleting all files in the folder.
|
|
// It returns ErrUninstallInvalidPluginDir is the plugin does not contain plugin.json nor dist/plugin.json.
|
|
func (f LocalFS) Remove() error {
|
|
// extra security check to ensure we only remove a directory that looks like a plugin
|
|
if _, err := os.Stat(filepath.Join(f.basePath, "plugin.json")); os.IsNotExist(err) {
|
|
if _, err = os.Stat(filepath.Join(f.basePath, "dist/plugin.json")); os.IsNotExist(err) {
|
|
return ErrUninstallInvalidPluginDir
|
|
}
|
|
}
|
|
return os.RemoveAll(f.basePath)
|
|
}
|
|
|
|
// staticFilesMap is a set-like map that contains files that can be accessed from a plugins.FS.
|
|
type staticFilesMap map[string]struct{}
|
|
|
|
// isAllowed returns true if the provided path is allowed.
|
|
// path is a string accepted by an FS Open() method.
|
|
func (a staticFilesMap) isAllowed(path string) bool {
|
|
_, ok := a[path]
|
|
return ok
|
|
}
|
|
|
|
// newStaticFilesMap creates a new staticFilesMap from a list of allowed file paths.
|
|
func newStaticFilesMap(files ...string) staticFilesMap {
|
|
m := staticFilesMap(make(map[string]struct{}, len(files)))
|
|
for _, k := range files {
|
|
m[k] = struct{}{}
|
|
}
|
|
return m
|
|
}
|
|
|
|
// StaticFS wraps an FS and allows accessing only the files in the allowList.
|
|
// This is a more secure implementation of a FS suitable for production environments.
|
|
// The keys of the allow list must be in the same format used by the underlying FS' Open() method.
|
|
type StaticFS struct {
|
|
FS
|
|
|
|
// staticFilesMap is a map of allowed paths (accepted by FS.Open())
|
|
staticFilesMap staticFilesMap
|
|
}
|
|
|
|
// NewStaticFS returns a new StaticFS that can access the files on an underlying FS,
|
|
// but only if they are also specified in a static list, which is constructed when creating the object
|
|
// by calling Files() on the underlying FS.
|
|
func NewStaticFS(fs FS) (StaticFS, error) {
|
|
files, err := fs.Files()
|
|
if err != nil {
|
|
return StaticFS{}, err
|
|
}
|
|
return StaticFS{
|
|
FS: fs,
|
|
staticFilesMap: newStaticFilesMap(files...),
|
|
}, nil
|
|
}
|
|
|
|
// Open checks that name is an allowed file and, if so, it returns a fs.File to access it, by calling the
|
|
// underlying FS' Open() method.
|
|
// If access is denied, the function returns ErrFileNotExist.
|
|
func (f StaticFS) Open(name string) (fs.File, error) {
|
|
// Ensure access to the file is allowed
|
|
if !f.staticFilesMap.isAllowed(name) {
|
|
return nil, ErrFileNotExist
|
|
}
|
|
// Use the wrapped FS to access the file
|
|
return f.FS.Open(name)
|
|
}
|
|
|
|
// Files returns a slice of all static file paths relative to the base path.
|
|
func (f StaticFS) Files() ([]string, error) {
|
|
files := make([]string, 0, len(f.staticFilesMap))
|
|
for fn := range f.staticFilesMap {
|
|
files = append(files, fn)
|
|
}
|
|
return files, nil
|
|
}
|
|
|
|
// LocalFile implements a fs.File for accessing the local filesystem.
|
|
type LocalFile struct {
|
|
f *os.File
|
|
path string
|
|
}
|
|
|
|
// Stat returns a FileInfo describing the named file.
|
|
// It returns ErrFileNotExist if the file does not exist, or ErrPluginFileRead if another error occurs.
|
|
func (p *LocalFile) Stat() (fs.FileInfo, error) {
|
|
fi, err := os.Stat(p.path)
|
|
if err != nil {
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
return nil, ErrFileNotExist
|
|
}
|
|
return nil, ErrPluginFileRead
|
|
}
|
|
return fi, nil
|
|
}
|
|
|
|
// Read reads up to len(b) bytes from the File and stores them in b.
|
|
// It returns the number of bytes read and any error encountered.
|
|
// At end of file, Read returns 0, io.EOF.
|
|
// If the file is already open, it is opened again, without closing it first.
|
|
// The file is not closed at the end of the read operation. If a non-nil error is returned, it
|
|
// must be manually closed by the caller by calling Close().
|
|
func (p *LocalFile) Read(b []byte) (int, error) {
|
|
if p.f != nil {
|
|
// File is already open, Read() can be called more than once.
|
|
// io.EOF is returned if the file has been read entirely.
|
|
return p.f.Read(b)
|
|
}
|
|
|
|
var err error
|
|
p.f, err = os.Open(p.path)
|
|
if err != nil {
|
|
if errors.Is(err, fs.ErrNotExist) {
|
|
return 0, ErrFileNotExist
|
|
}
|
|
return 0, ErrPluginFileRead
|
|
}
|
|
return p.f.Read(b)
|
|
}
|
|
|
|
// Close closes the file.
|
|
// If the file was never open, nil is returned.
|
|
// If the file is already closed, an error is returned.
|
|
func (p *LocalFile) Close() error {
|
|
if p.f != nil {
|
|
return p.f.Close()
|
|
}
|
|
p.f = nil
|
|
return nil
|
|
}
|