Skip to content

Commit

Permalink
fix(clouddriver): Throw SpinnakerHttpException when Http status is 40…
Browse files Browse the repository at this point in the history
…3 in DetermineSourceServerGroupTask (#4710)

* test(clouddriver): Demonstrate bug in DetermineSourceServerGroupTask

This task/class is suppose to throw a SpinnakerHttpException when a HTTP forbidden 403 error occurs, but instead it returns TaskResult in which the execution status and exception message is wrapped.

This task/class actually makes a couple of oortService API calls: 1 - oortService.getTargetServerGroup() - API to fetch the server group based on the target. 2 - oortService.getServerGroupsFromClusterTyped() - API to fetch server group based on server group name.

From the commit: 84a7106 , OortService uses SpinnakerRetrofitErrorHandler so, the expectation is to throw SpinnakerHttpException.

* fix(clouddriver): Throw SpinnakerHttpException when Http status is 403 in DetermineSourceServerGroupTask

This task earlier returns TaskResult with RUNNING status and exception message in context when Forbidden error 403 has occurred. With the new changes/fix it now throws SpinnakerHttpException in this scenario making it in compliance with the commit: 84a7106

Here is the stack trace that would demonstrate the bug and the fix -

Before Fix:

TaskResult(status=RUNNING, context={lastException=Status: 403, URL: http://localhost:39831/applications/testCluster/clusters/test/testCluster/aws/us-east-1/serverGroups/target/ancestor_asg_dynamic, Message: Forbidden
com.netflix.spinnaker.kork.retrofit.exceptions.SpinnakerHttpException: Status: 403, URL: http://localhost:39831/applications/testCluster/clusters/test/testCluster/aws/us-east-1/serverGroups/target/ancestor_asg_dynamic, Message: Forbidden
	at com.netflix.spinnaker.kork.retrofit.exceptions.SpinnakerRetrofitErrorHandler.handleError(SpinnakerRetrofitErrorHandler.java:55)
	at retrofit.RestAdapter$RestHandler.invoke(RestAdapter.java:242)
	at jdk.proxy3/jdk.proxy3.$Proxy21.getTargetServerGroup(Unknown Source)
	at com.netflix.spinnaker.orca.clouddriver.OortService$getTargetServerGroup.call(Unknown Source)
	at com.netflix.spinnaker.orca.clouddriver.pipeline.servergroup.support.TargetServerGroupResolver$_resolveByTarget_closure2.doCall(TargetServerGroupResolver.groovy:76)
	at com.netflix.spinnaker.orca.clouddriver.pipeline.servergroup.support.TargetServerGroupResolver$_resolveByTarget_closure2.doCall(TargetServerGroupResolver.groovy)
, attempt=2, consecutiveNotFound=0}, outputs={})

After Fix:

2024-04-24 16:48:25.900 ERROR   --- [    Test worker] .n.s.o.c.p.s.s.TargetServerGroupResolver : [] Unable to locate ancestor_asg_dynamic in test/us-east-1/testCluster
com.netflix.spinnaker.kork.retrofit.exceptions.SpinnakerHttpException: Status: 403, URL: http://localhost:32773/applications/testCluster/clusters/test/testCluster/aws/us-east-1/serverGroups/target/ancestor_asg_dynamic, Message: Forbidden
	at com.netflix.spinnaker.kork.retrofit.exceptions.SpinnakerRetrofitErrorHandler.handleError(SpinnakerRetrofitErrorHandler.java:55)
	at retrofit.RestAdapter$RestHandler.invoke(RestAdapter.java:242)
	at jdk.proxy3/jdk.proxy3.$Proxy21.getTargetServerGroup(Unknown Source)
	at com.netflix.spinnaker.orca.clouddriver.OortService$getTargetServerGroup.call(Unknown Source)
	at com.netflix.spinnaker.orca.clouddriver.pipeline.servergroup.support.TargetServerGroupResolver$_resolveByTarget_closure2.doCall(TargetServerGroupResolver.groovy:76)
	at com.netflix.spinnaker.orca.clouddriver.pipeline.servergroup.support.TargetServerGroupResolver$_resolveByTarget_closure2.doCall(TargetServerGroupResolver.groovy)

---------

Co-authored-by: Pranav-b-7 <[email protected]>
  • Loading branch information
Pranav-b-7 and Pranav-b-7 authored Apr 24, 2024
1 parent 3bc345b commit d2f5024
Show file tree
Hide file tree
Showing 2 changed files with 211 additions and 3 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@

package com.netflix.spinnaker.orca.kato.pipeline.strategy


import com.netflix.spinnaker.kork.retrofit.exceptions.SpinnakerHttpException
import com.netflix.spinnaker.orca.api.pipeline.RetryableTask
import com.netflix.spinnaker.orca.api.pipeline.TaskResult
import com.netflix.spinnaker.orca.api.pipeline.models.ExecutionStatus
Expand All @@ -26,7 +26,6 @@ import com.netflix.spinnaker.orca.kato.pipeline.support.StageData
import groovy.util.logging.Slf4j
import org.springframework.beans.factory.annotation.Autowired
import org.springframework.stereotype.Component
import retrofit.RetrofitError

import java.util.concurrent.TimeUnit

Expand Down Expand Up @@ -96,7 +95,7 @@ class DetermineSourceServerGroupTask implements RetryableTask {
consecutiveNotFound: isNotFound ? (stage.context.consecutiveNotFound ?: 0) + 1 : 0
]

if (lastException instanceof RetrofitError && lastException.response?.status == HTTP_FORBIDDEN) {
if (lastException instanceof SpinnakerHttpException && lastException.responseCode == HTTP_FORBIDDEN) {
// short-circuit on a 403 and allow the `RetrofitExceptionHandler` to handle and propagate any error messages
throw lastException;
}
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,209 @@
/*
* Copyright 2024 OpsMx, 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.kato.pipeline.strategy;

import static com.github.tomakehurst.wiremock.client.WireMock.aResponse;
import static com.github.tomakehurst.wiremock.client.WireMock.urlPathEqualTo;
import static com.github.tomakehurst.wiremock.core.WireMockConfiguration.wireMockConfig;
import static org.assertj.core.api.Assertions.assertThatThrownBy;

import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.github.tomakehurst.wiremock.client.WireMock;
import com.github.tomakehurst.wiremock.http.HttpHeaders;
import com.github.tomakehurst.wiremock.junit5.WireMockExtension;
import com.github.tomakehurst.wiremock.junit5.WireMockRuntimeInfo;
import com.netflix.spinnaker.kork.core.RetrySupport;
import com.netflix.spinnaker.kork.retrofit.exceptions.SpinnakerHttpException;
import com.netflix.spinnaker.kork.retrofit.exceptions.SpinnakerRetrofitErrorHandler;
import com.netflix.spinnaker.okhttp.SpinnakerRequestInterceptor;
import com.netflix.spinnaker.orca.clouddriver.CloudDriverService;
import com.netflix.spinnaker.orca.clouddriver.OortService;
import com.netflix.spinnaker.orca.clouddriver.model.Cluster;
import com.netflix.spinnaker.orca.clouddriver.model.ServerGroup;
import com.netflix.spinnaker.orca.clouddriver.pipeline.servergroup.support.TargetServerGroup;
import com.netflix.spinnaker.orca.clouddriver.pipeline.servergroup.support.TargetServerGroupResolver;
import com.netflix.spinnaker.orca.kato.pipeline.support.SourceResolver;
import com.netflix.spinnaker.orca.pipeline.model.PipelineExecutionImpl;
import com.netflix.spinnaker.orca.pipeline.model.StageExecutionImpl;
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
import org.jetbrains.annotations.NotNull;
import org.junit.jupiter.api.BeforeAll;
import org.junit.jupiter.api.BeforeEach;
import org.junit.jupiter.api.Test;
import org.junit.jupiter.api.extension.RegisterExtension;
import org.springframework.http.HttpStatus;
import retrofit.RestAdapter;
import retrofit.client.OkClient;
import retrofit.converter.JacksonConverter;

public class DetermineSourceServerGroupTaskTest {

private DetermineSourceServerGroupTask determineSourceServerGroupTask;
private SourceResolver sourceResolver;
private TargetServerGroupResolver resolver;

private static ObjectMapper objectMapper = new ObjectMapper();

private static OortService oortService;

private CloudDriverService cloudDriverService;

@RegisterExtension
private static WireMockExtension wireMock =
WireMockExtension.newInstance().options(wireMockConfig().dynamicPort()).build();

private static void simulateGETApiCall(String url, String body, HttpStatus httpStatus) {
wireMock.givenThat(
WireMock.get(urlPathEqualTo(url))
.willReturn(
aResponse()
.withHeaders(HttpHeaders.noHeaders())
.withStatus(httpStatus.value())
.withBody(body)));
}

@BeforeAll
public static void setupOnce(WireMockRuntimeInfo wmRuntimeInfo) {
OkClient okClient = new OkClient();
RestAdapter.LogLevel retrofitLogLevel = RestAdapter.LogLevel.NONE;

oortService =
new RestAdapter.Builder()
.setRequestInterceptor(new SpinnakerRequestInterceptor(true))
.setEndpoint(wmRuntimeInfo.getHttpBaseUrl())
.setClient(okClient)
.setLogLevel(retrofitLogLevel)
.setErrorHandler(SpinnakerRetrofitErrorHandler.getInstance())
.setConverter(new JacksonConverter(objectMapper))
.build()
.create(OortService.class);
}

@BeforeEach
public void setup() {

resolver = new TargetServerGroupResolver();
resolver.setMapper(objectMapper);
resolver.setOortService(oortService);
resolver.setRetrySupport(new RetrySupport());

cloudDriverService = new CloudDriverService(oortService, new ObjectMapper());
sourceResolver = new SourceResolver();
sourceResolver.setMapper(objectMapper);
sourceResolver.setResolver(resolver);
sourceResolver.setCloudDriverService(cloudDriverService);

determineSourceServerGroupTask = new DetermineSourceServerGroupTask();
determineSourceServerGroupTask.setSourceResolver(sourceResolver);
}

@Test
public void testForbiddenError() throws JsonProcessingException {

StageExecutionImpl stage = createStage();

ServerGroup serverGroup = createServerGroup();

Cluster cluster = createCluster(serverGroup);

// simulate getCluster() API call in SourceResolver
simulateGETApiCall(
"/applications/foo/clusters/test/foo/aws",
objectMapper.writeValueAsString(cluster),
HttpStatus.OK);

// simulate getTargetServerGroup API call in TargetServerGroupResolver
simulateGETApiCall(
"/applications/testCluster/clusters/test/testCluster/aws/us-east-1/serverGroups/target/ancestor_asg_dynamic",
objectMapper.writeValueAsString(serverGroup),
HttpStatus.FORBIDDEN);

assertThatThrownBy(() -> determineSourceServerGroupTask.execute(stage))
.isExactlyInstanceOf(SpinnakerHttpException.class)
.hasMessage(
"Status: 403, URL: http://localhost:"
+ wireMock.getPort()
+ "/applications/testCluster/clusters/test/testCluster/aws/us-east-1/serverGroups/target/ancestor_asg_dynamic, Message: Forbidden");
}

private @NotNull Cluster createCluster(ServerGroup serverGroup) {
Cluster cluster = new Cluster();
cluster.setName("testcluster");
cluster.setServerGroups(new ArrayList<>(List.of(serverGroup)));
return cluster;
}

private @NotNull ServerGroup createServerGroup() {
ServerGroup.Asg asg = new ServerGroup.Asg();

asg.setDesiredCapacity(10);
asg.setMaxSize(10);
asg.setMinSize(1);

ServerGroup.Process process = new ServerGroup.Process();
process.processName = "testProcess";

asg.setSuspendedProcesses(List.of(process));

ServerGroup serverGroup = new ServerGroup();
serverGroup.account = "test";
serverGroup.name = "test";
serverGroup.asg = asg;

ServerGroup.Image image = new ServerGroup.Image();
image.imageId = "image/image-1234";
image.name = "k8";
serverGroup.image = image;
return serverGroup;
}

private @NotNull StageExecutionImpl createStage() {
StageExecutionImpl stage =
new StageExecutionImpl(
PipelineExecutionImpl.newPipeline("orca"),
"deploy",
"deploy",
new HashMap<>(
Map.of(
"account", "test",
"application", "foo",
"asgName", "foo-test-v000",
"availabilityZones", Map.of("region", List.of("us-east-1")))));

stage.setContext(
Map.of(
"target",
TargetServerGroup.Params.Target.ancestor_asg_dynamic.name(),
"source",
Map.of(
"account",
"test",
"region",
"us-east-1",
"asgName",
"foo-test-v000",
"serverGroupName",
"test",
"clusterName",
"testCluster")));
return stage;
}
}

0 comments on commit d2f5024

Please sign in to comment.