675 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			675 lines
		
	
	
		
			25 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| import boto3
 | |
| import pytest
 | |
| import sure  # noqa # pylint: disable=unused-import
 | |
| from moto import mock_applicationautoscaling, mock_ecs
 | |
| from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID
 | |
| 
 | |
| DEFAULT_REGION = "us-east-1"
 | |
| DEFAULT_ECS_CLUSTER = "default"
 | |
| DEFAULT_ECS_TASK = "test_ecs_task"
 | |
| DEFAULT_ECS_SERVICE = "sample-webapp"
 | |
| DEFAULT_SERVICE_NAMESPACE = "ecs"
 | |
| DEFAULT_RESOURCE_ID = "service/{}/{}".format(DEFAULT_ECS_CLUSTER, DEFAULT_ECS_SERVICE)
 | |
| DEFAULT_SCALABLE_DIMENSION = "ecs:service:DesiredCount"
 | |
| DEFAULT_MIN_CAPACITY = 1
 | |
| DEFAULT_MAX_CAPACITY = 1
 | |
| DEFAULT_ROLE_ARN = "test:arn"
 | |
| DEFAULT_SUSPENDED_STATE = {
 | |
|     "DynamicScalingInSuspended": True,
 | |
|     "DynamicScalingOutSuspended": True,
 | |
|     "ScheduledScalingSuspended": True,
 | |
| }
 | |
| 
 | |
| 
 | |
| def _create_ecs_defaults(ecs, create_service=True):
 | |
|     _ = ecs.create_cluster(clusterName=DEFAULT_ECS_CLUSTER)
 | |
|     _ = ecs.register_task_definition(
 | |
|         family=DEFAULT_ECS_TASK,
 | |
|         containerDefinitions=[
 | |
|             {
 | |
|                 "name": "hello_world",
 | |
|                 "image": "docker/hello-world:latest",
 | |
|                 "cpu": 1024,
 | |
|                 "memory": 400,
 | |
|                 "essential": True,
 | |
|                 "environment": [
 | |
|                     {"name": "AWS_ACCESS_KEY_ID", "value": "SOME_ACCESS_KEY"}
 | |
|                 ],
 | |
|                 "logConfiguration": {"logDriver": "json-file"},
 | |
|             }
 | |
|         ],
 | |
|     )
 | |
|     if create_service:
 | |
|         _ = ecs.create_service(
 | |
|             cluster=DEFAULT_ECS_CLUSTER,
 | |
|             serviceName=DEFAULT_ECS_SERVICE,
 | |
|             taskDefinition=DEFAULT_ECS_TASK,
 | |
|             desiredCount=2,
 | |
|         )
 | |
| 
 | |
| 
 | |
| @mock_ecs
 | |
| @mock_applicationautoscaling
 | |
| def test_describe_scalable_targets_one_basic_ecs_success():
 | |
|     ecs = boto3.client("ecs", region_name=DEFAULT_REGION)
 | |
|     _create_ecs_defaults(ecs)
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     client.register_scalable_target(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE,
 | |
|         ResourceId=DEFAULT_RESOURCE_ID,
 | |
|         ScalableDimension=DEFAULT_SCALABLE_DIMENSION,
 | |
|     )
 | |
