Skip to content

Commit

Permalink
update to support map merging on anchors
Browse files Browse the repository at this point in the history
  • Loading branch information
ecrupper committed Dec 2, 2024
1 parent d408913 commit 0f29f8e
Show file tree
Hide file tree
Showing 26 changed files with 392 additions and 152 deletions.
2 changes: 1 addition & 1 deletion api/pipeline/template.go
Original file line number Diff line number Diff line change
Expand Up @@ -101,7 +101,7 @@ func GetTemplates(c *gin.Context) {
compiler := compiler.FromContext(c).Duplicate().WithCommit(p.GetCommit()).WithMetadata(m).WithRepo(r).WithUser(u)

// parse the pipeline configuration
pipeline, _, err := compiler.Parse(p.GetData(), p.GetType(), new(yaml.Template))
pipeline, _, _, err := compiler.Parse(p.GetData(), p.GetType(), new(yaml.Template))
if err != nil {
util.HandleError(c, http.StatusBadRequest, fmt.Errorf("unable to parse pipeline %s: %w", entry, err))

Expand Down
57 changes: 43 additions & 14 deletions api/types/pipeline.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,20 +10,21 @@ import (
//
// swagger:model Pipeline
type Pipeline struct {
ID *int64 `json:"id,omitempty"`
Repo *Repo `json:"repo,omitempty"`
Commit *string `json:"commit,omitempty"`
Flavor *string `json:"flavor,omitempty"`
Platform *string `json:"platform,omitempty"`
Ref *string `json:"ref,omitempty"`
Type *string `json:"type,omitempty"`
Version *string `json:"version,omitempty"`
ExternalSecrets *bool `json:"external_secrets,omitempty"`
InternalSecrets *bool `json:"internal_secrets,omitempty"`
Services *bool `json:"services,omitempty"`
Stages *bool `json:"stages,omitempty"`
Steps *bool `json:"steps,omitempty"`
Templates *bool `json:"templates,omitempty"`
ID *int64 `json:"id,omitempty"`
Repo *Repo `json:"repo,omitempty"`
Commit *string `json:"commit,omitempty"`
Flavor *string `json:"flavor,omitempty"`
Platform *string `json:"platform,omitempty"`
Ref *string `json:"ref,omitempty"`
Type *string `json:"type,omitempty"`
Version *string `json:"version,omitempty"`
ExternalSecrets *bool `json:"external_secrets,omitempty"`
InternalSecrets *bool `json:"internal_secrets,omitempty"`
Services *bool `json:"services,omitempty"`
Stages *bool `json:"stages,omitempty"`
Steps *bool `json:"steps,omitempty"`
Templates *bool `json:"templates,omitempty"`
Warnings *[]string `json:"warnings,omitempty"`
// swagger:strfmt base64
Data *[]byte `json:"data,omitempty"`
}
Expand Down Expand Up @@ -210,6 +211,19 @@ func (p *Pipeline) GetTemplates() bool {
return *p.Templates
}

// GetWarnings returns the Warnings field.
//
// When the provided Pipeline type is nil, or the field within
// the type is nil, it returns the zero value for the field.
func (p *Pipeline) GetWarnings() []string {
// return zero value if Pipeline type or Warnings field is nil
if p == nil || p.Warnings == nil {
return []string{}
}

return *p.Warnings
}

// GetData returns the Data field.
//
// When the provided Pipeline type is nil, or the field within
Expand Down Expand Up @@ -405,6 +419,19 @@ func (p *Pipeline) SetTemplates(v bool) {
p.Templates = &v
}

// SetWarnings sets the Warnings field.
//
// When the provided Pipeline type is nil, it
// will set nothing and immediately return.
func (p *Pipeline) SetWarnings(v []string) {
// return if Pipeline type is nil
if p == nil {
return
}

p.Warnings = &v
}

// SetData sets the Data field.
//
// When the provided Pipeline type is nil, it
Expand Down Expand Up @@ -436,6 +463,7 @@ func (p *Pipeline) String() string {
Templates: %t,
Type: %s,
Version: %s,
Warnings: %v,
}`,
p.GetCommit(),
p.GetData(),
Expand All @@ -452,5 +480,6 @@ func (p *Pipeline) String() string {
p.GetTemplates(),
p.GetType(),
p.GetVersion(),
p.GetWarnings(),
)
}
12 changes: 12 additions & 0 deletions api/types/pipeline_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,10 @@ func TestAPI_Pipeline_Getters(t *testing.T) {
t.Errorf("GetTemplates is %v, want %v", test.pipeline.GetTemplates(), test.want.GetTemplates())
}

if !reflect.DeepEqual(test.pipeline.GetWarnings(), test.want.GetWarnings()) {
t.Errorf("GetWarnings is %v, want %v", test.pipeline.GetWarnings(), test.want.GetWarnings())
}

if !reflect.DeepEqual(test.pipeline.GetData(), test.want.GetData()) {
t.Errorf("GetData is %v, want %v", test.pipeline.GetData(), test.want.GetData())
}
Expand Down Expand Up @@ -123,6 +127,7 @@ func TestAPI_Pipeline_Setters(t *testing.T) {
test.pipeline.SetStages(test.want.GetStages())
test.pipeline.SetSteps(test.want.GetSteps())
test.pipeline.SetTemplates(test.want.GetTemplates())
test.pipeline.SetWarnings(test.want.GetWarnings())
test.pipeline.SetData(test.want.GetData())

if test.pipeline.GetID() != test.want.GetID() {
Expand Down Expand Up @@ -181,6 +186,10 @@ func TestAPI_Pipeline_Setters(t *testing.T) {
t.Errorf("SetTemplates is %v, want %v", test.pipeline.GetTemplates(), test.want.GetTemplates())
}

if !reflect.DeepEqual(test.pipeline.GetWarnings(), test.want.GetWarnings()) {
t.Errorf("SetWarnings is %v, want %v", test.pipeline.GetWarnings(), test.want.GetWarnings())
}

if !reflect.DeepEqual(test.pipeline.GetData(), test.want.GetData()) {
t.Errorf("SetData is %v, want %v", test.pipeline.GetData(), test.want.GetData())
}
Expand All @@ -207,6 +216,7 @@ func TestAPI_Pipeline_String(t *testing.T) {
Templates: %t,
Type: %s,
Version: %s,
Warnings: %v,
}`,
p.GetCommit(),
p.GetData(),
Expand All @@ -223,6 +233,7 @@ func TestAPI_Pipeline_String(t *testing.T) {
p.GetTemplates(),
p.GetType(),
p.GetVersion(),
p.GetWarnings(),
)

// run test
Expand Down Expand Up @@ -252,6 +263,7 @@ func testPipeline() *Pipeline {
p.SetStages(false)
p.SetSteps(true)
p.SetTemplates(false)
p.SetWarnings([]string{"42:this is a warning"})
p.SetData(testPipelineData())

return p
Expand Down
2 changes: 1 addition & 1 deletion compiler/engine.go
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ type Engine interface {

// Parse defines a function that converts
// an object to a yaml configuration.
Parse(interface{}, string, *yaml.Template) (*yaml.Build, []byte, error)
Parse(interface{}, string, *yaml.Template) (*yaml.Build, []byte, []string, error)

// ParseRaw defines a function that converts
// an object to a string.
Expand Down
8 changes: 5 additions & 3 deletions compiler/native/compile.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ type ModifyResponse struct {

// Compile produces an executable pipeline from a yaml configuration.
func (c *client) Compile(ctx context.Context, v interface{}) (*pipeline.Build, *api.Pipeline, error) {
p, data, err := c.Parse(v, c.repo.GetPipelineType(), new(yaml.Template))
p, data, warnings, err := c.Parse(v, c.repo.GetPipelineType(), new(yaml.Template))
if err != nil {
return nil, nil, err
}
Expand All @@ -48,6 +48,7 @@ func (c *client) Compile(ctx context.Context, v interface{}) (*pipeline.Build, *
_pipeline := p.ToPipelineAPI()
_pipeline.SetData(data)
_pipeline.SetType(c.repo.GetPipelineType())
_pipeline.SetWarnings(warnings)

// create map of templates for easy lookup
templates := mapFromTemplates(p.Templates)
Expand Down Expand Up @@ -104,7 +105,7 @@ func (c *client) Compile(ctx context.Context, v interface{}) (*pipeline.Build, *

// CompileLite produces a partial of an executable pipeline from a yaml configuration.
func (c *client) CompileLite(ctx context.Context, v interface{}, ruleData *pipeline.RuleData, substitute bool) (*yaml.Build, *api.Pipeline, error) {
p, data, err := c.Parse(v, c.repo.GetPipelineType(), new(yaml.Template))
p, data, warnings, err := c.Parse(v, c.repo.GetPipelineType(), new(yaml.Template))
if err != nil {
return nil, nil, err
}
Expand All @@ -113,6 +114,7 @@ func (c *client) CompileLite(ctx context.Context, v interface{}, ruleData *pipel
_pipeline := p.ToPipelineAPI()
_pipeline.SetData(data)
_pipeline.SetType(c.repo.GetPipelineType())
_pipeline.SetWarnings(warnings)

if p.Metadata.RenderInline {
newPipeline, err := c.compileInline(ctx, p, c.GetTemplateDepth())
Expand Down Expand Up @@ -240,7 +242,7 @@ func (c *client) compileInline(ctx context.Context, p *yaml.Build, depth int) (*
// inject template name into variables
template.Variables["VELA_TEMPLATE_NAME"] = template.Name

parsed, _, err := c.Parse(bytes, format, template)
parsed, _, _, err := c.Parse(bytes, format, template)
if err != nil {
return nil, err
}
Expand Down
31 changes: 24 additions & 7 deletions compiler/native/compile_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import (
"net/http/httptest"
"os"
"path/filepath"
"reflect"
"testing"
"time"

Expand Down Expand Up @@ -2153,7 +2154,7 @@ func TestNative_Compile_LegacyMergeAnchor(t *testing.T) {
t.Errorf("Reading yaml file return err: %v", err)
}

got, _, err := compiler.Compile(context.Background(), yaml)
got, gotPipeline, err := compiler.Compile(context.Background(), yaml)
if err != nil {
t.Errorf("Compile returned err: %v", err)
}
Expand All @@ -2162,19 +2163,35 @@ func TestNative_Compile_LegacyMergeAnchor(t *testing.T) {
t.Errorf("Compile() mismatch (-want +got):\n%s", diff)
}

// run test on current version (should fail)
if !reflect.DeepEqual(gotPipeline.GetWarnings(), []string{"using legacy version. Upgrade to go-yaml v3"}) {
t.Errorf("Compile() returned warnings %v, want %v", gotPipeline.GetWarnings(), "blah")
}

// run test on current version
yaml, err = os.ReadFile("../types/yaml/buildkite/testdata/merge_anchor.yml") // has `version: "1"` instead of `version: "legacy"`
if err != nil {
t.Errorf("Reading yaml file return err: %v", err)
}

got, _, err = compiler.Compile(context.Background(), yaml)
if err == nil {
t.Errorf("Compile should have returned err")
got, gotPipeline, err = compiler.Compile(context.Background(), yaml)
if err != nil {
t.Errorf("Compile returned err: %v", err)
}

if got != nil {
t.Errorf("Compile is %v, want %v", got, nil)
// update version
want.Version = "1"

if diff := cmp.Diff(want, got); diff != "" {
t.Errorf("Compile() mismatch (-want +got):\n%s", diff)
}

if !reflect.DeepEqual(gotPipeline.GetWarnings(), []string{
"25:duplicate << keys in single YAML map",
"32:duplicate << keys in single YAML map",
"44:duplicate << keys in single YAML map",
"43:duplicate << keys in single YAML map",
}) {
t.Errorf("Compile() returned warnings %v, want %v", gotPipeline.GetWarnings(), "blah")
}
}

Expand Down
6 changes: 3 additions & 3 deletions compiler/native/expand.go
Original file line number Diff line number Diff line change
Expand Up @@ -131,7 +131,7 @@ func (c *client) ExpandSteps(ctx context.Context, s *yaml.Build, tmpls map[strin
// inject template name into variables
step.Template.Variables["VELA_TEMPLATE_NAME"] = step.Template.Name

tmplBuild, err := c.mergeTemplate(bytes, tmpl, step)
tmplBuild, _, err := c.mergeTemplate(bytes, tmpl, step)
if err != nil {
return s, err
}
Expand Down Expand Up @@ -345,7 +345,7 @@ func (c *client) getTemplate(ctx context.Context, tmpl *yaml.Template, name stri
}

//nolint:lll // ignore long line length due to input arguments
func (c *client) mergeTemplate(bytes []byte, tmpl *yaml.Template, step *yaml.Step) (*yaml.Build, error) {
func (c *client) mergeTemplate(bytes []byte, tmpl *yaml.Template, step *yaml.Step) (*yaml.Build, []string, error) {
switch tmpl.Format {
case constants.PipelineTypeGo, "golang", "":
//nolint:lll // ignore long line length due to return
Expand All @@ -355,7 +355,7 @@ func (c *client) mergeTemplate(bytes []byte, tmpl *yaml.Template, step *yaml.Ste
return starlark.Render(string(bytes), step.Name, step.Template.Name, step.Environment, step.Template.Variables, c.GetStarlarkExecLimit())
default:
//nolint:lll // ignore long line length due to return
return &yaml.Build{}, fmt.Errorf("format of %s is unsupported", tmpl.Format)
return &yaml.Build{}, nil, fmt.Errorf("format of %s is unsupported", tmpl.Format)
}
}

Expand Down
Loading

0 comments on commit 0f29f8e

Please sign in to comment.