blob: 66f5dc0ea8c5b78c3dc4173712f3a792abd1bd11 [file] [log] [blame] [edit]
/*
Copyright 2020 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 utils contains shared methods that can be used by different implementations of
// buildifier binary
package utils
import (
"os"
"path/filepath"
"strings"
"github.com/bazelbuild/buildtools/build"
"github.com/bazelbuild/buildtools/warn"
)
func isStarlarkFile(name string) bool {
ext := filepath.Ext(name)
switch ext {
case ".bzl", ".sky", ".star":
return true
}
switch ext {
case ".bazel", ".oss":
// BUILD.bazel or BUILD.foo.bazel should be treated as Starlark files, same for WORSKSPACE and MODULE
return strings.HasPrefix(name, "BUILD.") || strings.HasPrefix(name, "WORKSPACE.") || strings.HasPrefix(name, "MODULE.")
}
return name == "BUILD" || name == "WORKSPACE"
}
func skip(info os.FileInfo) bool {
return info.IsDir() && info.Name() == ".git"
}
// ExpandDirectories takes a list of file/directory names and returns a list with file names
// by traversing each directory recursively and searching for relevant Starlark files.
func ExpandDirectories(args *[]string) ([]string, error) {
files := []string{}
for _, arg := range *args {
info, err := os.Stat(arg)
if err != nil {
return []string{}, err
}
if !info.IsDir() {
files = append(files, arg)
continue
}
err = filepath.Walk(arg, func(path string, info os.FileInfo, err error) error {
if err != nil {
return err
}
if skip(info) {
return filepath.SkipDir
}
if !info.IsDir() && isStarlarkFile(info.Name()) {
files = append(files, path)
}
return nil
})
if err != nil {
return []string{}, err
}
}
return files, nil
}
// GetParser returns a parser for a given file type
func GetParser(inputType string) func(filename string, data []byte) (*build.File, error) {
switch inputType {
case "build":
return build.ParseBuild
case "bzl":
return build.ParseBzl
case "auto":
return build.Parse
case "workspace":
return build.ParseWorkspace
case "module":
return build.ParseModule
default:
return build.ParseDefault
}
}
// getFileReader returns a *FileReader object that reads files from the local
// filesystem if the workspace root is known.
func getFileReader(workspaceRoot string) *warn.FileReader {
if workspaceRoot == "" {
return nil
}
readFile := func(filename string) ([]byte, error) {
// Use OS-specific path separators
filename = strings.ReplaceAll(filename, "/", string(os.PathSeparator))
path := filepath.Join(workspaceRoot, filename)
return os.ReadFile(path)
}
return warn.NewFileReader(readFile)
}
// Lint calls the linter and returns a list of unresolved findings
func Lint(f *build.File, lint string, warningsList *[]string, verbose bool) []*warn.Finding {
fileReader := getFileReader(f.WorkspaceRoot)
switch lint {
case "warn":
return warn.FileWarnings(f, *warningsList, nil, warn.ModeWarn, fileReader)
case "fix":
warn.FixWarnings(f, *warningsList, verbose, fileReader)
}
return nil
}