// Copyright 2021 The Hugo Authors. All rights reserved. // // 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 hugofs import ( "io/fs" "os" "strings" "syscall" "time" "github.com/gohugoio/hugo/hugofs/glob" "github.com/spf13/afero" ) var _ FilesystemUnwrapper = (*filenameFilterFs)(nil) func newFilenameFilterFs(fs afero.Fs, base string, filter *glob.FilenameFilter) afero.Fs { return &filenameFilterFs{ fs: fs, base: base, filter: filter, } } // filenameFilterFs is a filesystem that filters by filename. type filenameFilterFs struct { base string fs afero.Fs filter *glob.FilenameFilter } func (fs *filenameFilterFs) UnwrapFilesystem() afero.Fs { return fs.fs } func (fs *filenameFilterFs) Open(name string) (afero.File, error) { fi, err := fs.fs.Stat(name) if err != nil { return nil, err } if !fs.filter.Match(name, fi.IsDir()) { return nil, os.ErrNotExist } f, err := fs.fs.Open(name) if err != nil { return nil, err } if !fi.IsDir() { return f, nil } return &filenameFilterDir{ File: f, base: fs.base, filter: fs.filter, }, nil } func (fs *filenameFilterFs) OpenFile(name string, flag int, perm os.FileMode) (afero.File, error) { return fs.Open(name) } func (fs *filenameFilterFs) Stat(name string) (os.FileInfo, error) { fi, err := fs.fs.Stat(name) if err != nil { return nil, err } if !fs.filter.Match(name, fi.IsDir()) { return nil, os.ErrNotExist } return fi, nil } type filenameFilterDir struct { afero.File base string filter *glob.FilenameFilter } func (f *filenameFilterDir) ReadDir(n int) ([]fs.DirEntry, error) { des, err := f.File.(fs.ReadDirFile).ReadDir(n) if err != nil { return nil, err } i := 0 for _, de := range des { fim := de.(FileMetaInfo) rel := strings.TrimPrefix(fim.Meta().Filename, f.base) if f.filter.Match(rel, de.IsDir()) { des[i] = de i++ } } return des[:i], nil } func (f *filenameFilterDir) Readdir(count int) ([]os.FileInfo, error) { panic("not supported: Use ReadDir") } func (f *filenameFilterDir) Readdirnames(count int) ([]string, error) { des, err := f.ReadDir(count) if err != nil { return nil, err } dirs := make([]string, len(des)) for i, d := range des { dirs[i] = d.Name() } return dirs, nil } func (fs *filenameFilterFs) Chmod(n string, m os.FileMode) error { return syscall.EPERM } func (fs *filenameFilterFs) Chtimes(n string, a, m time.Time) error { return syscall.EPERM } func (fs *filenameFilterFs) Chown(n string, uid, gid int) error { return syscall.EPERM } func (fs *filenameFilterFs) ReadDir(name string) ([]os.FileInfo, error) { panic("not implemented") } func (fs *filenameFilterFs) Remove(n string) error { return syscall.EPERM } func (fs *filenameFilterFs) RemoveAll(p string) error { return syscall.EPERM } func (fs *filenameFilterFs) Rename(o, n string) error { return syscall.EPERM } func (fs *filenameFilterFs) Create(n string) (afero.File, error) { return nil, syscall.EPERM } func (fs *filenameFilterFs) Name() string { return "FinameFilterFS" } func (fs *filenameFilterFs) Mkdir(n string, p os.FileMode) error { return syscall.EPERM } func (fs *filenameFilterFs) MkdirAll(n string, p os.FileMode) error { return syscall.EPERM }