blob: 680d8d1f5d777b9dcca5ed6f6a2c95bcbbe36038 [file]
// Copyright 2020, 2021, 2021 Google LLC
//
// 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
//
// https://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 runfiles
import (
"io/fs"
"os"
"path"
"path/filepath"
)
// Directory specifies the location of the runfiles directory. You can pass
// this as an option to [New]. If unset or empty, use the value of the
// environmental variable RUNFILES_DIR.
type Directory string
func (d Directory) new(sourceRepo SourceRepo) (*Runfiles, error) {
r := &Runfiles{
impl: d,
env: []string{
directoryVar + "=" + string(d),
legacyDirectoryVar + "=" + string(d),
},
sourceRepo: string(sourceRepo),
}
err := r.loadRepoMapping()
return r, err
}
func (d Directory) path(s string) (string, error) {
return filepath.Join(string(d), filepath.FromSlash(s)), nil
}
func (d Directory) open(name string) (fs.File, error) {
dirFS := os.DirFS(string(d))
f, err := dirFS.Open(name)
if err != nil {
return nil, err
}
return &resolvedFile{f.(*os.File), func(child string) (fs.FileInfo, error) {
return fs.Stat(dirFS, path.Join(name, child))
}}, nil
}
type resolvedFile struct {
fs.ReadDirFile
lstatChildAfterReadlink func(string) (fs.FileInfo, error)
}
func (f *resolvedFile) ReadDir(n int) ([]fs.DirEntry, error) {
entries, err := f.ReadDirFile.ReadDir(n)
if err != nil {
return nil, err
}
for i, entry := range entries {
// Bazel runfiles directories consist of symlinks to the real files, which may themselves
// be directories. We want fs.WalkDir to descend into these directories as it does with the
// manifest implementation. We do this by replacing the information about an entry that is
// a symlink by the info of the resolved file.
if entry.Type()&fs.ModeSymlink != 0 {
info, err := f.lstatChildAfterReadlink(entry.Name())
if err != nil {
return nil, err
}
entries[i] = renamedDirEntry{fs.FileInfoToDirEntry(info), entry.Name()}
}
}
return entries, nil
}