package libraries import ( "context" "fmt" "path" "testing" "github.com/databricks/cli/bundle" "github.com/databricks/cli/bundle/bundletest" "github.com/databricks/cli/bundle/config" "github.com/databricks/cli/bundle/config/resources" "github.com/databricks/cli/libs/diag" "github.com/databricks/cli/libs/dyn" "github.com/databricks/cli/libs/filer" "github.com/databricks/databricks-sdk-go/apierr" sdkconfig "github.com/databricks/databricks-sdk-go/config" "github.com/databricks/databricks-sdk-go/experimental/mocks" "github.com/databricks/databricks-sdk-go/service/catalog" "github.com/stretchr/testify/assert" "github.com/stretchr/testify/mock" "github.com/stretchr/testify/require" ) func TestFindVolumeInBundle(t *testing.T) { b := &bundle.Bundle{ Config: config.Root{ Resources: config.Resources{ Volumes: map[string]*resources.Volume{ "foo": { CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{ CatalogName: "main", Name: "my_volume", SchemaName: "my_schema", }, }, }, }, }, } bundletest.SetLocation(b, "resources.volumes.foo", []dyn.Location{ { File: "volume.yml", Line: 1, Column: 2, }, }) // volume is in DAB. path, locations, ok := findVolumeInBundle(b, "main", "my_schema", "my_volume") assert.True(t, ok) assert.Equal(t, []dyn.Location{{ File: "volume.yml", Line: 1, Column: 2, }}, locations) assert.Equal(t, dyn.MustPathFromString("resources.volumes.foo"), path) // wrong volume name _, _, ok = findVolumeInBundle(b, "main", "my_schema", "doesnotexist") assert.False(t, ok) // wrong schema name _, _, ok = findVolumeInBundle(b, "main", "doesnotexist", "my_volume") assert.False(t, ok) // wrong catalog name _, _, ok = findVolumeInBundle(b, "doesnotexist", "my_schema", "my_volume") assert.False(t, ok) // schema name is interpolated but does not have the right prefix. In this case // we should not match the volume. b.Config.Resources.Volumes["foo"].SchemaName = "${foo.bar.baz}" _, _, ok = findVolumeInBundle(b, "main", "my_schema", "my_volume") assert.False(t, ok) // schema name is interpolated. b.Config.Resources.Volumes["foo"].SchemaName = "${resources.schemas.my_schema.name}" path, locations, ok = findVolumeInBundle(b, "main", "valuedoesnotmatter", "my_volume") assert.True(t, ok) assert.Equal(t, []dyn.Location{{ File: "volume.yml", Line: 1, Column: 2, }}, locations) assert.Equal(t, dyn.MustPathFromString("resources.volumes.foo"), path) } func TestFilerForVolumeNotInBundle(t *testing.T) { b := &bundle.Bundle{ Config: config.Root{ Workspace: config.Workspace{ ArtifactPath: "/Volumes/main/my_schema/doesnotexist", }, }, } bundletest.SetLocation(b, "workspace.artifact_path", []dyn.Location{{File: "config.yml", Line: 1, Column: 2}}) m := mocks.NewMockWorkspaceClient(t) m.WorkspaceClient.Config = &sdkconfig.Config{} m.GetMockFilesAPI().EXPECT().GetDirectoryMetadataByDirectoryPath(mock.Anything, "/Volumes/main/my_schema/doesnotexist").Return(fmt.Errorf("error from API")) b.SetWorkpaceClient(m.WorkspaceClient) _, _, diags := filerForVolume(context.Background(), b) assert.Equal(t, diag.Diagnostics{ { Severity: diag.Error, Summary: "failed to fetch metadata for /Volumes/main/my_schema/doesnotexist: error from API", Locations: []dyn.Location{{File: "config.yml", Line: 1, Column: 2}}, Paths: []dyn.Path{dyn.MustPathFromString("workspace.artifact_path")}, }}, diags) } func TestFilerForVolumeInBundle(t *testing.T) { b := &bundle.Bundle{ Config: config.Root{ Workspace: config.Workspace{ ArtifactPath: "/Volumes/main/my_schema/my_volume", }, Resources: config.Resources{ Volumes: map[string]*resources.Volume{ "foo": { CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{ CatalogName: "main", Name: "my_volume", VolumeType: "MANAGED", SchemaName: "my_schema", }, }, }, }, }, } bundletest.SetLocation(b, "workspace.artifact_path", []dyn.Location{{File: "config.yml", Line: 1, Column: 2}}) bundletest.SetLocation(b, "resources.volumes.foo", []dyn.Location{{File: "volume.yml", Line: 1, Column: 2}}) m := mocks.NewMockWorkspaceClient(t) m.WorkspaceClient.Config = &sdkconfig.Config{} m.GetMockFilesAPI().EXPECT().GetDirectoryMetadataByDirectoryPath(mock.Anything, "/Volumes/main/my_schema/my_volume").Return(&apierr.APIError{ StatusCode: 404, Message: "error from API", }) b.SetWorkpaceClient(m.WorkspaceClient) _, _, diags := GetFilerForLibraries(context.Background(), b) assert.Equal(t, diag.Diagnostics{ { Severity: diag.Error, Summary: "failed to fetch metadata for /Volumes/main/my_schema/my_volume: error from API", Locations: []dyn.Location{{"config.yml", 1, 2}, {"volume.yml", 1, 2}}, Paths: []dyn.Path{dyn.MustPathFromString("workspace.artifact_path"), dyn.MustPathFromString("resources.volumes.foo")}, Detail: `You are using a UC volume in your artifact_path that is managed by this bundle but which has not been deployed yet. Please first deploy the UC volume using 'bundle deploy' and then switch over to using it in the artifact_path.`, }, }, diags) } func invalidVolumePaths() []string { return []string{ "/Volumes/", "/Volumes/main", "/Volumes/main/", "/Volumes/main//", "/Volumes/main//my_schema", "/Volumes/main/my_schema", "/Volumes/main/my_schema/", "/Volumes/main/my_schema//", "/Volumes//my_schema/my_volume", } } func TestFilerForVolumeWithInvalidVolumePaths(t *testing.T) { for _, p := range invalidVolumePaths() { b := &bundle.Bundle{ Config: config.Root{ Workspace: config.Workspace{ ArtifactPath: p, }, }, } bundletest.SetLocation(b, "workspace.artifact_path", []dyn.Location{{File: "config.yml", Line: 1, Column: 2}}) _, _, diags := GetFilerForLibraries(context.Background(), b) require.Equal(t, diags, diag.Diagnostics{{ Severity: diag.Error, Summary: fmt.Sprintf("expected UC volume path to be in the format /Volumes////..., got %s", p), Locations: []dyn.Location{{File: "config.yml", Line: 1, Column: 2}}, Paths: []dyn.Path{dyn.MustPathFromString("workspace.artifact_path")}, }}) } } func TestFilerForVolumeWithInvalidPrefix(t *testing.T) { b := &bundle.Bundle{ Config: config.Root{ Workspace: config.Workspace{ ArtifactPath: "/Volume/main/my_schema/my_volume", }, }, } _, _, diags := filerForVolume(context.Background(), b) require.EqualError(t, diags.Error(), "expected artifact_path to start with /Volumes/, got /Volume/main/my_schema/my_volume") } func TestFilerForVolumeWithValidVolumePaths(t *testing.T) { validPaths := []string{ "/Volumes/main/my_schema/my_volume", "/Volumes/main/my_schema/my_volume/", "/Volumes/main/my_schema/my_volume/a/b/c", "/Volumes/main/my_schema/my_volume/a/a/a", } for _, p := range validPaths { b := &bundle.Bundle{ Config: config.Root{ Workspace: config.Workspace{ ArtifactPath: p, }, }, } m := mocks.NewMockWorkspaceClient(t) m.WorkspaceClient.Config = &sdkconfig.Config{} m.GetMockFilesAPI().EXPECT().GetDirectoryMetadataByDirectoryPath(mock.Anything, "/Volumes/main/my_schema/my_volume").Return(nil) b.SetWorkpaceClient(m.WorkspaceClient) client, uploadPath, diags := filerForVolume(context.Background(), b) require.NoError(t, diags.Error()) assert.Equal(t, path.Join(p, ".internal"), uploadPath) assert.IsType(t, &filer.FilesClient{}, client) } } func TestExtractVolumeFromPath(t *testing.T) { catalogName, schemaName, volumeName, err := extractVolumeFromPath("/Volumes/main/my_schema/my_volume") require.NoError(t, err) assert.Equal(t, "main", catalogName) assert.Equal(t, "my_schema", schemaName) assert.Equal(t, "my_volume", volumeName) for _, p := range invalidVolumePaths() { _, _, _, err := extractVolumeFromPath(p) assert.EqualError(t, err, fmt.Sprintf("expected UC volume path to be in the format /Volumes////..., got %s", p)) } }