Skip to content

Commit

Permalink
Merge pull request #57 from laszlocph/depending-on-zero-step-pipe
Browse files Browse the repository at this point in the history
Depending on zero step pipe
  • Loading branch information
laszlocph authored Jul 22, 2019
2 parents a949750 + 3bb0f64 commit 063b7fc
Show file tree
Hide file tree
Showing 4 changed files with 159 additions and 19 deletions.
2 changes: 2 additions & 0 deletions server/build.go
Original file line number Diff line number Diff line change
Expand Up @@ -344,6 +344,7 @@ func PostApproval(c *gin.Context) {
store.UpdateBuild(c, build)
return
}
build = setBuildStepsOnBuild(b.Curr, buildItems)

err = store.FromContext(c).ProcCreate(build.Procs)
if err != nil {
Expand Down Expand Up @@ -558,6 +559,7 @@ func PostBuild(c *gin.Context) {
c.JSON(500, build)
return
}
build = setBuildStepsOnBuild(b.Curr, buildItems)

err = store.FromContext(c).ProcCreate(build.Procs)
if err != nil {
Expand Down
8 changes: 4 additions & 4 deletions server/hook.go
Original file line number Diff line number Diff line change
Expand Up @@ -177,7 +177,7 @@ func PostHook(c *gin.Context) {
return
}

if zeroSteps(*build, remoteYamlConfigs) {
if zeroSteps(build, remoteYamlConfigs) {
c.String(200, "Step conditions yield zero runnable steps")
return
}
Expand Down Expand Up @@ -261,6 +261,7 @@ func PostHook(c *gin.Context) {
store.UpdateBuild(c, build)
return
}
build = setBuildStepsOnBuild(b.Curr, buildItems)

err = store.FromContext(c).ProcCreate(build.Procs)
if err != nil {
Expand Down Expand Up @@ -298,11 +299,10 @@ func branchFiltered(build *model.Build, remoteYamlConfigs []*remote.FileMeta) bo
return true
}

// uses pass by value as procBuilder has side effects on build. Something to be fixed
func zeroSteps(build model.Build, remoteYamlConfigs []*remote.FileMeta) bool {
func zeroSteps(build *model.Build, remoteYamlConfigs []*remote.FileMeta) bool {
b := procBuilder{
Repo: &model.Repo{},
Curr: &build,
Curr: build,
Last: &model.Build{},
Netrc: &model.Netrc{},
Secs: []*model.Secret{},
Expand Down
47 changes: 43 additions & 4 deletions server/procBuilder.go
Original file line number Diff line number Diff line change
Expand Up @@ -128,17 +128,50 @@ func (b *procBuilder) Build() ([]*buildItem, error) {
item.Labels = map[string]string{}
}

b.Curr.Procs = append(b.Curr.Procs, proc)
items = append(items, item)
pidSequence++
}
}

setBuildSteps(b.Curr, items)
items = filterItemsWithMissingDependencies(items)

return items, nil
}

func filterItemsWithMissingDependencies(items []*buildItem) []*buildItem {
itemsToRemove := make([]*buildItem, 0)

for _, item := range items {
for _, dep := range item.DependsOn {
if !containsItemWithName(dep, items) {
itemsToRemove = append(itemsToRemove, item)
}
}
}

if len(itemsToRemove) > 0 {
filtered := make([]*buildItem, 0)
for _, item := range items {
if !containsItemWithName(item.Proc.Name, itemsToRemove) {
filtered = append(filtered, item)
}
}
// Recursive to handle transitive deps
return filterItemsWithMissingDependencies(filtered)
}

return items
}

func containsItemWithName(name string, items []*buildItem) bool {
for _, item := range items {
if name == item.Proc.Name {
return true
}
}
return false
}

func (b *procBuilder) envsubst_(y string, environ map[string]string) (string, error) {
return envsubst.Eval(y, func(name string) string {
env := environ[name]
Expand Down Expand Up @@ -214,8 +247,12 @@ func (b *procBuilder) toInternalRepresentation(parsed *yaml.Config, environ map[
).Compile(parsed)
}

func setBuildSteps(build *model.Build, buildItems []*buildItem) {
pcounter := len(buildItems)
func setBuildStepsOnBuild(build *model.Build, buildItems []*buildItem) *model.Build {
for _, item := range buildItems {
build.Procs = append(build.Procs, item.Proc)
}
pcounter := len(build.Procs)

for _, item := range buildItems {
for _, stage := range item.Config.Stages {
var gid int
Expand All @@ -239,6 +276,8 @@ func setBuildSteps(build *model.Build, buildItems []*buildItem) {
}
}
}

return build
}

// return the metadata from the cli context.
Expand Down
121 changes: 110 additions & 11 deletions server/procBuilder_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -70,13 +70,11 @@ func TestMultiPipeline(t *testing.T) {
pipeline:
xxx:
image: scratch
yyy: ${DRONE_COMMIT_MESSAGE}
`)},
&remote.FileMeta{Data: []byte(`
pipeline:
build:
image: scratch
yyy: ${DRONE_COMMIT_MESSAGE}
`)},
},
}
Expand All @@ -100,6 +98,16 @@ func TestDependsOn(t *testing.T) {
Regs: []*model.Registry{},
Link: "",
Yamls: []*remote.FileMeta{
&remote.FileMeta{Name: "lint", Data: []byte(`
pipeline:
build:
image: scratch
`)},
&remote.FileMeta{Name: "test", Data: []byte(`
pipeline:
build:
image: scratch
`)},
&remote.FileMeta{Data: []byte(`
pipeline:
deploy:
Expand All @@ -108,7 +116,6 @@ pipeline:
depends_on:
- lint
- test
- build
`)},
},
}
Expand All @@ -117,7 +124,7 @@ depends_on:
if err != nil {
t.Fatal(err)
}
if len(buildItems[0].DependsOn) != 3 {
if len(buildItems[0].DependsOn) != 2 {
t.Fatal("Should have 3 dependencies")
}
if buildItems[0].DependsOn[1] != "test" {
Expand Down Expand Up @@ -173,14 +180,12 @@ func TestBranchFilter(t *testing.T) {
pipeline:
xxx:
image: scratch
yyy: ${DRONE_COMMIT_MESSAGE}
branches: master
`)},
&remote.FileMeta{Data: []byte(`
pipeline:
build:
image: scratch
yyy: ${DRONE_COMMIT_MESSAGE}
`)},
},
}
Expand Down Expand Up @@ -224,7 +229,6 @@ pipeline:
when:
branch: notdev
image: scratch
yyy: ${DRONE_COMMIT_MESSAGE}
`)},
},
}
Expand All @@ -236,8 +240,103 @@ pipeline:
if len(buildItems) != 0 {
t.Fatal("Should not generate a build item if there are no steps")
}
if len(build.Procs) != 0 {
t.Fatal("Should not generate a build item if there are no steps")
}

func TestZeroStepsAsMultiPipelineDeps(t *testing.T) {
build := &model.Build{Branch: "dev"}

b := procBuilder{
Repo: &model.Repo{},
Curr: build,
Last: &model.Build{},
Netrc: &model.Netrc{},
Secs: []*model.Secret{},
Regs: []*model.Registry{},
Link: "",
Yamls: []*remote.FileMeta{
&remote.FileMeta{Name: "zerostep", Data: []byte(`
skip_clone: true
pipeline:
build:
when:
branch: notdev
image: scratch
`)},
&remote.FileMeta{Name: "justastep", Data: []byte(`
pipeline:
build:
image: scratch
`)},
&remote.FileMeta{Name: "shouldbefiltered", Data: []byte(`
pipeline:
build:
image: scratch
depends_on: [ zerostep ]
`)},
},
}

buildItems, err := b.Build()
if err != nil {
t.Fatal(err)
}
if len(buildItems) != 1 {
t.Fatal("Zerostep and the step that depends on it should not generate a build item")
}
if "justastep" != buildItems[0].Proc.Name {
t.Fatal("justastep should have been generated")
}
}

func TestZeroStepsAsMultiPipelineTransitiveDeps(t *testing.T) {
build := &model.Build{Branch: "dev"}

b := procBuilder{
Repo: &model.Repo{},
Curr: build,
Last: &model.Build{},
Netrc: &model.Netrc{},
Secs: []*model.Secret{},
Regs: []*model.Registry{},
Link: "",
Yamls: []*remote.FileMeta{
&remote.FileMeta{Name: "zerostep", Data: []byte(`
skip_clone: true
pipeline:
build:
when:
branch: notdev
image: scratch
`)},
&remote.FileMeta{Name: "justastep", Data: []byte(`
pipeline:
build:
image: scratch
`)},
&remote.FileMeta{Name: "shouldbefiltered", Data: []byte(`
pipeline:
build:
image: scratch
depends_on: [ zerostep ]
`)},
&remote.FileMeta{Name: "shouldbefilteredtoo", Data: []byte(`
pipeline:
build:
image: scratch
depends_on: [ shouldbefiltered ]
`)},
},
}

buildItems, err := b.Build()
if err != nil {
t.Fatal(err)
}
if len(buildItems) != 1 {
t.Fatal("Zerostep and the step that depends on it, and the one depending on it should not generate a build item")
}
if "justastep" != buildItems[0].Proc.Name {
t.Fatal("justastep should have been generated")
}
}

Expand All @@ -257,12 +356,12 @@ func TestTree(t *testing.T) {
pipeline:
build:
image: scratch
yyy: ${DRONE_COMMIT_MESSAGE}
`)},
},
}

_, err := b.Build()
buildItems, err := b.Build()
build = setBuildStepsOnBuild(build, buildItems)
if err != nil {
t.Fatal(err)
}
Expand Down

0 comments on commit 063b7fc

Please sign in to comment.