From ff987dcea557a2f0d72d97ccd23b368445a70fe5 Mon Sep 17 00:00:00 2001 From: Christos Arvanitis Date: Tue, 10 Sep 2024 19:49:22 +0300 Subject: [PATCH] chore(runJobStage): Fix logs annotation processing on manifest artifact (#4778) --- .../clouddriver/pipeline/job/RunJobStage.java | 21 +++ .../pipeline/job/RunJobStageTest.java | 158 ++++++++++++++++++ 2 files changed, 179 insertions(+) create mode 100644 orca-clouddriver/src/test/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStageTest.java diff --git a/orca-clouddriver/src/main/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStage.java b/orca-clouddriver/src/main/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStage.java index 17c5ca50ec..51c84b9e1b 100644 --- a/orca-clouddriver/src/main/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStage.java +++ b/orca-clouddriver/src/main/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStage.java @@ -18,6 +18,7 @@ import static java.util.Collections.emptyMap; +import com.fasterxml.jackson.core.type.TypeReference; import com.fasterxml.jackson.databind.ObjectMapper; import com.netflix.spinnaker.orca.api.pipeline.CancellableStage; import com.netflix.spinnaker.orca.api.pipeline.graph.StageDefinitionBuilder; @@ -29,6 +30,8 @@ import com.netflix.spinnaker.orca.clouddriver.tasks.job.MonitorJobTask; import com.netflix.spinnaker.orca.clouddriver.tasks.job.RunJobTask; import com.netflix.spinnaker.orca.clouddriver.tasks.job.WaitOnJobCompletion; +import com.netflix.spinnaker.orca.clouddriver.tasks.providers.kubernetes.Manifest; +import com.netflix.spinnaker.orca.clouddriver.tasks.providers.kubernetes.ManifestAnnotationExtractor; import com.netflix.spinnaker.orca.jackson.OrcaObjectMapper; import com.netflix.spinnaker.orca.pipeline.tasks.artifacts.BindProducedArtifactsTask; import java.util.*; @@ -83,9 +86,11 @@ public void taskGraph(@Nonnull StageExecution stage, @Nonnull TaskNode.Builder b @Override public void afterStages(@Nonnull StageExecution stage, @Nonnull StageGraphBuilder graph) { + Map executionLogs = getLogOutputFromManifestArtifact(stage); if (stage.getContext().getOrDefault("noOutput", "false").toString().equals("true")) { stage.setOutputs(emptyMap()); } + stage.getContext().putAll(executionLogs); } @Override @@ -155,4 +160,20 @@ private Optional getCloudProviderDecorator(StageExecution .filter(it -> it.supports(cloudProvider)) .findFirst()); } + + public Map getLogOutputFromManifestArtifact(StageExecution stage) { + Map outputs = new HashMap<>(); + Map execution = new HashMap<>(); + if (stage.getContext().containsKey("manifestArtifact")) { + List manifest = + objectMapper.convertValue( + stage.getContext().get("outputs.manifests"), new TypeReference>() {}); + String logTemplate = ManifestAnnotationExtractor.logs(manifest.get(0)); + if (logTemplate != null) { + execution.put("logs", logTemplate); + outputs.put("execution", execution); + } + } + return outputs; + } } diff --git a/orca-clouddriver/src/test/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStageTest.java b/orca-clouddriver/src/test/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStageTest.java new file mode 100644 index 0000000000..90620e0d62 --- /dev/null +++ b/orca-clouddriver/src/test/java/com/netflix/spinnaker/orca/clouddriver/pipeline/job/RunJobStageTest.java @@ -0,0 +1,158 @@ +/* + * Copyright 2024 Harness, Inc. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +package com.netflix.spinnaker.orca.clouddriver.pipeline.job; + +import static org.assertj.core.api.Assertions.assertThat; +import static org.assertj.core.api.Assertions.entry; + +import com.fasterxml.jackson.core.type.TypeReference; +import com.fasterxml.jackson.databind.ObjectMapper; +import com.google.common.collect.ImmutableMap; +import com.netflix.spinnaker.orca.api.pipeline.models.ExecutionStatus; +import com.netflix.spinnaker.orca.api.pipeline.models.ExecutionType; +import com.netflix.spinnaker.orca.api.pipeline.models.StageExecution; +import com.netflix.spinnaker.orca.clouddriver.tasks.job.DestroyJobTask; +import com.netflix.spinnaker.orca.clouddriver.tasks.providers.kubernetes.Manifest; +import com.netflix.spinnaker.orca.jackson.OrcaObjectMapper; +import com.netflix.spinnaker.orca.pipeline.graph.StageGraphBuilderImpl; +import com.netflix.spinnaker.orca.pipeline.model.PipelineExecutionImpl; +import com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl; +import java.util.List; +import java.util.Map; +import org.junit.jupiter.api.BeforeEach; +import org.junit.jupiter.api.Test; +import org.junit.jupiter.api.extension.ExtendWith; +import org.mockito.junit.jupiter.MockitoExtension; + +@ExtendWith(MockitoExtension.class) +class RunJobStageTest { + private RunJobStage runJobStage; + + private DestroyJobTask destroyJobTask; + + private static final String APPLICATION = "my-app"; + + private static final String ACCOUNT = "my-acct"; + + private static final String RUNJOB_STAGE_TYPE = "runJobManifest"; + + private static final ObjectMapper objectMapper = OrcaObjectMapper.getInstance(); + + @BeforeEach + void setUp() { + runJobStage = new RunJobStage(destroyJobTask, null); + } + + @Test + void testManifestArtifactWithLogAnnotation() { + RunJobStageContext runJobStageContext = new RunJobStageContext(); + Map context = + objectMapper.convertValue(runJobStageContext, new TypeReference>() {}); + context.put("account", ACCOUNT); + context.put("type", RUNJOB_STAGE_TYPE); + context.put("status", ExecutionStatus.SUCCEEDED); + context.put( + "manifestArtifact", + ImmutableMap.of("artifactAccount", "github", "reference", "my-artifact")); + + Manifest jobManifest = new Manifest(); + jobManifest.metadata.setAnnotations(Map.of("job.spinnaker.io/logs", "https://example.com/")); + context.put("outputs.manifests", List.of(jobManifest)); + StageExecutionImpl stage = + new StageExecutionImpl( + new PipelineExecutionImpl(ExecutionType.PIPELINE, APPLICATION), + RUNJOB_STAGE_TYPE, + context); + stage.setOutputs(context); + getAfterStages(stage); + + assertThat(stage.getContext()).extracting(s -> s.get("manifestArtifact")).isNotNull(); + assertThat(stage.getContext()).extracting(s -> s.get("outputs.manifests")).isNotNull(); + assertThat(stage.getContext()) + .extracting(s -> s.get("execution")) + .isEqualTo(ImmutableMap.of("logs", "https://example.com/")); + assertThat(stage.getOutputs()).isNotEmpty(); + } + + @Test + void testManifestArtifactWithOutLogAnnotation() { + RunJobStageContext runJobStageContext = new RunJobStageContext(); + Map context = + objectMapper.convertValue(runJobStageContext, new TypeReference>() {}); + context.put("account", ACCOUNT); + context.put("type", RUNJOB_STAGE_TYPE); + context.put("status", ExecutionStatus.SUCCEEDED); + context.put( + "manifestArtifact", + ImmutableMap.of("artifactAccount", "github", "reference", "my-artifact")); + + Manifest jobManifest = new Manifest(); + jobManifest.metadata.setAnnotations(Map.of("app", "name")); + context.put("outputs.manifests", List.of(jobManifest)); + StageExecutionImpl stage = + new StageExecutionImpl( + new PipelineExecutionImpl(ExecutionType.PIPELINE, APPLICATION), + RUNJOB_STAGE_TYPE, + context); + stage.setOutputs(context); + getAfterStages(stage); + + assertThat(stage.getContext()).extracting(s -> s.get("manifestArtifact")).isNotNull(); + assertThat(stage.getContext()).extracting(s -> s.get("outputs.manifests")).isNotNull(); + assertThat(stage.getContext()) + .doesNotContain(entry("execution", ImmutableMap.of("logs", "https://example.com/"))); + assertThat(stage.getOutputs()).isNotEmpty(); + } + + @Test + void testManifestArtifactLogAnnotationANDnoOutput() { + RunJobStageContext runJobStageContext = new RunJobStageContext(); + Map context = + objectMapper.convertValue(runJobStageContext, new TypeReference>() {}); + context.put("account", ACCOUNT); + context.put("type", RUNJOB_STAGE_TYPE); + context.put("status", ExecutionStatus.SUCCEEDED); + context.put( + "manifestArtifact", + ImmutableMap.of("artifactAccount", "github", "reference", "my-artifact")); + context.put("noOutput", true); + + Manifest jobManifest = new Manifest(); + jobManifest.metadata.setAnnotations(Map.of("job.spinnaker.io/logs", "https://example.com/")); + context.put("outputs.manifests", List.of(jobManifest)); + StageExecutionImpl stage = + new StageExecutionImpl( + new PipelineExecutionImpl(ExecutionType.PIPELINE, APPLICATION), + RUNJOB_STAGE_TYPE, + context); + stage.setOutputs(context); + getAfterStages(stage); + + assertThat(stage.getContext()).extracting(s -> s.get("manifestArtifact")).isNotNull(); + assertThat(stage.getContext()).extracting(s -> s.get("outputs.manifests")).isNotNull(); + assertThat(stage.getContext()) + .extracting(s -> s.get("execution")) + .isEqualTo(ImmutableMap.of("logs", "https://example.com/")); + assertThat(stage.getOutputs()).isEmpty(); + } + + private Iterable getAfterStages(StageExecutionImpl stage) { + StageGraphBuilderImpl graph = StageGraphBuilderImpl.afterStages(stage); + runJobStage.afterStages(stage, graph); + return graph.build(); + } +}