Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add DABs support for Unity Catalog volumes #1762

Merged
merged 78 commits into from
Dec 2, 2024
Merged
Show file tree
Hide file tree
Changes from 20 commits
Commits
Show all changes
78 commits
Select commit Hold shift + click to select a range
f772ce4
first comment
shreyas-goenka Sep 9, 2024
8f4f3ae
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Sep 9, 2024
ce5792c
fix convertor and add unit test
shreyas-goenka Sep 9, 2024
7c7abef
run as support
shreyas-goenka Sep 9, 2024
d04b6b0
-
shreyas-goenka Sep 9, 2024
9b66cd5
add apply target mode prefix functionality
shreyas-goenka Sep 9, 2024
4b22e2d
add conversion tests
shreyas-goenka Sep 9, 2024
88d0402
add inteprolation for volumes fields
shreyas-goenka Sep 9, 2024
6f9817e
add prompt and crud test for volumes
shreyas-goenka Sep 10, 2024
d47b0d6
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Sep 10, 2024
d180bab
add filer
shreyas-goenka Sep 15, 2024
fa54577
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Sep 15, 2024
73826ac
fix test and improve error
shreyas-goenka Sep 15, 2024
de7eb94
-
shreyas-goenka Sep 15, 2024
f10038a
-
shreyas-goenka Sep 15, 2024
df3bbad
add integration tests for artifacts portion:
shreyas-goenka Sep 15, 2024
aeab4ef
unit test for comparision of locatoin
shreyas-goenka Sep 15, 2024
aa2e16d
cleanup and add test
shreyas-goenka Sep 16, 2024
a90eb57
fix unit tests
shreyas-goenka Sep 16, 2024
39cb5e8
fix test on windows
shreyas-goenka Sep 16, 2024
13748f1
cleanup todos
shreyas-goenka Sep 16, 2024
bdecd08
-
shreyas-goenka Sep 16, 2024
274fd63
iter
shreyas-goenka Sep 16, 2024
d3d5d4c
-
shreyas-goenka Sep 16, 2024
227dfe9
fixes
shreyas-goenka Sep 16, 2024
e43f566
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Oct 14, 2024
f919e94
typo fix
shreyas-goenka Oct 14, 2024
9921263
separate GetFilerForLibraries tests
shreyas-goenka Oct 14, 2024
266c26c
fix tesT
shreyas-goenka Oct 14, 2024
a9b8575
fmt and fix test
shreyas-goenka Oct 14, 2024
c5a02ef
split into filer files
shreyas-goenka Oct 15, 2024
eb94cd6
-
shreyas-goenka Oct 15, 2024
3e3ddfd
fix test
shreyas-goenka Oct 15, 2024
d241c2b
add integration test for grant on volume
shreyas-goenka Oct 15, 2024
6192835
add custom prefixing behaviour for volumes
shreyas-goenka Oct 15, 2024
701b178
fmt
shreyas-goenka Oct 15, 2024
810da66
-
shreyas-goenka Oct 16, 2024
1a961eb
-
shreyas-goenka Oct 16, 2024
8a2fe49
merge
shreyas-goenka Oct 29, 2024
49b2cf2
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Oct 31, 2024
e32ebd0
use IsVolumesPath
shreyas-goenka Oct 31, 2024
6b12234
better message
shreyas-goenka Oct 31, 2024
f9287e0
address comments
shreyas-goenka Oct 31, 2024
250d426
rename to volume
shreyas-goenka Oct 31, 2024
1218178
-
shreyas-goenka Oct 31, 2024
68dc6c1
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Nov 18, 2024
4cc2790
remove prefixing for uc volumes
shreyas-goenka Nov 18, 2024
0406265
-
shreyas-goenka Nov 18, 2024
b0e527e
-
shreyas-goenka Nov 18, 2024
e6723de
undo containsUsername
shreyas-goenka Nov 18, 2024
f5ea8da
remove other mutator
shreyas-goenka Nov 18, 2024
ea6906e
add support for initializeUrl
shreyas-goenka Nov 18, 2024
76092cc
remove todo
shreyas-goenka Nov 18, 2024
039057f
fix renaming test
shreyas-goenka Nov 18, 2024
4f5f9ec
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Nov 29, 2024
5ec784b
merge
shreyas-goenka Nov 29, 2024
fcb8fff
-
shreyas-goenka Nov 29, 2024
e9a5544
-
shreyas-goenka Nov 29, 2024
5531e2a
-
shreyas-goenka Nov 29, 2024
3090588
Update bundle/libraries/filer.go
shreyas-goenka Nov 29, 2024
015f8cd
Update bundle/libraries/filer_volume.go
shreyas-goenka Nov 29, 2024
d0385a0
add extractVolumeFromPath
shreyas-goenka Nov 29, 2024
5ac2d67
-
shreyas-goenka Nov 29, 2024
9493795
address comments
shreyas-goenka Nov 29, 2024
e51b3a1
add const for internal
shreyas-goenka Nov 29, 2024
01c3830
add TestFilerForWorkspace
shreyas-goenka Nov 29, 2024
e5f5618
-
shreyas-goenka Nov 29, 2024
23e87d5
-
shreyas-goenka Nov 29, 2024
d1ec088
remove defensive bit
shreyas-goenka Nov 29, 2024
42bf6ae
revert bundletest move
shreyas-goenka Dec 2, 2024
5f2db1e
move .internal var
shreyas-goenka Dec 2, 2024
07f888c
remove UC
shreyas-goenka Dec 2, 2024
7d544f4
lowercase volumes
shreyas-goenka Dec 2, 2024
406c073
use apierr.ErrNotFound
shreyas-goenka Dec 2, 2024
3461018
better error message
shreyas-goenka Dec 2, 2024
8e25bb4
Merge remote-tracking branch 'origin' into feature/uc-volumes
shreyas-goenka Dec 2, 2024
8d790ef
address comments
shreyas-goenka Dec 2, 2024
d460bd6
fix
shreyas-goenka Dec 2, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
13 changes: 4 additions & 9 deletions bundle/artifacts/upload.go
Original file line number Diff line number Diff line change
Expand Up @@ -21,18 +21,13 @@ func (m *cleanUp) Name() string {
}

