2023-01-24 12:58:10 +00:00
|
|
|
package sync
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"path"
|
|
|
|
"strings"
|
|
|
|
|
2023-05-16 16:35:39 +00:00
|
|
|
"github.com/databricks/cli/libs/log"
|
2023-01-24 12:58:10 +00:00
|
|
|
"github.com/databricks/databricks-sdk-go"
|
|
|
|
"github.com/databricks/databricks-sdk-go/apierr"
|
2023-04-21 08:30:20 +00:00
|
|
|
"github.com/databricks/databricks-sdk-go/service/iam"
|
2023-01-24 12:58:10 +00:00
|
|
|
"github.com/databricks/databricks-sdk-go/service/workspace"
|
|
|
|
)
|
|
|
|
|
2023-04-21 08:30:20 +00:00
|
|
|
func repoPathForPath(me *iam.User, remotePath string) string {
|
2025-01-07 10:49:23 +00:00
|
|
|
base := path.Clean("/Repos/" + me.UserName)
|
2023-02-20 13:34:48 +00:00
|
|
|
remotePath = path.Clean(remotePath)
|
|
|
|
for strings.HasPrefix(path.Dir(remotePath), base) && path.Dir(remotePath) != base {
|
|
|
|
remotePath = path.Dir(remotePath)
|
|
|
|
}
|
|
|
|
return remotePath
|
|
|
|
}
|
|
|
|
|
|
|
|
// EnsureRemotePathIsUsable checks if the specified path is nested under
|
2023-01-24 12:58:10 +00:00
|
|
|
// expected base paths and if it is a directory or repository.
|
2023-07-30 07:19:49 +00:00
|
|
|
func EnsureRemotePathIsUsable(ctx context.Context, wsc *databricks.WorkspaceClient, remotePath string, me *iam.User) error {
|
|
|
|
var err error
|
|
|
|
|
|
|
|
// TODO: we should cache CurrentUser.Me at the SDK level
|
|
|
|
// for now we let clients pass in any existing user they might already have
|
|
|
|
if me == nil {
|
|
|
|
me, err = wsc.CurrentUser.Me(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-01-24 12:58:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure that the remote path exists.
|
|
|
|
// If it is a repo, it has to exist.
|
|
|
|
// If it is a workspace path, it may not exist.
|
2023-02-20 13:34:48 +00:00
|
|
|
info, err := wsc.Workspace.GetStatusByPath(ctx, remotePath)
|
2023-01-24 12:58:10 +00:00
|
|
|
if err != nil {
|
|
|
|
// We only deal with 404s below.
|
|
|
|
if !apierr.IsMissing(err) {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-02-20 13:34:48 +00:00
|
|
|
// If the path is nested under a repo, the repo has to exist.
|
|
|
|
if strings.HasPrefix(remotePath, "/Repos/") {
|
|
|
|
repoPath := repoPathForPath(me, remotePath)
|
|
|
|
_, err = wsc.Workspace.GetStatusByPath(ctx, repoPath)
|
|
|
|
if err != nil && apierr.IsMissing(err) {
|
|
|
|
return fmt.Errorf("%s does not exist; please create it first", repoPath)
|
2023-01-24 12:58:10 +00:00
|
|
|
}
|
2023-02-20 13:34:48 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// The workspace path doesn't exist. Create it and try again.
|
|
|
|
err = wsc.Workspace.MkdirsByPath(ctx, remotePath)
|
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to create directory at %s: %w", remotePath, err)
|
|
|
|
}
|
|
|
|
info, err = wsc.Workspace.GetStatusByPath(ctx, remotePath)
|
|
|
|
if err != nil {
|
2023-01-24 12:58:10 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-17 14:17:31 +00:00
|
|
|
log.Debugf(
|
|
|
|
ctx,
|
|
|
|
"Path %s has type %s (ID: %d)",
|
2023-01-24 12:58:10 +00:00
|
|
|
info.Path,
|
|
|
|
strings.ToLower(info.ObjectType.String()),
|
|
|
|
info.ObjectId,
|
|
|
|
)
|
|
|
|
|
|
|
|
// We expect the object at path to be a directory or a repo.
|
|
|
|
switch info.ObjectType {
|
|
|
|
case workspace.ObjectTypeDirectory:
|
|
|
|
return nil
|
|
|
|
case workspace.ObjectTypeRepo:
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-02-20 13:34:48 +00:00
|
|
|
return fmt.Errorf("%s points to a %s", remotePath, strings.ToLower(info.ObjectType.String()))
|
2023-01-24 12:58:10 +00:00
|
|
|
}
|