mirror of https://github.com/databricks/cli.git
209 lines
5.7 KiB
Go
209 lines
5.7 KiB
Go
package mutator
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"net/url"
|
|
"os"
|
|
"path"
|
|
"path/filepath"
|
|
"strings"
|
|
|
|
"github.com/databricks/cli/bundle"
|
|
"github.com/databricks/cli/libs/diag"
|
|
"github.com/databricks/cli/libs/dyn"
|
|
"github.com/databricks/cli/libs/notebook"
|
|
)
|
|
|
|
type ErrIsNotebook struct {
|
|
path string
|
|
}
|
|
|
|
func (err ErrIsNotebook) Error() string {
|
|
return fmt.Sprintf("file at %s is a notebook", err.path)
|
|
}
|
|
|
|
type ErrIsNotNotebook struct {
|
|
path string
|
|
}
|
|
|
|
func (err ErrIsNotNotebook) Error() string {
|
|
return fmt.Sprintf("file at %s is not a notebook", err.path)
|
|
}
|
|
|
|
type translatePaths struct {
|
|
seen map[string]string
|
|
}
|
|
|
|
// TranslatePaths converts paths to local notebook files into paths in the workspace file system.
|
|
func TranslatePaths() bundle.Mutator {
|
|
return &translatePaths{}
|
|
}
|
|
|
|
func (m *translatePaths) Name() string {
|
|
return "TranslatePaths"
|
|
}
|
|
|
|
type rewriteFunc func(literal, localFullPath, localRelPath, remotePath string) (string, error)
|
|
|
|
// rewritePath converts a given relative path from the loaded config to a new path based on the passed rewriting function
|
|
//
|
|
// It takes these arguments:
|
|
// - The argument `dir` is the directory relative to which the given relative path is.
|
|
// - The given relative path is both passed and written back through `*p`.
|
|
// - The argument `fn` is a function that performs the actual rewriting logic.
|
|
// This logic is different between regular files or notebooks.
|
|
//
|
|
// The function returns an error if it is impossible to rewrite the given relative path.
|
|
func (m *translatePaths) rewritePath(
|
|
dir string,
|
|
b *bundle.Bundle,
|
|
p *string,
|
|
fn rewriteFunc,
|
|
) error {
|
|
// We assume absolute paths point to a location in the workspace
|
|
if path.IsAbs(filepath.ToSlash(*p)) {
|
|
return nil
|
|
}
|
|
|
|
url, err := url.Parse(*p)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// If the file path has scheme, it's a full path and we don't need to transform it
|
|
if url.Scheme != "" {
|
|
return nil
|
|
}
|
|
|
|
// Local path is relative to the directory the resource was defined in.
|
|
localPath := filepath.Join(dir, filepath.FromSlash(*p))
|
|
if interp, ok := m.seen[localPath]; ok {
|
|
*p = interp
|
|
return nil
|
|
}
|
|
|
|
// Remote path must be relative to the bundle root.
|
|
localRelPath, err := filepath.Rel(b.Config.Path, localPath)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if strings.HasPrefix(localRelPath, "..") {
|
|
return fmt.Errorf("path %s is not contained in bundle root path", localPath)
|
|
}
|
|
|
|
// Prefix remote path with its remote root path.
|
|
remotePath := path.Join(b.Config.Workspace.FilePath, filepath.ToSlash(localRelPath))
|
|
|
|
// Convert local path into workspace path via specified function.
|
|
interp, err := fn(*p, localPath, localRelPath, filepath.ToSlash(remotePath))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
*p = interp
|
|
m.seen[localPath] = interp
|
|
return nil
|
|
}
|
|
|
|
func translateNotebookPath(literal, localFullPath, localRelPath, remotePath string) (string, error) {
|
|
nb, _, err := notebook.Detect(localFullPath)
|
|
if os.IsNotExist(err) {
|
|
return "", fmt.Errorf("notebook %s not found", literal)
|
|
}
|
|
if err != nil {
|
|
return "", fmt.Errorf("unable to determine if %s is a notebook: %w", localFullPath, err)
|
|
}
|
|
if !nb {
|
|
return "", ErrIsNotNotebook{localFullPath}
|
|
}
|
|
|
|
// Upon import, notebooks are stripped of their extension.
|
|
return strings.TrimSuffix(remotePath, filepath.Ext(localFullPath)), nil
|
|
}
|
|
|
|
func translateFilePath(literal, localFullPath, localRelPath, remotePath string) (string, error) {
|
|
nb, _, err := notebook.Detect(localFullPath)
|
|
if os.IsNotExist(err) {
|
|
return "", fmt.Errorf("file %s not found", literal)
|
|
}
|
|
if err != nil {
|
|
return "", fmt.Errorf("unable to determine if %s is not a notebook: %w", localFullPath, err)
|
|
}
|
|
if nb {
|
|
return "", ErrIsNotebook{localFullPath}
|
|
}
|
|
return remotePath, nil
|
|
}
|
|
|
|
func translateDirectoryPath(literal, localFullPath, localRelPath, remotePath string) (string, error) {
|
|
info, err := os.Stat(localFullPath)
|
|
if err != nil {
|
|
return "", err
|
|
}
|
|
if !info.IsDir() {
|
|
return "", fmt.Errorf("%s is not a directory", localFullPath)
|
|
}
|
|
return remotePath, nil
|
|
}
|
|
|
|
func translateNoOp(literal, localFullPath, localRelPath, remotePath string) (string, error) {
|
|
return localRelPath, nil
|
|
}
|
|
|
|
func (m *translatePaths) rewriteValue(b *bundle.Bundle, p dyn.Path, v dyn.Value, fn rewriteFunc, dir string) (dyn.Value, error) {
|
|
out := v.MustString()
|
|
err := m.rewritePath(dir, b, &out, fn)
|
|
if err != nil {
|
|
if target := (&ErrIsNotebook{}); errors.As(err, target) {
|
|
return dyn.InvalidValue, fmt.Errorf(`expected a file for "%s" but got a notebook: %w`, p, target)
|
|
}
|
|
if target := (&ErrIsNotNotebook{}); errors.As(err, target) {
|
|
return dyn.InvalidValue, fmt.Errorf(`expected a notebook for "%s" but got a file: %w`, p, target)
|
|
}
|
|
return dyn.InvalidValue, err
|
|
}
|
|
|
|
return dyn.NewValue(out, v.Location()), nil
|
|
}
|
|
|
|
func (m *translatePaths) rewriteRelativeTo(b *bundle.Bundle, p dyn.Path, v dyn.Value, fn rewriteFunc, dir, fallback string) (dyn.Value, error) {
|
|
nv, err := m.rewriteValue(b, p, v, fn, dir)
|
|
if err == nil {
|
|
return nv, nil
|
|
}
|
|
|
|
// If we failed to rewrite the path, try to rewrite it relative to the fallback directory.
|
|
if fallback != "" {
|
|
nv, nerr := m.rewriteValue(b, p, v, fn, fallback)
|
|
if nerr == nil {
|
|
// TODO: Emit a warning that this path should be rewritten.
|
|
return nv, nil
|
|
}
|
|
}
|
|
|
|
return dyn.InvalidValue, err
|
|
}
|
|
|
|
func (m *translatePaths) Apply(_ context.Context, b *bundle.Bundle) diag.Diagnostics {
|
|
m.seen = make(map[string]string)
|
|
|
|
err := b.Config.Mutate(func(v dyn.Value) (dyn.Value, error) {
|
|
var err error
|
|
for _, fn := range []func(*bundle.Bundle, dyn.Value) (dyn.Value, error){
|
|
m.applyJobTranslations,
|
|
m.applyPipelineTranslations,
|
|
m.applyArtifactTranslations,
|
|
} {
|
|
v, err = fn(b, v)
|
|
if err != nil {
|
|
return dyn.InvalidValue, err
|
|
}
|
|
}
|
|
return v, nil
|
|
})
|
|
|
|
return diag.FromErr(err)
|
|
}
|