-
Notifications
You must be signed in to change notification settings - Fork 57
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 configurable presets for name prefixes, tags, etc. #1490
Merged
lennartkats-db
merged 29 commits into
databricks:main
from
lennartkats-db:cp-mutator-settings
Aug 19, 2024
Merged
Changes from 20 commits
Commits
Show all changes
29 commits
Select commit
Hold shift + click to select a range
e5ac74d
Initial draft of customizable transformation
lennartkats-db 6eaee84
Refactor to one function per transformer
lennartkats-db 601c32b
WIP
lennartkats-db b16c18c
Remove enabled fields
lennartkats-db 901097a
Add config merging & test
lennartkats-db a815f30
e2e fixes
lennartkats-db 13630dd
Cleanup
lennartkats-db e3b0435
Use PAUSED/UNPAUSED instead of a boolean
lennartkats-db 10a1ffc
Rename to transform for now
lennartkats-db 405e202
Merge remote-tracking branch 'databricks/main' into cp-mutator-settings
lennartkats-db 7323d02
Cleanup
lennartkats-db 4dc5f41
Add stricter validations
lennartkats-db 82e1d49
Cleanup
lennartkats-db 6d75e84
Simply mutations, no need for dyn here
lennartkats-db 40b004e
Cleanup
lennartkats-db 29a23cf
Merge remote-tracking branch 'databricks/main' into cp-mutator-settings
lennartkats-db b353a2f
Rename to presets
lennartkats-db f636e09
Allow tags to merge instead of override
lennartkats-db 347e24e
Fix test
lennartkats-db 3e003c0
Pause continuous pipelines when 'mode: development' is used
lennartkats-db 40f3bb4
Use extension configuration
lennartkats-db b1427b3
Address reviewer comments, fix names
lennartkats-db f2553ff
Merge remote-tracking branch 'databricks/main' into cp-mutator-settings
lennartkats-db fb902c9
Fix regression in main
lennartkats-db 6159c3c
Merge remote-tracking branch 'databricks/main' into cp-mutator-settings
lennartkats-db b4564f2
Use bundle.Apply() for tests
lennartkats-db 70d8988
Add assertion
lennartkats-db a073f84
Merge remote-tracking branch 'databricks/main' into cp-mutator-settings
lennartkats-db b9e3278
Merge remote-tracking branch 'databricks/main' into cp-mutator-settings
lennartkats-db File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,211 @@ | ||
package mutator | ||
|
||
import ( | ||
"context" | ||
"path" | ||
"slices" | ||
"sort" | ||
"strings" | ||
|
||
"github.com/databricks/cli/bundle" | ||
"github.com/databricks/cli/bundle/config" | ||
"github.com/databricks/cli/libs/diag" | ||
"github.com/databricks/cli/libs/textutil" | ||
"github.com/databricks/databricks-sdk-go/service/catalog" | ||
"github.com/databricks/databricks-sdk-go/service/jobs" | ||
"github.com/databricks/databricks-sdk-go/service/ml" | ||
) | ||
|
||
type applyPresets struct{} | ||
|
||
// Apply all presets, e.g. the prefix presets that | ||
// adds a prefix to all names of all resources. | ||
func ApplyPresets() *applyPresets { | ||
return &applyPresets{} | ||
} | ||
|
||
type Tag struct { | ||
Key string | ||
Value string | ||
} | ||
|
||
func (m *applyPresets) Name() string { | ||
return "ApplyPresets" | ||
} | ||
|
||
func (m *applyPresets) Apply(ctx context.Context, b *bundle.Bundle) diag.Diagnostics { | ||
if d := validatePauseStatus(b); d != nil { | ||
return d | ||
} | ||
if d := validateTags(b); d != nil { | ||
return d | ||
} | ||
|
||
r := b.Config.Resources | ||
t := b.Config.Presets | ||
prefix := t.Prefix | ||
tags := toTagArray(t.Tags) | ||
|
||
// Jobs presets: Prefix, Tags, JobsMaxConcurrentRuns, TriggerPauseStatus | ||
for _, j := range r.Jobs { | ||
j.Name = prefix + j.Name | ||
if j.Tags == nil { | ||
j.Tags = make(map[string]string) | ||
} | ||
for _, tag := range tags { | ||
if j.Tags[tag.Key] == "" { | ||
j.Tags[tag.Key] = tag.Value | ||
} | ||
} | ||
if j.MaxConcurrentRuns == 0 { | ||
j.MaxConcurrentRuns = t.JobsMaxConcurrentRuns | ||
} | ||
if t.TriggerPauseStatus != "" { | ||
paused := jobs.PauseStatusPaused | ||
if t.TriggerPauseStatus == config.Unpaused { | ||
paused = jobs.PauseStatusUnpaused | ||
} | ||
|
||
if j.Schedule != nil && j.Schedule.PauseStatus == "" { | ||
j.Schedule.PauseStatus = paused | ||
} | ||
if j.Continuous != nil && j.Continuous.PauseStatus == "" { | ||
j.Continuous.PauseStatus = paused | ||
} | ||
if j.Trigger != nil && j.Trigger.PauseStatus == "" { | ||
j.Trigger.PauseStatus = paused | ||
} | ||
} | ||
} | ||
|
||
// Pipelines presets: Prefix, PipelinesDevelopment | ||
for i := range r.Pipelines { | ||
r.Pipelines[i].Name = prefix + r.Pipelines[i].Name | ||
if config.IsExplicitlyEnabled(t.PipelinesDevelopment) { | ||
r.Pipelines[i].Development = true | ||
} | ||
if t.TriggerPauseStatus == config.Paused { | ||
r.Pipelines[i].Continuous = false | ||
} | ||
|
||
// As of 2024-06, pipelines don't yet support tags | ||
} | ||
|
||
// Models presets: Prefix, Tags | ||
for _, m := range r.Models { | ||
m.Name = prefix + m.Name | ||
for _, t := range tags { | ||
exists := slices.ContainsFunc(m.Tags, func(modelTag ml.ModelTag) bool { | ||
return modelTag.Key == t.Key | ||
}) | ||
if !exists { | ||
// Only add this tag if the resource didn't include any tag that overrides its value. | ||
m.Tags = append(m.Tags, ml.ModelTag{Key: t.Key, Value: t.Value}) | ||
} | ||
} | ||
} | ||
|
||
// Experiments presets: Prefix, Tags | ||
for _, e := range r.Experiments { | ||
filepath := e.Name | ||
dir := path.Dir(filepath) | ||
base := path.Base(filepath) | ||
if dir == "." { | ||
e.Name = prefix + base | ||
} else { | ||
e.Name = dir + "/" + prefix + base | ||
} | ||
for _, t := range tags { | ||
exists := false | ||
for _, experimentTag := range e.Tags { | ||
if experimentTag.Key == t.Key { | ||
exists = true | ||
break | ||
} | ||
} | ||
if !exists { | ||
e.Tags = append(e.Tags, ml.ExperimentTag{Key: t.Key, Value: t.Value}) | ||
} | ||
} | ||
} | ||
|
||
// Model serving endpoint presets: Prefix | ||
for i := range r.ModelServingEndpoints { | ||
r.ModelServingEndpoints[i].Name = normalizePrefix(prefix) + r.ModelServingEndpoints[i].Name | ||
|
||
// As of 2024-06, model serving endpoints don't yet support tags | ||
} | ||
|
||
// Registered models presets: Prefix | ||
for i := range r.RegisteredModels { | ||
r.RegisteredModels[i].Name = normalizePrefix(prefix) + r.RegisteredModels[i].Name | ||
|
||
// As of 2024-06, registered models don't yet support tags | ||
} | ||
|
||
// Quality monitors presets: Prefix | ||
if t.TriggerPauseStatus == config.Paused { | ||
for i := range r.QualityMonitors { | ||
// Remove all schedules from monitors, since they don't support pausing/unpausing. | ||
// Quality monitors might support the "pause" property in the future, so at the | ||
// CLI level we do respect that property if it is set to "unpaused." | ||
if r.QualityMonitors[i].Schedule != nil && r.QualityMonitors[i].Schedule.PauseStatus != catalog.MonitorCronSchedulePauseStatusUnpaused { | ||
r.QualityMonitors[i].Schedule = nil | ||
} | ||
} | ||
} | ||
|
||
return nil | ||
} | ||
|
||
func validatePauseStatus(b *bundle.Bundle) diag.Diagnostics { | ||
p := b.Config.Presets.TriggerPauseStatus | ||
if p == "" || p == config.Paused || p == config.Unpaused { | ||
return nil | ||
} | ||
return diag.Diagnostics{{ | ||
Summary: "Invalid value for trigger_pause_status, should be PAUSED or UNPAUSED", | ||
pietern marked this conversation as resolved.
Show resolved
Hide resolved
|
||
Severity: diag.Error, | ||
Location: b.Config.GetLocation("presets.trigger_pause_status"), | ||
}} | ||
} | ||
|
||
func validateTags(b *bundle.Bundle) diag.Diagnostics { | ||
tags := b.Config.Presets.Tags | ||
for key := range tags { | ||
// Tags keys starting with a "~" are reserved. | ||
// They may someday be used to indicate that a tag should be removed. | ||
// For example {"~dev:": nil} could indicate that the "dev" tag should be removed. | ||
lennartkats-db marked this conversation as resolved.
Show resolved
Hide resolved
|
||
if strings.HasPrefix(key, "~") { | ||
return diag.Diagnostics{{ | ||
Summary: "Invalid tag key: " + key, | ||
Severity: diag.Error, | ||
}} | ||
} | ||
} | ||
return nil | ||
} | ||
|
||
// toTagArray converts a map of tags to an array of tags. | ||
// We sort tags so ensure stable ordering. | ||
func toTagArray(tags map[string]string) []Tag { | ||
var tagArray []Tag | ||
if tags == nil { | ||
return tagArray | ||
} | ||
for key, value := range tags { | ||
tagArray = append(tagArray, Tag{Key: key, Value: value}) | ||
} | ||
sort.Slice(tagArray, func(i, j int) bool { | ||
return tagArray[i].Key < tagArray[j].Key | ||
}) | ||
return tagArray | ||
} | ||
|
||
// normalizePrefix prefixes strings like '[dev lennart] ' to 'dev_lennart_'. | ||
// We leave unicode letters and numbers but remove all "special characters." | ||
func normalizePrefix(prefix string) string { | ||
prefix = strings.ReplaceAll(prefix, "[", "") | ||
prefix = strings.ReplaceAll(prefix, "] ", "_") | ||
return textutil.NormalizeString(prefix) | ||
} |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
It could be "foobar" and be treated as paused. To improve readability you can use a switch/case on
t.TriggerPauseStatus
here.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
We have a validation for the
TriggerPauseStatus
property above though, so we would show a fatal error if a user uses "foobar". And this way the code here can be a bit more concise.