Skip to content

Commit

Permalink
slight directive execution engine refactoring
Browse files Browse the repository at this point in the history
Signed-off-by: Kent Rancourt <[email protected]>
  • Loading branch information
krancour committed Sep 9, 2024
1 parent ea439c1 commit d0fef18
Show file tree
Hide file tree
Showing 8 changed files with 97 additions and 60 deletions.
20 changes: 12 additions & 8 deletions internal/directives/copy_directive.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,29 +34,33 @@ func (d *copyDirective) Name() string {
}

func (d *copyDirective) Run(ctx context.Context, stepCtx *StepContext) (Result, error) {
failure := Result{Status: StatusFailure}
// Validate the configuration against the JSON Schema.
if err := validate(d.schemaLoader, gojsonschema.NewGoLoader(stepCtx.Config), d.Name()); err != nil {
return ResultFailure, err
return failure, err
}

// Convert the configuration into a typed object.
cfg, err := configToStruct[CopyConfig](stepCtx.Config)
if err != nil {
return ResultFailure, fmt.Errorf("could not convert config into %s config: %w", d.Name(), err)
return failure, fmt.Errorf("could not convert config into %s config: %w", d.Name(), err)
}

return d.run(ctx, stepCtx, cfg)
if err = d.run(ctx, stepCtx, cfg); err != nil {
return failure, err
}
return Result{Status: StatusSuccess}, nil
}

func (d *copyDirective) run(ctx context.Context, stepCtx *StepContext, cfg CopyConfig) (Result, error) {
func (d *copyDirective) run(ctx context.Context, stepCtx *StepContext, cfg CopyConfig) error {
// Secure join the paths to prevent path traversal attacks.
inPath, err := securejoin.SecureJoin(stepCtx.WorkDir, cfg.InPath)
if err != nil {
return ResultFailure, fmt.Errorf("could not secure join inPath %q: %w", cfg.InPath, err)
return fmt.Errorf("could not secure join inPath %q: %w", cfg.InPath, err)
}
outPath, err := securejoin.SecureJoin(stepCtx.WorkDir, cfg.OutPath)
if err != nil {
return ResultFailure, fmt.Errorf("could not secure join outPath %q: %w", cfg.OutPath, err)
return fmt.Errorf("could not secure join outPath %q: %w", cfg.OutPath, err)
}

// Perform the copy operation.
Expand All @@ -70,9 +74,9 @@ func (d *copyDirective) run(ctx context.Context, stepCtx *StepContext, cfg CopyC
},
}
if err = copy.Copy(inPath, outPath, opts); err != nil {
return ResultFailure, fmt.Errorf("failed to copy %q to %q: %w", cfg.InPath, cfg.OutPath, err)
return fmt.Errorf("failed to copy %q to %q: %w", cfg.InPath, cfg.OutPath, err)
}
return ResultSuccess, nil
return nil
}

