Feature: Dockerless Batch (#5100)
This commit is contained in:
		
							parent
							
								
									c2727a7c20
								
							
						
					
					
						commit
						beb05662e4
					
				| @ -12,6 +12,8 @@ | |||||||
| batch | batch | ||||||
| ===== | ===== | ||||||
| 
 | 
 | ||||||
|  | .. autoclass:: moto.batch.models.BatchBackend | ||||||
|  | 
 | ||||||
| |start-h3| Example usage |end-h3| | |start-h3| Example usage |end-h3| | ||||||
| 
 | 
 | ||||||
| .. sourcecode:: python | .. sourcecode:: python | ||||||
| @ -28,21 +30,6 @@ batch | |||||||
| - [X] cancel_job | - [X] cancel_job | ||||||
| - [X] create_compute_environment | - [X] create_compute_environment | ||||||
| - [X] create_job_queue | - [X] create_job_queue | ||||||
|    |  | ||||||
|         Create a job queue |  | ||||||
| 
 |  | ||||||
|         :param queue_name: Queue name |  | ||||||
|         :type queue_name: str |  | ||||||
|         :param priority: Queue priority |  | ||||||
|         :type priority: int |  | ||||||
|         :param state: Queue state |  | ||||||
|         :type state: string |  | ||||||
|         :param compute_env_order: Compute environment list |  | ||||||
|         :type compute_env_order: list of dict |  | ||||||
|         :return: Tuple of Name, ARN |  | ||||||
|         :rtype: tuple of str |  | ||||||
|          |  | ||||||
| 
 |  | ||||||
| - [ ] create_scheduling_policy | - [ ] create_scheduling_policy | ||||||
| - [X] delete_compute_environment | - [X] delete_compute_environment | ||||||
| - [X] delete_job_queue | - [X] delete_job_queue | ||||||
| @ -83,20 +70,5 @@ batch | |||||||
| - [X] untag_resource | - [X] untag_resource | ||||||
| - [X] update_compute_environment | - [X] update_compute_environment | ||||||
| - [X] update_job_queue | - [X] update_job_queue | ||||||
|    |  | ||||||
|         Update a job queue |  | ||||||
| 
 |  | ||||||
|         :param queue_name: Queue name |  | ||||||
|         :type queue_name: str |  | ||||||
|         :param priority: Queue priority |  | ||||||
|         :type priority: int |  | ||||||
|         :param state: Queue state |  | ||||||
|         :type state: string |  | ||||||
|         :param compute_env_order: Compute environment list |  | ||||||
|         :type compute_env_order: list of dict |  | ||||||
|         :return: Tuple of Name, ARN |  | ||||||
|         :rtype: tuple of str |  | ||||||
|          |  | ||||||
| 
 |  | ||||||
| - [ ] update_scheduling_policy | - [ ] update_scheduling_policy | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -50,6 +50,12 @@ mock_lambda = lazy_load( | |||||||
|     ".awslambda", "mock_lambda", boto3_name="lambda", backend="lambda_backends" |     ".awslambda", "mock_lambda", boto3_name="lambda", backend="lambda_backends" | ||||||
| ) | ) | ||||||
| mock_batch = lazy_load(".batch", "mock_batch") | mock_batch = lazy_load(".batch", "mock_batch") | ||||||
|  | mock_batch_simple = lazy_load( | ||||||
|  |     ".batch_simple", | ||||||
|  |     "mock_batch_simple", | ||||||
|  |     boto3_name="batch", | ||||||
|  |     backend="batch_simple_backends", | ||||||
|  | ) | ||||||
| mock_budgets = lazy_load(".budgets", "mock_budgets") | mock_budgets = lazy_load(".budgets", "mock_budgets") | ||||||
| mock_cloudformation = lazy_load(".cloudformation", "mock_cloudformation") | mock_cloudformation = lazy_load(".cloudformation", "mock_cloudformation") | ||||||
| mock_cloudfront = lazy_load(".cloudfront", "mock_cloudfront") | mock_cloudfront = lazy_load(".cloudfront", "mock_cloudfront") | ||||||
|  | |||||||
| @ -5,7 +5,6 @@ import datetime | |||||||
| import time | import time | ||||||
| import uuid | import uuid | ||||||
| import logging | import logging | ||||||
| import docker |  | ||||||
| import threading | import threading | ||||||
| import dateutil.parser | import dateutil.parser | ||||||
| from sys import platform | from sys import platform | ||||||
| @ -568,6 +567,8 @@ class Job(threading.Thread, BaseModel, DockerModel, ManagedState): | |||||||
|         :return: |         :return: | ||||||
|         """ |         """ | ||||||
|         try: |         try: | ||||||
|  |             import docker | ||||||
|  | 
 | ||||||
