Skip to content

Commit

Permalink
feat(orca) Place produced artifacts in global context
Browse files Browse the repository at this point in the history
  • Loading branch information
brujoand committed Oct 11, 2017
1 parent 3c89a96 commit 50e5dbc
Show file tree
Hide file tree
Showing 12 changed files with 246 additions and 11 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@

package com.netflix.spinnaker.orca.bakery.api

import com.netflix.spinnaker.orca.pipeline.model.Artifact
import groovy.transform.CompileStatic
import groovy.transform.EqualsAndHashCode
import groovy.transform.ToString
Expand All @@ -35,5 +36,6 @@ class Bake {
String ami
String amiName
String imageName
Artifact artifact
// TODO(duftler): Add a cloudProviderType property here? Will be straightforward once rosco is backed by redis.
}
Original file line number Diff line number Diff line change
Expand Up @@ -127,10 +127,12 @@ class BakeStage implements StageDefinitionBuilder, RestartableStage {
it.parentStageId == stage.parentStageId && it.status == ExecutionStatus.RUNNING
}

def relatedBakeStages = stage.execution.stages.findAll {
it.type == PIPELINE_CONFIG_TYPE && bakeInitializationStages*.id.contains(it.parentStageId)
}

def globalContext = [
deploymentDetails: stage.execution.stages.findAll {
it.type == PIPELINE_CONFIG_TYPE && bakeInitializationStages*.id.contains(it.parentStageId) && (it.context.ami || it.context.imageId)
}.collect { Stage bakeStage ->
deploymentDetails: relatedBakeStages.findAll{it.context.ami || it.context.imageId}.collect { Stage bakeStage ->
def deploymentDetails = [:]
["ami", "imageId", "amiSuffix", "baseLabel", "baseOs", "refId", "storeType", "vmType", "region", "package", "cloudProviderType", "cloudProvider"].each {
if (bakeStage.context.containsKey(it)) {
Expand All @@ -139,6 +141,9 @@ class BakeStage implements StageDefinitionBuilder, RestartableStage {
}

return deploymentDetails
},
artifacts: relatedBakeStages.findAll{it.context.artifact}.collect { Stage bakeStage ->
return bakeStage.context.get("artifact") ?: [:]
}
]
new TaskResult(ExecutionStatus.SUCCEEDED, [:], globalContext)
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@ import com.netflix.spinnaker.orca.Task
import com.netflix.spinnaker.orca.TaskResult
import com.netflix.spinnaker.orca.bakery.api.BakeStatus
import com.netflix.spinnaker.orca.bakery.api.BakeryService
import com.netflix.spinnaker.orca.pipeline.model.Artifact
import com.netflix.spinnaker.orca.pipeline.model.Stage
import groovy.transform.CompileStatic
import org.springframework.beans.factory.annotation.Autowired
Expand All @@ -39,7 +40,7 @@ class CompletedBakeTask implements Task {
def bakeStatus = stage.context.status as BakeStatus
def bake = bakery.lookupBake(region, bakeStatus.resourceId).toBlocking().first()
// This treatment of ami allows both the aws and gce bake results to be propagated.
def results = [ami: bake.ami ?: bake.imageName, imageId: bake.ami ?: bake.imageName]
def results = [ami: bake.ami ?: bake.imageName, imageId: bake.ami ?: bake.imageName, artifact: bake.artifact ?: new Artifact()]
/**
* TODO:
* It would be good to standardize on the key here. "imageId" works for all providers.
Expand All @@ -49,6 +50,7 @@ class CompletedBakeTask implements Task {
if (bake.imageName || bake.amiName) {
results.imageName = bake.imageName ?: bake.amiName
}

new TaskResult(ExecutionStatus.SUCCEEDED, results)
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@
package com.netflix.spinnaker.orca.bakery.pipeline

import com.netflix.spinnaker.orca.ExecutionStatus
import com.netflix.spinnaker.orca.pipeline.model.Artifact
import com.netflix.spinnaker.orca.pipeline.model.Stage
import groovy.time.TimeCategory
import spock.lang.Specification
Expand Down Expand Up @@ -108,6 +109,33 @@ class BakeStageSpec extends Specification {
}
}

def "should include provided artifacts"() {
given:
def pipeline = pipeline {
stage {
id = "1"
type = "bake"
context = [
"region": "global",
]
status = ExecutionStatus.RUNNING
}
}

def bakeStage = pipeline.stageById("1")
def parallelStages = new BakeStage().parallelStages(bakeStage)
parallelStages.each { it.context.artifact = new Artifact(name: "myArtifact") }
pipeline.stages.addAll(parallelStages)

when:
def taskResult = new BakeStage.CompleteParallelBakeTask().execute(pipeline.stageById("1"))

then:
with(taskResult.outputs) {
artifacts[0].name == "myArtifact"
}
}

private
static List<Map> deployAz(String cloudProvider, String prefix, String... regions) {
if (prefix == "clusters") {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -20,6 +20,7 @@ import com.netflix.spinnaker.orca.ExecutionStatus
import com.netflix.spinnaker.orca.bakery.api.Bake
import com.netflix.spinnaker.orca.bakery.api.BakeStatus
import com.netflix.spinnaker.orca.bakery.api.BakeryService
import com.netflix.spinnaker.orca.pipeline.model.Artifact
import com.netflix.spinnaker.orca.pipeline.model.Pipeline
import com.netflix.spinnaker.orca.pipeline.model.Stage
import retrofit.RetrofitError
Expand All @@ -44,10 +45,10 @@ class CompletedBakeTaskSpec extends Specification {
null
)

def "finds the AMI created by a bake"() {
def "finds the AMI and artifact created by a bake"() {
given:
task.bakery = Stub(BakeryService) {
lookupBake(region, bakeId) >> Observable.from(new Bake(id: bakeId, ami: ami))
lookupBake(region, bakeId) >> Observable.from(new Bake(id: bakeId, ami: ami, artifact: artifact))
}

and:
Expand All @@ -59,11 +60,13 @@ class CompletedBakeTaskSpec extends Specification {
then:
result.status == ExecutionStatus.SUCCEEDED
result.context.ami == ami
result.context.artifact.reference == ami

where:
region = "us-west-1"
bakeId = "b-5af233wjj78mwt2f420wt8ey3w"
ami = "ami-280c3b6d"
artifact = new Artifact(reference: ami)
}

def "fails if the bake is not found"() {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import com.netflix.spinnaker.orca.TaskResult
import com.netflix.spinnaker.orca.clouddriver.OortService
import com.netflix.spinnaker.orca.clouddriver.pipeline.servergroup.support.Location
import com.netflix.spinnaker.orca.clouddriver.tasks.AbstractCloudProviderAwareTask
import com.netflix.spinnaker.orca.pipeline.model.Artifact
import com.netflix.spinnaker.orca.pipeline.model.Stage
import groovy.transform.Canonical
import groovy.util.logging.Slf4j
Expand Down Expand Up @@ -229,10 +230,45 @@ class FindImageFromClusterTask extends AbstractCloudProviderAwareTask implements
}
}.flatten()

List<Artifact> artifacts = imageSummaries.collect { placement, summaries ->
Artifact artifact = new Artifact()
summaries.findResults { summary ->
if (config.imageNamePattern && !(summary.imageName ==~ config.imageNamePattern)) {
return null
}
def location = "global"

if (placement.type == Location.Type.REGION) {
location = placement.value
} else if (placement.type == Location.Type.ZONE) {
location = placement.value
}

def metadata = [
sourceServerGroup: summary.serverGroupName,
refId: stage.refId
]

try {
metadata.putAll(summary.image ?: [:])
metadata.putAll(summary.buildInfo ?: [:])
} catch (Exception e) {
log.error("Unable to merge server group image/build info (summary: ${summary})", e)
}

artifact.metadata = metadata
artifact.name = summary.imageName
artifact.type = "${cloudProvider}/image/${location}"
artifact.reference = "${summary.imageId}"
}
return artifact
}.flatten()

return new TaskResult(ExecutionStatus.SUCCEEDED, [
amiDetails: deploymentDetails
], [
deploymentDetails: deploymentDetails
deploymentDetails: deploymentDetails,
artifacts: artifacts
])
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,8 +16,10 @@

package com.netflix.spinnaker.orca.clouddriver.tasks.image;

import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import com.fasterxml.jackson.annotation.JsonProperty;
Expand All @@ -26,6 +28,7 @@
import com.netflix.spinnaker.orca.RetryableTask;
import com.netflix.spinnaker.orca.TaskResult;
import com.netflix.spinnaker.orca.clouddriver.tasks.AbstractCloudProviderAwareTask;
import com.netflix.spinnaker.orca.pipeline.model.Artifact;
import com.netflix.spinnaker.orca.pipeline.model.Stage;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;
Expand Down Expand Up @@ -53,13 +56,38 @@ public TaskResult execute(Stage stage) {
throw new IllegalStateException("Could not find tagged image for package: " + stageData.packageName + " and tags: " + stageData.tags);
}

List<Artifact> artifacts = new ArrayList<>();
imageDetails.forEach(imageDetail -> artifacts.add(generateArtifactFrom(imageDetail, cloudProvider)));

Map<String, Object> globalOutputs = new HashMap<>();
globalOutputs.put("deploymentDetails", imageDetails);
globalOutputs.put("artifacts", artifacts);

return new TaskResult(
ExecutionStatus.SUCCEEDED,
Collections.singletonMap("amiDetails", imageDetails),
Collections.singletonMap("deploymentDetails", imageDetails)
globalOutputs
);

}

private Artifact generateArtifactFrom(ImageFinder.ImageDetails imageDetails, String cloudProvider) {
Map<String, String> metadata = new HashMap<>();
try {
ImageFinder.JenkinsDetails jenkinsDetails = imageDetails.getJenkins();
metadata.put("build_info_url", jenkinsDetails.get("host"));
metadata.put("build_number", jenkinsDetails.get("number"));
} catch (Exception e) {
// This is either all or nothing
}

Artifact artifact = new Artifact();
artifact.setName(imageDetails.getImageName());
artifact.setReference(imageDetails.getImageId());
artifact.setType(cloudProvider + "/image/" + imageDetails.getRegion());
artifact.setMetadata(metadata);

return artifact;
}

@Override
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ public interface ImageFinder {
interface ImageDetails {
String getImageId();
String getImageName();
String getRegion();
JenkinsDetails getJenkins();
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -144,6 +144,11 @@ public String getImageName() {
return (String) super.get("imageName");
}

@Override
public String getRegion() {
return (String) super.get("region");
}

@Override
public JenkinsDetails getJenkins() {
return (JenkinsDetails) super.get("jenkins");
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -132,6 +132,11 @@ public String getImageName() {
return (String) super.get("imageName");
}

@Override
public String getRegion() {
return (String) super.get("region");
}

@Override
public JenkinsDetails getJenkins() {
return (JenkinsDetails) super.get("jenkins");
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,13 @@ import spock.lang.Subject
class FindImageFromTagTaskSpec extends Specification {

def imageFinder = Mock(ImageFinder)
def imageDetails = Mock(ImageFinder.ImageDetails)
Stage stage = new Stage<>(new Pipeline("orca"), "", [packageName: 'myPackage', tags: ['foo': 'bar']])

@Subject
def task = new FindImageFromTagsTask(imageFinders: [imageFinder])

def "Not finding an images should throw IllegalStateException"() {
def "Not finding images should throw IllegalStateException"() {
when:
task.execute(stage)

Expand All @@ -25,14 +26,18 @@ class FindImageFromTagTaskSpec extends Specification {
1 * imageFinder.getCloudProvider() >> 'aws'
}

def "Finding an images should set task state to SUCCEEDED"() {
def "Finding images should set task state to SUCCEEDED"() {
when:
def result = task.execute(stage)

then:
result.status == ExecutionStatus.SUCCEEDED

1 * imageFinder.getCloudProvider() >> 'aws'
1 * imageFinder.byTags(stage, stage.context.packageName, stage.context.tags) >> [[:]]
1 * imageFinder.byTags(stage, stage.context.packageName, stage.context.tags) >> [imageDetails]
1 * imageDetails.getImageName() >> "somename"
1 * imageDetails.getImageId() >> "someId"
1 * imageDetails.getJenkins() >> new ImageFinder.JenkinsDetails("somehost", "somename", "42")
}

}
Loading

0 comments on commit 50e5dbc

Please sign in to comment.