2024-10-18 15:42:05 +00:00
|
|
|
package internal
|
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/base64"
|
|
|
|
"testing"
|
|
|
|
|
|
|
|
"github.com/databricks/cli/internal/acc"
|
2024-12-12 12:35:38 +00:00
|
|
|
"github.com/databricks/cli/internal/testutil"
|
2024-10-18 15:42:05 +00:00
|
|
|
"github.com/databricks/cli/libs/dyn"
|
|
|
|
"github.com/databricks/cli/libs/dyn/convert"
|
|
|
|
"github.com/databricks/cli/libs/dyn/merge"
|
|
|
|
"github.com/databricks/databricks-sdk-go/apierr"
|
|
|
|
"github.com/databricks/databricks-sdk-go/service/dashboards"
|
|
|
|
"github.com/databricks/databricks-sdk-go/service/workspace"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/assert"
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Verify that importing a dashboard through the Workspace API retains the identity of the underying resource,
|
|
|
|
// as well as properties exclusively accessible through the dashboards API.
|
|
|
|
func TestAccDashboardAssumptions_WorkspaceImport(t *testing.T) {
|
|
|
|
ctx, wt := acc.WorkspaceTest(t)
|
|
|
|
|
|
|
|
t.Parallel()
|
|
|
|
|
|
|
|
dashboardName := "New Dashboard"
|
|
|
|
dashboardPayload := []byte(`{"pages":[{"name":"2506f97a","displayName":"New Page"}]}`)
|
2024-12-12 12:35:38 +00:00
|
|
|
warehouseId := testutil.GetEnvOrSkipTest(t, "TEST_DEFAULT_WAREHOUSE_ID")
|
2024-10-18 15:42:05 +00:00
|
|
|
|
2024-12-12 21:28:04 +00:00
|
|
|
dir := acc.TemporaryWorkspaceDir(wt, "dashboard-assumptions-")
|
2024-10-18 15:42:05 +00:00
|
|
|
|
|
|
|
dashboard, err := wt.W.Lakeview.Create(ctx, dashboards.CreateDashboardRequest{
|
2024-11-13 13:40:53 +00:00
|
|
|
Dashboard: &dashboards.Dashboard{
|
|
|
|
DisplayName: dashboardName,
|
|
|
|
ParentPath: dir,
|
|
|
|
SerializedDashboard: string(dashboardPayload),
|
|
|
|
WarehouseId: warehouseId,
|
|
|
|
},
|
2024-10-18 15:42:05 +00:00
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
t.Logf("Dashboard ID (per Lakeview API): %s", dashboard.DashboardId)
|
|
|
|
|
|
|
|
// Overwrite the dashboard via the workspace API.
|
|
|
|
{
|
|
|
|
err := wt.W.Workspace.Import(ctx, workspace.Import{
|
|
|
|
Format: workspace.ImportFormatAuto,
|
|
|
|
Path: dashboard.Path,
|
|
|
|
Content: base64.StdEncoding.EncodeToString(dashboardPayload),
|
|
|
|
Overwrite: true,
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cross-check consistency with the workspace object.
|
|
|
|
{
|
|
|
|
obj, err := wt.W.Workspace.GetStatusByPath(ctx, dashboard.Path)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Confirm that the resource ID included in the response is equal to the dashboard ID.
|
|
|
|
require.Equal(t, dashboard.DashboardId, obj.ResourceId)
|
|
|
|
t.Logf("Dashboard ID (per workspace object status): %s", obj.ResourceId)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to overwrite the dashboard via the Lakeview API (and expect failure).
|
|
|
|
{
|
|
|
|
_, err := wt.W.Lakeview.Create(ctx, dashboards.CreateDashboardRequest{
|
2024-11-13 13:40:53 +00:00
|
|
|
Dashboard: &dashboards.Dashboard{
|
|
|
|
DisplayName: dashboardName,
|
|
|
|
ParentPath: dir,
|
|
|
|
SerializedDashboard: string(dashboardPayload),
|
|
|
|
},
|
2024-10-18 15:42:05 +00:00
|
|
|
})
|
|
|
|
require.ErrorIs(t, err, apierr.ErrResourceAlreadyExists)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Retrieve the dashboard object and confirm that only select fields were updated by the import.
|
|
|
|
{
|
|
|
|
previousDashboard := dashboard
|
|
|
|
currentDashboard, err := wt.W.Lakeview.Get(ctx, dashboards.GetDashboardRequest{
|
|
|
|
DashboardId: dashboard.DashboardId,
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Convert the dashboard object to a [dyn.Value] to make comparison easier.
|
|
|
|
previous, err := convert.FromTyped(previousDashboard, dyn.NilValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
current, err := convert.FromTyped(currentDashboard, dyn.NilValue)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Collect updated paths.
|
|
|
|
var updatedFieldPaths []string
|
|
|
|
_, err = merge.Override(previous, current, merge.OverrideVisitor{
|
|
|
|
VisitDelete: func(basePath dyn.Path, left dyn.Value) error {
|
|
|
|
assert.Fail(t, "unexpected delete operation")
|
|
|
|
return nil
|
|
|
|
},
|
|
|
|
VisitInsert: func(basePath dyn.Path, right dyn.Value) (dyn.Value, error) {
|
|
|
|
assert.Fail(t, "unexpected insert operation")
|
|
|
|
return right, nil
|
|
|
|
},
|
|
|
|
VisitUpdate: func(basePath dyn.Path, left, right dyn.Value) (dyn.Value, error) {
|
|
|
|
updatedFieldPaths = append(updatedFieldPaths, basePath.String())
|
|
|
|
return right, nil
|
|
|
|
},
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Confirm that only the expected fields have been updated.
|
|
|
|
assert.ElementsMatch(t, []string{
|
|
|
|
"etag",
|
|
|
|
"update_time",
|
|
|
|
}, updatedFieldPaths)
|
|
|
|
}
|
|
|
|
}
|