|             self.advance() |             self.advance() | ||||||
|             while self.status == "SUBMITTED": |             while self.status == "SUBMITTED": | ||||||
|                 # Wait until we've moved onto state 'PENDING' |                 # Wait until we've moved onto state 'PENDING' | ||||||
| @ -817,6 +818,14 @@ class Job(threading.Thread, BaseModel, DockerModel, ManagedState): | |||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class BatchBackend(BaseBackend): | class BatchBackend(BaseBackend): | ||||||
|  |     """ | ||||||
|  |     Batch-jobs are executed inside a Docker-container. Everytime the `submit_job`-method is called, a new Docker container is started. | ||||||
|  |     A job is marked as 'Success' when the Docker-container exits without throwing an error. | ||||||
|  | 
 | ||||||
|  |     Use `@mock_batch_simple` instead if you do not want to use a Docker-container. | ||||||
|  |     With this decorator, jobs are simply marked as 'Success' without trying to execute any commands/scripts. | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|     def __init__(self, region_name=None): |     def __init__(self, region_name=None): | ||||||
|         super().__init__() |         super().__init__() | ||||||
|         self.region_name = region_name |         self.region_name = region_name | ||||||
| @ -1296,20 +1305,6 @@ class BatchBackend(BaseBackend): | |||||||
|     def create_job_queue( |     def create_job_queue( | ||||||
|         self, queue_name, priority, state, compute_env_order, tags=None |         self, queue_name, priority, state, compute_env_order, tags=None | ||||||
|     ): |     ): | ||||||
|         """ |  | ||||||
|         Create a job queue |  | ||||||
| 
 |  | ||||||
|         :param queue_name: Queue name |  | ||||||
|         :type queue_name: str |  | ||||||
|         :param priority: Queue priority |  | ||||||
|         :type priority: int |  | ||||||
|         :param state: Queue state |  | ||||||
|         :type state: string |  | ||||||
|         :param compute_env_order: Compute environment list |  | ||||||
|         :type compute_env_order: list of dict |  | ||||||
|         :return: Tuple of Name, ARN |  | ||||||
|         :rtype: tuple of str |  | ||||||
|         """ |  | ||||||
|         for variable, var_name in ( |         for variable, var_name in ( | ||||||
|             (queue_name, "jobQueueName"), |             (queue_name, "jobQueueName"), | ||||||
|             (priority, "priority"), |             (priority, "priority"), | ||||||
| @ -1380,20 +1375,6 @@ class BatchBackend(BaseBackend): | |||||||
|         return result |         return result | ||||||
| 
 | 
 | ||||||
|     def update_job_queue(self, queue_name, priority, state, compute_env_order): |     def update_job_queue(self, queue_name, priority, state, compute_env_order): | ||||||
|         """ |  | ||||||
|         Update a job queue |  | ||||||
| 
 |  | ||||||
|         :param queue_name: Queue name |  | ||||||
|         :type queue_name: str |  | ||||||
|         :param priority: Queue priority |  | ||||||
|         :type priority: int |  | ||||||
|         :param state: Queue state |  | ||||||
|         :type state: string |  | ||||||
|         :param compute_env_order: Compute environment list |  | ||||||
|         :type compute_env_order: list of dict |  | ||||||
|         :return: Tuple of Name, ARN |  | ||||||
|         :rtype: tuple of str |  | ||||||
|         """ |  | ||||||
|         if queue_name is None: |         if queue_name is None: | ||||||
|             raise ClientException("jobQueueName must be provided") |             raise ClientException("jobQueueName must be provided") | ||||||
| 
 | 
 | ||||||
|  | |||||||
							
								
								
									
										5
									
								
								moto/batch_simple/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								moto/batch_simple/__init__.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | from .models import batch_simple_backends | ||||||
|  | from ..core.models import base_decorator | ||||||
|  | 
 | ||||||
|  | batch_backend = batch_simple_backends["us-east-1"] | ||||||
|  | mock_batch_simple = base_decorator(batch_simple_backends) | ||||||
							
								
								
									
										85
									
								
								moto/batch_simple/models.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										85
									
								
								moto/batch_simple/models.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,85 @@ | |||||||
|  | from ..batch.models import batch_backends, BaseBackend, Job, ClientException | ||||||
|  | from ..core.utils import BackendDict | ||||||
|  | 
 | ||||||
|  | import datetime | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class BatchSimpleBackend(BaseBackend): | ||||||
|  |     """ | ||||||
|  |     Implements a Batch-Backend that does not use Docker containers. Submitted Jobs are simply marked as Success | ||||||
|  |     Use the `@mock_batch_simple`-decorator to use this class. | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|  |     def __init__(self, region_name=None): | ||||||
|  |         self.region_name = region_name | ||||||
|  | 
 | ||||||
|  |     @property | ||||||
|  |     def backend(self): | ||||||
|  |         return batch_backends[self.region_name] | ||||||
|  | 
 | ||||||
|  |     def __getattribute__(self, name): | ||||||
|  |         """ | ||||||
|  |         Magic part that makes this class behave like a wrapper around the regular batch_backend | ||||||
|  |         We intercept calls to `submit_job` and replace this with our own (non-Docker) implementation | ||||||
|  |         Every other method call is send through to batch_backend | ||||||
|  |         """ | ||||||
|  |         if name in [ | ||||||
|  |             "backend", | ||||||
|  |             "region_name", | ||||||
|  |             "urls", | ||||||
|  |             "_url_module", | ||||||
|  |             "__class__", | ||||||
|  |             "url_bases", | ||||||
|  |         ]: | ||||||
|  |             return object.__getattribute__(self, name) | ||||||
|  |         if name in ["submit_job"]: | ||||||
|  | 
 | ||||||
