mirror of https://github.com/databricks/cli.git
217 lines
4.8 KiB
Go
217 lines
4.8 KiB
Go
package libraries
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/databricks/cli/bundle"
|
|
"github.com/databricks/cli/libs/diag"
|
|
"github.com/databricks/cli/libs/dyn"
|
|
)
|
|
|
|
type expand struct{}
|
|
|
|
func matchError(p dyn.Path, l []dyn.Location, message string) diag.Diagnostic {
|
|
return diag.Diagnostic{
|
|
Severity: diag.Error,
|
|
Summary: message,
|
|
Locations: l,
|
|
Paths: []dyn.Path{p},
|
|
}
|
|
}
|
|
|
|
func getLibDetails(v dyn.Value) (string, string, bool) {
|
|
m := v.MustMap()
|
|
whl, ok := m.GetByString("whl")
|
|
if ok {
|
|
return whl.MustString(), "whl", true
|
|
}
|
|
|
|
jar, ok := m.GetByString("jar")
|
|
if ok {
|
|
return jar.MustString(), "jar", true
|
|
}
|
|
|
|
return "", "", false
|
|
}
|
|
|
|
func findMatches(b *bundle.Bundle, path string) ([]string, error) {
|
|
matches, err := filepath.Glob(filepath.Join(b.SyncRootPath, path))
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if len(matches) == 0 {
|
|
if isGlobPattern(path) {
|
|
return nil, fmt.Errorf("no files match pattern: %s", path)
|
|
} else {
|
|
return nil, fmt.Errorf("file doesn't exist %s", path)
|
|
}
|
|
}
|
|
|
|
// We make the matched path relative to the sync root path before storing it
|
|
// to allow upload mutator to distinguish between local and remote paths
|
|
for i, match := range matches {
|
|
matches[i], err = filepath.Rel(b.SyncRootPath, match)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
}
|
|
|
|
return matches, nil
|
|
}
|
|
|
|
// Checks if the path is a glob pattern
|
|
// It can contain *, [] or ? characters
|
|
func isGlobPattern(path string) bool {
|
|
return strings.ContainsAny(path, "*?[")
|
|
}
|
|
|
|
func expandLibraries(b *bundle.Bundle, p dyn.Path, v dyn.Value) (diag.Diagnostics, []dyn.Value) {
|
|
var output []dyn.Value
|
|
var diags diag.Diagnostics
|
|
|
|
libs := v.MustSequence()
|
|
for i, lib := range libs {
|
|
lp := p.Append(dyn.Index(i))
|
|
path, libType, supported := getLibDetails(lib)
|
|
if !supported || !IsLibraryLocal(path) {
|
|
output = append(output, lib)
|
|
continue
|
|
}
|
|
|
|
lp = lp.Append(dyn.Key(libType))
|
|
|
|
matches, err := findMatches(b, path)
|
|
if err != nil {
|
|
diags = diags.Append(matchError(lp, lib.Locations(), err.Error()))
|
|
continue
|
|
}
|
|
|
|
for _, match := range matches {
|
|
output = append(output, dyn.NewValue(map[string]dyn.Value{
|
|
libType: dyn.V(match),
|
|
}, lib.Locations()))
|
|
}
|
|
}
|
|
|
|
return diags, output
|
|
}
|
|
|
|
func expandEnvironmentDeps(b *bundle.Bundle, p dyn.Path, v dyn.Value) (diag.Diagnostics, []dyn.Value) {
|
|
var output []dyn.Value
|
|
var diags diag.Diagnostics
|
|
|
|
deps := v.MustSequence()
|
|
for i, dep := range deps {
|
|
lp := p.Append(dyn.Index(i))
|
|
path := dep.MustString()
|
|
if !IsLibraryLocal(path) {
|
|
output = append(output, dep)
|
|
continue
|
|
}
|
|
|
|
matches, err := findMatches(b, path)
|
|
if err != nil {
|
|
diags = diags.Append(matchError(lp, dep.Locations(), err.Error()))
|
|
continue
|
|
}
|
|
|
|
for _, match := range matches {
|
|
output = append(output, dyn.NewValue(match, dep.Locations()))
|
|
}
|
|
}
|
|
|
|
return diags, output
|
|
}
|
|
|
|
type expandPattern struct {
|
|
pattern dyn.Pattern
|
|
fn func(b *bundle.Bundle, p dyn.Path, v dyn.Value) (diag.Diagnostics, []dyn.Value)
|
|
}
|
|
|
|
var taskLibrariesPattern = dyn.NewPattern(
|
|
dyn.Key("resources"),
|
|
dyn.Key("jobs"),
|
|
dyn.AnyKey(),
|
|
dyn.Key("tasks"),
|
|
dyn.AnyIndex(),
|
|
dyn.Key("libraries"),
|
|
)
|
|
|
|
var forEachTaskLibrariesPattern = dyn.NewPattern(
|
|
dyn.Key("resources"),
|
|
dyn.Key("jobs"),
|
|
dyn.AnyKey(),
|
|
dyn.Key("tasks"),
|
|
dyn.AnyIndex(),
|
|
dyn.Key("for_each_task"),
|
|
dyn.Key("task"),
|
|
dyn.Key("libraries"),
|
|
)
|
|
|
|
var envDepsPattern = dyn.NewPattern(
|
|
dyn.Key("resources"),
|
|
dyn.Key("jobs"),
|
|
dyn.AnyKey(),
|
|
dyn.Key("environments"),
|
|
dyn.AnyIndex(),
|
|
dyn.Key("spec"),
|
|
dyn.Key("dependencies"),
|
|
)
|
|
|
|
func (e *expand) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
expanders := []expandPattern{
|
|
{
|
|
pattern: taskLibrariesPattern,
|
|
fn: expandLibraries,
|
|
},
|
|
{
|
|
pattern: forEachTaskLibrariesPattern,
|
|
fn: expandLibraries,
|
|
},
|
|
{
|
|
pattern: envDepsPattern,
|
|
fn: expandEnvironmentDeps,
|
|
},
|
|
}
|
|
|
|
var diags diag.Diagnostics
|
|
|
|
err := b.Config.Mutate(func(v dyn.Value) (dyn.Value, error) {
|
|
var err error
|
|
for _, expander := range expanders {
|
|
v, err = dyn.MapByPattern(v, expander.pattern, func(p dyn.Path, lv dyn.Value) (dyn.Value, error) {
|
|
d, output := expander.fn(b, p, lv)
|
|
diags = diags.Extend(d)
|
|
return dyn.V(output), nil
|
|
})
|
|
if err != nil {
|
|
return dyn.InvalidValue, err
|
|
}
|
|
}
|
|
|
|
return v, nil
|
|
})
|
|
if err != nil {
|
|
diags = diags.Extend(diag.FromErr(err))
|
|
}
|
|
|
|
return diags
|
|
}
|
|
|
|
func (e *expand) Name() string {
|
|
return "libraries.ExpandGlobReferences"
|
|
}
|
|
|
|
// ExpandGlobReferences expands any glob references in the libraries or environments section
|
|
// to corresponding local paths.
|
|
// We only expand local paths (i.e. paths that are relative to the sync root path).
|
|
// After expanding we make the paths relative to the sync root path to allow upload mutator later in the chain to
|
|
// distinguish between local and remote paths.
|
|
func ExpandGlobReferences() bundle.Mutator {
|
|
return &expand{}
|
|
}
|