func (m *cleanUp) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
uploadPath, err := libraries.GetUploadBasePath(b)
if err != nil {
return diag.FromErr(err)
}

client, err := libraries.GetFilerForLibraries(b.WorkspaceClient(), uploadPath)
if err != nil {
return diag.FromErr(err)
client, uploadPath, diags := libraries.GetFilerForLibraries(ctx, b)
if diags.HasError() {
return diags
}

// We intentionally ignore the error because it is not critical to the deployment
err = client.Delete(ctx, ".", filer.DeleteRecursively)
err := client.Delete(ctx, ".", filer.DeleteRecursively)
if err != nil {
log.Errorf(ctx, "failed to delete %s: %v", uploadPath, err)
}
Expand Down
7 changes: 7 additions & 0 deletions bundle/config/mutator/apply_presets.go
Original file line number Diff line number Diff line change
Expand Up @@ -160,6 +160,13 @@ func (m *applyPresets) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnos
// the Databricks UI and via the SQL API.
}

// Apply the prefix to volumes
shreyas-goenka marked this conversation as resolved.
Show resolved Hide resolved
for i := range r.Volumes {
r.Volumes[i].Name = normalizePrefix(prefix) + r.Volumes[i].Name
shreyas-goenka marked this conversation as resolved.
Show resolved Hide resolved
// HTTP API for volumes doesn't yet support tags. It's only supported in
// the Databricks UI and via the SQL API.
}

return nil
}

Expand Down
56 changes: 56 additions & 0 deletions bundle/config/mutator/apply_presets_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -125,6 +125,62 @@ func TestApplyPresetsPrefixForUcSchema(t *testing.T) {
}
}

func TestApplyPresetsPrefixForUcVolumes(t *testing.T) {
tests := []struct {
name string
prefix string
volume *resources.Volume
want string
}{
{
name: "add prefix to volume",
prefix: "[prefix]",
volume: &resources.Volume{
CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{
Name: "volume1",
},
},
want: "prefix_volume1",
},
{
name: "add empty prefix to volume",
prefix: "",
volume: &resources.Volume{
CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{
Name: "volume1",
},
},
want: "volume1",
},
}

for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
b := &bundle.Bundle{
Config: config.Root{
Resources: config.Resources{
Volumes: map[string]*resources.Volume{
"volume1": tt.volume,
},
},
Presets: config.Presets{
NamePrefix: tt.prefix,
},
},
}

ctx := context.Background()
diag := bundle.Apply(ctx, b, mutator.ApplyPresets())

if diag.HasError() {
t.Fatalf("unexpected error: %v", diag)
}

require.Equal(t, tt.want, b.Config.Resources.Volumes["volume1"].Name)
})
}
}