// sanitizePathError sanitizes the path in a path error to be relative to the
Expand Down
21 changes: 10 additions & 11 deletions internal/directives/copy_directive_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ func Test_copyDirective_run(t *testing.T) {
name string
setupFiles func(*testing.T) string
cfg CopyConfig
assertions func(*testing.T, string, Result, error)
assertions func(*testing.T, string, error)
}{
{
name: "succeeds copying file",
Expand All @@ -32,8 +32,7 @@ func Test_copyDirective_run(t *testing.T) {
InPath: "input.txt",
OutPath: "output.txt",
},
assertions: func(t *testing.T, workDir string, result Result, err error) {
assert.Equal(t, ResultSuccess, result)
assertions: func(t *testing.T, workDir string, err error) {
assert.NoError(t, err)

outPath := filepath.Join(workDir, "output.txt")
Expand Down Expand Up @@ -64,8 +63,7 @@ func Test_copyDirective_run(t *testing.T) {
InPath: "input/",
OutPath: "output/",
},
assertions: func(t *testing.T, workDir string, result Result, err error) {
assert.Equal(t, ResultSuccess, result)
assertions: func(t *testing.T, workDir string, err error) {
assert.NoError(t, err)

outDir := filepath.Join(workDir, "output")
Expand Down Expand Up @@ -102,8 +100,7 @@ func Test_copyDirective_run(t *testing.T) {
InPath: "input/",
OutPath: "output/",
},
assertions: func(t *testing.T, workDir string, result Result, err error) {
assert.Equal(t, ResultSuccess, result)
assertions: func(t *testing.T, workDir string, err error) {
assert.NoError(t, err)

outDir := filepath.Join(workDir, "output")
Expand All @@ -127,9 +124,8 @@ func Test_copyDirective_run(t *testing.T) {
cfg: CopyConfig{
InPath: "input.txt",
},
assertions: func(t *testing.T, _ string, result Result, err error) {
assertions: func(t *testing.T, _ string, err error) {
require.ErrorContains(t, err, "failed to copy")
assert.Equal(t, ResultFailure, result)
},
},
}
Expand All @@ -139,8 +135,11 @@ func Test_copyDirective_run(t *testing.T) {
workDir := tt.setupFiles(t)

d := &copyDirective{}
result, err := d.run(context.Background(), &StepContext{WorkDir: workDir}, tt.cfg)
tt.assertions(t, workDir, result, err)
tt.assertions(
t,
workDir,
d.run(context.Background(), &StepContext{WorkDir: workDir}, tt.cfg),
)
})
}
}
Expand Down
34 changes: 28 additions & 6 deletions internal/directives/directive.go
Original file line number Diff line number Diff line change
Expand Up @@ -90,6 +90,17 @@ func (s State) Get(key string) (any, bool) {
return value, ok
}

// DeepCopy returns a deep copy of the state.
func (s *State) DeepCopy() State {
if s == nil {
return nil
}
// TODO(hidde): we piggyback on the runtime package for now, as we expect
// the configuration to originate from a Kubernetes API object. We should
// consider writing our own implementation in the future.
return runtime.DeepCopyJSON(*s)
}

// Config is a map of configuration values that can be passed to a step.
// The keys and values are arbitrary, and the step is responsible for
// interpreting them.
Expand All @@ -106,16 +117,27 @@ func (c Config) DeepCopy() Config {
return runtime.DeepCopyJSON(c)
}

// Result is a type that represents the result of a Directive.
type Result string
// Status is a type that represents the high-level outcome of a directive
// execution.
type Status string

const (
// ResultSuccess is the result of a successful directive.
ResultSuccess Result = "Success"
// ResultFailure is the result of a failed directive.
ResultFailure Result = "Failure"
// StatusSuccess is the result of a successful directive execution.
StatusSuccess Status = "Success"
// StatusFailure is the result of a failed directive execution.
StatusFailure Status = "Failure"
)

// Result represents the outcome of a directive execution, including its status
// (e.g. Success or Failure) and any output (State) that the execution engine
// executing the directive must append to the shared state.
type Result struct {
// Status is the high-level outcome of the directive execution.
Status Status
// Output is the output of the directive execution.
Output State
}

// Directive is an interface that a directive must implement. A directive is
// a responsible for executing a specific action, and may modify the provided
// context to allow subsequent directives to access the results of its
Expand Down
23 changes: 15 additions & 8 deletions internal/directives/engine.go
Original file line number Diff line number Diff line change
Expand Up @@ -45,11 +45,11 @@ func NewEngine(
}

// Execute runs the provided list of directives in sequence.
func (e *Engine) Execute(ctx context.Context, steps []Step) (Result, error) {
func (e *Engine) Execute(ctx context.Context, steps []Step) (Status, error) {
// TODO(hidde): allow the workDir to be restored from a previous execution.
workDir, err := os.MkdirTemp("", "run-")
if err != nil {
return ResultFailure, fmt.Errorf("temporary working directory creation failed: %w", err)
return StatusFailure, fmt.Errorf("temporary working directory creation failed: %w", err)
}
defer os.RemoveAll(workDir)

Expand All @@ -59,16 +59,18 @@ func (e *Engine) Execute(ctx context.Context, steps []Step) (Result, error) {
for _, d := range steps {
select {
case <-ctx.Done():
return ResultFailure, ctx.Err()
return StatusFailure, ctx.Err()
default:
reg, err := e.registry.GetDirectiveRegistration(d.Directive)
if err != nil {
return ResultFailure, fmt.Errorf("failed to get step %q: %w", d.Directive, err)
return StatusFailure, fmt.Errorf("failed to get step %q: %w", d.Directive, err)
}

stateCopy := state.DeepCopy()

stepCtx := &StepContext{
WorkDir: workDir,
SharedState: state,
SharedState: stateCopy,
Alias: d.Alias,
Config: d.Config.DeepCopy(),
}
Expand All @@ -83,10 +85,15 @@ func (e *Engine) Execute(ctx context.Context, steps []Step) (Result, error) {
stepCtx.ArgoCDClient = e.argoCDClient
}

if result, err := reg.Directive.Run(ctx, stepCtx); err != nil {
return result, fmt.Errorf("failed to run step %q: %w", d.Directive, err)
result, err := reg.Directive.Run(ctx, stepCtx)
if err != nil {
return result.Status, fmt.Errorf("failed to run step %q: %w", d.Directive, err)
}

if d.Alias != "" {
state[d.Alias] = result.Output
}
}
}
return ResultSuccess, nil
return StatusSuccess, nil
}
38 changes: 20 additions & 18 deletions internal/directives/engine_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,12 +10,14 @@ import (
)

func TestEngine_Execute(t *testing.T) {
failureResult := Result{Status: StatusFailure}
successResult := Result{Status: StatusSuccess}
tests := []struct {
name string
directives []Step
initRegistry func() DirectiveRegistry
ctx context.Context
assertions func(t *testing.T, result Result, err error)
assertions func(t *testing.T, status Status, err error)
}{
{
name: "success: single directive",
Expand All @@ -27,15 +29,15 @@ func TestEngine_Execute(t *testing.T) {
registry.RegisterDirective(
&mockDirective{
name: "mock",
runResult: ResultSuccess,
runResult: successResult,
},
nil,
)
return registry
},
ctx: context.Background(),
assertions: func(t *testing.T, result Result, err error) {
assert.Equal(t, ResultSuccess, result)
assertions: func(t *testing.T, status Status, err error) {
assert.Equal(t, StatusSuccess, status)
assert.NoError(t, err)
},
},
Expand All @@ -50,22 +52,22 @@ func TestEngine_Execute(t *testing.T) {
registry.RegisterDirective(
&mockDirective{
name: "mock1",
runResult: ResultSuccess,
runResult: successResult,
},
nil,
)
registry.RegisterDirective(
&mockDirective{
name: "mock2",
runResult: ResultSuccess,
runResult: successResult,
},
nil,
)
return registry
},
ctx: context.Background(),
assertions: func(t *testing.T, result Result, err error) {
assert.Equal(t, ResultSuccess, result)
assertions: func(t *testing.T, status Status, err error) {
assert.Equal(t, StatusSuccess, status)
assert.NoError(t, err)
},
},
Expand All @@ -78,8 +80,8 @@ func TestEngine_Execute(t *testing.T) {
return make(DirectiveRegistry)
},
ctx: context.Background(),
assertions: func(t *testing.T, result Result, err error) {
assert.Equal(t, ResultFailure, result)
assertions: func(t *testing.T, status Status, err error) {
assert.Equal(t, StatusFailure, status)
assert.ErrorContains(t, err, "not found")
},
},
Expand All @@ -93,16 +95,16 @@ func TestEngine_Execute(t *testing.T) {
registry.RegisterDirective(
&mockDirective{
name: "failing",
runResult: ResultFailure,
runResult: failureResult,
runErr: errors.New("something went wrong"),
},
nil,
)
return registry
},
ctx: context.Background(),
assertions: func(t *testing.T, result Result, err error) {
assert.Equal(t, ResultFailure, result)
assertions: func(t *testing.T, status Status, err error) {
assert.Equal(t, StatusFailure, status)
assert.ErrorContains(t, err, "something went wrong")
},
},
Expand All @@ -119,7 +121,7 @@ func TestEngine_Execute(t *testing.T) {
name: "mock",
runFunc: func(ctx context.Context, _ *StepContext) (Result, error) {
<-ctx.Done() // Wait for context to be canceled
return ResultSuccess, nil
return successResult, nil
},
},
nil,
Expand All @@ -134,17 +136,17 @@ func TestEngine_Execute(t *testing.T) {
}()
return ctx
}(),
assertions: func(t *testing.T, result Result, err error) {
assert.Equal(t, ResultFailure, result)
assertions: func(t *testing.T, status Status, err error) {
assert.Equal(t, StatusFailure, status)
assert.ErrorIs(t, err, context.Canceled)
},
},
}
for _, tt := range tests {
t.Run(tt.name, func(t *testing.T) {
engine := NewEngine(tt.initRegistry(), nil, nil, nil)
result, err := engine.Execute(tt.ctx, tt.directives)
tt.assertions(t, result, err)
status, err := engine.Execute(tt.ctx, tt.directives)
tt.assertions(t, status, err)
})
}
}
7 changes: 4 additions & 3 deletions internal/directives/git_clone_directive.go
Original file line number Diff line number Diff line change
Expand Up @@ -41,18 +41,19 @@ func (g *gitCloneDirective) Run(
_ context.Context,
stepCtx *StepContext,
) (Result, error) {
failure := Result{Status: StatusFailure}
// Validate the configuration against the JSON Schema
if err := validate(
g.schemaLoader,
gojsonschema.NewGoLoader(stepCtx.Config),
"git-clone",
); err != nil {
return ResultFailure, err
return failure, err
}
if _, err := configToStruct[GitCloneConfig](stepCtx.Config); err != nil {
return ResultFailure,
return failure,
fmt.Errorf("could not convert config into git-clone config: %w", err)
}
// TODO: Add implementation here
return ResultSuccess, nil
return Result{Status: StatusSuccess}, nil
}
7 changes: 4 additions & 3 deletions internal/directives/git_commit_directive.go
Original file line number Diff line number Diff line change
Expand Up @@ -35,18 +35,19 @@ func (g *gitCommitDirective) Run(
_ context.Context,
stepCtx *StepContext,
) (Result, error) {
failure := Result{Status: StatusFailure}
// Validate the configuration against the JSON Schema
if err := validate(
g.schemaLoader,
gojsonschema.NewGoLoader(stepCtx.Config),
"git-commit",
); err != nil {
return ResultFailure, err
return failure, err
}
if _, err := configToStruct[GitCommitConfig](stepCtx.Config); err != nil {
return ResultFailure,
return failure,
fmt.Errorf("could not convert config into git-commit config: %w", err)
}
// TODO: Add implementation here
return ResultSuccess, nil
return Result{Status: StatusSuccess}, nil
}
Loading

0 comments on commit d0fef18

Please sign in to comment.