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

Resolve variables in a loop #2164

Merged
merged 9 commits into from
Jan 16, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
12 changes: 12 additions & 0 deletions acceptance/bundle/variables/complex-cross-ref/databricks.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,12 @@
bundle:
name: complex-cross-ref

variables:
a:
default:
a_1: 500
a_2: ${var.b.b_2}
b:
default:
b_1: ${var.a.a_1}
b_2: 2.5
22 changes: 22 additions & 0 deletions acceptance/bundle/variables/complex-cross-ref/output.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,22 @@
{
"a": {
"default": {
"a_1": 500,
"a_2": 2.5
},
"value": {
"a_1": 500,
"a_2": 2.5
}
},
"b": {
"default": {
"b_1": 500,
"b_2": 2.5
},
"value": {
"b_1": 500,
"b_2": 2.5
}
}
}
1 change: 1 addition & 0 deletions acceptance/bundle/variables/complex-cross-ref/script
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
$CLI bundle validate -o json | jq .variables
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
bundle:
name: cycle

variables:
a:
default:
hello: ${var.a}
9 changes: 9 additions & 0 deletions acceptance/bundle/variables/complex-cycle-self/output.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
Warning: Detected unresolved variables after 11 resolution rounds

Name: cycle
Target: default
Workspace:
User: $USERNAME
Path: /Workspace/Users/$USERNAME/.bundle/cycle/default

Found 1 warning
1 change: 1 addition & 0 deletions acceptance/bundle/variables/complex-cycle-self/script
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
$CLI bundle validate
10 changes: 10 additions & 0 deletions acceptance/bundle/variables/complex-cycle/databricks.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
bundle:
name: cycle

variables:
a:
default:
hello: ${var.b}
b:
default:
hello: ${var.a}
9 changes: 9 additions & 0 deletions acceptance/bundle/variables/complex-cycle/output.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
Warning: Detected unresolved variables after 11 resolution rounds

Name: cycle
Target: default
Workspace:
User: $USERNAME
Path: /Workspace/Users/$USERNAME/.bundle/cycle/default

Found 1 warning
1 change: 1 addition & 0 deletions acceptance/bundle/variables/complex-cycle/script
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
$CLI bundle validate
Original file line number Diff line number Diff line change
@@ -1,3 +1,3 @@
{
"spark.databricks.sql.initial.catalog.name": "${var.catalog}"
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can you add a test for cases when there are nested cycle dependencies?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

See acceptance/bundle/variables/complex-cycle/databricks.yml

"spark.databricks.sql.initial.catalog.name": "hive_metastore"
}
8 changes: 8 additions & 0 deletions acceptance/bundle/variables/cycle/databricks.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
bundle:
name: cycle

variables:
a:
Copy link
Contributor

@andrewnester andrewnester Jan 16, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I meant a deeper level of nesting for cycle :) something like


variables:
  a:
    default: ${var.complex.a}
  b:
    default: ${var.a}
  c: 
    default: ${var.b}
  complex:
    default: 
        a: ${var.c}
        b: foo

default: ${var.b}
b:
default: ${var.a}
14 changes: 14 additions & 0 deletions acceptance/bundle/variables/cycle/output.txt
Original file line number Diff line number Diff line change
@@ -0,0 +1,14 @@
Error: cycle detected in field resolution: variables.a.default -> var.b -> var.a -> var.b

{
"a": {
"default": "${var.b}",
"value": "${var.b}"
},
"b": {
"default": "${var.a}",
"value": "${var.a}"
}
}

Exit code: 1
1 change: 1 addition & 0 deletions acceptance/bundle/variables/cycle/script
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
$CLI bundle validate -o json | jq .variables
66 changes: 60 additions & 6 deletions bundle/config/mutator/resolve_variable_references.go
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ package mutator
import (
"context"
"errors"
"fmt"

"github.com/databricks/cli/bundle"
"github.com/databricks/cli/bundle/config"
Expand All @@ -13,15 +14,37 @@ import (
"github.com/databricks/cli/libs/dyn/dynvar"
)

/*
For pathological cases, output and time grow exponentially.

On my laptop, timings for acceptance/bundle/variables/complex-cycle:
rounds time

9 0.10s
10 0.13s
11 0.27s
12 0.68s
13 1.98s
14 6.28s
15 21.70s
16 78.16s
*/
const maxResolutionRounds = 11

type resolveVariableReferences struct {
prefixes []string
pattern dyn.Pattern
lookupFn func(dyn.Value, dyn.Path, *bundle.Bundle) (dyn.Value, error)
skipFn func(dyn.Value) bool
prefixes []string
pattern dyn.Pattern
lookupFn func(dyn.Value, dyn.Path, *bundle.Bundle) (dyn.Value, error)
skipFn func(dyn.Value) bool
extraRounds int
}

func ResolveVariableReferences(prefixes ...string) bundle.Mutator {
return &resolveVariableReferences{prefixes: prefixes, lookupFn: lookup}
return &resolveVariableReferences{
prefixes: prefixes,
lookupFn: lookup,
extraRounds: maxResolutionRounds - 1,
}
}

func ResolveVariableReferencesInLookup() bundle.Mutator {
Expand Down Expand Up @@ -86,7 +109,36 @@ func (m *resolveVariableReferences) Apply(ctx context.Context, b *bundle.Bundle)
varPath := dyn.NewPath(dyn.Key("var"))

var diags diag.Diagnostics
maxRounds := 1 + m.extraRounds

for round := range maxRounds {
hasUpdates, newDiags := m.resolveOnce(b, prefixes, varPath)

diags = diags.Extend(newDiags)

if diags.HasError() {
break
}

if !hasUpdates {
break
}

if round >= maxRounds-1 {
diags = diags.Append(diag.Diagnostic{
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

shall we break here as well?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Well it's the last stmt in the last loop iteration, so it does not matter.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

it's not necessarily last one if I pass extraRounds bigger than maxResolutionRounds

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

good catch! it's a bug. c3ac311

Severity: diag.Warning,
Summary: fmt.Sprintf("Detected unresolved variables after %d resolution rounds", round+1),
// Would be nice to include names of the variables there, but that would complicate things more
})
break
}
}
return diags
}

func (m *resolveVariableReferences) resolveOnce(b *bundle.Bundle, prefixes []dyn.Path, varPath dyn.Path) (bool, diag.Diagnostics) {
var diags diag.Diagnostics
hasUpdates := false
err := b.Config.Mutate(func(root dyn.Value) (dyn.Value, error) {
// Synthesize a copy of the root that has all fields that are present in the type
// but not set in the dynamic value set to their corresponding empty value.
Expand Down Expand Up @@ -129,6 +181,7 @@ func (m *resolveVariableReferences) Apply(ctx context.Context, b *bundle.Bundle)
if m.skipFn != nil && m.skipFn(v) {
return dyn.InvalidValue, dynvar.ErrSkipResolution
}
hasUpdates = true
return m.lookupFn(normalized, path, b)
}
}
Expand All @@ -149,5 +202,6 @@ func (m *resolveVariableReferences) Apply(ctx context.Context, b *bundle.Bundle)
if err != nil {
diags = diags.Extend(diag.FromErr(err))
}
return diags

return hasUpdates, diags
}
5 changes: 0 additions & 5 deletions bundle/phases/initialize.go
Original file line number Diff line number Diff line change
Expand Up @@ -66,11 +66,6 @@ func Initialize() bundle.Mutator {
"workspace",
"variables",
),
mutator.ResolveVariableReferences(
"bundle",
"workspace",
"variables",
),

mutator.MergeJobClusters(),
mutator.MergeJobParameters(),
Expand Down
Loading