func TestApplyPresetsTags(t *testing.T) {
tests := []struct {
name string
Expand Down
7 changes: 7 additions & 0 deletions bundle/config/mutator/process_target_mode_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -119,6 +119,9 @@ func mockBundle(mode config.Mode) *bundle.Bundle {
Schemas: map[string]*resources.Schema{
"schema1": {CreateSchema: &catalog.CreateSchema{Name: "schema1"}},
},
Volumes: map[string]*resources.Volume{
"volume1": {CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{Name: "volume1"}},
},
},
},
// Use AWS implementation for testing.
Expand Down Expand Up @@ -281,6 +284,8 @@ func TestProcessTargetModeDefault(t *testing.T) {
assert.Equal(t, "servingendpoint1", b.Config.Resources.ModelServingEndpoints["servingendpoint1"].Name)
assert.Equal(t, "registeredmodel1", b.Config.Resources.RegisteredModels["registeredmodel1"].Name)
assert.Equal(t, "qualityMonitor1", b.Config.Resources.QualityMonitors["qualityMonitor1"].TableName)
assert.Equal(t, "schema1", b.Config.Resources.Schemas["schema1"].Name)
assert.Equal(t, "volume1", b.Config.Resources.Volumes["volume1"].Name)
}

func TestProcessTargetModeProduction(t *testing.T) {
Expand Down Expand Up @@ -322,6 +327,8 @@ func TestProcessTargetModeProduction(t *testing.T) {
assert.Equal(t, "servingendpoint1", b.Config.Resources.ModelServingEndpoints["servingendpoint1"].Name)
assert.Equal(t, "registeredmodel1", b.Config.Resources.RegisteredModels["registeredmodel1"].Name)
assert.Equal(t, "qualityMonitor1", b.Config.Resources.QualityMonitors["qualityMonitor1"].TableName)
assert.Equal(t, "schema1", b.Config.Resources.Schemas["schema1"].Name)
assert.Equal(t, "volume1", b.Config.Resources.Volumes["volume1"].Name)
}

func TestProcessTargetModeProductionOkForPrincipal(t *testing.T) {
Expand Down
2 changes: 2 additions & 0 deletions bundle/config/mutator/run_as_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -40,6 +40,7 @@ func allResourceTypes(t *testing.T) []string {
"quality_monitors",
"registered_models",
"schemas",
"volumes",
},
resourceTypes,
)
Expand Down Expand Up @@ -138,6 +139,7 @@ func TestRunAsErrorForUnsupportedResources(t *testing.T) {
"registered_models",
"experiments",
"schemas",
"volumes",
}

base := config.Root{
Expand Down
1 change: 1 addition & 0 deletions bundle/config/resources.go
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,7 @@ type Resources struct {
RegisteredModels map[string]*resources.RegisteredModel `json:"registered_models,omitempty"`
QualityMonitors map[string]*resources.QualityMonitor `json:"quality_monitors,omitempty"`
Schemas map[string]*resources.Schema `json:"schemas,omitempty"`
Volumes map[string]*resources.Volume `json:"volumes,omitempty"`
}

type ConfigResource interface {
Expand Down
30 changes: 30 additions & 0 deletions bundle/config/resources/volume.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,30 @@
package resources

import (
"github.com/databricks/databricks-sdk-go/marshal"
"github.com/databricks/databricks-sdk-go/service/catalog"
)

type Volume struct {
// List of grants to apply on this schema.
shreyas-goenka marked this conversation as resolved.
Show resolved Hide resolved
Grants []Grant `json:"grants,omitempty"`

// TODO: Confirm the accuracy of this comment.
// Full name of the schema (catalog_name.schema_name.volume_name). This value is read from
shreyas-goenka marked this conversation as resolved.
Show resolved Hide resolved
// the terraform state after deployment succeeds.
ID string `json:"id,omitempty" bundle:"readonly"`

// TODO: Are there fields in the edit API or terraform that are not in this struct?
// If so call it out in the PR.
*catalog.CreateVolumeRequestContent
shreyas-goenka marked this conversation as resolved.
Show resolved Hide resolved

ModifiedStatus ModifiedStatus `json:"modified_status,omitempty" bundle:"internal"`
}

func (v *Volume) UnmarshalJSON(b []byte) error {
return marshal.Unmarshal(b, v)
}

func (v Volume) MarshalJSON() ([]byte, error) {
return marshal.Marshal(v)
}
15 changes: 15 additions & 0 deletions bundle/deploy/terraform/convert.go
Original file line number Diff line number Diff line change
Expand Up @@ -394,6 +394,16 @@ func TerraformToBundle(state *resourcesState, config *config.Root) error {
}
cur.ID = instance.Attributes.ID
config.Resources.Schemas[resource.Name] = cur
case "databricks_volume":
if config.Resources.Volumes == nil {
config.Resources.Volumes = make(map[string]*resources.Volume)
}
cur := config.Resources.Volumes[resource.Name]
if cur == nil {
cur = &resources.Volume{ModifiedStatus: resources.ModifiedStatusDeleted}
}
cur.ID = instance.Attributes.ID
config.Resources.Volumes[resource.Name] = cur
case "databricks_permissions":
case "databricks_grants":
// Ignore; no need to pull these back into the configuration.
Expand Down Expand Up @@ -443,6 +453,11 @@ func TerraformToBundle(state *resourcesState, config *config.Root) error {
src.ModifiedStatus = resources.ModifiedStatusCreated
}
}
for _, src := range config.Resources.Volumes {
if src.ModifiedStatus == "" && src.ID == "" {
src.ModifiedStatus = resources.ModifiedStatusCreated
}
}

return nil
}
56 changes: 56 additions & 0 deletions bundle/deploy/terraform/convert_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -663,6 +663,14 @@ func TestTerraformToBundleEmptyLocalResources(t *testing.T) {
{Attributes: stateInstanceAttributes{ID: "1"}},
},
},
{
Type: "databricks_volume",
Mode: "managed",
Name: "test_volume",
Instances: []stateResourceInstance{
{Attributes: stateInstanceAttributes{ID: "1"}},
},
},
},
}
err := TerraformToBundle(&tfState, &config)
Expand Down Expand Up @@ -692,6 +700,9 @@ func TestTerraformToBundleEmptyLocalResources(t *testing.T) {
assert.Equal(t, "1", config.Resources.Schemas["test_schema"].ID)
assert.Equal(t, resources.ModifiedStatusDeleted, config.Resources.Schemas["test_schema"].ModifiedStatus)

assert.Equal(t, "1", config.Resources.Volumes["test_volume"].ID)
assert.Equal(t, resources.ModifiedStatusDeleted, config.Resources.Volumes["test_volume"].ModifiedStatus)

AssertFullResourceCoverage(t, &config)
}

