2024-10-15 13:22:41 +00:00
package libraries
import (
"context"
"fmt"
"path"
"testing"
"github.com/databricks/cli/bundle"
2024-10-31 16:52:45 +00:00
"github.com/databricks/cli/bundle/bundletest"
2024-10-15 13:22:41 +00:00
"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"
2024-11-29 18:43:25 +00:00
"github.com/databricks/databricks-sdk-go/apierr"
2024-10-15 13:22:41 +00:00
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 )
2024-10-15 16:04:20 +00:00
// 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 )
2024-10-15 13:22:41 +00:00
// schema name is interpolated.
2024-10-15 16:04:20 +00:00
b . Config . Resources . Volumes [ "foo" ] . SchemaName = "${resources.schemas.my_schema.name}"
2024-10-15 13:22:41 +00:00
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" ,
} ,
} ,
}
2024-11-29 18:43:25 +00:00
bundletest . SetLocation ( b , "workspace.artifact_path" , [ ] dyn . Location { { File : "config.yml" , Line : 1 , Column : 2 } } )
2024-10-15 13:22:41 +00:00
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 )
2024-11-29 18:43:25 +00:00
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 )
2024-10-15 13:22:41 +00:00
}
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" ,
} ,
} ,
} ,
} ,
} ,
}
2024-11-29 18:43:25 +00:00
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 } } )
2024-10-15 13:22:41 +00:00
m := mocks . NewMockWorkspaceClient ( t )
m . WorkspaceClient . Config = & sdkconfig . Config { }
2024-11-29 18:43:25 +00:00
m . GetMockFilesAPI ( ) . EXPECT ( ) . GetDirectoryMetadataByDirectoryPath ( mock . Anything , "/Volumes/main/my_schema/my_volume" ) . Return ( & apierr . APIError {
StatusCode : 404 ,
Message : "error from API" ,
} )
2024-10-15 13:22:41 +00:00
b . SetWorkpaceClient ( m . WorkspaceClient )
_ , _ , diags := GetFilerForLibraries ( context . Background ( ) , b )
2024-11-29 18:43:25 +00:00
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
2024-11-29 19:00:41 +00:00
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 . ` ,
2024-11-29 18:43:25 +00:00
} ,
} , diags )
2024-10-15 13:22:41 +00:00
}
2024-11-29 01:45:33 +00:00
func invalidVolumePaths ( ) [ ] string {
return [ ] string {
2024-10-15 13:22:41 +00:00
"/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" ,
}
2024-11-29 01:45:33 +00:00
}
2024-10-15 13:22:41 +00:00
2024-11-29 01:45:33 +00:00
func TestFilerForVolumeWithInvalidVolumePaths ( t * testing . T ) {
for _ , p := range invalidVolumePaths ( ) {
2024-10-15 13:22:41 +00:00
b := & bundle . Bundle {
Config : config . Root {
Workspace : config . Workspace {
ArtifactPath : p ,
} ,
} ,
}
2024-11-29 01:45:33 +00:00
bundletest . SetLocation ( b , "workspace.artifact_path" , [ ] dyn . Location { { File : "config.yml" , Line : 1 , Column : 2 } } )
2024-10-15 13:22:41 +00:00
_ , _ , diags := GetFilerForLibraries ( context . Background ( ) , b )
2024-11-29 01:45:33 +00:00
require . Equal ( t , diags , diag . Diagnostics { {
Severity : diag . Error ,
Summary : fmt . Sprintf ( "expected UC volume path to be in the format /Volumes/<catalog>/<schema>/<volume>/..., got %s" , p ) ,
Locations : [ ] dyn . Location { { File : "config.yml" , Line : 1 , Column : 2 } } ,
Paths : [ ] dyn . Path { dyn . MustPathFromString ( "workspace.artifact_path" ) } ,
} } )
2024-10-15 13:22:41 +00:00
}
}
2024-10-15 13:27:58 +00:00
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" )
}
2024-11-29 01:46:31 +00:00
func TestFilerForVolumeWithValidVolumePaths ( t * testing . T ) {
2024-10-15 13:22:41 +00:00
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 )
}
}
2024-11-29 01:45:33 +00:00
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/<catalog>/<schema>/<volume>/..., got %s" , p ) )
}
}