diff --git a/.changeset/blue-pumpkins-sniff.md b/.changeset/blue-pumpkins-sniff.md new file mode 100644 index 00000000000..0a7576f328e --- /dev/null +++ b/.changeset/blue-pumpkins-sniff.md @@ -0,0 +1,5 @@ +--- +"chainlink": minor +--- + +#internal move workflow validation to common repo diff --git a/core/services/workflows/engine.go b/core/services/workflows/engine.go index 2b497057ada..447339b5e7f 100644 --- a/core/services/workflows/engine.go +++ b/core/services/workflows/engine.go @@ -14,6 +14,7 @@ import ( "github.com/smartcontractkit/chainlink-common/pkg/services" "github.com/smartcontractkit/chainlink-common/pkg/types/core" "github.com/smartcontractkit/chainlink-common/pkg/values" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" "github.com/smartcontractkit/chainlink/v2/core/logger" p2ptypes "github.com/smartcontractkit/chainlink/v2/core/services/p2p/types" "github.com/smartcontractkit/chainlink/v2/core/services/workflows/store" @@ -29,6 +30,11 @@ type donInfo struct { PeerID func() *p2ptypes.PeerID } +type stepRequest struct { + stepRef string + state store.WorkflowExecution +} + // Engine handles the lifecycle of a single workflow and its executions. type Engine struct { services.StateMachine @@ -103,11 +109,11 @@ func (e *Engine) resolveWorkflowCapabilities(ctx context.Context) error { // - fetching the capability // - register the capability to this workflow // - initializing the step's executionStrategy - capabilityRegistrationErr := e.workflow.walkDo(keywordTrigger, func(s *step) error { + capabilityRegistrationErr := e.workflow.walkDo(workflows.KeywordTrigger, func(s *step) error { // The graph contains a dummy step for triggers, but // we handle triggers separately since there might be more than one // trigger registered to a workflow. - if s.Ref == keywordTrigger { + if s.Ref == workflows.KeywordTrigger { return nil } @@ -441,13 +447,13 @@ func (e *Engine) startExecution(ctx context.Context, executionID string, event v e.logger.Debugw("executing on a trigger event", "event", event, "executionID", executionID) ec := &store.WorkflowExecution{ Steps: map[string]*store.WorkflowExecutionStep{ - keywordTrigger: { + workflows.KeywordTrigger: { Outputs: &store.StepOutput{ Value: event, }, Status: store.StatusCompleted, ExecutionID: executionID, - Ref: keywordTrigger, + Ref: workflows.KeywordTrigger, }, }, WorkflowID: e.workflow.id, @@ -463,7 +469,7 @@ func (e *Engine) startExecution(ctx context.Context, executionID string, event v // Find the tasks we need to fire when a trigger has fired and enqueue them. // This consists of a) nodes without a dependency and b) nodes which depend // on a trigger - triggerDependents, err := e.workflow.dependents(keywordTrigger) + triggerDependents, err := e.workflow.dependents(workflows.KeywordTrigger) if err != nil { return err } @@ -492,7 +498,7 @@ func (e *Engine) handleStepUpdate(ctx context.Context, stepUpdate store.Workflow // we've completed the workflow. if len(stepDependents) == 0 { workflowCompleted := true - err := e.workflow.walkDo(keywordTrigger, func(s *step) error { + err := e.workflow.walkDo(workflows.KeywordTrigger, func(s *step) error { step, ok := state.Steps[s.Ref] // The step is missing from the state, // which means it hasn't been processed yet. @@ -543,7 +549,7 @@ func (e *Engine) handleStepUpdate(ctx context.Context, stepUpdate store.Workflow func (e *Engine) queueIfReady(state store.WorkflowExecution, step *step) { // Check if all dependencies are completed for the current step var waitingOnDependencies bool - for _, dr := range step.dependencies { + for _, dr := range step.Vertex.Dependencies { stepState, ok := state.Steps[dr] if !ok { waitingOnDependencies = true @@ -701,8 +707,8 @@ func (e *Engine) Close() error { close(e.stopCh) e.wg.Wait() - err := e.workflow.walkDo(keywordTrigger, func(s *step) error { - if s.Ref == keywordTrigger { + err := e.workflow.walkDo(workflows.KeywordTrigger, func(s *step) error { + if s.Ref == workflows.KeywordTrigger { return nil } diff --git a/core/services/workflows/engine_test.go b/core/services/workflows/engine_test.go index 1ad7a3c2ae2..6abd241e66c 100644 --- a/core/services/workflows/engine_test.go +++ b/core/services/workflows/engine_test.go @@ -13,6 +13,7 @@ import ( "github.com/smartcontractkit/chainlink-common/pkg/capabilities" "github.com/smartcontractkit/chainlink-common/pkg/values" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" coreCap "github.com/smartcontractkit/chainlink/v2/core/capabilities" "github.com/smartcontractkit/chainlink/v2/core/internal/testutils" "github.com/smartcontractkit/chainlink/v2/core/internal/testutils/pgtest" @@ -547,13 +548,13 @@ func TestEngine_ResumesPendingExecutions(t *testing.T) { dbstore := store.NewDBStore(pgtest.NewSqlxDB(t), clockwork.NewFakeClock()) ec := &store.WorkflowExecution{ Steps: map[string]*store.WorkflowExecutionStep{ - keywordTrigger: { + workflows.KeywordTrigger: { Outputs: &store.StepOutput{ Value: resp, }, Status: store.StatusCompleted, ExecutionID: "", - Ref: keywordTrigger, + Ref: workflows.KeywordTrigger, }, }, WorkflowID: "", @@ -602,13 +603,13 @@ func TestEngine_TimesOutOldExecutions(t *testing.T) { dbstore := store.NewDBStore(pgtest.NewSqlxDB(t), clock) ec := &store.WorkflowExecution{ Steps: map[string]*store.WorkflowExecutionStep{ - keywordTrigger: { + workflows.KeywordTrigger: { Outputs: &store.StepOutput{ Value: resp, }, Status: store.StatusCompleted, ExecutionID: "", - Ref: keywordTrigger, + Ref: workflows.KeywordTrigger, }, }, WorkflowID: "", diff --git a/core/services/workflows/models.go b/core/services/workflows/models.go index dadadc8ba0e..ac157e04b40 100644 --- a/core/services/workflows/models.go +++ b/core/services/workflows/models.go @@ -1,49 +1,15 @@ package workflows import ( - "errors" "fmt" "github.com/dominikbraun/graph" "github.com/smartcontractkit/chainlink-common/pkg/capabilities" "github.com/smartcontractkit/chainlink-common/pkg/values" - "github.com/smartcontractkit/chainlink/v2/core/services/workflows/store" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" ) -type stepRequest struct { - stepRef string - state store.WorkflowExecution -} - -// StepDefinition is the parsed representation of a step in a workflow. -// -// Within the workflow spec, they are called "Capability Properties". -type StepDefinition struct { - ID string `json:"id" jsonschema:"required"` - Ref string `json:"ref,omitempty" jsonschema:"pattern=^[a-z0-9_]+$"` - Inputs map[string]any `json:"inputs,omitempty"` - Config map[string]any `json:"config" jsonschema:"required"` - - CapabilityType capabilities.CapabilityType `json:"-"` -} - -// WorkflowSpec is the parsed representation of a workflow. -type WorkflowSpec struct { - Triggers []StepDefinition `json:"triggers" jsonschema:"required"` - Actions []StepDefinition `json:"actions,omitempty"` - Consensus []StepDefinition `json:"consensus" jsonschema:"required"` - Targets []StepDefinition `json:"targets" jsonschema:"required"` -} - -func (w *WorkflowSpec) Steps() []StepDefinition { - s := []StepDefinition{} - s = append(s, w.Actions...) - s = append(s, w.Consensus...) - s = append(s, w.Targets...) - return s -} - // workflow is a directed graph of nodes, where each node is a step. // // triggers are special steps that are stored separately, they're @@ -55,7 +21,7 @@ type workflow struct { triggers []*triggerCapability - spec *WorkflowSpec + spec *workflows.WorkflowSpec } func (w *workflow) walkDo(start string, do func(s *step) error) error { @@ -108,144 +74,29 @@ func (w *workflow) dependents(start string) ([]*step, error) { // step wraps a Vertex with additional context for execution that is mutated by the engine type step struct { - Vertex + workflows.Vertex capability capabilities.CallbackCapability config *values.Map executionStrategy executionStrategy } -type Vertex struct { - StepDefinition - dependencies []string -} - -// DependencyGraph is an intermediate representation of a workflow wherein all the graph -// vertices are represented and validated. It is a static representation of the workflow dependencies. -type DependencyGraph struct { - ID string - graph.Graph[string, *Vertex] - - Triggers []*StepDefinition - - Spec *WorkflowSpec -} - -// VID is an identifier for a Vertex that can be used to uniquely identify it in a graph. -// it represents the notion `hash` in the graph package AddVertex method. -// we refrain from naming it `hash` to avoid confusion with the hash function. -func (v *Vertex) VID() string { - return v.Ref -} - type triggerCapability struct { - StepDefinition + workflows.StepDefinition trigger capabilities.TriggerCapability config *values.Map } -const ( - keywordTrigger = "trigger" -) - func Parse(yamlWorkflow string) (*workflow, error) { - wf2, err := ParseDepedencyGraph(yamlWorkflow) + wf2, err := workflows.ParseDependencyGraph(yamlWorkflow) if err != nil { return nil, err } return createWorkflow(wf2) } -func ParseDepedencyGraph(yamlWorkflow string) (*DependencyGraph, error) { - spec, err := ParseWorkflowSpecYaml(yamlWorkflow) - if err != nil { - return nil, err - } - - // Construct and validate the graph. We instantiate an - // empty graph with just one starting entry: `trigger`. - // This provides the starting point for our graph and - // points to all dependent steps. - // Note: all triggers are represented by a single step called - // `trigger`. This is because for workflows with multiple triggers - // only one trigger will have started the workflow. - stepHash := func(s *Vertex) string { - return s.VID() - } - g := graph.New( - stepHash, - graph.PreventCycles(), - graph.Directed(), - ) - err = g.AddVertex(&Vertex{ - StepDefinition: StepDefinition{Ref: keywordTrigger}, - }) - if err != nil { - return nil, err - } - - // Next, let's populate the other entries in the graph. - for _, s := range spec.Steps() { - // TODO: The workflow format spec doesn't always require a `Ref` - // to be provided (triggers and targets don't have a `Ref` for example). - // To handle this, we default the `Ref` to the type, but ideally we - // should find a better long-term way to handle this. - if s.Ref == "" { - s.Ref = s.ID - } - - innerErr := g.AddVertex(&Vertex{StepDefinition: s}) - if innerErr != nil { - return nil, fmt.Errorf("cannot add vertex %s: %w", s.Ref, innerErr) - } - } - - stepRefs, err := g.AdjacencyMap() - if err != nil { - return nil, err - } - - // Next, let's iterate over the steps and populate - // any edges. - for stepRef := range stepRefs { - step, innerErr := g.Vertex(stepRef) - if innerErr != nil { - return nil, innerErr - } - - refs, innerErr := findRefs(step.Inputs) - if innerErr != nil { - return nil, innerErr - } - step.dependencies = refs - - if stepRef != keywordTrigger && len(refs) == 0 { - return nil, errors.New("all non-trigger steps must have a dependent ref") - } - - for _, r := range refs { - innerErr = g.AddEdge(r, step.Ref) - if innerErr != nil { - return nil, innerErr - } - } - } - - triggerSteps := []*StepDefinition{} - for _, t := range spec.Triggers { - tt := t - triggerSteps = append(triggerSteps, &tt) - } - wf := &DependencyGraph{ - Spec: &spec, - Graph: g, - Triggers: triggerSteps, - } - return wf, err -} - // createWorkflow converts a StaticWorkflow to an executable workflow // by adding metadata to the vertices that is owned by the workflow runtime. -func createWorkflow(wf2 *DependencyGraph) (*workflow, error) { +func createWorkflow(wf2 *workflows.DependencyGraph) (*workflow, error) { out := &workflow{ id: wf2.ID, triggers: []*triggerCapability{}, diff --git a/core/services/workflows/models_test.go b/core/services/workflows/models_test.go index 0964b13d277..6bc74ab109a 100644 --- a/core/services/workflows/models_test.go +++ b/core/services/workflows/models_test.go @@ -5,6 +5,8 @@ import ( "github.com/stretchr/testify/assert" "github.com/stretchr/testify/require" + + "github.com/smartcontractkit/chainlink-common/pkg/workflows" ) func TestParse_Graph(t *testing.T) { @@ -41,7 +43,7 @@ targets: consensus_output: $(a-consensus.outputs) `, graph: map[string]map[string]struct{}{ - keywordTrigger: { + workflows.KeywordTrigger: { "an-action": struct{}{}, "a-consensus": struct{}{}, }, @@ -175,7 +177,7 @@ targets: consensus_output: $(a-consensus.outputs) `, graph: map[string]map[string]struct{}{ - keywordTrigger: { + workflows.KeywordTrigger: { "an-action": struct{}{}, }, "an-action": { @@ -240,3 +242,13 @@ targets: }) } } + +func TestParsesIntsCorrectly(t *testing.T) { + wf, err := Parse(hardcodedWorkflow) + require.NoError(t, err) + + n, err := wf.Vertex("evm_median") + require.NoError(t, err) + + assert.Equal(t, int64(3600), n.Config["aggregation_config"].(map[string]any)["0x1111111111111111111100000000000000000000000000000000000000000000"].(map[string]any)["heartbeat"]) +} diff --git a/core/services/workflows/models_yaml.go b/core/services/workflows/models_yaml.go deleted file mode 100644 index 90d3f109c06..00000000000 --- a/core/services/workflows/models_yaml.go +++ /dev/null @@ -1,342 +0,0 @@ -package workflows - -import ( - "bytes" - "encoding/json" - "fmt" - "slices" - "strings" - - "github.com/invopop/jsonschema" - "github.com/shopspring/decimal" - "sigs.k8s.io/yaml" - - "github.com/smartcontractkit/chainlink-common/pkg/capabilities" -) - -func GenerateJsonSchema() ([]byte, error) { - schema := jsonschema.Reflect(&workflowSpecYaml{}) - - return json.MarshalIndent(schema, "", " ") -} - -func ParseWorkflowSpecYaml(data string) (WorkflowSpec, error) { - w := workflowSpecYaml{} - err := yaml.Unmarshal([]byte(data), &w) - - return w.toWorkflowSpec(), err -} - -// workflowSpecYaml is the YAML representation of a workflow spec. -// -// It allows for multiple ways of defining a workflow spec, which we later -// convert to a single representation, `workflowSpec`. -type workflowSpecYaml struct { - // Triggers define a starting condition for the workflow, based on specific events or conditions. - Triggers []stepDefinitionYaml `json:"triggers" jsonschema:"required"` - // Actions represent a discrete operation within the workflow, potentially transforming input data. - Actions []stepDefinitionYaml `json:"actions,omitempty"` - // Consensus encapsulates the logic for aggregating and validating the results from various nodes. - Consensus []stepDefinitionYaml `json:"consensus" jsonschema:"required"` - // Targets represents the final step of the workflow, delivering the processed data to a specified location. - Targets []stepDefinitionYaml `json:"targets" jsonschema:"required"` -} - -// toWorkflowSpec converts a workflowSpecYaml to a workflowSpec. -// -// We support multiple ways of defining a workflow spec yaml, -// but internally we want to work with a single representation. -func (w workflowSpecYaml) toWorkflowSpec() WorkflowSpec { - triggers := make([]StepDefinition, 0, len(w.Triggers)) - for _, t := range w.Triggers { - sd := t.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeTrigger - triggers = append(triggers, sd) - } - - actions := make([]StepDefinition, 0, len(w.Actions)) - for _, a := range w.Actions { - sd := a.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeAction - actions = append(actions, sd) - } - - consensus := make([]StepDefinition, 0, len(w.Consensus)) - for _, c := range w.Consensus { - sd := c.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeConsensus - consensus = append(consensus, sd) - } - - targets := make([]StepDefinition, 0, len(w.Targets)) - for _, t := range w.Targets { - sd := t.toStepDefinition() - sd.CapabilityType = capabilities.CapabilityTypeTarget - targets = append(targets, sd) - } - - return WorkflowSpec{ - Triggers: triggers, - Actions: actions, - Consensus: consensus, - Targets: targets, - } -} - -type mapping map[string]any - -func (m *mapping) UnmarshalJSON(b []byte) error { - mp := map[string]any{} - - d := json.NewDecoder(bytes.NewReader(b)) - d.UseNumber() - - err := d.Decode(&mp) - if err != nil { - return err - } - - nm, err := convertNumbers(mp) - if err != nil { - return err - } - - *m = (mapping)(nm) - return err -} - -func convertNumber(el any) (any, error) { - switch elv := el.(type) { - case json.Number: - if strings.Contains(elv.String(), ".") { - f, err := elv.Float64() - if err == nil { - return decimal.NewFromFloat(f), nil - } - } - - return elv.Int64() - default: - return el, nil - } -} - -func convertNumbers(m map[string]any) (map[string]any, error) { - nm := map[string]any{} - for k, v := range m { - switch tv := v.(type) { - case map[string]any: - cm, err := convertNumbers(tv) - if err != nil { - return nil, err - } - - nm[k] = cm - case []any: - na := make([]any, len(tv)) - for i, v := range tv { - cv, err := convertNumber(v) - if err != nil { - return nil, err - } - - na[i] = cv - } - - nm[k] = na - default: - cv, err := convertNumber(v) - if err != nil { - return nil, err - } - - nm[k] = cv - } - } - - return nm, nil -} - -func (m mapping) MarshalJSON() ([]byte, error) { - return json.Marshal(map[string]any(m)) -} - -// stepDefinitionYaml is the YAML representation of a step in a workflow. -// -// It allows for multiple ways of defining a step, which we later -// convert to a single representation, `stepDefinition`. -type stepDefinitionYaml struct { - // A universally unique name for a capability will be defined under the “id” property. The uniqueness will, eventually, be enforced in the Capability Registry. - // - // Semver must be used to specify the version of the Capability at the end of the id field. Capability versions must be immutable. - // - // Initially, we will require major versions. This will ease upgrades early on while we develop the infrastructure. - // - // Eventually, we might support minor version and specific version pins. This will allow workflow authors to have flexibility when selecting the version, and node operators will be able to determine when they should update their capabilities. - // - // There are two ways to specify an id - using a string as a fully qualified ID or a structured table. When using a table, labels are ordered alphanumerically and joined into a string following a - // {name}:{label1_key}_{label1_value}:{label2_key}_{label2_value}@{version} - // pattern. - // - // The “id” supports [a-z0-9_-:] characters followed by an @ and [semver regex] at the end. - // - // Validation must throw an error if: - // - // Unsupported characters are used. - // (For Keystone only.) More specific than a major version is specified. - // - // Example (string) - // id: read_chain:chain_ethereum:network_mainnet@1 - // - // Example (table) - // - // id: - // name: read_chain - // version: 1 - // labels: - // chain: ethereum - // network: mainnet - // - // [semver regex]: https://semver.org/#is-there-a-suggested-regular-expression-regex-to-check-a-semver-string - ID stepDefinitionID `json:"id" jsonschema:"required"` - - // Actions and Consensus capabilities have a required “ref” property that must be unique within a Workflow file (not universally) This property enables referencing outputs and is required because Actions and Consensus always need to be referenced in the following phases. Triggers can optionally specify if they need to be referenced. - // - // The “ref” supports [a-z0-9_] characters. - // - // Validation must throw an error if: - // - Unsupported characters are used. - // - The same “ref” appears in the workflow multiple times. - // - “ref” is used on a Target capability. - // - “ref” has a circular reference. - // - // NOTE: Should introduce a custom validator to cover trigger case - Ref string `json:"ref,omitempty" jsonschema:"pattern=^[a-z0-9_-]+$"` - - // Capabilities can specify an additional optional ”inputs” property. It allows specifying a dependency on the result of one or more other capabilities. These are always runtime values that cannot be provided upfront. It takes a map of the argument name internal to the capability and an explicit reference to the values. - // - // References are specified using the [id].[ref].[path_to_value] pattern. - // - // The interpolation of “inputs” is allowed - // - // Validation must throw an error if: - // - Input reference cannot be resolved. - // - Input is defined on triggers - // NOTE: Should introduce a custom validator to cover trigger case - Inputs mapping `json:"inputs,omitempty"` - - // The configuration of a Capability will be done using the “config” property. Each capability is responsible for defining an external interface used during setup. This interface may be unique or identical, meaning multiple Capabilities might use the same configuration properties. - // - // The interpolation of “inputs” - // - // Interpolation of self inputs is allowed from within the “config” property. - // - // Example - // targets: - // - id: write_polygon_mainnet@1 - // inputs: - // report: - // - consensus.evm_median.outputs.report - // config: - // address: "0xaabbcc" - // method: "updateFeedValues(report bytes, role uint8)" - // params: [$(inputs.report), 1] - Config mapping `json:"config" jsonschema:"required"` -} - -// toStepDefinition converts a stepDefinitionYaml to a stepDefinition. -// -// `stepDefinition` is the converged representation of a step in a workflow. -func (s stepDefinitionYaml) toStepDefinition() StepDefinition { - return StepDefinition{ - Ref: s.Ref, - ID: s.ID.String(), - Inputs: s.Inputs, - Config: s.Config, - } -} - -// stepDefinitionID represents both the string and table representations of the "id" field in a stepDefinition. -type stepDefinitionID struct { - idStr string - idTable *stepDefinitionTableID -} - -func (s stepDefinitionID) String() string { - if s.idStr != "" { - return s.idStr - } - - return s.idTable.String() -} - -func (s *stepDefinitionID) UnmarshalJSON(data []byte) error { - // Unmarshal the JSON data into a map to determine if it's a string or a table - var m string - err := json.Unmarshal(data, &m) - if err == nil { - s.idStr = m - return nil - } - - // If the JSON data is a table, unmarshal it into a stepDefinitionTableID - var table stepDefinitionTableID - err = json.Unmarshal(data, &table) - if err != nil { - return err - } - s.idTable = &table - return nil -} - -func (s *stepDefinitionID) MarshalJSON() ([]byte, error) { - if s.idStr != "" { - return json.Marshal(s.idStr) - } - - return json.Marshal(s.idTable) -} - -// JSONSchema returns the JSON schema for a stepDefinitionID. -// -// The schema is a oneOf schema that allows either a string or a table. -func (stepDefinitionID) JSONSchema() *jsonschema.Schema { - reflector := jsonschema.Reflector{DoNotReference: true, ExpandedStruct: true} - tableSchema := reflector.Reflect(&stepDefinitionTableID{}) - stringSchema := &jsonschema.Schema{ - ID: "string", - Pattern: "^[a-z0-9_\\-:]+@(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$", - } - - return &jsonschema.Schema{ - Title: "id", - OneOf: []*jsonschema.Schema{ - stringSchema, - tableSchema, - }, - } -} - -// stepDefinitionTableID is the structured representation of a stepDefinitionID. -type stepDefinitionTableID struct { - Name string `json:"name"` - Version string `json:"version" jsonschema:"pattern=(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$"` - Labels map[string]string `json:"labels"` -} - -// String returns the string representation of a stepDefinitionTableID. -// -// It follows the format: -// -// {name}:{label1_key}_{label1_value}:{label2_key}_{label2_value}@{version} -// -// where labels are ordered alphanumerically. -func (s stepDefinitionTableID) String() string { - labels := make([]string, 0, len(s.Labels)) - for k, v := range s.Labels { - labels = append(labels, fmt.Sprintf("%s_%s", k, v)) - } - slices.Sort(labels) - - return fmt.Sprintf("%s:%s@%s", s.Name, strings.Join(labels, ":"), s.Version) -} diff --git a/core/services/workflows/models_yaml_test.go b/core/services/workflows/models_yaml_test.go deleted file mode 100644 index 5fa326dda5d..00000000000 --- a/core/services/workflows/models_yaml_test.go +++ /dev/null @@ -1,259 +0,0 @@ -package workflows - -import ( - "encoding/json" - "fmt" - "os" - "testing" - - "github.com/google/go-cmp/cmp" - "github.com/santhosh-tekuri/jsonschema/v5" - "github.com/shopspring/decimal" - "github.com/stretchr/testify/assert" - "github.com/stretchr/testify/require" - - "sigs.k8s.io/yaml" -) - -var fixtureDir = "./testdata/fixtures/workflows/" - -// yamlFixtureReaderObj reads a yaml fixture file and returns the parsed object -func yamlFixtureReaderObj(t *testing.T, testCase string) func(name string) any { - testFixtureReader := yamlFixtureReaderBytes(t, testCase) - - return func(name string) any { - testFileBytes := testFixtureReader(name) - - var testFileYaml any - err := yaml.Unmarshal(testFileBytes, &testFileYaml) - require.NoError(t, err) - - return testFileYaml - } -} - -// yamlFixtureReaderBytes reads a yaml fixture file and returns the bytes -func yamlFixtureReaderBytes(t *testing.T, testCase string) func(name string) []byte { - return func(name string) []byte { - testFileBytes, err := os.ReadFile(fmt.Sprintf(fixtureDir+"%s/%s.yaml", testCase, name)) - require.NoError(t, err) - - return testFileBytes - } -} - -var transformJSON = cmp.FilterValues(func(x, y []byte) bool { - return json.Valid(x) && json.Valid(y) -}, cmp.Transformer("ParseJSON", func(in []byte) (out interface{}) { - if err := json.Unmarshal(in, &out); err != nil { - panic(err) // should never occur given previous filter to ensure valid JSON - } - return out -})) - -func TestWorkflowSpecMarshalling(t *testing.T) { - t.Parallel() - fixtureReader := yamlFixtureReaderBytes(t, "marshalling") - - t.Run("Type coercion", func(t *testing.T) { - workflowBytes := fixtureReader("workflow_1") - - spec := workflowSpecYaml{} - err := yaml.Unmarshal(workflowBytes, &spec) - require.NoError(t, err) - - // Test that our workflowSpec still keeps all of the original data - var rawSpec interface{} - err = yaml.Unmarshal(workflowBytes, &rawSpec) - require.NoError(t, err) - - workflowspecJson, err := json.MarshalIndent(spec, "", " ") - require.NoError(t, err) - rawWorkflowSpecJson, err := json.MarshalIndent(rawSpec, "", " ") - require.NoError(t, err) - - if diff := cmp.Diff(rawWorkflowSpecJson, workflowspecJson, transformJSON); diff != "" { - t.Errorf("ParseWorkflowWorkflowSpecFromString() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - - // Spot check some fields - consensusConfig := spec.Consensus[0].Config - v, ok := consensusConfig["aggregation_config"] - require.True(t, ok, "expected aggregation_config to be present in consensus config") - - // the type of the keys present in v should be string rather than a number - // this is because JSON keys are always strings - _, ok = v.(map[string]any) - require.True(t, ok, "expected map[string]interface{} but got %T", v) - - // Make sure we dont have any weird type coercion with possible boolean values - booleanCoercions, ok := spec.Triggers[0].Config["boolean_coercion"].(map[string]any) - require.True(t, ok, "expected boolean_coercion to be present in triggers config") - - // check bools - bools, ok := booleanCoercions["bools"] - require.True(t, ok, "expected bools to be present in boolean_coercions") - for _, v := range bools.([]interface{}) { - _, ok = v.(bool) - require.True(t, ok, "expected bool but got %T", v) - } - - // check strings - strings, ok := booleanCoercions["strings"] - require.True(t, ok, "expected strings to be present in boolean_coercions") - for _, v := range strings.([]interface{}) { - _, ok = v.(string) - require.True(t, ok, "expected string but got %T", v) - } - - // check numbers - numbers, ok := booleanCoercions["numbers"] - require.True(t, ok, "expected numbers to be present in boolean_coercions") - for _, v := range numbers.([]interface{}) { - _, ok = v.(int64) - require.True(t, ok, "expected int64 but got %T", v) - } - }) - - t.Run("Table and string capability id", func(t *testing.T) { - workflowBytes := fixtureReader("workflow_2") - - spec := workflowSpecYaml{} - err := yaml.Unmarshal(workflowBytes, &spec) - require.NoError(t, err) - - // Test that our workflowSpec still keeps all of the original data - var rawSpec interface{} - err = yaml.Unmarshal(workflowBytes, &rawSpec) - require.NoError(t, err) - - workflowspecJson, err := json.MarshalIndent(spec, "", " ") - require.NoError(t, err) - rawWorkflowSpecJson, err := json.MarshalIndent(rawSpec, "", " ") - require.NoError(t, err) - - if diff := cmp.Diff(rawWorkflowSpecJson, workflowspecJson, transformJSON); diff != "" { - t.Errorf("ParseWorkflowWorkflowSpecFromString() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - }) - - t.Run("Yaml spec to spec", func(t *testing.T) { - expectedSpecPath := fixtureDir + "marshalling/" + "workflow_2_spec.json" - workflowBytes := fixtureReader("workflow_2") - - workflowYaml := &workflowSpecYaml{} - err := yaml.Unmarshal(workflowBytes, workflowYaml) - require.NoError(t, err) - - workflowSpec := workflowYaml.toWorkflowSpec() - workflowSpecBytes, err := json.MarshalIndent(workflowSpec, "", " ") - require.NoError(t, err) - - // change this to update golden file - shouldUpdateWorkflowSpec := false - if shouldUpdateWorkflowSpec { - err = os.WriteFile(expectedSpecPath, workflowSpecBytes, 0600) - require.NoError(t, err) - } - - expectedSpecBytes, err := os.ReadFile(expectedSpecPath) - require.NoError(t, err) - diff := cmp.Diff(expectedSpecBytes, workflowSpecBytes, transformJSON) - if diff != "" { - t.Errorf("WorkflowYamlSpecToWorkflowSpec() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - }) -} - -func TestJsonSchema(t *testing.T) { - t.Parallel() - t.Run("GenerateJsonSchema", func(t *testing.T) { - expectedSchemaPath := fixtureDir + "workflow_schema.json" - generatedSchema, err := GenerateJsonSchema() - require.NoError(t, err) - - // change this to update golden file - shouldUpdateSchema := false - if shouldUpdateSchema { - err = os.WriteFile(expectedSchemaPath, generatedSchema, 0600) - require.NoError(t, err) - } - - expectedSchema, err := os.ReadFile(expectedSchemaPath) - require.NoError(t, err) - diff := cmp.Diff(expectedSchema, generatedSchema, transformJSON) - if diff != "" { - t.Errorf("GenerateJsonSchema() mismatch (-want +got):\n%s", diff) - t.FailNow() - } - }) - - t.Run("ValidateJsonSchema", func(t *testing.T) { - generatedSchema, err := GenerateJsonSchema() - require.NoError(t, err) - - // test version regex - // for keystone, we should support major versions only along with prereleases and build metadata - t.Run("version", func(t *testing.T) { - readVersionFixture := yamlFixtureReaderObj(t, "versioning") - failingFixture1 := readVersionFixture("failing_1") - failingFixture2 := readVersionFixture("failing_2") - passingFixture1 := readVersionFixture("passing_1") - jsonSchema, err := jsonschema.CompileString("github.com/smartcontractkit/chainlink", string(generatedSchema)) - require.NoError(t, err) - - err = jsonSchema.Validate(failingFixture1) - require.Error(t, err) - - err = jsonSchema.Validate(failingFixture2) - require.Error(t, err) - - err = jsonSchema.Validate(passingFixture1) - require.NoError(t, err) - }) - - // test ref regex - t.Run("ref", func(t *testing.T) { - readRefFixture := yamlFixtureReaderObj(t, "references") - failingFixture1 := readRefFixture("failing_1") - passingFixture1 := readRefFixture("passing_1") - jsonSchema, err := jsonschema.CompileString("github.com/smartcontractkit/chainlink", string(generatedSchema)) - require.NoError(t, err) - - err = jsonSchema.Validate(failingFixture1) - require.Error(t, err) - - err = jsonSchema.Validate(passingFixture1) - require.NoError(t, err) - }) - }) -} - -func TestParsesIntsCorrectly(t *testing.T) { - wf, err := Parse(hardcodedWorkflow) - require.NoError(t, err) - - n, err := wf.Vertex("evm_median") - require.NoError(t, err) - - assert.Equal(t, int64(3600), n.Config["aggregation_config"].(map[string]any)["0x1111111111111111111100000000000000000000000000000000000000000000"].(map[string]any)["heartbeat"]) -} - -func TestMappingCustomType(t *testing.T) { - m := mapping(map[string]any{}) - data := ` -{ - "foo": 100, - "bar": 100.00, - "baz": { "gnat": 11.10 } -}` - - err := m.UnmarshalJSON([]byte(data)) - require.NoError(t, err) - assert.Equal(t, int64(100), m["foo"], m) - assert.Equal(t, decimal.NewFromFloat(100.00), m["bar"], m) - assert.Equal(t, decimal.NewFromFloat(11.10), m["baz"].(map[string]any)["gnat"], m) -} diff --git a/core/services/workflows/state.go b/core/services/workflows/state.go index 4026a59be0b..218022eae36 100644 --- a/core/services/workflows/state.go +++ b/core/services/workflows/state.go @@ -2,13 +2,13 @@ package workflows import ( "fmt" - "regexp" "strconv" "strings" "github.com/smartcontractkit/chainlink/v2/core/services/workflows/store" "github.com/smartcontractkit/chainlink-common/pkg/values" + "github.com/smartcontractkit/chainlink-common/pkg/workflows" ) // copyState returns a deep copy of the input executionState @@ -118,19 +118,15 @@ func interpolateKey(key string, state store.WorkflowExecution) (any, error) { return val, nil } -var ( - interpolationTokenRe = regexp.MustCompile(`^\$\((\S+)\)$`) -) - // findAndInterpolateAllKeys takes an `input` any value, and recursively // identifies any values that should be replaced from `state`. // // A value `v` should be replaced if it is wrapped as follows: `$(v)`. func findAndInterpolateAllKeys(input any, state store.WorkflowExecution) (any, error) { - return deepMap( + return workflows.DeepMap( input, func(el string) (any, error) { - matches := interpolationTokenRe.FindStringSubmatch(el) + matches := workflows.InterpolationTokenRe.FindStringSubmatch(el) if len(matches) < 2 { return el, nil } @@ -140,92 +136,3 @@ func findAndInterpolateAllKeys(input any, state store.WorkflowExecution) (any, e }, ) } - -// findRefs takes an `inputs` map and returns a list of all the step references -// contained within it. -func findRefs(inputs map[string]any) ([]string, error) { - refs := []string{} - _, err := deepMap( - inputs, - // This function is called for each string in the map - // for each string, we iterate over each match of the interpolation token - // - if there are no matches, return no reference - // - if there is one match, return the reference - // - if there are multiple matches (in the case of a multi-part state reference), return just the step ref - func(el string) (any, error) { - matches := interpolationTokenRe.FindStringSubmatch(el) - if len(matches) < 2 { - return el, nil - } - - m := matches[1] - parts := strings.Split(m, ".") - if len(parts) < 1 { - return nil, fmt.Errorf("invalid ref %s", m) - } - - refs = append(refs, parts[0]) - return el, nil - }, - ) - return refs, err -} - -// deepMap recursively applies a transformation function -// over each string within: -// -// - a map[string]any -// - a []any -// - a string -func deepMap(input any, transform func(el string) (any, error)) (any, error) { - // in the case of a string, simply apply the transformation - // in the case of a map, recurse and apply the transformation to each value - // in the case of a list, recurse and apply the transformation to each element - switch tv := input.(type) { - case string: - nv, err := transform(tv) - if err != nil { - return nil, err - } - - return nv, nil - case mapping: - // coerce mapping to map[string]any - mp := map[string]any(tv) - - nm := map[string]any{} - for k, v := range mp { - nv, err := deepMap(v, transform) - if err != nil { - return nil, err - } - - nm[k] = nv - } - return nm, nil - case map[string]any: - nm := map[string]any{} - for k, v := range tv { - nv, err := deepMap(v, transform) - if err != nil { - return nil, err - } - - nm[k] = nv - } - return nm, nil - case []any: - a := []any{} - for _, el := range tv { - ne, err := deepMap(el, transform) - if err != nil { - return nil, err - } - - a = append(a, ne) - } - return a, nil - } - - return nil, fmt.Errorf("cannot traverse item %+v of type %T", input, input) -} diff --git a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_1.yaml b/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_1.yaml deleted file mode 100644 index 9a9870af875..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_1.yaml +++ /dev/null @@ -1,88 +0,0 @@ - triggers: - - id: mercury-trigger@1 - ref: report_data - config: - boolean_coercion: - bools: - - y - - n - - yes - - no - - Y - - N - - YES - - NO - - No - - Yes - - TRUE - - FALSE - - True - - False - - true - - false - strings: - - TruE - - FalsE - - "true" - - "false" - - "TRUE" - - "FALSE" - - t - - f - - "T" - - "F" - - "t" - - "f" - - "1" - - "0" - - "yes" - - "no" - - "y" - - "n" - - "YES" - - "NO" - - "Y" - - "N" - numbers: - - 1 - - 0 - feed_ids: - - 123 # ETHUSD - - 456 # LINKUSD - - 789 # USDBTC - - # no actions - - consensus: - - id: offchain_reporting@1 - inputs: - observations: - - triggers.report_data.outputs - config: - aggregation_method: data_feeds_2_0 - aggregation_config: - 123: # ETHUSD - deviation: "0.005" - heartbeat: 24h - test: - 456: # LINKUSD - deviation: "0.001" - heartbeat: 24h - 789: # USDBTC - deviation: "0.002" - heartbeat: 6h - encoder: EVM - encoder_config: - abi: "mercury_reports bytes[]" - - targets: - - id: write_polygon_mainnet@1 - inputs: - report: - - consensus.evm_median.outputs.report - config: - address: "0xaabbcc" - method: "updateFeedValues(report bytes, role uint8)" - params: [$(inputs.report), 1] - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2.yaml b/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2.yaml deleted file mode 100644 index be40a91daa0..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2.yaml +++ /dev/null @@ -1,28 +0,0 @@ - triggers: - - id: on_mercury_report@1 - ref: report_data - config: {} - - # no actions - - consensus: - - id: - name: trigger_test - version: "2" - labels: - chain: ethereum - aaShouldBeFirst: "true" - network: mainnet - config: {} - inputs: - observations: - - triggers.report_data.outputs - - targets: - - id: write_polygon_mainnet@1 - config: {} - inputs: - report: - - consensus.evm_median.outputs.report - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2_spec.json b/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2_spec.json deleted file mode 100644 index 000fa469218..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/marshalling/workflow_2_spec.json +++ /dev/null @@ -1,31 +0,0 @@ -{ - "triggers": [ - { - "id": "on_mercury_report@1", - "ref": "report_data", - "config": {} - } - ], - "consensus": [ - { - "id": "trigger_test:aaShouldBeFirst_true:chain_ethereum:network_mainnet@2", - "inputs": { - "observations": [ - "triggers.report_data.outputs" - ] - }, - "config": {} - } - ], - "targets": [ - { - "id": "write_polygon_mainnet@1", - "inputs": { - "report": [ - "consensus.evm_median.outputs.report" - ] - }, - "config": {} - } - ] -} \ No newline at end of file diff --git a/core/services/workflows/testdata/fixtures/workflows/references/failing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/references/failing_1.yaml deleted file mode 100644 index b3c984e9892..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/references/failing_1.yaml +++ /dev/null @@ -1,15 +0,0 @@ -triggers: -- id: trigger_test@1 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting=1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/references/passing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/references/passing_1.yaml deleted file mode 100644 index cb2f424e981..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/references/passing_1.yaml +++ /dev/null @@ -1,15 +0,0 @@ -triggers: -- id: trigger_test@1 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting_1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/versioning/failing_1.yaml deleted file mode 100644 index 2e41eeb9898..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_1.yaml +++ /dev/null @@ -1,16 +0,0 @@ -# Should fail since version is more specific than major -triggers: - - id: trigger_test@1.0 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting_1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_2.yaml b/core/services/workflows/testdata/fixtures/workflows/versioning/failing_2.yaml deleted file mode 100644 index 36cd5b68b6b..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/versioning/failing_2.yaml +++ /dev/null @@ -1,17 +0,0 @@ - -# Should fail since version is more specific than major -triggers: - - id: trigger_test@1.0.0 - config: {} - -consensus: - - id: offchain_reporting@1 - ref: offchain_reporting_1 - config: {} - -targets: - - id: write_polygon_mainnet@1 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/versioning/passing_1.yaml b/core/services/workflows/testdata/fixtures/workflows/versioning/passing_1.yaml deleted file mode 100644 index 4579c2899b9..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/versioning/passing_1.yaml +++ /dev/null @@ -1,15 +0,0 @@ - triggers: - - id: trigger_test@1 - config: {} - - consensus: - - id: offchain_reporting@1-beta.1 - ref: offchain_reporting_1 - config: {} - - targets: - - id: write_polygon_mainnet@1-alpha+sha246er3 - ref: write_polygon_mainnet_1 - config: {} - -# yaml-language-server: $schema=../workflow_schema.json diff --git a/core/services/workflows/testdata/fixtures/workflows/workflow_schema.json b/core/services/workflows/testdata/fixtures/workflows/workflow_schema.json deleted file mode 100644 index f9f9fd88646..00000000000 --- a/core/services/workflows/testdata/fixtures/workflows/workflow_schema.json +++ /dev/null @@ -1,103 +0,0 @@ -{ - "$schema": "https://json-schema.org/draft/2020-12/schema", - "$id": "https://github.com/smartcontractkit/chainlink/v2/core/services/workflows/workflow-spec-yaml", - "$ref": "#/$defs/workflowSpecYaml", - "$defs": { - "mapping": { - "type": "object" - }, - "stepDefinitionID": { - "oneOf": [ - { - "$id": "string", - "pattern": "^[a-z0-9_\\-:]+@(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$" - }, - { - "$schema": "https://json-schema.org/draft/2020-12/schema", - "$id": "https://github.com/smartcontractkit/chainlink/v2/core/services/workflows/step-definition-table-id", - "properties": { - "name": { - "type": "string" - }, - "version": { - "type": "string", - "pattern": "(0|[1-9]\\d*)(?:-((?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*)(?:\\.(?:0|[1-9]\\d*|\\d*[a-zA-Z-][0-9a-zA-Z-]*))*))?(?:\\+([0-9a-zA-Z-]+(?:\\.[0-9a-zA-Z-]+)*))?$" - }, - "labels": { - "additionalProperties": { - "type": "string" - }, - "type": "object" - } - }, - "additionalProperties": false, - "type": "object", - "required": [ - "name", - "version", - "labels" - ] - } - ], - "title": "id" - }, - "stepDefinitionYaml": { - "properties": { - "id": { - "$ref": "#/$defs/stepDefinitionID" - }, - "ref": { - "type": "string", - "pattern": "^[a-z0-9_-]+$" - }, - "inputs": { - "$ref": "#/$defs/mapping" - }, - "config": { - "$ref": "#/$defs/mapping" - } - }, - "additionalProperties": false, - "type": "object", - "required": [ - "id", - "config" - ] - }, - "workflowSpecYaml": { - "properties": { - "triggers": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - }, - "actions": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - }, - "consensus": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - }, - "targets": { - "items": { - "$ref": "#/$defs/stepDefinitionYaml" - }, - "type": "array" - } - }, - "additionalProperties": false, - "type": "object", - "required": [ - "triggers", - "consensus", - "targets" - ] - } - } -} \ No newline at end of file diff --git a/go.mod b/go.mod index 4dc34e0fd61..8ba80e67eba 100644 --- a/go.mod +++ b/go.mod @@ -112,7 +112,6 @@ require ( google.golang.org/protobuf v1.33.0 gopkg.in/guregu/null.v4 v4.0.0 gopkg.in/natefinch/lumberjack.v2 v2.2.1 - sigs.k8s.io/yaml v1.4.0 ) require ( @@ -210,7 +209,7 @@ require ( github.com/golang/protobuf v1.5.4 // indirect github.com/golang/snappy v0.0.5-0.20220116011046-fa5810519dcb // indirect github.com/google/btree v1.1.2 // indirect - github.com/google/go-cmp v0.6.0 + github.com/google/go-cmp v0.6.0 // indirect github.com/google/go-tpm v0.9.0 // indirect github.com/google/gofuzz v1.2.0 // indirect github.com/gorilla/context v1.1.1 // indirect @@ -237,7 +236,7 @@ require ( github.com/huin/goupnp v1.3.0 // indirect github.com/imdario/mergo v0.3.16 // indirect github.com/inconshreveable/mousetrap v1.1.0 // indirect - github.com/invopop/jsonschema v0.12.0 + github.com/invopop/jsonschema v0.12.0 // indirect github.com/jackc/chunkreader/v2 v2.0.1 // indirect github.com/jackc/pgio v1.0.0 // indirect github.com/jackc/pgpassfile v1.0.0 // indirect @@ -279,7 +278,7 @@ require ( github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 // indirect github.com/rivo/uniseg v0.4.4 // indirect github.com/russross/blackfriday/v2 v2.1.0 // indirect - github.com/santhosh-tekuri/jsonschema/v5 v5.3.1 + github.com/santhosh-tekuri/jsonschema/v5 v5.3.1 // indirect github.com/sasha-s/go-deadlock v0.3.1 // indirect github.com/sethvargo/go-retry v0.2.4 // indirect github.com/shirou/gopsutil v3.21.11+incompatible // indirect @@ -337,6 +336,7 @@ require ( gopkg.in/yaml.v3 v3.0.1 // indirect pgregory.net/rapid v0.5.5 // indirect rsc.io/tmplfunc v0.0.3 // indirect + sigs.k8s.io/yaml v1.4.0 // indirect ) replace (