2023-11-13 11:29:40 +00:00
|
|
|
package permissions
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
|
|
|
"slices"
|
|
|
|
"strings"
|
|
|
|
|
|
|
|
"github.com/databricks/cli/bundle"
|
2024-03-25 14:18:47 +00:00
|
|
|
"github.com/databricks/cli/libs/diag"
|
2023-11-13 11:29:40 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const CAN_MANAGE = "CAN_MANAGE"
|
|
|
|
const CAN_VIEW = "CAN_VIEW"
|
|
|
|
const CAN_RUN = "CAN_RUN"
|
|
|
|
|
|
|
|
var allowedLevels = []string{CAN_MANAGE, CAN_VIEW, CAN_RUN}
|
|
|
|
var levelsMap = map[string](map[string]string){
|
|
|
|
"jobs": {
|
|
|
|
CAN_MANAGE: "CAN_MANAGE",
|
|
|
|
CAN_VIEW: "CAN_VIEW",
|
|
|
|
CAN_RUN: "CAN_MANAGE_RUN",
|
|
|
|
},
|
|
|
|
"pipelines": {
|
|
|
|
CAN_MANAGE: "CAN_MANAGE",
|
|
|
|
CAN_VIEW: "CAN_VIEW",
|
|
|
|
CAN_RUN: "CAN_RUN",
|
|
|
|
},
|
|
|
|
"mlflow_experiments": {
|
|
|
|
CAN_MANAGE: "CAN_MANAGE",
|
|
|
|
CAN_VIEW: "CAN_READ",
|
|
|
|
},
|
|
|
|
"mlflow_models": {
|
|
|
|
CAN_MANAGE: "CAN_MANAGE",
|
|
|
|
CAN_VIEW: "CAN_READ",
|
|
|
|
},
|
|
|
|
"model_serving_endpoints": {
|
|
|
|
CAN_MANAGE: "CAN_MANAGE",
|
|
|
|
CAN_VIEW: "CAN_VIEW",
|
|
|
|
CAN_RUN: "CAN_QUERY",
|
|
|
|
},
|
2024-09-03 08:35:41 +00:00
|
|
|
"dashboards": {
|
|
|
|
CAN_MANAGE: "CAN_MANAGE",
|
|
|
|
CAN_VIEW: "CAN_READ",
|
|
|
|
},
|
2023-11-13 11:29:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type bundlePermissions struct{}
|
|
|
|
|
|
|
|
func ApplyBundlePermissions() bundle.Mutator {
|
|
|
|
return &bundlePermissions{}
|
|
|
|
}
|
|
|
|
|
2024-03-25 14:18:47 +00:00
|
|
|
func (m *bundlePermissions) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnostics {
|
2023-11-13 11:29:40 +00:00
|
|
|
err := validate(b)
|
|
|
|
if err != nil {
|
2024-03-25 14:18:47 +00:00
|
|
|
return diag.FromErr(err)
|
2023-11-13 11:29:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
applyForJobs(ctx, b)
|
|
|
|
applyForPipelines(ctx, b)
|
|
|
|
applyForMlModels(ctx, b)
|
|
|
|
applyForMlExperiments(ctx, b)
|
|
|
|
applyForModelServiceEndpoints(ctx, b)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func validate(b *bundle.Bundle) error {
|
|
|
|
for _, p := range b.Config.Permissions {
|
|
|
|
if !slices.Contains(allowedLevels, p.Level) {
|
|
|
|
return fmt.Errorf("invalid permission level: %s, allowed values: [%s]", p.Level, strings.Join(allowedLevels, ", "))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func applyForJobs(ctx context.Context, b *bundle.Bundle) {
|
2023-12-22 14:45:53 +00:00
|
|
|
for key, job := range b.Config.Resources.Jobs {
|
2023-11-13 11:29:40 +00:00
|
|
|
job.Permissions = append(job.Permissions, convert(
|
|
|
|
ctx,
|
|
|
|
b.Config.Permissions,
|
|
|
|
job.Permissions,
|
2023-12-22 14:45:53 +00:00
|
|
|
key,
|
2023-11-13 11:29:40 +00:00
|
|
|
levelsMap["jobs"],
|
|
|
|
)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func applyForPipelines(ctx context.Context, b *bundle.Bundle) {
|
2023-12-22 14:45:53 +00:00
|
|
|
for key, pipeline := range b.Config.Resources.Pipelines {
|
2023-11-13 11:29:40 +00:00
|
|
|
pipeline.Permissions = append(pipeline.Permissions, convert(
|
|
|
|
ctx,
|
|
|
|
b.Config.Permissions,
|
|
|
|
pipeline.Permissions,
|
2023-12-22 14:45:53 +00:00
|
|
|
key,
|
2023-11-13 11:29:40 +00:00
|
|
|
levelsMap["pipelines"],
|
|
|
|
)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func applyForMlExperiments(ctx context.Context, b *bundle.Bundle) {
|
2023-12-22 14:45:53 +00:00
|
|
|
for key, experiment := range b.Config.Resources.Experiments {
|
2023-11-13 11:29:40 +00:00
|
|
|
experiment.Permissions = append(experiment.Permissions, convert(
|
|
|
|
ctx,
|
|
|
|
b.Config.Permissions,
|
|
|
|
experiment.Permissions,
|
2023-12-22 14:45:53 +00:00
|
|
|
key,
|
2023-11-13 11:29:40 +00:00
|
|
|
levelsMap["mlflow_experiments"],
|
|
|
|
)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func applyForMlModels(ctx context.Context, b *bundle.Bundle) {
|
2023-12-22 14:45:53 +00:00
|
|
|
for key, model := range b.Config.Resources.Models {
|
2023-11-13 11:29:40 +00:00
|
|
|
model.Permissions = append(model.Permissions, convert(
|
|
|
|
ctx,
|
|
|
|
b.Config.Permissions,
|
|
|
|
model.Permissions,
|
2023-12-22 14:45:53 +00:00
|
|
|
key,
|
2023-11-13 11:29:40 +00:00
|
|
|
levelsMap["mlflow_models"],
|
|
|
|
)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func applyForModelServiceEndpoints(ctx context.Context, b *bundle.Bundle) {
|
2023-12-22 14:45:53 +00:00
|
|
|
for key, model := range b.Config.Resources.ModelServingEndpoints {
|
2023-11-13 11:29:40 +00:00
|
|
|
model.Permissions = append(model.Permissions, convert(
|
|
|
|
ctx,
|
|
|
|
b.Config.Permissions,
|
|
|
|
model.Permissions,
|
2023-12-22 14:45:53 +00:00
|
|
|
key,
|
2023-11-13 11:29:40 +00:00
|
|
|
levelsMap["model_serving_endpoints"],
|
|
|
|
)...)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *bundlePermissions) Name() string {
|
|
|
|
return "ApplyBundlePermissions"
|
|
|
|
}
|