package mutator import ( "context" "fmt" "os" "path/filepath" "strings" "github.com/databricks/cli/bundle" "github.com/databricks/cli/bundle/config" "golang.org/x/exp/slices" ) const ExtraIncludePathsKey string = "DATABRICKS_BUNDLE_INCLUDES" // Get extra include paths from environment variable func GetExtraIncludePaths() []string { value, exists := os.LookupEnv(ExtraIncludePathsKey) if !exists { return nil } return strings.Split(value, string(os.PathListSeparator)) } type processRootIncludes struct{} // ProcessRootIncludes expands the patterns in the configuration's include list // into a list of mutators for each matching file. func ProcessRootIncludes() bundle.Mutator { return &processRootIncludes{} } func (m *processRootIncludes) Name() string { return "ProcessRootIncludes" } func (m *processRootIncludes) Apply(ctx context.Context, b *bundle.Bundle) error { var out []bundle.Mutator // Map with files we've already seen to avoid loading them twice. var seen = map[string]bool{} for _, file := range config.FileNames { seen[file] = true } // Maintain list of files in order of files being loaded. // This is stored in the bundle configuration for observability. var files []string // Converts extra include paths from environment variable to relative paths for _, extraIncludePath := range GetExtraIncludePaths() { if filepath.IsAbs(extraIncludePath) { rel, err := filepath.Rel(b.Config.Path, extraIncludePath) if err != nil { return fmt.Errorf("unable to include file '%s': %w", extraIncludePath, err) } extraIncludePath = rel } b.Config.Include = append(b.Config.Include, extraIncludePath) } // For each glob, find all files to load. // Ordering of the list of globs is maintained in the output. // For matches that appear in multiple globs, only the first is kept. for _, entry := range b.Config.Include { // Include paths must be relative. if filepath.IsAbs(entry) { return fmt.Errorf("%s: includes must be relative paths", entry) } // Anchor includes to the bundle root path. matches, err := filepath.Glob(filepath.Join(b.Config.Path, entry)) if err != nil { return err } // If the entry is not a glob pattern and no matches found, // return an error because the file defined is not found if len(matches) == 0 && !strings.ContainsAny(entry, "*?[") { return fmt.Errorf("%s defined in 'include' section does not match any files", entry) } // Filter matches to ones we haven't seen yet. var includes []string for _, match := range matches { rel, err := filepath.Rel(b.Config.Path, match) if err != nil { return err } if _, ok := seen[rel]; ok { continue } seen[rel] = true includes = append(includes, rel) } // Add matches to list of mutators to return. slices.Sort(includes) files = append(files, includes...) for _, include := range includes { out = append(out, ProcessInclude(filepath.Join(b.Config.Path, include), include)) } } // Swap out the original includes list with the expanded globs. b.Config.Include = files return bundle.Apply(ctx, b, bundle.Seq(out...)) }