| // Copyright 2020 The Fuchsia 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 checklicenses |
| |
| import ( |
| "context" |
| "encoding/json" |
| "fmt" |
| "io" |
| "log" |
| "os" |
| "path/filepath" |
| "runtime/trace" |
| "strings" |
| "sync" |
| ) |
| |
| // FileTree is an in memory representation of the state of the repository. |
| type FileTree struct { |
| Name string `json:"name"` |
| Path string `json:"path"` |
| SingleLicenseFiles map[string][]*License `json:"project licenses"` |
| Files []*File `json:"files"` |
| Children map[string]*FileTree `json:"children"` |
| Parent *FileTree `json:"-"` |
| StrictAnalysis bool `json:"-"` |
| |
| sync.RWMutex |
| } |
| |
| // NewFileTree returns an instance of FileTree, given the input configuration |
| // file. |
| func NewFileTree(ctx context.Context, root string, parent *FileTree, config *Config, metrics *Metrics) *FileTree { |
| defer trace.StartRegion(ctx, "NewFileTree").End() |
| ft := FileTree{ |
| Children: make(map[string]*FileTree), |
| SingleLicenseFiles: make(map[string][]*License), |
| } |
| |
| abs, _ := filepath.Abs(root) |
| ft.Name = filepath.Base(abs) |
| ft.Path = abs |
| ft.Parent = parent |
| |
| // If config.StrictAnalysis is true, we ignore all LICENSE files in the fuchsia directory. |
| if parent == nil { |
| ft.StrictAnalysis = config.StrictAnalysis |
| } else { |
| ft.StrictAnalysis = parent.StrictAnalysis |
| } |
| |
| // If we are at a boundary where licenses change (e.g. "third_party" or "prebuilt" dirs), |
| // turn off strict analysis. We don't have enough control over 3p repositories to enforce |
| // having strict LICENSE information in all source files. |
| for _, dirName := range config.StopLicensePropagation { |
| if ft.Name == dirName { |
| ft.StrictAnalysis = false |
| break |
| } |
| } |
| |
| for _, customProjectLicense := range config.CustomProjectLicenses { |
| if strings.HasSuffix(root, customProjectLicense.ProjectRoot) { |
| metrics.increment("num_single_license_files") |
| licLocation := filepath.Join(root, customProjectLicense.LicenseLocation) |
| ft.SingleLicenseFiles[licLocation] = []*License{} |
| break |
| } |
| } |
| |
| err := filepath.Walk(root, func(path string, info os.FileInfo, err error) error { |
| if err != nil { |
| return err |
| } |
| if info.IsDir() { |
| for _, skipDir := range config.SkipDirs { |
| if info.Name() == skipDir || path == skipDir { |
| log.Printf("skipping: %s", path) |
| return filepath.SkipDir |
| } |
| } |
| if path != root { |
| child := NewFileTree(ctx, path, &ft, config, metrics) |
| ft.Children[path] = child |
| return filepath.SkipDir |
| } |
| } |
| |
| if info.Size() == 0 { |
| // An empty file has no content to copyright. Skip. |
| return nil |
| } |
| for _, skipFile := range config.SkipFiles { |
| if strings.ToLower(info.Name()) == skipFile || strings.ToLower(path) == skipFile { |
| log.Printf("skipping: %s", path) |
| return nil |
| } |
| } |
| newFile, err := NewFile(path, &ft) |
| |
| // TODO(jcecil): a file named LICENSE in the fuchsia tree will be |
| // entirely skipped when running in strict analysis mode, since it |
| // doesn't have a valid text extension. We should still analyze |
| // these files, even if we don't add them as SingleLicenseFiles. |
| if hasLowerPrefix(info.Name(), config.SingleLicenseFiles) && !ft.StrictAnalysis { |
| metrics.increment("num_single_license_files") |
| ft.SingleLicenseFiles[path] = []*License{} |
| return nil |
| } |
| extensions := config.TextExtensionList |
| if ft.StrictAnalysis { |
| extensions = config.StrictTextExtensionList |
| } |
| if hasExt(info.Name(), extensions) { |
| metrics.increment("num_non_single_license_files") |
| if err == nil { |
| ft.Files = append(ft.Files, newFile) |
| } |
| } else { |
| log.Printf("ignoring: %s", path) |
| metrics.increment("num_extensions_excluded") |
| } |
| return nil |
| }) |
| if err != nil { |
| // TODO(jcecil): This must be an error. |
| fmt.Printf("error while traversing directory '%v", err) |
| return nil |
| } |
| |
| return &ft |
| } |
| |
| func (ft *FileTree) propagateProjectLicenses(config *Config) { |
| propagate := true |
| for _, dirName := range config.StopLicensePropagation { |
| if ft.Name == dirName { |
| propagate = false |
| break |
| } |
| } |
| |
| if propagate && ft.Parent != nil { |
| for key, val := range ft.Parent.SingleLicenseFiles { |
| ft.SingleLicenseFiles[key] = val |
| } |
| } |
| |
| for _, child := range ft.Children { |
| child.propagateProjectLicenses(config) |
| } |
| } |
| |
| func (ft *FileTree) getSingleLicenseFileIterator() <-chan *FileTree { |
| ch := make(chan *FileTree, 1) |
| go func() { |
| var curr *FileTree |
| var q []*FileTree |
| q = append(q, ft) |
| var pos int |
| for len(q) > 0 { |
| pos = len(q) - 1 |
| curr = q[pos] |
| q = q[:pos] |
| if len(curr.SingleLicenseFiles) > 0 { |
| ch <- curr |
| } |
| curr.RLock() |
| for _, child := range curr.Children { |
| q = append(q, child) |
| } |
| curr.RUnlock() |
| } |
| close(ch) |
| }() |
| return ch |
| } |
| |
| func (ft *FileTree) getFileIterator() <-chan *File { |
| ch := make(chan *File, 1) |
| go func() { |
| var curr *FileTree |
| var q []*FileTree |
| q = append(q, ft) |
| var pos int |
| for len(q) > 0 { |
| pos = len(q) - 1 |
| curr = q[pos] |
| q = q[:pos] |
| for _, file := range curr.Files { |
| ch <- file |
| } |
| curr.RLock() |
| for _, child := range curr.Children { |
| q = append(q, child) |
| } |
| curr.RUnlock() |
| } |
| close(ch) |
| }() |
| return ch |
| } |
| |
| // Maps are used in FileTree to prevent duplicate values (since go doesn't have sets). |
| // However, Maps make the final JSON object difficult to read. |
| // Define a custom MarshalJSON function to convert the internal Maps into slices. |
| func (ft *FileTree) MarshalJSON() ([]byte, error) { |
| type Alias FileTree |
| childrenList := []*FileTree{} |
| fileList := []string{} |
| |
| for _, c := range ft.Children { |
| childrenList = append(childrenList, c) |
| } |
| |
| for _, f := range ft.Files { |
| fileList = append(fileList, f.Name) |
| } |
| |
| return json.Marshal(&struct { |
| *Alias |
| Children []*FileTree `json:"children"` |
| }{ |
| Alias: (*Alias)(ft), |
| Children: childrenList, |
| }) |
| } |
| |
| func (ft *FileTree) saveTreeState(filename string) error { |
| jsonString, err := json.MarshalIndent(ft, "", " ") |
| if err != nil { |
| return fmt.Errorf("error marshalling the file tree: %v\n", err) |
| } |
| |
| file, err := os.Create(filename) |
| if err != nil { |
| return err |
| } |
| defer file.Close() |
| _, err = io.WriteString(file, string(jsonString)) |
| if err != nil { |
| return err |
| } |
| return nil |
| } |
| |
| func (ft *FileTree) Equal(other *FileTree) bool { |
| if ft.Name != other.Name { |
| return false |
| } |
| if ft.Path != other.Path { |
| return false |
| } |
| if ft.Parent != other.Parent { |
| return false |
| } |
| if ft.StrictAnalysis != other.StrictAnalysis { |
| return false |
| } |
| |
| if len(ft.SingleLicenseFiles) != len(other.SingleLicenseFiles) { |
| return false |
| } |
| for k := range ft.SingleLicenseFiles { |
| left := ft.SingleLicenseFiles[k] |
| right := other.SingleLicenseFiles[k] |
| if len(left) != len(right) { |
| return false |
| } |
| for i := range left { |
| if left[i] != right[i] { |
| return false |
| } |
| } |
| } |
| |
| if len(ft.Files) != len(other.Files) { |
| return false |
| } |
| for i := range ft.Files { |
| if !ft.Files[i].Equal(other.Files[i]) { |
| return false |
| } |
| } |
| |
| if len(ft.Children) != len(other.Children) { |
| return false |
| } |
| for k := range ft.Children { |
| if ft.Children[k] != other.Children[k] { |
| return false |
| } |
| } |
| |
| return true |
| } |
| |
| // hasExt returns true if path has one of the extensions in the list. |
| func hasExt(path string, exts []string) bool { |
| if ext := filepath.Ext(path); ext != "" { |
| ext = ext[1:] |
| for _, e := range exts { |
| if e == ext { |
| return true |
| } |
| } |
| } |
| return false |
| } |
| |
| // hasLowerPrefix returns true if the name has one of files as a prefix in |
| // lower case. |
| func hasLowerPrefix(name string, files []string) bool { |
| name = strings.ToLower(name) |
| for _, f := range files { |
| if strings.HasPrefix(name, f) { |
| return true |
| } |
| } |
| return false |
| } |