|  |             def newfunc(*args, **kwargs): | ||||||
|  |                 attr = object.__getattribute__(self, name) | ||||||
|  |                 return attr(*args, **kwargs) | ||||||
|  | 
 | ||||||
|  |             return newfunc | ||||||
|  |         else: | ||||||
|  |             return object.__getattribute__(self.backend, name) | ||||||
|  | 
 | ||||||
|  |     def submit_job( | ||||||
|  |         self, | ||||||
|  |         job_name, | ||||||
|  |         job_def_id, | ||||||
|  |         job_queue, | ||||||
|  |         depends_on=None, | ||||||
|  |         container_overrides=None, | ||||||
|  |         timeout=None, | ||||||
|  |     ): | ||||||
|  |         # Look for job definition | ||||||
|  |         job_def = self.get_job_definition(job_def_id) | ||||||
|  |         if job_def is None: | ||||||
|  |             raise ClientException( | ||||||
|  |                 "Job definition {0} does not exist".format(job_def_id) | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |         queue = self.get_job_queue(job_queue) | ||||||
|  |         if queue is None: | ||||||
|  |             raise ClientException("Job queue {0} does not exist".format(job_queue)) | ||||||
|  | 
 | ||||||
|  |         job = Job( | ||||||
|  |             job_name, | ||||||
|  |             job_def, | ||||||
|  |             queue, | ||||||
|  |             log_backend=self.logs_backend, | ||||||
|  |             container_overrides=container_overrides, | ||||||
|  |             depends_on=depends_on, | ||||||
|  |             all_jobs=self._jobs, | ||||||
|  |             timeout=timeout, | ||||||
|  |         ) | ||||||
|  |         self.backend._jobs[job.job_id] = job | ||||||
|  | 
 | ||||||
|  |         # We don't want to actually run the job - just mark it as succeeded | ||||||
|  |         job.job_started_at = datetime.datetime.now() | ||||||
|  |         job._start_attempt() | ||||||
|  |         job._mark_stopped(success=True) | ||||||
|  | 
 | ||||||
|  |         return job_name, job.job_id | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | batch_simple_backends = BackendDict(BatchSimpleBackend, "batch") | ||||||
							
								
								
									
										12
									
								
								moto/batch_simple/responses.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								moto/batch_simple/responses.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | from ..batch.responses import BatchResponse | ||||||
|  | from .models import batch_simple_backends | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class BatchSimpleResponse(BatchResponse): | ||||||
|  |     @property | ||||||
|  |     def batch_backend(self): | ||||||
|  |         """ | ||||||
|  |         :return: Batch Backend | ||||||
|  |         :rtype: moto.batch.models.BatchBackend | ||||||
|  |         """ | ||||||
|  |         return batch_simple_backends[self.region] | ||||||
							
								
								
									
										6
									
								
								moto/batch_simple/urls.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										6
									
								
								moto/batch_simple/urls.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,6 @@ | |||||||
|  | from ..batch.urls import url_bases as batch_url_bases | ||||||
|  | from ..batch.urls import url_paths as batch_url_paths | ||||||
|  | from .responses import BatchSimpleResponse | ||||||
|  | 
 | ||||||
|  | url_bases = batch_url_bases.copy() | ||||||
|  | url_paths = {k: BatchSimpleResponse.dispatch for k in batch_url_paths} | ||||||
| @ -1,4 +1,3 @@ | |||||||
| import docker |  | ||||||
| import functools | import functools | ||||||
| import requests.adapters | import requests.adapters | ||||||
| 
 | 
 | ||||||
| @ -17,6 +16,8 @@ class DockerModel: | |||||||
|         if self.__docker_client is None: |         if self.__docker_client is None: | ||||||
|             # We should only initiate the Docker Client at runtime. |             # We should only initiate the Docker Client at runtime. | ||||||
|             # The docker.from_env() call will fall if Docker is not running |             # The docker.from_env() call will fall if Docker is not running | ||||||
|  |             import docker | ||||||
|  | 
 | ||||||
|             self.__docker_client = docker.from_env() |             self.__docker_client = docker.from_env() | ||||||
| 
 | 
 | ||||||
|             # Unfortunately mocking replaces this method w/o fallback enabled, so we |             # Unfortunately mocking replaces this method w/o fallback enabled, so we | ||||||
|  | |||||||
							
								
								
									
										0
									
								
								tests/test_batch_simple/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										0
									
								
								tests/test_batch_simple/__init__.py
									
									
									
									
									
										Normal file
									
								
							
							
								
								
									
										278
									
								
								tests/test_batch_simple/test_batch_cloudformation.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										278
									
								
								tests/test_batch_simple/test_batch_cloudformation.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,278 @@ | |||||||
|  | import boto3 | ||||||
|  | import json | ||||||
|  | import sure  # noqa # pylint: disable=unused-import | ||||||
|  | from moto import mock_iam, mock_ec2, mock_ecs, mock_cloudformation | ||||||
|  | from moto import mock_batch_simple as mock_batch_without_docker | ||||||
|  | from uuid import uuid4 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Copy of test_batch/test_batch_cloudformation | ||||||
|  | # Except that we verify this behaviour still works without docker | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | DEFAULT_REGION = "eu-central-1" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def _get_clients(): | ||||||
|  |     return ( | ||||||
|  |         boto3.client("ec2", region_name=DEFAULT_REGION), | ||||||
|  |         boto3.client("iam", region_name=DEFAULT_REGION), | ||||||
|  |         boto3.client("ecs", region_name=DEFAULT_REGION), | ||||||
|  |         boto3.client("logs", region_name=DEFAULT_REGION), | ||||||
|  |         boto3.client("batch", region_name=DEFAULT_REGION), | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def _setup(ec2_client, iam_client): | ||||||
|  |     """ | ||||||
|  |     Do prerequisite setup | ||||||
|  |     :return: VPC ID, Subnet ID, Security group ID, IAM Role ARN | ||||||
|  |     :rtype: tuple | ||||||
|  |     """ | ||||||
|  |     resp = ec2_client.create_vpc(CidrBlock="172.30.0.0/24") | ||||||
|  |     vpc_id = resp["Vpc"]["VpcId"] | ||||||
|  |     resp = ec2_client.create_subnet( | ||||||
|  |         AvailabilityZone="eu-central-1a", CidrBlock="172.30.0.0/25", VpcId=vpc_id | ||||||
|  |     ) | ||||||
|  |     subnet_id = resp["Subnet"]["SubnetId"] | ||||||
|  |     resp = ec2_client.create_security_group( | ||||||
|  |         Description="test_sg_desc", GroupName=str(uuid4())[0:6], VpcId=vpc_id | ||||||
|  |     ) | ||||||
|  |     sg_id = resp["GroupId"] | ||||||
|  | 
 | ||||||
|  |     role_name = str(uuid4())[0:6] | ||||||
|  |     resp = iam_client.create_role( | ||||||
|  |         RoleName=role_name, AssumeRolePolicyDocument="some_policy" | ||||||
|  |     ) | ||||||
|  |     iam_arn = resp["Role"]["Arn"] | ||||||
|  |     iam_client.create_instance_profile(InstanceProfileName=role_name) | ||||||
|  |     iam_client.add_role_to_instance_profile( | ||||||
|  |         InstanceProfileName=role_name, RoleName=role_name | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     return vpc_id, subnet_id, sg_id, iam_arn | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @mock_cloudformation() | ||||||
|  | @mock_ec2 | ||||||
|  | @mock_ecs | ||||||
|  | @mock_iam | ||||||
|  | @mock_batch_without_docker | ||||||
|  | def test_create_env_cf(): | ||||||
|  |     ec2_client, iam_client, _, _, _ = _get_clients() | ||||||
|  |     _, subnet_id, sg_id, iam_arn = _setup(ec2_client, iam_client) | ||||||
|  | 
 | ||||||
|  |     create_environment_template = { | ||||||
|  |         "Resources": { | ||||||
|  |             "ComputeEnvironment": { | ||||||
|  |                 "Type": "AWS::Batch::ComputeEnvironment", | ||||||
|  |                 "Properties": { | ||||||
|  |                     "Type": "MANAGED", | ||||||
|  |                     "ComputeResources": { | ||||||
|  |                         "Type": "EC2", | ||||||
|  |                         "MinvCpus": 0, | ||||||
|  |                         "DesiredvCpus": 0, | ||||||
|  |                         "MaxvCpus": 64, | ||||||
|  |                         "InstanceTypes": ["optimal"], | ||||||
|  |                         "Subnets": [subnet_id], | ||||||
|  |                         "SecurityGroupIds": [sg_id], | ||||||
|  |                         "InstanceRole": iam_arn.replace("role", "instance-profile"), | ||||||
|  |                     }, | ||||||
|  |                     "ServiceRole": iam_arn, | ||||||
|  |                 }, | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     cf_json = json.dumps(create_environment_template) | ||||||
|  | 
 | ||||||
|  |     cf_conn = boto3.client("cloudformation", DEFAULT_REGION) | ||||||
|  |     stack_name = str(uuid4())[0:6] | ||||||
|  |     stack_id = cf_conn.create_stack(StackName=stack_name, TemplateBody=cf_json)[ | ||||||
|  |         "StackId" | ||||||
|  |     ] | ||||||
|  | 
 | ||||||
|  |     stack_resources = cf_conn.list_stack_resources(StackName=stack_id) | ||||||
|  | 
 | ||||||
|  |     stack_resources["StackResourceSummaries"][0]["ResourceStatus"].should.equal( | ||||||
|  |         "CREATE_COMPLETE" | ||||||
|  |     ) | ||||||
|  |     # Spot checks on the ARN | ||||||
|  |     stack_resources["StackResourceSummaries"][0]["PhysicalResourceId"].startswith( | ||||||
|  |         "arn:aws:batch:" | ||||||
|  |     ) | ||||||
|  |     stack_resources["StackResourceSummaries"][0]["PhysicalResourceId"].should.contain( | ||||||
|  |         stack_name | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @mock_cloudformation() | ||||||
|  | @mock_ec2 | ||||||
|  | @mock_ecs | ||||||
|  | @mock_iam | ||||||
|  | @mock_batch_without_docker | ||||||
|  | def test_create_job_queue_cf(): | ||||||
|  |     ec2_client, iam_client, _, _, _ = _get_clients() | ||||||
|  |     _, subnet_id, sg_id, iam_arn = _setup(ec2_client, iam_client) | ||||||
|  | 
 | ||||||
|  |     create_environment_template = { | ||||||
|  |         "Resources": { | ||||||
|  |             "ComputeEnvironment": { | ||||||
|  |                 "Type": "AWS::Batch::ComputeEnvironment", | ||||||
|  |                 "Properties": { | ||||||
|  |                     "Type": "MANAGED", | ||||||
|  |                     "ComputeResources": { | ||||||
|  |                         "Type": "EC2", | ||||||
|  |                         "MinvCpus": 0, | ||||||
|  |                         "DesiredvCpus": 0, | ||||||
|  |                         "MaxvCpus": 64, | ||||||
|  |                         "InstanceTypes": ["optimal"], | ||||||
|  |                         "Subnets": [subnet_id], | ||||||
|  |                         "SecurityGroupIds": [sg_id], | ||||||
|  |                         "InstanceRole": iam_arn.replace("role", "instance-profile"), | ||||||
|  |                     }, | ||||||
|  |                     "ServiceRole": iam_arn, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |             "JobQueue": { | ||||||
|  |                 "Type": "AWS::Batch::JobQueue", | ||||||
|  |                 "Properties": { | ||||||
|  |                     "Priority": 1, | ||||||
|  |                     "ComputeEnvironmentOrder": [ | ||||||
|  |                         { | ||||||
|  |                             "Order": 1, | ||||||
|  |                             "ComputeEnvironment": {"Ref": "ComputeEnvironment"}, | ||||||
|  |                         } | ||||||
|  |                     ], | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     cf_json = json.dumps(create_environment_template) | ||||||
|  | 
 | ||||||
|  |     cf_conn = boto3.client("cloudformation", DEFAULT_REGION) | ||||||
|  |     stack_name = str(uuid4())[0:6] | ||||||
|  |     stack_id = cf_conn.create_stack(StackName=stack_name, TemplateBody=cf_json)[ | ||||||
|  |         "StackId" | ||||||
|  |     ] | ||||||
|  | 
 | ||||||
|  |     stack_resources = cf_conn.list_stack_resources(StackName=stack_id) | ||||||
|  |     len(stack_resources["StackResourceSummaries"]).should.equal(2) | ||||||
|  | 
 | ||||||
|  |     job_queue_resource = list( | ||||||
|  |         filter( | ||||||
|  |             lambda item: item["ResourceType"] == "AWS::Batch::JobQueue", | ||||||
|  |             stack_resources["StackResourceSummaries"], | ||||||
|  |         ) | ||||||
|  |     )[0] | ||||||
|  | 
 | ||||||
|  |     job_queue_resource["ResourceStatus"].should.equal("CREATE_COMPLETE") | ||||||
|  |     # Spot checks on the ARN | ||||||
|  |     job_queue_resource["PhysicalResourceId"].startswith("arn:aws:batch:") | ||||||
|  |     job_queue_resource["PhysicalResourceId"].should.contain(stack_name) | ||||||
|  |     job_queue_resource["PhysicalResourceId"].should.contain("job-queue/") | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @mock_cloudformation | ||||||
|  | @mock_ec2 | ||||||
|  | @mock_ecs | ||||||
|  | @mock_iam | ||||||
|  | @mock_batch_without_docker | ||||||
|  | def test_create_job_def_cf(): | ||||||
|  |     ec2_client, iam_client, _, _, _ = _get_clients() | ||||||
|  |     _, subnet_id, sg_id, iam_arn = _setup(ec2_client, iam_client) | ||||||
|  | 
 | ||||||
|  |     create_environment_template = { | ||||||
|  |         "Resources": { | ||||||
|  |             "ComputeEnvironment": { | ||||||
|  |                 "Type": "AWS::Batch::ComputeEnvironment", | ||||||
|  |                 "Properties": { | ||||||
|  |                     "Type": "MANAGED", | ||||||
|  |                     "ComputeResources": { | ||||||
|  |                         "Type": "EC2", | ||||||
|  |                         "MinvCpus": 0, | ||||||
|  |                         "DesiredvCpus": 0, | ||||||
|  |                         "MaxvCpus": 64, | ||||||
|  |                         "InstanceTypes": ["optimal"], | ||||||
|  |                         "Subnets": [subnet_id], | ||||||
|  |                         "SecurityGroupIds": [sg_id], | ||||||
|  |                         "InstanceRole": iam_arn.replace("role", "instance-profile"), | ||||||
|  |                     }, | ||||||
|  |                     "ServiceRole": iam_arn, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |             "JobQueue": { | ||||||
|  |                 "Type": "AWS::Batch::JobQueue", | ||||||
|  |                 "Properties": { | ||||||
|  |                     "Priority": 1, | ||||||
|  |                     "ComputeEnvironmentOrder": [ | ||||||
|  |                         { | ||||||
|  |                             "Order": 1, | ||||||
|  |                             "ComputeEnvironment": {"Ref": "ComputeEnvironment"}, | ||||||
|  |                         } | ||||||
|  |                     ], | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |             "JobDefinition": { | ||||||
|  |                 "Type": "AWS::Batch::JobDefinition", | ||||||
|  |                 "Properties": { | ||||||
|  |                     "Type": "container", | ||||||
|  |                     "ContainerProperties": { | ||||||
|  |                         "Image": { | ||||||
|  |                             "Fn::Join": [ | ||||||
|  |                                 "", | ||||||
|  |                                 [ | ||||||
|  |                                     "137112412989.dkr.ecr.", | ||||||
|  |                                     {"Ref": "AWS::Region"}, | ||||||
|  |                                     ".amazonaws.com/amazonlinux:latest", | ||||||
|  |                                 ], | ||||||
|  |                             ] | ||||||
|  |                         }, | ||||||
|  |                         "ResourceRequirements": [ | ||||||
|  |                             {"Type": "MEMORY", "Value": 2000}, | ||||||
|  |                             {"Type": "VCPU", "Value": 2}, | ||||||
|  |                         ], | ||||||
|  |                         "Command": ["echo", "Hello world"], | ||||||
|  |                         "LinuxParameters": {"Devices": [{"HostPath": "test-path"}]}, | ||||||
|  |                     }, | ||||||
|  |                     "RetryStrategy": {"Attempts": 1}, | ||||||
|  |                 }, | ||||||
|  |             }, | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |     cf_json = json.dumps(create_environment_template) | ||||||
|  | 
 | ||||||
|  |     cf_conn = boto3.client("cloudformation", DEFAULT_REGION) | ||||||
|  |     stack_name = str(uuid4())[0:6] | ||||||
|  |     stack_id = cf_conn.create_stack(StackName=stack_name, TemplateBody=cf_json)[ | ||||||
|  |         "StackId" | ||||||
|  |     ] | ||||||
|  | 
 | ||||||
|  |     stack_resources = cf_conn.list_stack_resources(StackName=stack_id) | ||||||
|  |     len(stack_resources["StackResourceSummaries"]).should.equal(3) | ||||||
|  | 
 | ||||||
|  |     job_def_resource = list( | ||||||
|  |         filter( | ||||||
|  |             lambda item: item["ResourceType"] == "AWS::Batch::JobDefinition", | ||||||
|  |             stack_resources["StackResourceSummaries"], | ||||||
|  |         ) | ||||||
|  |     )[0] | ||||||
|  | 
 | ||||||
|  |     job_def_resource["ResourceStatus"].should.equal("CREATE_COMPLETE") | ||||||
|  |     # Spot checks on the ARN | ||||||
|  |     job_def_resource["PhysicalResourceId"].startswith("arn:aws:batch:") | ||||||
|  |     job_def_resource["PhysicalResourceId"].should.contain(f"{stack_name}-JobDef") | ||||||
|  |     job_def_resource["PhysicalResourceId"].should.contain("job-definition/") | ||||||
|  | 
 | ||||||
|  |     # Test the linux parameter device host path | ||||||
|  |     # This ensures that batch is parsing the parameter dictionaries | ||||||
|  |     # correctly by recursively converting the first character of all | ||||||
|  |     # dict keys to lowercase. | ||||||
|  |     batch_conn = boto3.client("batch", DEFAULT_REGION) | ||||||
|  |     response = batch_conn.describe_job_definitions( | ||||||
|  |         jobDefinitions=[job_def_resource["PhysicalResourceId"]] | ||||||
|  |     ) | ||||||
|  |     job_def_linux_device_host_path = response.get("jobDefinitions")[0][ | ||||||
|  |         "containerProperties" | ||||||
|  |     ]["linuxParameters"]["devices"][0]["hostPath"] | ||||||
|  | 
 | ||||||
|  |     job_def_linux_device_host_path.should.equal("test-path") | ||||||
							
								
								
									
										101
									
								
								tests/test_batch_simple/test_batch_compute_envs.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										101
									
								
								tests/test_batch_simple/test_batch_compute_envs.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,101 @@ | |||||||
|  | from ..test_batch import _get_clients, _setup | ||||||
|  | import sure  # noqa # pylint: disable=unused-import | ||||||
|  | from moto import mock_batch_simple, mock_iam, mock_ec2, mock_ecs, settings | ||||||
|  | from uuid import uuid4 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Copy of test_batch/test_batch_cloudformation | ||||||
|  | # Except that we verify this behaviour still works without docker | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @mock_ec2 | ||||||
|  | @mock_ecs | ||||||
|  | @mock_iam | ||||||
|  | @mock_batch_simple | ||||||
|  | def test_create_managed_compute_environment(): | ||||||
|  |     ec2_client, iam_client, ecs_client, _, batch_client = _get_clients() | ||||||
|  |     _, subnet_id, sg_id, iam_arn = _setup(ec2_client, iam_client) | ||||||
|  | 
 | ||||||
|  |     compute_name = str(uuid4()) | ||||||
|  |     resp = batch_client.create_compute_environment( | ||||||
|  |         computeEnvironmentName=compute_name, | ||||||
|  |         type="MANAGED", | ||||||
|  |         state="ENABLED", | ||||||
|  |         computeResources={ | ||||||
|  |             "type": "EC2", | ||||||
|  |             "minvCpus": 5, | ||||||
|  |             "maxvCpus": 10, | ||||||
|  |             "desiredvCpus": 5, | ||||||
|  |             "instanceTypes": ["t2.small", "t2.medium"], | ||||||
|  |             "imageId": "some_image_id", | ||||||
|  |             "subnets": [subnet_id], | ||||||
|  |             "securityGroupIds": [sg_id], | ||||||
|  |             "ec2KeyPair": "string", | ||||||
|  |             "instanceRole": iam_arn.replace("role", "instance-profile"), | ||||||
|  |             "tags": {"string": "string"}, | ||||||
|  |             "bidPercentage": 123, | ||||||
|  |             "spotIamFleetRole": "string", | ||||||
|  |         }, | ||||||
|  |         serviceRole=iam_arn, | ||||||
|  |     ) | ||||||
|  |     resp.should.contain("computeEnvironmentArn") | ||||||
|  |     resp["computeEnvironmentName"].should.equal(compute_name) | ||||||
|  | 
 | ||||||
|  |     our_env = batch_client.describe_compute_environments( | ||||||
|  |         computeEnvironments=[compute_name] | ||||||
|  |     )["computeEnvironments"][0] | ||||||
|  | 
 | ||||||
|  |     # Given a t2.medium is 2 vcpu and t2.small is 1, therefore 2 mediums and 1 small should be created | ||||||
|  |     if not settings.TEST_SERVER_MODE: | ||||||
|  |         # Can't verify this in ServerMode, as other tests may have created instances | ||||||
|  |         resp = ec2_client.describe_instances() | ||||||
|  |         resp.should.contain("Reservations") | ||||||
|  |         len(resp["Reservations"]).should.equal(3) | ||||||
|  | 
 | ||||||
|  |     # Should have created 1 ECS cluster | ||||||
|  |     all_clusters = ecs_client.list_clusters()["clusterArns"] | ||||||
|  |     all_clusters.should.contain(our_env["ecsClusterArn"]) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @mock_ec2 | ||||||
|  | @mock_ecs | ||||||
|  | @mock_iam | ||||||
|  | @mock_batch_simple | ||||||
|  | def test_create_managed_compute_environment_with_instance_family(): | ||||||
|  |     """ | ||||||
|  |     The InstanceType parameter can have multiple values: | ||||||
|  |     instance_type     t2.small | ||||||
|  |     instance_family   t2       <-- What we're testing here | ||||||
|  |     'optimal' | ||||||
|  |     unknown value | ||||||
|  |     """ | ||||||
|  |     ec2_client, iam_client, _, _, batch_client = _get_clients() | ||||||
|  |     _, subnet_id, sg_id, iam_arn = _setup(ec2_client, iam_client) | ||||||
|  | 
 | ||||||
|  |     compute_name = str(uuid4()) | ||||||
|  |     batch_client.create_compute_environment( | ||||||
|  |         computeEnvironmentName=compute_name, | ||||||
|  |         type="MANAGED", | ||||||
|  |         state="ENABLED", | ||||||
|  |         computeResources={ | ||||||
|  |             "type": "EC2", | ||||||
|  |             "minvCpus": 5, | ||||||
|  |             "maxvCpus": 10, | ||||||
|  |             "desiredvCpus": 5, | ||||||
|  |             "instanceTypes": ["t2"], | ||||||
|  |             "imageId": "some_image_id", | ||||||
|  |             "subnets": [subnet_id], | ||||||
|  |             "securityGroupIds": [sg_id], | ||||||
|  |             "ec2KeyPair": "string", | ||||||
|  |             "instanceRole": iam_arn.replace("role", "instance-profile"), | ||||||
|  |             "tags": {"string": "string"}, | ||||||
|  |             "bidPercentage": 123, | ||||||
|  |             "spotIamFleetRole": "string", | ||||||
|  |         }, | ||||||
|  |         serviceRole=iam_arn, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     our_env = batch_client.describe_compute_environments( | ||||||
|  |         computeEnvironments=[compute_name] | ||||||
|  |     )["computeEnvironments"][0] | ||||||
|  |     our_env["computeResources"]["instanceTypes"].should.equal(["t2"]) | ||||||
							
								
								
									
										112
									
								
								tests/test_batch_simple/test_batch_jobs.py
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										112
									
								
								tests/test_batch_simple/test_batch_jobs.py
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,112 @@ | |||||||
|  | from ..test_batch import _get_clients, _setup | ||||||
|  | 
 | ||||||
|  | import sure  # noqa # pylint: disable=unused-import | ||||||
|  | from moto import mock_iam, mock_ec2, mock_ecs, mock_logs | ||||||
|  | from moto import mock_batch_simple | ||||||
|  | from uuid import uuid4 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Copy of test_batch/test_batch_jobs | ||||||
|  | # Except that we verify this behaviour still works without docker | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @mock_logs | ||||||
|  | @mock_ec2 | ||||||
|  | @mock_ecs | ||||||
|  | @mock_iam | ||||||
|  | @mock_batch_simple | ||||||
|  | def test_submit_job_by_name(): | ||||||
|  |     ec2_client, iam_client, _, _, batch_client = _get_clients() | ||||||
|  |     _, _, _, iam_arn = _setup(ec2_client, iam_client) | ||||||
|  | 
 | ||||||
|  |     compute_name = str(uuid4()) | ||||||
|  |     resp = batch_client.create_compute_environment( | ||||||
|  |         computeEnvironmentName=compute_name, | ||||||
|  |         type="UNMANAGED", | ||||||
|  |         state="ENABLED", | ||||||
|  |         serviceRole=iam_arn, | ||||||
|  |     ) | ||||||
|  |     arn = resp["computeEnvironmentArn"] | ||||||
|  | 
 | ||||||
|  |     resp = batch_client.create_job_queue( | ||||||
|  |         jobQueueName=str(uuid4()), | ||||||
|  |         state="ENABLED", | ||||||
|  |         priority=123, | ||||||
|  |         computeEnvironmentOrder=[{"order": 123, "computeEnvironment": arn}], | ||||||
|  |     ) | ||||||
|  |     queue_arn = resp["jobQueueArn"] | ||||||
|  | 
 | ||||||
|  |     job_definition_name = f"sleep10_{str(uuid4())[0:6]}" | ||||||
|  | 
 | ||||||
|  |     resp = batch_client.register_job_definition( | ||||||
|  |         jobDefinitionName=job_definition_name, | ||||||
|  |         type="container", | ||||||
|  |         containerProperties={ | ||||||
|  |             "image": "busybox", | ||||||
|  |             "vcpus": 1, | ||||||
|  |             "memory": 512, | ||||||
|  |             "command": ["sleep", "10"], | ||||||
|  |         }, | ||||||
|  |     ) | ||||||
|  |     job_definition_arn = resp["jobDefinitionArn"] | ||||||
|  | 
 | ||||||
|  |     resp = batch_client.submit_job( | ||||||
|  |         jobName="test1", jobQueue=queue_arn, jobDefinition=job_definition_name | ||||||
|  |     ) | ||||||
|  |     job_id = resp["jobId"] | ||||||
|  | 
 | ||||||
|  |     resp_jobs = batch_client.describe_jobs(jobs=[job_id]) | ||||||
|  | 
 | ||||||
|  |     len(resp_jobs["jobs"]).should.equal(1) | ||||||
|  |     job = resp_jobs["jobs"][0] | ||||||
|  | 
 | ||||||
|  |     job["jobId"].should.equal(job_id) | ||||||
|  |     job["jobQueue"].should.equal(queue_arn) | ||||||
|  |     job["jobDefinition"].should.equal(job_definition_arn) | ||||||
|  |     job["status"].should.equal("SUCCEEDED") | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @mock_batch_simple | ||||||
|  | def test_update_job_definition(): | ||||||
|  |     _, _, _, _, batch_client = _get_clients() | ||||||
|  | 
 | ||||||
|  |     tags = [ | ||||||
|  |         {"Foo1": "bar1", "Baz1": "buzz1"}, | ||||||
|  |         {"Foo2": "bar2", "Baz2": "buzz2"}, | ||||||
|  |     ] | ||||||
|  | 
 | ||||||
|  |     container_props = { | ||||||
|  |         "image": "amazonlinux", | ||||||
|  |         "memory": 1024, | ||||||
|  |         "vcpus": 2, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     job_def_name = str(uuid4())[0:6] | ||||||
|  |     batch_client.register_job_definition( | ||||||
|  |         jobDefinitionName=job_def_name, | ||||||
|  |         type="container", | ||||||
|  |         tags=tags[0], | ||||||
|  |         parameters={}, | ||||||
|  |         containerProperties=container_props, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     container_props["memory"] = 2048 | ||||||
|  |     batch_client.register_job_definition( | ||||||
|  |         jobDefinitionName=job_def_name, | ||||||
|  |         type="container", | ||||||
|  |         tags=tags[1], | ||||||
|  |         parameters={}, | ||||||
|  |         containerProperties=container_props, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     job_defs = batch_client.describe_job_definitions(jobDefinitionName=job_def_name)[ | ||||||
|  |         "jobDefinitions" | ||||||
|  |     ] | ||||||
|  |     job_defs.should.have.length_of(2) | ||||||
|  | 
 | ||||||
|  |     job_defs[0]["containerProperties"]["memory"].should.equal(1024) | ||||||
|  |     job_defs[0]["tags"].should.equal(tags[0]) | ||||||
|  |     job_defs[0].shouldnt.have.key("timeout") | ||||||
|  | 
 | ||||||
|  |     job_defs[1]["containerProperties"]["memory"].should.equal(2048) | ||||||
|  |     job_defs[1]["tags"].should.equal(tags[1]) | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user