Expand Down Expand Up @@ -754,6 +765,13 @@ func TestTerraformToBundleEmptyRemoteResources(t *testing.T) {
},
},
},
Volumes: map[string]*resources.Volume{
"test_volume": {
CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{
Name: "test_volume",
},
},
},
},
}
var tfState = resourcesState{
Expand Down Expand Up @@ -786,6 +804,9 @@ func TestTerraformToBundleEmptyRemoteResources(t *testing.T) {
assert.Equal(t, "", config.Resources.Schemas["test_schema"].ID)
assert.Equal(t, resources.ModifiedStatusCreated, config.Resources.Schemas["test_schema"].ModifiedStatus)

assert.Equal(t, "", config.Resources.Volumes["test_volume"].ID)
assert.Equal(t, resources.ModifiedStatusCreated, config.Resources.Volumes["test_volume"].ModifiedStatus)

AssertFullResourceCoverage(t, &config)
}

Expand Down Expand Up @@ -888,6 +909,18 @@ func TestTerraformToBundleModifiedResources(t *testing.T) {
},
},
},
Volumes: map[string]*resources.Volume{
"test_volume": {
CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{
Name: "test_volume",
},
},
"test_volume_new": {
CreateVolumeRequestContent: &catalog.CreateVolumeRequestContent{
Name: "test_volume_new",
},
},
},
},
}
var tfState = resourcesState{
Expand Down Expand Up @@ -1020,6 +1053,22 @@ func TestTerraformToBundleModifiedResources(t *testing.T) {
{Attributes: stateInstanceAttributes{ID: "2"}},
},
},
{
Type: "databricks_volume",
Mode: "managed",
Name: "test_volume",
Instances: []stateResourceInstance{
{Attributes: stateInstanceAttributes{ID: "1"}},
},
},
{
Type: "databricks_volume",
Mode: "managed",
Name: "test_volume_old",
Instances: []stateResourceInstance{
{Attributes: stateInstanceAttributes{ID: "2"}},
},
},
shreyas-goenka marked this conversation as resolved.
Show resolved Hide resolved
},
}
err := TerraformToBundle(&tfState, &config)
Expand Down Expand Up @@ -1081,6 +1130,13 @@ func TestTerraformToBundleModifiedResources(t *testing.T) {
assert.Equal(t, "", config.Resources.Schemas["test_schema_new"].ID)
assert.Equal(t, resources.ModifiedStatusCreated, config.Resources.Schemas["test_schema_new"].ModifiedStatus)

assert.Equal(t, "1", config.Resources.Volumes["test_volume"].ID)
assert.Equal(t, "", config.Resources.Volumes["test_volume"].ModifiedStatus)
assert.Equal(t, "2", config.Resources.Volumes["test_volume_old"].ID)
assert.Equal(t, resources.ModifiedStatusDeleted, config.Resources.Volumes["test_volume_old"].ModifiedStatus)
assert.Equal(t, "", config.Resources.Volumes["test_volume_new"].ID)
assert.Equal(t, resources.ModifiedStatusCreated, config.Resources.Volumes["test_volume_new"].ModifiedStatus)

AssertFullResourceCoverage(t, &config)
}