|     response = client.describe_scalable_targets(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     len(response["ScalableTargets"]).should.equal(1)
 | |
|     t = response["ScalableTargets"][0]
 | |
|     t.should.have.key("ServiceNamespace").which.should.equal(DEFAULT_SERVICE_NAMESPACE)
 | |
|     t.should.have.key("ResourceId").which.should.equal(DEFAULT_RESOURCE_ID)
 | |
|     t.should.have.key("ScalableDimension").which.should.equal(
 | |
|         DEFAULT_SCALABLE_DIMENSION
 | |
|     )
 | |
|     t.should.have.key("CreationTime").which.should.be.a("datetime.datetime")
 | |
| 
 | |
| 
 | |
| @mock_ecs
 | |
| @mock_applicationautoscaling
 | |
| def test_describe_scalable_targets_one_full_ecs_success():
 | |
|     ecs = boto3.client("ecs", region_name=DEFAULT_REGION)
 | |
|     _create_ecs_defaults(ecs)
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     register_scalable_target(client)
 | |
|     response = client.describe_scalable_targets(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     len(response["ScalableTargets"]).should.equal(1)
 | |
|     t = response["ScalableTargets"][0]
 | |
|     t.should.have.key("ServiceNamespace").which.should.equal(DEFAULT_SERVICE_NAMESPACE)
 | |
|     t.should.have.key("ResourceId").which.should.equal(DEFAULT_RESOURCE_ID)
 | |
|     t.should.have.key("ScalableDimension").which.should.equal(
 | |
|         DEFAULT_SCALABLE_DIMENSION
 | |
|     )
 | |
|     t.should.have.key("MinCapacity").which.should.equal(DEFAULT_MIN_CAPACITY)
 | |
|     t.should.have.key("MaxCapacity").which.should.equal(DEFAULT_MAX_CAPACITY)
 | |
|     t.should.have.key("RoleARN").which.should.equal(DEFAULT_ROLE_ARN)
 | |
|     t.should.have.key("CreationTime").which.should.be.a("datetime.datetime")
 | |
|     t.should.have.key("SuspendedState")
 | |
|     t["SuspendedState"]["DynamicScalingInSuspended"].should.equal(
 | |
|         DEFAULT_SUSPENDED_STATE["DynamicScalingInSuspended"]
 | |
|     )
 | |
| 
 | |
| 
 | |
| @mock_ecs
 | |
| @mock_applicationautoscaling
 | |
| def test_describe_scalable_targets_only_return_ecs_targets():
 | |
|     ecs = boto3.client("ecs", region_name=DEFAULT_REGION)
 | |
|     _create_ecs_defaults(ecs, create_service=False)
 | |
|     _ = ecs.create_service(
 | |
|         cluster=DEFAULT_ECS_CLUSTER,
 | |
|         serviceName="test1",
 | |
|         taskDefinition=DEFAULT_ECS_TASK,
 | |
|         desiredCount=2,
 | |
|     )
 | |
|     _ = ecs.create_service(
 | |
|         cluster=DEFAULT_ECS_CLUSTER,
 | |
|         serviceName="test2",
 | |
|         taskDefinition=DEFAULT_ECS_TASK,
 | |
|         desiredCount=2,
 | |
|     )
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     register_scalable_target(
 | |
|         client,
 | |
|         ServiceNamespace="ecs",
 | |
|         ResourceId="service/{}/test1".format(DEFAULT_ECS_CLUSTER),
 | |
|     )
 | |
|     register_scalable_target(
 | |
|         client,
 | |
|         ServiceNamespace="ecs",
 | |
|         ResourceId="service/{}/test2".format(DEFAULT_ECS_CLUSTER),
 | |
|     )
 | |
|     register_scalable_target(
 | |
|         client,
 | |
|         ServiceNamespace="elasticmapreduce",
 | |
|         ResourceId="instancegroup/j-2EEZNYKUA1NTV/ig-1791Y4E1L8YI0",
 | |
|         ScalableDimension="elasticmapreduce:instancegroup:InstanceCount",
 | |
|     )
 | |
|     response = client.describe_scalable_targets(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     len(response["ScalableTargets"]).should.equal(2)
 | |
| 
 | |
| 
 | |
| @mock_ecs
 | |
| @mock_applicationautoscaling
 | |
| def test_describe_scalable_targets_next_token_success():
 | |
|     ecs = boto3.client("ecs", region_name=DEFAULT_REGION)
 | |
|     _create_ecs_defaults(ecs, create_service=False)
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     for i in range(0, 100):
 | |
|         _ = ecs.create_service(
 | |
|             cluster=DEFAULT_ECS_CLUSTER,
 | |
|             serviceName=str(i),
 | |
|             taskDefinition=DEFAULT_ECS_TASK,
 | |
|             desiredCount=2,
 | |
|         )
 | |
|         register_scalable_target(
 | |
|             client,
 | |
|             ServiceNamespace="ecs",
 | |
|             ResourceId="service/{}/{}".format(DEFAULT_ECS_CLUSTER, i),
 | |
|         )
 | |
|     response = client.describe_scalable_targets(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     len(response["ScalableTargets"]).should.equal(50)
 | |
|     response["ScalableTargets"][0]["ResourceId"].should.equal("service/default/0")
 | |
|     response.should.have.key("NextToken").which.should.equal("49")
 | |
|     response = client.describe_scalable_targets(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE, NextToken=str(response["NextToken"])
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     len(response["ScalableTargets"]).should.equal(50)
 | |
|     response["ScalableTargets"][0]["ResourceId"].should.equal("service/default/50")
 | |
|     response.should_not.have.key("NextToken")
 | |
| 
 | |
| 
 | |
| def register_scalable_target(client, **kwargs):
 | |
|     """Build a default scalable target object for use in tests."""
 | |
|     return client.register_scalable_target(
 | |
|         ServiceNamespace=kwargs.get("ServiceNamespace", DEFAULT_SERVICE_NAMESPACE),
 | |
|         ResourceId=kwargs.get("ResourceId", DEFAULT_RESOURCE_ID),
 | |
|         ScalableDimension=kwargs.get("ScalableDimension", DEFAULT_SCALABLE_DIMENSION),
 | |
|         MinCapacity=kwargs.get("MinCapacity", DEFAULT_MIN_CAPACITY),
 | |
|         MaxCapacity=kwargs.get("MaxCapacity", DEFAULT_MAX_CAPACITY),
 | |
|         RoleARN=kwargs.get("RoleARN", DEFAULT_ROLE_ARN),
 | |
|         SuspendedState=kwargs.get("SuspendedState", DEFAULT_SUSPENDED_STATE),
 | |
|     )
 | |
| 
 | |
| 
 | |
| @mock_ecs
 | |
| @mock_applicationautoscaling
 | |
| def test_register_scalable_target_resource_id_variations():
 | |
| 
 | |
|     # Required to register an ECS target in moto
 | |
|     ecs = boto3.client("ecs", region_name=DEFAULT_REGION)
 | |
|     _create_ecs_defaults(ecs)
 | |
| 
 | |
|     # See https://docs.aws.amazon.com/AWSCloudFormation/latest/UserGuide/aws-resource-applicationautoscaling-scalabletarget.html
 | |
|     resource_id_variations = [
 | |
|         (
 | |
|             DEFAULT_SERVICE_NAMESPACE,
 | |
|             DEFAULT_RESOURCE_ID,
 | |
|             DEFAULT_SCALABLE_DIMENSION,
 | |
|         ),  # ECS
 | |
|         (
 | |
|             "ec2",
 | |
|             "spot-fleet-request/sfr-73fbd2ce-aa30-494c-8788-1cee4EXAMPLE",
 | |
|             "ec2:spot-fleet-request:TargetCapacity",
 | |
|         ),
 | |
|         (
 | |
|             "elasticmapreduce",
 | |
|             "instancegroup/j-2EEZNYKUA1NTV/ig-1791Y4E1L8YI0",
 | |
|             "elasticmapreduce:instancegroup:InstanceCount",
 | |
|         ),
 | |
|         ("appstream", "fleet/sample-fleet", "appstream:fleet:DesiredCapacity"),
 | |
|         ("dynamodb", "table/my-table", "dynamodb:table:ReadCapacityUnits"),
 | |
|         (
 | |
|             "dynamodb",
 | |
|             "table/my-table/index/my-table-index",
 | |
|             "dynamodb:index:ReadCapacityUnits",
 | |
|         ),
 | |
|         ("rds", "cluster:my-db-cluster", "rds:cluster:ReadReplicaCount"),
 | |
|         (
 | |
|             "sagemaker",
 | |
|             "endpoint/MyEndPoint/variant/MyVariant",
 | |
|             "sagemaker:variant:DesiredInstanceCount",
 | |
|         ),
 | |
|         (
 | |
|             "comprehend",
 | |
|             "arn:aws:comprehend:us-west-2:123456789012:document-classifier-endpoint/EXAMPLE",
 | |
|             "comprehend:document-classifier-endpoint:DesiredInferenceUnits",
 | |
|         ),
 | |
|         (
 | |
|             "lambda",
 | |
|             "function:my-function:prod",
 | |
|             "lambda:function:ProvisionedConcurrency",
 | |
|         ),
 | |
|         (
 | |
|             "cassandra",
 | |
|             "keyspace/mykeyspace/table/mytable",
 | |
|             "cassandra:table:ReadCapacityUnits",
 | |
|         ),
 | |
|         (
 | |
|             "custom-resource",
 | |
|             "https://test-endpoint.amazon.com/ScalableDimension/test-resource",
 | |
|             "custom-resource:ResourceType:Property",
 | |
|         ),
 | |
|     ]
 | |
| 
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     for namespace, resource_id, scalable_dimension in resource_id_variations:
 | |
|         client.register_scalable_target(
 | |
|             ServiceNamespace=namespace,
 | |
|             ResourceId=resource_id,
 | |
|             ScalableDimension=scalable_dimension,
 | |
|             MinCapacity=1,
 | |
|             MaxCapacity=8,
 | |
|         )
 | |
|         response = client.describe_scalable_targets(ServiceNamespace=namespace)
 | |
|         response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|         num_targets = 2 if namespace == "dynamodb" and "index" in resource_id else 1
 | |
|         len(response["ScalableTargets"]).should.equal(num_targets)
 | |
|         t = response["ScalableTargets"][-1]
 | |
|         t.should.have.key("ServiceNamespace").which.should.equal(namespace)
 | |
|         t.should.have.key("ResourceId").which.should.equal(resource_id)
 | |
|         t.should.have.key("ScalableDimension").which.should.equal(scalable_dimension)
 | |
|         t.should.have.key("CreationTime").which.should.be.a("datetime.datetime")
 | |
| 
 | |
| 
 | |
| @mock_ecs
 | |
| @mock_applicationautoscaling
 | |
| def test_register_scalable_target_updates_existing_target():
 | |
|     ecs = boto3.client("ecs", region_name=DEFAULT_REGION)
 | |
|     _create_ecs_defaults(ecs)
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     register_scalable_target(client)
 | |
| 
 | |
|     updated_min_capacity = 3
 | |
|     updated_max_capacity = 10
 | |
|     updated_suspended_state = {
 | |
|         "DynamicScalingInSuspended": False,
 | |
|         "DynamicScalingOutSuspended": False,
 | |
|         "ScheduledScalingSuspended": False,
 | |
|     }
 | |
| 
 | |
|     client.register_scalable_target(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE,
 | |
|         ResourceId=DEFAULT_RESOURCE_ID,
 | |
|         ScalableDimension=DEFAULT_SCALABLE_DIMENSION,
 | |
|         MinCapacity=updated_min_capacity,
 | |
|         MaxCapacity=updated_max_capacity,
 | |
|         SuspendedState=updated_suspended_state,
 | |
|     )
 | |
|     response = client.describe_scalable_targets(
 | |
|         ServiceNamespace=DEFAULT_SERVICE_NAMESPACE
 | |
|     )
 | |
| 
 | |
|     len(response["ScalableTargets"]).should.equal(1)
 | |
|     t = response["ScalableTargets"][0]
 | |
|     t.should.have.key("MinCapacity").which.should.equal(updated_min_capacity)
 | |
|     t.should.have.key("MaxCapacity").which.should.equal(updated_max_capacity)
 | |
|     t.should.have.key("SuspendedState")
 | |
|     t["SuspendedState"]["DynamicScalingInSuspended"].should.equal(
 | |
|         updated_suspended_state["DynamicScalingInSuspended"]
 | |
|     )
 | |
|     t["SuspendedState"]["DynamicScalingOutSuspended"].should.equal(
 | |
|         updated_suspended_state["DynamicScalingOutSuspended"]
 | |
|     )
 | |
|     t["SuspendedState"]["ScheduledScalingSuspended"].should.equal(
 | |
|         updated_suspended_state["ScheduledScalingSuspended"]
 | |
|     )
 | |
| 
 | |
| 
 | |
| @pytest.mark.parametrize(
 | |
|     ["policy_type", "policy_body_kwargs"],
 | |
|     [
 | |
|         [
 | |
|             "TargetTrackingScaling",
 | |
|             {
 | |
|                 "TargetTrackingScalingPolicyConfiguration": {
 | |
|                     "TargetValue": 70.0,
 | |
|                     "PredefinedMetricSpecification": {
 | |
|                         "PredefinedMetricType": "SageMakerVariantInvocationsPerInstance"
 | |
|                     },
 | |
|                 }
 | |
|             },
 | |
|         ],
 | |
|         [
 | |
|             "TargetTrackingScaling",
 | |
|             {
 | |
|                 "StepScalingPolicyConfiguration": {
 | |
|                     "AdjustmentType": "ChangeCapacity",
 | |
|                     "StepAdjustments": [{"ScalingAdjustment": 10}],
 | |
|                     "MinAdjustmentMagnitude": 2,
 | |
|                 },
 | |
|             },
 | |
|         ],
 | |
|     ],
 | |
| )
 | |
| @mock_applicationautoscaling
 | |
| def test_put_scaling_policy(policy_type, policy_body_kwargs):
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     namespace = "sagemaker"
 | |
|     resource_id = "endpoint/MyEndPoint/variant/MyVariant"
 | |
|     scalable_dimension = "sagemaker:variant:DesiredInstanceCount"
 | |
| 
 | |
|     client.register_scalable_target(
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|         MinCapacity=1,
 | |
|         MaxCapacity=8,
 | |
|     )
 | |
| 
 | |
|     policy_name = "MyPolicy"
 | |
| 
 | |
|     with pytest.raises(client.exceptions.ValidationException) as e:
 | |
|         client.put_scaling_policy(
 | |
|             PolicyName=policy_name,
 | |
|             ServiceNamespace=namespace,
 | |
|             ResourceId=resource_id,
 | |
|             ScalableDimension=scalable_dimension,
 | |
|             PolicyType="ABCDEFG",
 | |
|             **policy_body_kwargs,
 | |
|         )
 | |
|     e.value.response["Error"]["Message"].should.match(
 | |
|         r"Unknown policy type .* specified."
 | |
|     )
 | |
| 
 | |
|     response = client.put_scaling_policy(
 | |
|         PolicyName=policy_name,
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|         PolicyType=policy_type,
 | |
|         **policy_body_kwargs,
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     response["PolicyARN"].should.match(
 | |
|         r"arn:aws:autoscaling:.*1:scalingPolicy:.*:resource/{}/{}:policyName/{}".format(
 | |
|             namespace, resource_id, policy_name
 | |
|         )
 | |
|     )
 | |
| 
 | |
| 
 | |
| @mock_applicationautoscaling
 | |
| def test_describe_scaling_policies():
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     namespace = "sagemaker"
 | |
|     resource_id = "endpoint/MyEndPoint/variant/MyVariant"
 | |
|     scalable_dimension = "sagemaker:variant:DesiredInstanceCount"
 | |
| 
 | |
|     client.register_scalable_target(
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|         MinCapacity=1,
 | |
|         MaxCapacity=8,
 | |
|     )
 | |
| 
 | |
|     policy_name = "MyPolicy"
 | |
|     policy_type = "TargetTrackingScaling"
 | |
|     policy_body = {
 | |
|         "TargetValue": 70.0,
 | |
|         "PredefinedMetricSpecification": {
 | |
|             "PredefinedMetricType": "SageMakerVariantInvocationsPerInstance"
 | |
|         },
 | |
|     }
 | |
| 
 | |
|     response = client.put_scaling_policy(
 | |
|         PolicyName=policy_name,
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|         PolicyType=policy_type,
 | |
|         TargetTrackingScalingPolicyConfiguration=policy_body,
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
| 
 | |
|     response = client.describe_scaling_policies(
 | |
|         PolicyNames=[policy_name],
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     policy = response["ScalingPolicies"][0]
 | |
|     policy["PolicyName"].should.equal(policy_name)
 | |
|     policy["ServiceNamespace"].should.equal(namespace)
 | |
|     policy["ResourceId"].should.equal(resource_id)
 | |
|     policy["ScalableDimension"].should.equal(scalable_dimension)
 | |
|     policy["PolicyType"].should.equal(policy_type)
 | |
|     policy["TargetTrackingScalingPolicyConfiguration"].should.equal(policy_body)
 | |
|     policy["PolicyARN"].should.match(
 | |
|         r"arn:aws:autoscaling:.*1:scalingPolicy:.*:resource/{}/{}:policyName/{}".format(
 | |
|             namespace, resource_id, policy_name
 | |
|         )
 | |
|     )
 | |
|     policy.should.have.key("CreationTime").which.should.be.a("datetime.datetime")
 | |
| 
 | |
| 
 | |
| @mock_applicationautoscaling
 | |
| def test_delete_scaling_policies():
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     namespace = "sagemaker"
 | |
|     resource_id = "endpoint/MyEndPoint/variant/MyVariant"
 | |
|     scalable_dimension = "sagemaker:variant:DesiredInstanceCount"
 | |
| 
 | |
|     client.register_scalable_target(
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|         MinCapacity=1,
 | |
|         MaxCapacity=8,
 | |
|     )
 | |
| 
 | |
|     policy_name = "MyPolicy"
 | |
|     policy_type = "TargetTrackingScaling"
 | |
|     policy_body = {
 | |
|         "TargetValue": 70.0,
 | |
|         "PredefinedMetricSpecification": {
 | |
|             "PredefinedMetricType": "SageMakerVariantInvocationsPerInstance"
 | |
|         },
 | |
|     }
 | |
| 
 | |
|     with pytest.raises(client.exceptions.ValidationException) as e:
 | |
|         client.delete_scaling_policy(
 | |
|             PolicyName=policy_name,
 | |
|             ServiceNamespace=namespace,
 | |
|             ResourceId=resource_id,
 | |
|             ScalableDimension=scalable_dimension,
 | |
|         )
 | |
|     e.value.response["Error"]["Message"].should.match(r"No scaling policy found .*")
 | |
| 
 | |
|     response = client.put_scaling_policy(
 | |
|         PolicyName=policy_name,
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|         PolicyType=policy_type,
 | |
|         TargetTrackingScalingPolicyConfiguration=policy_body,
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
| 
 | |
|     response = client.delete_scaling_policy(
 | |
|         PolicyName=policy_name,
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
| 
 | |
|     response = client.describe_scaling_policies(
 | |
|         PolicyNames=[policy_name],
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|     )
 | |
|     response["ResponseMetadata"]["HTTPStatusCode"].should.equal(200)
 | |
|     len(response["ScalingPolicies"]).should.equal(0)
 | |
| 
 | |
| 
 | |
| @mock_applicationautoscaling
 | |
| def test_deregister_scalable_target():
 | |
|     client = boto3.client("application-autoscaling", region_name=DEFAULT_REGION)
 | |
|     namespace = "sagemaker"
 | |
|     resource_id = "endpoint/MyEndPoint/variant/MyVariant"
 | |
|     scalable_dimension = "sagemaker:variant:DesiredInstanceCount"
 | |
| 
 | |
|     client.register_scalable_target(
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|         MinCapacity=1,
 | |
|         MaxCapacity=8,
 | |
|     )
 | |
| 
 | |
|     response = client.describe_scalable_targets(ServiceNamespace=namespace)
 | |
|     len(response["ScalableTargets"]).should.equal(1)
 | |
| 
 | |
|     client.deregister_scalable_target(
 | |
|         ServiceNamespace=namespace,
 | |
|         ResourceId=resource_id,
 | |
|         ScalableDimension=scalable_dimension,
 | |
|     )
 | |
| 
 | |
|     response = client.describe_scalable_targets(ServiceNamespace=namespace)
 | |
|     len(response["ScalableTargets"]).should.equal(0)
 | |
| 
 | |
|     with pytest.raises(client.exceptions.ValidationException) as e:
 | |
|         client.deregister_scalable_target(
 | |
|             ServiceNamespace=namespace,
 | |
|             ResourceId=resource_id,
 | |
|             ScalableDimension=scalable_dimension,
 | |
|         )
 | |
|     e.value.response["Error"]["Message"].should.match(r"No scalable target found .*")
 | |
| 
 | |
| 
 | |
| @mock_applicationautoscaling
 | |
| def test_delete_scheduled_action():
 | |
|     client = boto3.client("application-autoscaling", region_name="eu-west-1")
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ecs")
 | |
|     resp.should.have.key("ScheduledActions").length_of(0)
 | |
| 
 | |
|     for i in range(3):
 | |
|         client.put_scheduled_action(
 | |
|             ServiceNamespace="ecs",
 | |
|             ScheduledActionName=f"ecs_action_{i}",
 | |
|             ResourceId=f"ecs:cluster:{i}",
 | |
|             ScalableDimension="ecs:service:DesiredCount",
 | |
|         )
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ecs")
 | |
|     resp.should.have.key("ScheduledActions").length_of(3)
 | |
| 
 | |
|     client.delete_scheduled_action(
 | |
|         ServiceNamespace="ecs",
 | |
|         ScheduledActionName="ecs_action_0",
 | |
|         ResourceId="ecs:cluster:0",
 | |
|         ScalableDimension="ecs:service:DesiredCount",
 | |
|     )
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ecs")
 | |
|     resp.should.have.key("ScheduledActions").length_of(2)
 | |
| 
 | |
| 
 | |
| @mock_applicationautoscaling
 | |
| def test_describe_scheduled_actions():
 | |
|     client = boto3.client("application-autoscaling", region_name="eu-west-1")
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ecs")
 | |
|     resp.should.have.key("ScheduledActions").length_of(0)
 | |
| 
 | |
|     for i in range(3):
 | |
|         client.put_scheduled_action(
 | |
|             ServiceNamespace="ecs",
 | |
|             ScheduledActionName=f"ecs_action_{i}",
 | |
|             ResourceId=f"ecs:cluster:{i}",
 | |
|             ScalableDimension="ecs:service:DesiredCount",
 | |
|         )
 | |
|         client.put_scheduled_action(
 | |
|             ServiceNamespace="dynamodb",
 | |
|             ScheduledActionName=f"ddb_action_{i}",
 | |
|             ResourceId=f"table/table_{i}",
 | |
|             ScalableDimension="dynamodb:table:ReadCapacityUnits",
 | |
|         )
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ecs")
 | |
|     resp.should.have.key("ScheduledActions").length_of(3)
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ec2")
 | |
|     resp.should.have.key("ScheduledActions").length_of(0)
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(
 | |
|         ServiceNamespace="dynamodb", ScheduledActionNames=["ddb_action_0"]
 | |
|     )
 | |
|     resp.should.have.key("ScheduledActions").length_of(1)
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(
 | |
|         ServiceNamespace="dynamodb",
 | |
|         ResourceId="table/table_0",
 | |
|         ScalableDimension="dynamodb:table:ReadCapacityUnits",
 | |
|     )
 | |
|     resp.should.have.key("ScheduledActions").length_of(1)
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(
 | |
|         ServiceNamespace="dynamodb",
 | |
|         ResourceId="table/table_0",
 | |
|         ScalableDimension="dynamodb:table:WriteCapacityUnits",
 | |
|     )
 | |
|     resp.should.have.key("ScheduledActions").length_of(0)
 | |
| 
 | |
| 
 | |
| @mock_applicationautoscaling
 | |
| def test_put_scheduled_action():
 | |
|     client = boto3.client("application-autoscaling", region_name="ap-southeast-1")
 | |
|     client.put_scheduled_action(
 | |
|         ServiceNamespace="ecs",
 | |
|         ScheduledActionName="action_name",
 | |
|         ResourceId="ecs:cluster:x",
 | |
|         ScalableDimension="ecs:service:DesiredCount",
 | |
|     )
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ecs")
 | |
|     resp.should.have.key("ScheduledActions").length_of(1)
 | |
| 
 | |
|     action = resp["ScheduledActions"][0]
 | |
|     action.should.have.key("ScheduledActionName").equals("action_name")
 | |
|     action.should.have.key("ScheduledActionARN").equals(
 | |
|         f"arn:aws:autoscaling:ap-southeast-1:{ACCOUNT_ID}:scheduledAction:ecs:scheduledActionName/action_name"
 | |
|     )
 | |
|     action.should.have.key("ServiceNamespace").equals("ecs")
 | |
|     action.should.have.key("ResourceId").equals("ecs:cluster:x")
 | |
|     action.should.have.key("ScalableDimension").equals("ecs:service:DesiredCount")
 | |
|     action.should.have.key("CreationTime")
 | |
|     action.shouldnt.have.key("ScalableTargetAction")
 | |
| 
 | |
| 
 | |
| @mock_applicationautoscaling
 | |
| def test_put_scheduled_action__use_update():
 | |
|     client = boto3.client("application-autoscaling", region_name="ap-southeast-1")
 | |
|     client.put_scheduled_action(
 | |
|         ServiceNamespace="ecs",
 | |
|         ScheduledActionName="action_name",
 | |
|         ResourceId="ecs:cluster:x",
 | |
|         ScalableDimension="ecs:service:DesiredCount",
 | |
|     )
 | |
|     client.put_scheduled_action(
 | |
|         ServiceNamespace="ecs",
 | |
|         ScheduledActionName="action_name_updated",
 | |
|         ResourceId="ecs:cluster:x",
 | |
|         ScalableDimension="ecs:service:DesiredCount",
 | |
|         ScalableTargetAction={
 | |
|             "MinCapacity": 12,
 | |
|             "MaxCapacity": 23,
 | |
|         },
 | |
|     )
 | |
| 
 | |
|     resp = client.describe_scheduled_actions(ServiceNamespace="ecs")
 | |
|     resp.should.have.key("ScheduledActions").length_of(1)
 | |
| 
 | |
|     action = resp["ScheduledActions"][0]
 | |
|     action.should.have.key("ScheduledActionName").equals("action_name_updated")
 | |
|     action.should.have.key("ScheduledActionARN").equals(
 | |
|         f"arn:aws:autoscaling:ap-southeast-1:{ACCOUNT_ID}:scheduledAction:ecs:scheduledActionName/action_name"
 | |
|     )
 | |
|     action.should.have.key("ServiceNamespace").equals("ecs")
 | |
|     action.should.have.key("ResourceId").equals("ecs:cluster:x")
 | |
|     action.should.have.key("ScalableDimension").equals("ecs:service:DesiredCount")
 | |
|     action.should.have.key("CreationTime")
 | |
|     action.should.have.key("ScalableTargetAction").equals(
 | |
|         {"MaxCapacity": 23, "MinCapacity": 12}
 | |
|     )
 |