package software.amazon.events.rule; import com.fasterxml.jackson.core.JsonProcessingException; import com.fasterxml.jackson.core.type.TypeReference; import com.fasterxml.jackson.databind.ObjectMapper; import org.junit.jupiter.api.AfterEach; import org.junit.jupiter.api.BeforeEach; import org.junit.jupiter.api.Test; import org.junit.jupiter.api.extension.ExtendWith; import org.mockito.Mock; import org.mockito.junit.jupiter.MockitoExtension; import software.amazon.awssdk.services.cloudwatchevents.CloudWatchEventsClient; import software.amazon.awssdk.services.cloudwatchevents.model.InvalidEventPatternException; import software.amazon.awssdk.services.cloudwatchevents.model.PutRuleRequest; import software.amazon.awssdk.services.cloudwatchevents.model.PutRuleResponse; import software.amazon.awssdk.services.cloudwatchevents.model.PutTargetsRequest; import software.amazon.awssdk.services.cloudwatchevents.model.PutTargetsResponse; import software.amazon.awssdk.services.cloudwatchevents.model.PutTargetsResultEntry; import software.amazon.awssdk.services.cloudwatchevents.model.RemoveTargetsRequest; import software.amazon.awssdk.services.cloudwatchevents.model.RemoveTargetsResponse; import software.amazon.awssdk.services.cloudwatchevents.model.RemoveTargetsResultEntry; import software.amazon.cloudformation.proxy.AmazonWebServicesClientProxy; import software.amazon.cloudformation.proxy.HandlerErrorCode; import software.amazon.cloudformation.proxy.OperationStatus; import software.amazon.cloudformation.proxy.ProgressEvent; import software.amazon.cloudformation.proxy.ProxyClient; import software.amazon.cloudformation.proxy.ResourceHandlerRequest; import java.time.Duration; import java.util.ArrayList; import java.util.Collection; import java.util.HashMap; import java.util.HashSet; import java.util.Map; import java.util.Set; import static org.assertj.core.api.Assertions.assertThat; import static org.mockito.ArgumentMatchers.any; import static org.mockito.Mockito.atLeastOnce; import static org.mockito.Mockito.mock; import static org.mockito.Mockito.verify; import static org.mockito.Mockito.verifyNoMoreInteractions; import static org.mockito.Mockito.when; @ExtendWith(MockitoExtension.class) public class UpdateHandlerTest extends AbstractTestBase { @Mock private AmazonWebServicesClientProxy proxy; @Mock private ProxyClient proxyClient; @Mock CloudWatchEventsClient sdkClient; private static final ObjectMapper MAPPER = new ObjectMapper(); @BeforeEach public void setup() { System.setProperty("aws.region", "us-west-2"); proxy = new AmazonWebServicesClientProxy(logger, MOCK_CREDENTIALS, () -> Duration.ofSeconds(600).toMillis()); sdkClient = mock(CloudWatchEventsClient.class); proxyClient = MOCK_PROXY(proxy, sdkClient); } //@AfterEach FIXME This seems like a bad fix public void tear_down() { verify(sdkClient, atLeastOnce()).serviceName(); verifyNoMoreInteractions(sdkClient); } @Test public void handleRequest_SimpleSuccess() { final UpdateHandler handler = new UpdateHandler(); String eventPatternString = String.join("", "{", " \"source\": [", " \"aws.s3\"", " ],", " \"detail-type\": [", " \"Object created\"", " ],", " \"detail\": {", " \"bucket\": {", " \"name\": [", " \"testcdkstack-bucket43879c71-r2j3dsw4wp4z\"", " ]", " }", " }", "}"); Map eventMapperMap; try { eventMapperMap = MAPPER.readValue(eventPatternString, new TypeReference>(){}); } catch (JsonProcessingException e) { throw new RuntimeException(e); } // MODEL Set targets = new HashSet<>(); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId1") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId1") .build()); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId2") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId2") .build()); final ResourceModel model = ResourceModel.builder() .name(EVENT_RULE_NAME) .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(targets) .build(); Set previousTargets = new HashSet<>(); previousTargets.add(targets.iterator().next()); previousTargets.add(software.amazon.events.rule.Target.builder() .id("ToDeleteId") .arn("ToDeleteArn") .build()); final ResourceModel previousModel = ResourceModel.builder() .name("TestRule") .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(previousTargets) .build(); // MOCK /* describeRule putRule listTargetsByRule removeTargets listTargetsByRule putTargets listTargetsByRule */ final PutRuleResponse putRuleResponse = PutRuleResponse.builder() .ruleArn(EVENT_RULE_ARN_DEFAULT_BUS) .build(); final RemoveTargetsResponse removeTargetsResponse = RemoveTargetsResponse.builder() .build(); final PutTargetsResponse putTargetsResponse = PutTargetsResponse.builder() .build(); when(proxyClient.client().putRule(any(PutRuleRequest.class))) .thenReturn(putRuleResponse); when(proxyClient.client().removeTargets(any(RemoveTargetsRequest.class))) .thenReturn(removeTargetsResponse); when(proxyClient.client().putTargets(any(PutTargetsRequest.class))) .thenReturn(putTargetsResponse); // RUN final ResourceHandlerRequest request = ResourceHandlerRequest.builder() .awsAccountId(SOURCE_ACCOUNT_ID) .desiredResourceState(model) .previousResourceState(previousModel) .build(); CallbackContext context = new CallbackContext(); ProgressEvent response; response = handler.handleRequest(proxy, request, context, proxyClient, logger); assertThat(response.getStatus()).isEqualTo(OperationStatus.IN_PROGRESS); response = handler.handleRequest(proxy, request, context, proxyClient, logger); // ASSERT assertThat(response).isNotNull(); assertThat(response.getStatus()).isEqualTo(OperationStatus.SUCCESS); assertThat(response.getCallbackDelaySeconds()).isEqualTo(0); assertThat(response.getResourceModel()).isEqualTo(request.getDesiredResourceState()); assertThat(response.getResourceModels()).isNull(); assertThat(response.getMessage()).isNull(); assertThat(response.getErrorCode()).isNull(); } @Test public void handleRequest_NoTargets() { final UpdateHandler handler = new UpdateHandler(); String eventPatternString = String.join("", "{", " \"source\": [", " \"aws.s3\"", " ],", " \"detail-type\": [", " \"Object created\"", " ],", " \"detail\": {", " \"bucket\": {", " \"name\": [", " \"testcdkstack-bucket43879c71-r2j3dsw4wp4z\"", " ]", " }", " }", "}"); Map eventMapperMap; try { eventMapperMap = MAPPER.readValue(eventPatternString, new TypeReference>(){}); } catch (JsonProcessingException e) { throw new RuntimeException(e); } // MODEL final ResourceModel model = ResourceModel.builder() .name(EVENT_RULE_NAME) .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .build(); final ResourceModel previousModel = ResourceModel.builder() .name(EVENT_RULE_NAME) .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .build(); // MOCK /* describeRule putRule listTargetsByRule removeTargets putTargets */ final PutRuleResponse putRuleResponse = PutRuleResponse.builder() .ruleArn(EVENT_RULE_ARN_DEFAULT_BUS) .build(); when(proxyClient.client().putRule(any(PutRuleRequest.class))) .thenReturn(putRuleResponse); // RUN final ResourceHandlerRequest request = ResourceHandlerRequest.builder() .awsAccountId(SOURCE_ACCOUNT_ID) .desiredResourceState(model) .previousResourceState(previousModel) .build(); final ProgressEvent response = handler.handleRequest(proxy, request, new CallbackContext(), proxyClient, logger); // ASSERT assertThat(response).isNotNull(); assertThat(response.getStatus()).isEqualTo(OperationStatus.SUCCESS); assertThat(response.getCallbackDelaySeconds()).isEqualTo(0); assertThat(response.getResourceModel()).isEqualTo(request.getDesiredResourceState()); assertThat(response.getResourceModels()).isNull(); assertThat(response.getMessage()).isNull(); assertThat(response.getErrorCode()).isNull(); } @Test public void handleRequest_CreateTargetsFail() { final UpdateHandler handler = new UpdateHandler(); String eventPatternString = String.join("", "{", " \"source\": [", " \"aws.s3\"", " ],", " \"detail-type\": [", " \"Object created\"", " ],", " \"detail\": {", " \"bucket\": {", " \"name\": [", " \"testcdkstack-bucket43879c71-r2j3dsw4wp4z\"", " ]", " }", " }", "}"); Map eventMapperMap; try { eventMapperMap = MAPPER.readValue(eventPatternString, new TypeReference>(){}); } catch (JsonProcessingException e) { throw new RuntimeException(e); } // MODEL Set targets = new HashSet<>(); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId") .build()); final ResourceModel model = ResourceModel.builder() .name(EVENT_RULE_NAME) .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(targets) .build(); final ResourceModel previousModel = ResourceModel.builder() .name("TestRule") .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .build(); // MOCK /* describeRule putRule listTargetsByRule removeTargets putTargets */ final PutRuleResponse putRuleResponse = PutRuleResponse.builder() .ruleArn(EVENT_RULE_ARN_DEFAULT_BUS) .build(); final Collection PutTargetsResultEntries = new ArrayList<>(); for (software.amazon.events.rule.Target target : model.getTargets()) { PutTargetsResultEntries.add(PutTargetsResultEntry.builder() .targetId(target.getId()) .build()); } final PutTargetsResponse putTargetsResponse = PutTargetsResponse.builder() .failedEntries(PutTargetsResultEntries) .build(); when(proxyClient.client().putRule(any(PutRuleRequest.class))) .thenReturn(putRuleResponse); when(proxyClient.client().putTargets(any(PutTargetsRequest.class))) .thenReturn(putTargetsResponse); // RUN final ResourceHandlerRequest request = ResourceHandlerRequest.builder() .awsAccountId(SOURCE_ACCOUNT_ID) .desiredResourceState(model) .previousResourceState(previousModel) .build(); CallbackContext context = new CallbackContext(); ProgressEvent response; response = handler.handleRequest(proxy, request, context, proxyClient, logger); // ASSERT assertThat(response).isNotNull(); assertThat(response.getStatus()).isEqualTo(OperationStatus.FAILED); assertThat(response.getCallbackDelaySeconds()).isEqualTo(0); assertThat(response.getResourceModel()).isEqualTo(request.getDesiredResourceState()); assertThat(response.getResourceModels()).isNull(); assertThat(response.getMessage()).isEqualTo("Target(s) failed to create/update"); assertThat(response.getErrorCode()).isEqualTo(HandlerErrorCode.GeneralServiceException); } @Test public void handleRequest_RemoveTargetsFail() { final UpdateHandler handler = new UpdateHandler(); String eventPatternString = String.join("", "{", " \"source\": [", " \"aws.s3\"", " ],", " \"detail-type\": [", " \"Object created\"", " ],", " \"detail\": {", " \"bucket\": {", " \"name\": [", " \"testcdkstack-bucket43879c71-r2j3dsw4wp4z\"", " ]", " }", " }", "}"); Map eventMapperMap; try { eventMapperMap = MAPPER.readValue(eventPatternString, new TypeReference>(){}); } catch (JsonProcessingException e) { throw new RuntimeException(e); } // MODEL Set targets = new HashSet<>(); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId") .build()); final ResourceModel model = ResourceModel.builder() .name(EVENT_RULE_NAME) .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(targets) .build(); Set previousTargets = new HashSet<>(targets); String targetToRemoveId = "ToDeleteId"; previousTargets.add(software.amazon.events.rule.Target.builder() .id(targetToRemoveId) .arn("ToDeleteArn") .build()); final ResourceModel previousModel = ResourceModel.builder() .name("TestRule") .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(previousTargets) .build(); // MOCK /* describeRule putRule describeRule listTargetsByRule removeTargets putTargets */ final PutRuleResponse putRuleResponse = PutRuleResponse.builder() .ruleArn(EVENT_RULE_ARN_DEFAULT_BUS) .build(); final Collection removeTargetsResultEntries = new ArrayList<>(); removeTargetsResultEntries.add(RemoveTargetsResultEntry.builder() .targetId(targetToRemoveId) .build()); final RemoveTargetsResponse removeTargetsResponse = RemoveTargetsResponse.builder() .failedEntries(removeTargetsResultEntries) .build(); when(proxyClient.client().putRule(any(PutRuleRequest.class))) .thenReturn(putRuleResponse); when(proxyClient.client().removeTargets(any(RemoveTargetsRequest.class))) .thenReturn(removeTargetsResponse); // RUN final ResourceHandlerRequest request = ResourceHandlerRequest.builder() .awsAccountId(SOURCE_ACCOUNT_ID) .desiredResourceState(model) .previousResourceState(previousModel) .build(); CallbackContext context = new CallbackContext(); ProgressEvent response; response = handler.handleRequest(proxy, request, context, proxyClient, logger); // ASSERT assertThat(response).isNotNull(); assertThat(response.getStatus()).isEqualTo(OperationStatus.FAILED); assertThat(response.getCallbackDelaySeconds()).isEqualTo(0); assertThat(response.getResourceModel()).isEqualTo(request.getDesiredResourceState()); assertThat(response.getResourceModels()).isNull(); assertThat(response.getMessage()).isEqualTo("Target(s) failed to be removed"); assertThat(response.getErrorCode()).isEqualTo(HandlerErrorCode.GeneralServiceException); } @Test public void handleRequest_TranslatorTest() { final UpdateHandler handler = new UpdateHandler(); String eventPatternString = String.join("", "{", " \"source\": [", " \"aws.s3\"", " ],", " \"detail-type\": [", " \"Object created\"", " ],", " \"detail\": {", " \"bucket\": {", " \"name\": [", " \"testcdkstack-bucket43879c71-r2j3dsw4wp4z\"", " ]", " }", " }", "}"); Map eventMapperMap; try { eventMapperMap = MAPPER.readValue(eventPatternString, new TypeReference>(){}); } catch (JsonProcessingException e) { throw new RuntimeException(e); } // MODEL Set targets = new HashSet<>(); Set securityGroups = new HashSet<>(); Set subnets = new HashSet<>(); Map headerParameters = new HashMap<>(); Set pathParameterValues = new HashSet<>(); Map queryStringParameters = new HashMap<>(); Map inputPathsMap = new HashMap<>(); Set runCommandTargets = new HashSet<>(); Set runCommandTargetsValues = new HashSet<>(); Set sageMakerPipelineParameters = new HashSet<>(); Set tags = new HashSet<>(); Set placementStrategies = new HashSet<>(); Set placementConstraints = new HashSet<>(); Set capacityProviderStrategy = new HashSet<>(); securityGroups.add("SECURITY_GROUP"); subnets.add("SUBNET"); headerParameters.put("HEADER_PARAMETER_KEY", "HEADER_PARAMETER_VALUE"); pathParameterValues.add("PATH_PARAMETER_VALUE"); queryStringParameters.put("QUERY_STRING_PARAMETER_KEY", "QUERY_STRING_PARAMETER_VALUE"); inputPathsMap.put("INPUT_PATH_KEY", "INPUT_PATH_VALUE"); runCommandTargetsValues.add("RUN_COMMAND_TARGETS_VALUE"); runCommandTargets.add(software.amazon.events.rule.RunCommandTarget.builder() .key("RUN_COMMAND_TARGET_KEY") .values(runCommandTargetsValues) .build()); sageMakerPipelineParameters.add(SageMakerPipelineParameter.builder() .name("SAGEMAKER_PIPELINE_PARAMETER_NAME") .value("SAGEMAKER_PIPELINE_PARAMETER_VALUE") .build()); tags.add(software.amazon.events.rule.Tag.builder() .key("TAG_KEY") .value("TAG_VALUE") .build()); placementStrategies.add(software.amazon.events.rule.PlacementStrategy.builder() .field("PLACEMENT_STRATEGY_FIELD") .type("PLACEMENT_STRATEGY_TYPE") .build()); placementConstraints.add(software.amazon.events.rule.PlacementConstraint.builder() .expression("PLACEMENT_CONSTRAINT_EXPRESSION") .type("PLACEMENT_CONSTRAINT_TYPE") .build()); capacityProviderStrategy.add(software.amazon.events.rule.CapacityProviderStrategyItem.builder() .base(1) .capacityProvider("CAPACITY_PROVIDER_STRATEGY_CAPACITY") .weight(1) .build()); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId") .batchParameters(software.amazon.events.rule.BatchParameters.builder() .arrayProperties(software.amazon.events.rule.BatchArrayProperties.builder() .size(1) .build()) .retryStrategy(software.amazon.events.rule.BatchRetryStrategy.builder() .attempts(1) .build()) .build()) .deadLetterConfig(software.amazon.events.rule.DeadLetterConfig.builder() .arn("ARN") .build()) .ecsParameters(software.amazon.events.rule.EcsParameters.builder() .networkConfiguration(software.amazon.events.rule.NetworkConfiguration.builder() .awsVpcConfiguration(software.amazon.events.rule.AwsVpcConfiguration.builder() .assignPublicIp("UNKNOWN_TO_SDK_VERSION") .securityGroups(securityGroups) .subnets(subnets) .build()) .build()) .group("GROUP") .launchType("UNKNOWN_TO_SDK_VERSION") .platformVersion("PLATFORM_VERSION") .taskCount(1) .taskDefinitionArn("TASK_DEFINITION_ARN") .tagList(tags) .placementStrategies(placementStrategies) .placementConstraints(placementConstraints) .capacityProviderStrategy(capacityProviderStrategy) .build()) .httpParameters(software.amazon.events.rule.HttpParameters.builder() .headerParameters(headerParameters) .pathParameterValues(pathParameterValues) .queryStringParameters(queryStringParameters) .build()) .inputTransformer(software.amazon.events.rule.InputTransformer.builder() .inputPathsMap(inputPathsMap) .inputTemplate("INPUT_TEMPLATE") .build()) .kinesisParameters(software.amazon.events.rule.KinesisParameters.builder() .partitionKeyPath("PARTITION_KEY_PATH") .build()) .redshiftDataParameters(software.amazon.events.rule.RedshiftDataParameters.builder() .database("DATABASE") .dbUser("DB_USER") .secretManagerArn("SECRET_MANAGER_ARN") .sql("SQL") .statementName("STATEMENT_NAME") .withEvent(true) .build()) .retryPolicy(software.amazon.events.rule.RetryPolicy.builder() .maximumEventAgeInSeconds(1) .maximumRetryAttempts(1) .build()) .runCommandParameters(software.amazon.events.rule.RunCommandParameters.builder() .runCommandTargets(runCommandTargets) .build()) .sqsParameters(software.amazon.events.rule.SqsParameters.builder() .messageGroupId("MESSAGE_GROUP_ID") .build()) .sageMakerPipelineParameters(software.amazon.events.rule.SageMakerPipelineParameters.builder() .pipelineParameterList(sageMakerPipelineParameters) .build()) .build()); final ResourceModel model = ResourceModel.builder() .name("TestRule") .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(targets) .build(); Set previousTargets = new HashSet<>(targets); String targetToRemoveId = "ToDeleteId"; previousTargets.add(software.amazon.events.rule.Target.builder() .id(targetToRemoveId) .arn("ToDeleteArn") .build()); final ResourceModel previousModel = ResourceModel.builder() .name("TestRule") .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(previousTargets) .build(); // MOCK /* describeRule putRule listTargetsByRule removeTargets listTargetsByRule putTargets listTargetsByRule describeRule listTargetsByRule */ final PutRuleResponse putRuleResponse = PutRuleResponse.builder() .ruleArn(EVENT_RULE_ARN_DEFAULT_BUS) .build(); final RemoveTargetsResponse removeTargetsResponse = RemoveTargetsResponse.builder() .build(); final PutTargetsResponse putTargetsResponse = PutTargetsResponse.builder() .build(); when(proxyClient.client().putRule(any(PutRuleRequest.class))) .thenReturn(putRuleResponse); when(proxyClient.client().removeTargets(any(RemoveTargetsRequest.class))) .thenReturn(removeTargetsResponse); when(proxyClient.client().putTargets(any(PutTargetsRequest.class))) .thenReturn(putTargetsResponse); // RUN final ResourceHandlerRequest request = ResourceHandlerRequest.builder() .awsAccountId(SOURCE_ACCOUNT_ID) .desiredResourceState(model) .previousResourceState(previousModel) .build(); CallbackContext context = new CallbackContext(); ProgressEvent response; response = handler.handleRequest(proxy, request, context, proxyClient, logger); assertThat(response.getStatus()).isEqualTo(OperationStatus.IN_PROGRESS); response = handler.handleRequest(proxy, request, context, proxyClient, logger); // ASSERT assertThat(response).isNotNull(); assertThat(response.getStatus()).isEqualTo(OperationStatus.SUCCESS); assertThat(response.getCallbackDelaySeconds()).isEqualTo(0); assertThat(response.getResourceModel()).isEqualTo(request.getDesiredResourceState()); assertThat(response.getResourceModels()).isNull(); assertThat(response.getMessage()).isNull(); assertThat(response.getErrorCode()).isNull(); } @Test public void handleRequest_TranslatorTestNulls() { final UpdateHandler handler = new UpdateHandler(); String eventPatternString = String.join("", "{", " \"source\": [", " \"aws.s3\"", " ],", " \"detail-type\": [", " \"Object created\"", " ],", " \"detail\": {", " \"bucket\": {", " \"name\": [", " \"testcdkstack-bucket43879c71-r2j3dsw4wp4z\"", " ]", " }", " }", "}"); Map eventMapperMap; try { eventMapperMap = MAPPER.readValue(eventPatternString, new TypeReference>(){}); } catch (JsonProcessingException e) { throw new RuntimeException(e); } // MODEL Set targets = new HashSet<>(); Set securityGroups = new HashSet<>(); Set subnets = new HashSet<>(); Map headerParameters = new HashMap<>(); Set pathParameterValues = new HashSet<>(); Map queryStringParameters = new HashMap<>(); Map inputPathsMap = new HashMap<>(); Set runCommandTargets = new HashSet<>(); Set runCommandTargetsValues = new HashSet<>(); securityGroups.add("SECURITY_GROUP"); subnets.add("SUBNET"); headerParameters.put("HEADER_PARAMETER_KEY", "HEADER_PARAMETER_VALUE"); pathParameterValues.add("PATH_PARAMETER_VALUE"); queryStringParameters.put("QUERY_STRING_PARAMETER_KEY", "QUERY_STRING_PARAMETER_VALUE"); inputPathsMap.put("INPUT_PATH_KEY", "INPUT_PATH_VALUE"); runCommandTargetsValues.add("RUN_COMMAND_TARGETS_VALUE"); runCommandTargets.add(software.amazon.events.rule.RunCommandTarget.builder() .key("RUN_COMMAND_TARGET_KEY") .values(runCommandTargetsValues) .build()); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId") .batchParameters(software.amazon.events.rule.BatchParameters.builder() .build()) .deadLetterConfig(software.amazon.events.rule.DeadLetterConfig.builder() .arn("ARN") .build()) .ecsParameters(software.amazon.events.rule.EcsParameters.builder() .group("GROUP") .launchType("EC2") .platformVersion("PLATFORM_VERSION") .taskCount(1) .taskDefinitionArn("TASK_DEFINITION_ARN") .build()) .httpParameters(software.amazon.events.rule.HttpParameters.builder() .headerParameters(headerParameters) .pathParameterValues(pathParameterValues) .queryStringParameters(queryStringParameters) .build()) .inputTransformer(software.amazon.events.rule.InputTransformer.builder() .inputPathsMap(inputPathsMap) .inputTemplate("INPUT_TEMPLATE") .build()) .kinesisParameters(software.amazon.events.rule.KinesisParameters.builder() .partitionKeyPath("PARTITION_KEY_PATH") .build()) .redshiftDataParameters(software.amazon.events.rule.RedshiftDataParameters.builder() .database("DATABASE") .dbUser("DB_USER") .secretManagerArn("SECRET_MANAGER_ARN") .sql("SQL") .statementName("STATEMENT_NAME") .withEvent(true) .build()) .retryPolicy(software.amazon.events.rule.RetryPolicy.builder() .maximumEventAgeInSeconds(1) .maximumRetryAttempts(1) .build()) .sqsParameters(software.amazon.events.rule.SqsParameters.builder() .messageGroupId("MESSAGE_GROUP_ID") .build()) .build()); final ResourceModel model = ResourceModel.builder() .name(EVENT_RULE_NAME) .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(targets) .build(); Set previousTargets = new HashSet<>(targets); String targetToRemoveId = "ToDeleteId"; previousTargets.add(software.amazon.events.rule.Target.builder() .id(targetToRemoveId) .arn("ToDeleteArn") .build()); final ResourceModel previousModel = ResourceModel.builder() .name("TestRule") .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(previousTargets) .build(); // MOCK /* describeRule putRule listTargetsByRule removeTargets putTargets describeRule listTargetsByRule */ final PutRuleResponse putRuleResponse = PutRuleResponse.builder() .ruleArn(EVENT_RULE_ARN_DEFAULT_BUS) .build(); final RemoveTargetsResponse removeTargetsResponse = RemoveTargetsResponse.builder() .build(); final PutTargetsResponse putTargetsResponse = PutTargetsResponse.builder() .build(); when(proxyClient.client().putRule(any(PutRuleRequest.class))) .thenReturn(putRuleResponse); when(proxyClient.client().removeTargets(any(RemoveTargetsRequest.class))) .thenReturn(removeTargetsResponse); when(proxyClient.client().putTargets(any(PutTargetsRequest.class))) .thenReturn(putTargetsResponse); // RUN final ResourceHandlerRequest request = ResourceHandlerRequest.builder() .awsAccountId(SOURCE_ACCOUNT_ID) .desiredResourceState(model) .previousResourceState(previousModel) .build(); CallbackContext context = new CallbackContext(); ProgressEvent response; response = handler.handleRequest(proxy, request, context, proxyClient, logger); assertThat(response.getStatus()).isEqualTo(OperationStatus.IN_PROGRESS); response = handler.handleRequest(proxy, request, context, proxyClient, logger); // ASSERT assertThat(response).isNotNull(); assertThat(response.getStatus()).isEqualTo(OperationStatus.SUCCESS); assertThat(response.getCallbackDelaySeconds()).isEqualTo(0); assertThat(response.getResourceModel()).isEqualTo(request.getDesiredResourceState()); assertThat(response.getResourceModels()).isNull(); assertThat(response.getMessage()).isNull(); assertThat(response.getErrorCode()).isNull(); } @Test public void handleRequest_ErrorTest() { final UpdateHandler handler = new UpdateHandler(); String eventPatternString = String.join("", "{", " \"source\": [", " \"aws.s3\"", " ],", " \"detail-type\": [", " \"Object created\"", " ],", " \"detail\": {", " \"bucket\": {", " \"name\": [", " \"testcdkstack-bucket43879c71-r2j3dsw4wp4z\"", " ]", " }", " }", "}"); Map eventMapperMap; try { eventMapperMap = MAPPER.readValue(eventPatternString, new TypeReference>(){}); } catch (JsonProcessingException e) { throw new RuntimeException(e); } // MODEL Set targets = new HashSet<>(); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId1") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId1") .build()); targets.add(software.amazon.events.rule.Target.builder() .id("TestLambdaFunctionId2") .arn("arn:aws:lambda:us-west-2:123456789123:function:TestLambdaFunctionId2") .build()); final ResourceModel model = ResourceModel.builder() .name(EVENT_RULE_NAME) .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(targets) .build(); Set previousTargets = new HashSet<>(targets); final ResourceModel previousModel = ResourceModel.builder() .name("TestRule") .description("TestDescription") .eventPattern(eventMapperMap) .state("ENABLED") .targets(previousTargets) .build(); // MOCK /* describeRule putRule listTargetsByRule removeTargets listTargetsByRule putTargets listTargetsByRule */ when(proxyClient.client().putRule(any(PutRuleRequest.class))) .thenThrow(InvalidEventPatternException.class); // RUN final ResourceHandlerRequest request = ResourceHandlerRequest.builder() .awsAccountId(SOURCE_ACCOUNT_ID) .desiredResourceState(model) .previousResourceState(previousModel) .build(); CallbackContext context = new CallbackContext(); ProgressEvent response; response = handler.handleRequest(proxy, request, context, proxyClient, logger); // ASSERT assertThat(response).isNotNull(); assertThat(response.getStatus()).isEqualTo(OperationStatus.FAILED); } }