From ea7c8a0811d71404d9eeccb469623dc3b6aa216b Mon Sep 17 00:00:00 2001 From: William Dumont Date: Thu, 9 Jan 2025 10:08:31 +0100 Subject: [PATCH] Refactor of the value cache (#2151) * Merge scope hierarchy for evaluation * get rid of parent scope * refactor value_cache * review feedback * review feedback * make post evaluation errors more explicit --- .../runtime/internal/controller/loader.go | 52 ++-- .../internal/controller/value_cache.go | 227 +++++++++-------- .../internal/controller/value_cache_test.go | 228 ++++++++++++++++-- syntax/vm/vm.go | 23 +- 4 files changed, 369 insertions(+), 161 deletions(-) diff --git a/internal/runtime/internal/controller/loader.go b/internal/runtime/internal/controller/loader.go index 9af5919722..8ba483093a 100644 --- a/internal/runtime/internal/controller/loader.go +++ b/internal/runtime/internal/controller/loader.go @@ -151,7 +151,9 @@ func (l *Loader) Apply(options ApplyOptions) diag.Diagnostics { l.cm.controllerEvaluation.Set(1) defer l.cm.controllerEvaluation.Set(0) - l.cache.SetScope(options.ArgScope) + if options.ArgScope != nil { + l.cache.UpdateScopeVariables(options.ArgScope.Variables) + } for key, value := range options.Args { l.cache.CacheModuleArgument(key, value) @@ -168,7 +170,7 @@ func (l *Loader) Apply(options ApplyOptions) diag.Diagnostics { var ( components = make([]ComponentNode, 0) - componentIDs = make([]ComponentID, 0) + componentIDs = make(map[string]ComponentID) services = make([]*ServiceNode, 0, len(l.services)) ) @@ -202,7 +204,7 @@ func (l *Loader) Apply(options ApplyOptions) diag.Diagnostics { switch n := n.(type) { case ComponentNode: components = append(components, n) - componentIDs = append(componentIDs, n.ID()) + componentIDs[n.ID().String()] = n.ID() if err = l.evaluate(logger, n); err != nil { var evalDiags diag.Diagnostics @@ -262,7 +264,15 @@ func (l *Loader) Apply(options ApplyOptions) diag.Diagnostics { l.componentNodes = components l.serviceNodes = services l.graph = &newGraph - l.cache.SyncIDs(componentIDs) + err := l.cache.SyncIDs(componentIDs) + if err != nil { + diags.Add(diag.Diagnostic{ + Severity: diag.SeverityLevelError, + Message: fmt.Sprintf("Failed to update the internal cache with the new list of components: %s", err), + }) + // return it now because there is no point to process further + return diags + } l.blocks = options.ComponentBlocks if l.globals.OnExportsChange != nil && l.cache.ExportChangeIndex() != l.moduleExportIndex { l.moduleExportIndex = l.cache.ExportChangeIndex() @@ -617,7 +627,7 @@ func (l *Loader) wireGraphEdges(g *dag.Graph) diag.Diagnostics { // Finally, wire component references. l.cache.mut.RLock() - refs, nodeDiags := ComponentReferences(n, g, l.log, l.cache.scope, l.globals.MinStability) + refs, nodeDiags := ComponentReferences(n, g, l.log, l.cache.GetContext(), l.globals.MinStability) l.cache.mut.RUnlock() for _, ref := range refs { g.AddEdge(dag.Edge{From: n, To: ref.Target}) @@ -647,7 +657,7 @@ func (l *Loader) wireCustomComponentNode(g *dag.Graph, cc *CustomComponentNode) // Variables returns the Variables the Loader exposes for other components to // reference. func (l *Loader) Variables() map[string]interface{} { - return l.cache.BuildContext().Variables + return l.cache.GetContext().Variables } // Components returns the current set of loaded components. @@ -705,7 +715,10 @@ func (l *Loader) EvaluateDependants(ctx context.Context, updatedNodes []*QueuedN switch parentNode := parent.Node.(type) { case ComponentNode: // Make sure we're in-sync with the current exports of parent. - l.cache.CacheExports(parentNode.ID(), parentNode.Exports()) + err := l.cache.CacheExports(parentNode.ID(), parentNode.Exports()) + if err != nil { + level.Error(l.log).Log("msg", "failed to cache exports during evaluation", "err", err) + } case *ImportConfigNode: // Update the scope with the imported content. l.componentNodeManager.customComponentReg.updateImportContent(parentNode) @@ -794,7 +807,7 @@ func (l *Loader) concurrentEvalFn(n dag.Node, spanCtx context.Context, tracer tr var err error switch n := n.(type) { case BlockNode: - ectx := l.cache.BuildContext() + ectx := l.cache.GetContext() // RLock before evaluate to prevent Evaluating while the config is being reloaded l.mut.RLock() @@ -833,22 +846,33 @@ func (l *Loader) concurrentEvalFn(n dag.Node, spanCtx context.Context, tracer tr // evaluate constructs the final context for the BlockNode and // evaluates it. mut must be held when calling evaluate. func (l *Loader) evaluate(logger log.Logger, bn BlockNode) error { - ectx := l.cache.BuildContext() + ectx := l.cache.GetContext() err := bn.Evaluate(ectx) return l.postEvaluate(logger, bn, err) } // postEvaluate is called after a node has been evaluated. It updates the caches and logs any errors. // mut must be held when calling postEvaluate. +// The evaluation err is passed as an argument to allow shadowing it with an error that could be more relevant to the user +// but cannot be determined before the evaluation (for example, we must evaluate the argument node to see if it's optional before +// raising an error when a value is missing). When err is not nil, this function must return an error. func (l *Loader) postEvaluate(logger log.Logger, bn BlockNode, err error) error { switch c := bn.(type) { case ComponentNode: - // Always update the cache both the arguments and exports, since both might - // change when a component gets re-evaluated. We also want to cache the arguments and exports in case of an error - l.cache.CacheArguments(c.ID(), c.Arguments()) - l.cache.CacheExports(c.ID(), c.Exports()) + // Always update the cached exports, since that it might change when a component gets re-evaluated. + // We also want to cache it in case of an error + err2 := l.cache.CacheExports(c.ID(), c.Exports()) + if err2 != nil { + if err != nil { + level.Error(logger).Log("msg", "evaluation and exports caching failed", "eval err", err, "caching err", err2) + return errors.Join(err, err2) + } else { + level.Error(logger).Log("msg", "failed to cache exports after evaluation", "err", err2) + return err2 + } + } case *ArgumentConfigNode: - if _, found := l.cache.moduleArguments[c.Label()]; !found { + if _, found := l.cache.GetModuleArgument(c.Label()); !found { if c.Optional() { l.cache.CacheModuleArgument(c.Label(), c.Default()) } else { diff --git a/internal/runtime/internal/controller/value_cache.go b/internal/runtime/internal/controller/value_cache.go index 6aae014ba9..009c592d74 100644 --- a/internal/runtime/internal/controller/value_cache.go +++ b/internal/runtime/internal/controller/value_cache.go @@ -1,6 +1,7 @@ package controller import ( + "fmt" "reflect" "sync" @@ -8,18 +9,17 @@ import ( "github.com/grafana/alloy/syntax/vm" ) -// valueCache caches component arguments and exports to expose as variables for -// Alloy expressions. -// -// The current state of valueCache can then be built into a *vm.Scope for other -// components to be evaluated. +// This special keyword is used to expose the argument values to the custom components. +const argumentLabel = "argument" + +// valueCache caches exports and module arguments to expose as variables for Alloy expressions. +// It also caches module exports to expose them to the parent loader. +// The exports are stored directly in the scope which is used to evaluate Alloy expressions. type valueCache struct { mut sync.RWMutex - components map[string]ComponentID // NodeID -> ComponentID - args map[string]interface{} // NodeID -> component arguments value - exports map[string]interface{} // NodeID -> component exports value - moduleArguments map[string]any // key -> module arguments value - moduleExports map[string]any // name -> value for the value of module exports + componentIds map[string]ComponentID // NodeID -> ComponentID + moduleExports map[string]any // Export label -> Export value + moduleArguments map[string]any // Argument label -> Map with the key "value" that points to the Argument value moduleChangedIndex int // Everytime a change occurs this is incremented scope *vm.Scope // scope provides additional context for the nodes in the module } @@ -27,50 +27,54 @@ type valueCache struct { // newValueCache creates a new ValueCache. func newValueCache() *valueCache { return &valueCache{ - components: make(map[string]ComponentID), - args: make(map[string]interface{}), - exports: make(map[string]interface{}), - moduleArguments: make(map[string]any), + componentIds: make(map[string]ComponentID, 0), moduleExports: make(map[string]any), + moduleArguments: make(map[string]any), + scope: vm.NewScope(make(map[string]any)), } } -func (vc *valueCache) SetScope(scope *vm.Scope) { - vc.mut.Lock() - defer vc.mut.Unlock() - vc.scope = scope -} - -// CacheArguments will cache the provided arguments by the given id. args may -// be nil to store an empty object. -func (vc *valueCache) CacheArguments(id ComponentID, args component.Arguments) { +// UpdateScopeVariables updates the Variables map of the scope with a deep copy of the provided map. +func (vc *valueCache) UpdateScopeVariables(variables map[string]any) { + if variables == nil { + return + } vc.mut.Lock() defer vc.mut.Unlock() - - nodeID := id.String() - vc.components[nodeID] = id - - var argsVal interface{} = make(map[string]interface{}) - if args != nil { - argsVal = args - } - vc.args[nodeID] = argsVal + vc.scope.Variables = deepCopyMap(variables) } // CacheExports will cache the provided exports using the given id. exports may // be nil to store an empty object. -func (vc *valueCache) CacheExports(id ComponentID, exports component.Exports) { +func (vc *valueCache) CacheExports(id ComponentID, exports component.Exports) error { vc.mut.Lock() defer vc.mut.Unlock() - nodeID := id.String() - vc.components[nodeID] = id + variables := vc.scope.Variables + // Build nested maps. + for i := 0; i < len(id)-1; i++ { + t := id[i] + if _, ok := variables[t]; !ok { + variables[t] = make(map[string]any) + } else if _, ok := variables[t].(map[string]any); !ok { + return fmt.Errorf("expected a map but found a value for %q when trying to cache the export for %s", t, id.String()) + } + variables = variables[t].(map[string]any) + } - var exportsVal interface{} = make(map[string]interface{}) + var exportsVal any = make(map[string]any) if exports != nil { exportsVal = exports } - vc.exports[nodeID] = exportsVal + variables[id[len(id)-1]] = exportsVal + return nil +} + +func (vc *valueCache) GetModuleArgument(key string) (any, bool) { + vc.mut.RLock() + defer vc.mut.RUnlock() + v, exist := vc.moduleArguments[key] + return v, exist } // CacheModuleArgument will cache the provided exports using the given id. @@ -78,11 +82,9 @@ func (vc *valueCache) CacheModuleArgument(key string, value any) { vc.mut.Lock() defer vc.mut.Unlock() - if value == nil { - vc.moduleArguments[key] = nil - } else { - vc.moduleArguments[key] = value - } + keyMap := make(map[string]any) + keyMap["value"] = value + vc.moduleArguments[key] = keyMap } // CacheModuleExportValue saves the value to the map @@ -130,107 +132,102 @@ func (vc *valueCache) ExportChangeIndex() int { return vc.moduleChangedIndex } -// SyncIDs will remove any cached values for any Component ID which is not in -// ids. SyncIDs should be called with the current set of components after the -// graph is updated. -func (vc *valueCache) SyncIDs(ids []ComponentID) { - expectMap := make(map[string]ComponentID, len(ids)) - for _, id := range ids { - expectMap[id.String()] = id - } - +// SyncIDs will remove any cached values for any Component ID from the graph which is not in ids. +// SyncIDs should be called with the current set of components after the graph is updated. +func (vc *valueCache) SyncIDs(ids map[string]ComponentID) error { vc.mut.Lock() defer vc.mut.Unlock() - for id := range vc.components { - if _, keep := expectMap[id]; keep { - continue + // Find the components that should be removed. + cleanupIds := make([]ComponentID, 0) + for name, id := range vc.componentIds { + if _, exist := ids[name]; !exist { + cleanupIds = append(cleanupIds, id) } - delete(vc.components, id) - delete(vc.args, id) - delete(vc.exports, id) } -} - -// SyncModuleArgs will remove any cached values for any args no longer in the map. -func (vc *valueCache) SyncModuleArgs(args map[string]any) { - vc.mut.Lock() - defer vc.mut.Unlock() - for id := range vc.moduleArguments { - if _, keep := args[id]; keep { - continue + // Remove the component exports from the scope. + for _, id := range cleanupIds { + err := cleanup(vc.scope.Variables, id) + if err != nil { + return fmt.Errorf("failed to sync component %s: %w", id.String(), err) } - delete(vc.moduleArguments, id) } + vc.componentIds = ids + return nil } -// BuildContext builds a vm.Scope based on the current set of cached values. -// The arguments and exports for the same ID are merged into one object. -func (vc *valueCache) BuildContext() *vm.Scope { - vc.mut.RLock() - defer vc.mut.RUnlock() +// cleanup removes the ComponentID path from the map +func cleanup(m map[string]any, id ComponentID) error { + // Start with the index "0". It refers to the first part of the componentID and it's used for recursion. + return cleanupFromIndex(m, id, 0) +} - scope := vm.NewScopeWithParent(vc.scope, make(map[string]interface{})) +func cleanupFromIndex(m map[string]any, id ComponentID, index int) error { - // First, partition components by Alloy block name. - var componentsByBlockName = make(map[string][]ComponentID) - for _, id := range vc.components { - blockName := id[0] - componentsByBlockName[blockName] = append(componentsByBlockName[blockName], id) + if _, ok := m[id[index]]; !ok { + return nil } - // Then, convert each partition into a single value. - for blockName, ids := range componentsByBlockName { - scope.Variables[blockName] = vc.buildValue(ids, 1) + if index == len(id)-1 { + delete(m, id[index]) // Remove the component's exports. + return nil } - // Add module arguments to the scope. - if len(vc.moduleArguments) > 0 { - scope.Variables["argument"] = make(map[string]any) + if _, ok := m[id[index]].(map[string]any); !ok { + return fmt.Errorf("expected a map but found a value for %q", id[index]) } - for key, value := range vc.moduleArguments { - keyMap := make(map[string]any) - keyMap["value"] = value + nextM := m[id[index]].(map[string]any) - switch args := scope.Variables["argument"].(type) { - case map[string]any: - args[key] = keyMap - } + err := cleanupFromIndex(nextM, id, index+1) + if err != nil { + return err } - return scope + // Delete if the map at this level is empty. + // If you only have one Prometheus component and you remove it, it will cleanup the full Prometheus path. + // If you have one Prometheus relabel and one Prometheus scrape, and you remove the relabel, it will cleanup the relabel path. + if len(nextM) == 0 { + delete(m, id[index]) + } + return nil } -// buildValue recursively converts the set of user components into a single -// value. offset is used to determine which element in the userComponentName -// we're looking at. -func (vc *valueCache) buildValue(from []ComponentID, offset int) interface{} { - // We can't recurse anymore; return the node directly. - if len(from) == 1 && offset >= len(from[0]) { - name := from[0].String() +// SyncModuleArgs will remove any cached values for any args no longer in the map. +func (vc *valueCache) SyncModuleArgs(args map[string]any) { + vc.mut.Lock() + defer vc.mut.Unlock() - // TODO(rfratto): should we allow arguments to be returned so users can - // reference arguments as well as exports? - exports, ok := vc.exports[name] - if !ok { - exports = make(map[string]interface{}) + for arg := range vc.moduleArguments { + if _, ok := args[arg]; !ok { + delete(vc.moduleArguments, arg) } - return exports } +} - attrs := make(map[string]interface{}) +// GetContext returns a scope that can be used for evaluation. +func (vc *valueCache) GetContext() *vm.Scope { + vc.mut.RLock() + defer vc.mut.RUnlock() + vars := deepCopyMap(vc.scope.Variables) - // First, partition the components by their label. - var componentsByLabel = make(map[string][]ComponentID) - for _, id := range from { - blockName := id[offset] - componentsByLabel[blockName] = append(componentsByLabel[blockName], id) + // Add module arguments if there are any. + if len(vc.moduleArguments) > 0 { + vars[argumentLabel] = deepCopyMap(vc.moduleArguments) } - // Then, convert each partition into a single value. - for label, ids := range componentsByLabel { - attrs[label] = vc.buildValue(ids, offset+1) + return vm.NewScope(vars) +} + +func deepCopyMap(original map[string]any) map[string]any { + newMap := make(map[string]any, len(original)) + for key, value := range original { + switch v := value.(type) { + case map[string]any: + newMap[key] = deepCopyMap(v) + default: + newMap[key] = v + } } - return attrs + return newMap } diff --git a/internal/runtime/internal/controller/value_cache_test.go b/internal/runtime/internal/controller/value_cache_test.go index eebcd9b642..e31baf6a18 100644 --- a/internal/runtime/internal/controller/value_cache_test.go +++ b/internal/runtime/internal/controller/value_cache_test.go @@ -3,6 +3,7 @@ package controller import ( "testing" + "github.com/grafana/alloy/syntax/vm" "github.com/stretchr/testify/require" ) @@ -53,15 +54,12 @@ func TestValueCache(t *testing.T) { // For now, only exports are placed in generated objects, which is why the // bar values are empty and the foo object only contains the exports. - vc.CacheArguments(ComponentID{"foo"}, fooArgs{Something: true}) - vc.CacheExports(ComponentID{"foo"}, fooExports{SomethingElse: true}) - vc.CacheArguments(ComponentID{"bar", "label_a"}, barArgs{Number: 12}) - vc.CacheArguments(ComponentID{"bar", "label_b"}, barArgs{Number: 34}) + require.NoError(t, vc.CacheExports(ComponentID{"foo"}, fooExports{SomethingElse: true})) - res := vc.BuildContext() + res := vc.GetContext() var ( - expectKeys = []string{"foo", "bar"} + expectKeys = []string{"foo"} actualKeys []string ) for varName := range res.Variables { @@ -72,12 +70,7 @@ func TestValueCache(t *testing.T) { type object = map[string]interface{} expectFoo := fooExports{SomethingElse: true} - expectBar := object{ - "label_a": object{}, // no exports for bar - "label_b": object{}, // no exports for bar - } require.Equal(t, expectFoo, res.Variables["foo"]) - require.Equal(t, expectBar, res.Variables["bar"]) } func TestExportValueCache(t *testing.T) { @@ -156,21 +149,228 @@ func TestModuleArgumentCache(t *testing.T) { vc.CacheModuleArgument("arg", tc.argValue) // Build the scope and validate it - res := vc.BuildContext() + res := vc.GetContext() expected := map[string]any{"arg": map[string]any{"value": tc.argValue}} require.Equal(t, expected, res.Variables["argument"]) // Sync arguments where the arg shouldn't change syncArgs := map[string]any{"arg": tc.argValue} vc.SyncModuleArgs(syncArgs) - res = vc.BuildContext() + res = vc.GetContext() require.Equal(t, expected, res.Variables["argument"]) // Sync arguments where the arg should clear out syncArgs = map[string]any{} vc.SyncModuleArgs(syncArgs) - res = vc.BuildContext() + res = vc.GetContext() require.Equal(t, map[string]any{}, res.Variables) }) } } + +func TestScope(t *testing.T) { + vc := newValueCache() + vc.scope = vm.NewScope( + map[string]any{ + "test": map[string]any{ + "scope": barArgs{Number: 13}, + }, + }, + ) + require.NoError(t, vc.CacheExports(ComponentID{"foo", "bar"}, barArgs{Number: 12})) + res := vc.GetContext() + + expected := map[string]any{ + "test": map[string]any{ + "scope": barArgs{ + Number: 13, + }, + }, + "foo": map[string]any{ + "bar": barArgs{ + Number: 12, + }, + }, + } + require.Equal(t, expected, res.Variables) +} + +func TestScopeSameNamespace(t *testing.T) { + vc := newValueCache() + vc.scope = vm.NewScope( + map[string]any{ + "test": map[string]any{ + "scope": barArgs{Number: 13}, + }, + }, + ) + require.NoError(t, vc.CacheExports(ComponentID{"test", "bar"}, barArgs{Number: 12})) + res := vc.GetContext() + + expected := map[string]any{ + "test": map[string]any{ + "scope": barArgs{ + Number: 13, + }, + "bar": barArgs{ + Number: 12, + }, + }, + } + require.Equal(t, expected, res.Variables) +} + +func TestScopeOverride(t *testing.T) { + vc := newValueCache() + vc.scope = vm.NewScope( + map[string]any{ + "test": map[string]any{ + "scope": barArgs{Number: 13}, + }, + }, + ) + require.NoError(t, vc.CacheExports(ComponentID{"test", "scope"}, barArgs{Number: 12})) + res := vc.GetContext() + + expected := map[string]any{ + "test": map[string]any{ + "scope": barArgs{ + Number: 12, + }, + }, + } + require.Equal(t, expected, res.Variables) +} + +func TestScopePathOverrideError(t *testing.T) { + vc := newValueCache() + vc.scope = vm.NewScope( + map[string]any{ + "test": barArgs{Number: 13}, + }, + ) + require.ErrorContains(t, + vc.CacheExports(ComponentID{"test", "scope"}, barArgs{Number: 12}), + "expected a map but found a value for \"test\" when trying to cache the export for test.scope", + ) +} + +func TestScopeComplex(t *testing.T) { + vc := newValueCache() + vc.scope = vm.NewScope( + map[string]any{ + "test": map[string]any{ + "cp1": map[string]any{ + "scope": barArgs{Number: 13}, + }, + "cp2": barArgs{Number: 12}, + }, + }, + ) + require.NoError(t, vc.CacheExports(ComponentID{"test", "cp1", "foo"}, barArgs{Number: 12})) + require.NoError(t, vc.CacheExports(ComponentID{"test", "cp1", "bar", "fizz"}, barArgs{Number: 2})) + res := vc.GetContext() + + expected := map[string]any{ + "test": map[string]any{ + "cp1": map[string]any{ + "scope": barArgs{Number: 13}, + "foo": barArgs{Number: 12}, + "bar": map[string]any{ + "fizz": barArgs{Number: 2}, + }, + }, + "cp2": barArgs{Number: 12}, + }, + } + require.Equal(t, expected, res.Variables) +} + +func TestSyncIds(t *testing.T) { + vc := newValueCache() + vc.scope = vm.NewScope( + map[string]any{ + "test": map[string]any{ + "cp1": map[string]any{ + "scope": barArgs{Number: 13}, + }, + "cp2": barArgs{Number: 12}, + }, + "test2": map[string]any{ + "cp1": map[string]any{ + "scope": barArgs{Number: 13}, + }, + }, + "test3": 5, + }, + ) + require.NoError(t, vc.CacheExports(ComponentID{"test", "cp1", "bar", "fizz"}, barArgs{Number: 2})) + originalIds := map[string]ComponentID{ + "test.cp1": {"test", "cp1"}, + "test.cp2": {"test", "cp2"}, + "test2.cp1": {"test2", "cp1"}, + "test3": {"test3"}, + } + require.NoError(t, vc.SyncIDs(originalIds)) + require.Equal(t, originalIds, vc.componentIds) + + newIds := map[string]ComponentID{ + "test.cp1": {"test", "cp1"}, + "test2.cp1": {"test2", "cp1"}, + } + require.NoError(t, vc.SyncIDs(newIds)) + require.Equal(t, newIds, vc.componentIds) + expected := map[string]any{ + "test": map[string]any{ + "cp1": map[string]any{ + "scope": barArgs{Number: 13}, + "bar": map[string]any{ + "fizz": barArgs{Number: 2}, + }, + }, + }, + "test2": map[string]any{ + "cp1": map[string]any{ + "scope": barArgs{Number: 13}, + }, + }, + } + res := vc.GetContext() + require.Equal(t, expected, res.Variables) +} + +func TestSyncIdsError(t *testing.T) { + vc := newValueCache() + componentID := ComponentID{"test", "cp1", "bar", "fizz"} + ids := map[string]ComponentID{"test.cp1.bar.fizz": componentID} + require.NoError(t, vc.CacheExports(componentID, barArgs{Number: 2})) + require.NoError(t, vc.SyncIDs(ids)) + require.Equal(t, ids, vc.componentIds) + + // Modify the map manually + vc.componentIds = map[string]ComponentID{"test.cp1.bar.fizz.foo": {"test", "cp1", "bar", "fizz", "foo"}} + require.ErrorContains(t, + vc.SyncIDs(ids), + "failed to sync component test.cp1.bar.fizz.foo: expected a map but found a value for \"fizz\"", + ) +} + +func TestCapsule(t *testing.T) { + vc := newValueCache() + bar := barArgs{Number: 13} + vc.scope = vm.NewScope( + map[string]any{ + "test": map[string]any{ + "scope": &bar, + }, + }, + ) + res := vc.GetContext() + + expected := map[string]any{ + "test": map[string]any{ + "scope": &bar, + }, + } + require.Equal(t, expected, res.Variables) +} diff --git a/syntax/vm/vm.go b/syntax/vm/vm.go index f8d0e44341..8bb7d6a999 100644 --- a/syntax/vm/vm.go +++ b/syntax/vm/vm.go @@ -36,7 +36,7 @@ func New(node ast.Node) *Evaluator { // // Each call to Evaluate may provide a different scope with new values for // available variables. If a variable used by the Evaluator's node isn't -// defined in scope or any of the parent scopes, Evaluate will return an error. +// defined in scope, Evaluate will return an error. func (vm *Evaluator) Evaluate(scope *Scope, v interface{}) (err error) { // Track a map that allows us to associate values with ast.Nodes so we can // return decorated error messages. @@ -455,11 +455,6 @@ func (vm *Evaluator) evaluateExpr(scope *Scope, assoc map[value.Value]ast.Node, // A Scope exposes a set of variables available to use during evaluation. type Scope struct { - // Parent optionally points to a parent Scope containing more variable. - // Variables defined in children scopes take precedence over variables of the - // same name found in parent scopes. - Parent *Scope - // Variables holds the list of available variable names that can be used when // evaluating a node. // @@ -475,23 +470,15 @@ func NewScope(variables map[string]interface{}) *Scope { } } -func NewScopeWithParent(parent *Scope, variables map[string]interface{}) *Scope { - return &Scope{ - Parent: parent, - Variables: variables, - } -} - -// Lookup looks up a named identifier from the scope, all of the scope's -// parents, and the stdlib. +// Lookup looks up a named identifier from the scope and the stdlib. func (s *Scope) Lookup(name string) (interface{}, bool) { - // Traverse the scope first, then fall back to stdlib. - for s != nil { + // Check the scope first. + if s != nil { if val, ok := s.Variables[name]; ok { return val, true } - s = s.Parent } + // Falls back to the stdlib. if ident, ok := stdlib.Identifiers[name]; ok { return ident, true }