Expand Down
2 changes: 2 additions & 0 deletions bundle/deploy/terraform/interpolate.go
Original file line number Diff line number Diff line change
Expand Up @@ -58,6 +58,8 @@ func (m *interpolateMutator) Apply(ctx context.Context, b *bundle.Bundle) diag.D
path = dyn.NewPath(dyn.Key("databricks_quality_monitor")).Append(path[2:]...)
case dyn.Key("schemas"):
path = dyn.NewPath(dyn.Key("databricks_schema")).Append(path[2:]...)
case dyn.Key("volumes"):
path = dyn.NewPath(dyn.Key("databricks_volume")).Append(path[2:]...)
default:
// Trigger "key not found" for unknown resource types.
return dyn.GetByPath(root, path)
Expand Down
2 changes: 2 additions & 0 deletions bundle/deploy/terraform/interpolate_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@ func TestInterpolate(t *testing.T) {
"other_model_serving": "${resources.model_serving_endpoints.other_model_serving.id}",
"other_registered_model": "${resources.registered_models.other_registered_model.id}",
"other_schema": "${resources.schemas.other_schema.id}",
"other_volume": "${resources.volumes.other_volume.id}",
},
Tasks: []jobs.Task{
{
Expand Down Expand Up @@ -67,6 +68,7 @@ func TestInterpolate(t *testing.T) {
assert.Equal(t, "${databricks_model_serving.other_model_serving.id}", j.Tags["other_model_serving"])
assert.Equal(t, "${databricks_registered_model.other_registered_model.id}", j.Tags["other_registered_model"])
assert.Equal(t, "${databricks_schema.other_schema.id}", j.Tags["other_schema"])
assert.Equal(t, "${databricks_volume.other_volume.id}", j.Tags["other_volume"])

m := b.Config.Resources.Models["my_model"]
assert.Equal(t, "my_model", m.Model.Name)
Expand Down
49 changes: 49 additions & 0 deletions bundle/deploy/terraform/tfdyn/convert_volume.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
package tfdyn

import (
"context"
"fmt"

"github.com/databricks/cli/bundle/internal/tf/schema"
"github.com/databricks/cli/libs/dyn"
"github.com/databricks/cli/libs/dyn/convert"
"github.com/databricks/cli/libs/log"
)

// TODO: Articulate the consequences of deleting a UC volume in the prompt message that
// is displayed.
// TODO: What sort of interpolation should be allowed at `artifact_path`? Should it be
// ${volumes.foo.id} or ${volumes.foo.name} or something else?
func convertVolumeResource(ctx context.Context, vin dyn.Value) (dyn.Value, error) {
// Normalize the output value to the target schema.
vout, diags := convert.Normalize(schema.ResourceVolume{}, vin)
for _, diag := range diags {
log.Debugf(ctx, "volume normalization diagnostic: %s", diag.Summary)
}

return vout, nil
}

type volumeConverter struct{}

func (volumeConverter) Convert(ctx context.Context, key string, vin dyn.Value, out *schema.Resources) error {
vout, err := convertVolumeResource(ctx, vin)
if err != nil {
return err
}

// Add the converted resource to the output.
out.Volume[key] = vout.AsAny()

// Configure grants for this resource.
if grants := convertGrantsResource(ctx, vin); grants != nil {
grants.Volume = fmt.Sprintf("${databricks_volume.%s.id}", key)
shreyas-goenka marked this conversation as resolved.
Show resolved Hide resolved
out.Grants["volume_"+key] = grants
}

return nil
}

func init() {
registerConverter("volumes", volumeConverter{})
}
Loading
Loading