Run black on moto & test directories.
This commit is contained in:
parent
c820395dbf
commit
96e5b1993d
507 changed files with 52541 additions and 47814 deletions
|
|
@ -11,24 +11,23 @@ from moto.core import BaseBackend, BaseModel
|
|||
from moto.ec2 import ec2_backends
|
||||
from moto.ecr.exceptions import ImageNotFoundException, RepositoryNotFoundException
|
||||
|
||||
DEFAULT_REGISTRY_ID = '012345678910'
|
||||
DEFAULT_REGISTRY_ID = "012345678910"
|
||||
|
||||
|
||||
class BaseObject(BaseModel):
|
||||
|
||||
def camelCase(self, key):
|
||||
words = []
|
||||
for i, word in enumerate(key.split('_')):
|
||||
for i, word in enumerate(key.split("_")):
|
||||
if i > 0:
|
||||
words.append(word.title())
|
||||
else:
|
||||
words.append(word)
|
||||
return ''.join(words)
|
||||
return "".join(words)
|
||||
|
||||
def gen_response_object(self):
|
||||
response_object = dict()
|
||||
for key, value in self.__dict__.items():
|
||||
if '_' in key:
|
||||
if "_" in key:
|
||||
response_object[self.camelCase(key)] = value
|
||||
else:
|
||||
response_object[key] = value
|
||||
|
|
@ -40,15 +39,16 @@ class BaseObject(BaseModel):
|
|||
|
||||
|
||||
class Repository(BaseObject):
|
||||
|
||||
def __init__(self, repository_name):
|
||||
self.registry_id = DEFAULT_REGISTRY_ID
|
||||
self.arn = 'arn:aws:ecr:us-east-1:{0}:repository/{1}'.format(
|
||||
self.registry_id, repository_name)
|
||||
self.arn = "arn:aws:ecr:us-east-1:{0}:repository/{1}".format(
|
||||
self.registry_id, repository_name
|
||||
)
|
||||
self.name = repository_name
|
||||
# self.created = datetime.utcnow()
|
||||
self.uri = '{0}.dkr.ecr.us-east-1.amazonaws.com/{1}'.format(
|
||||
self.registry_id, repository_name)
|
||||
self.uri = "{0}.dkr.ecr.us-east-1.amazonaws.com/{1}".format(
|
||||
self.registry_id, repository_name
|
||||
)
|
||||
self.images = []
|
||||
|
||||
@property
|
||||
|
|
@ -59,38 +59,45 @@ class Repository(BaseObject):
|
|||
def response_object(self):
|
||||
response_object = self.gen_response_object()
|
||||
|
||||
response_object['registryId'] = self.registry_id
|
||||
response_object['repositoryArn'] = self.arn
|
||||
response_object['repositoryName'] = self.name
|
||||
response_object['repositoryUri'] = self.uri
|
||||
response_object["registryId"] = self.registry_id
|
||||
response_object["repositoryArn"] = self.arn
|
||||
response_object["repositoryName"] = self.name
|
||||
response_object["repositoryUri"] = self.uri
|
||||
# response_object['createdAt'] = self.created
|
||||
del response_object['arn'], response_object['name'], response_object['images']
|
||||
del response_object["arn"], response_object["name"], response_object["images"]
|
||||
return response_object
|
||||
|
||||
@classmethod
|
||||
def create_from_cloudformation_json(cls, resource_name, cloudformation_json, region_name):
|
||||
properties = cloudformation_json['Properties']
|
||||
def create_from_cloudformation_json(
|
||||
cls, resource_name, cloudformation_json, region_name
|
||||
):
|
||||
properties = cloudformation_json["Properties"]
|
||||
|
||||
ecr_backend = ecr_backends[region_name]
|
||||
return ecr_backend.create_repository(
|
||||
# RepositoryName is optional in CloudFormation, thus create a random
|
||||
# name if necessary
|
||||
repository_name=properties.get(
|
||||
'RepositoryName', 'ecrrepository{0}'.format(int(random() * 10 ** 6))),
|
||||
"RepositoryName", "ecrrepository{0}".format(int(random() * 10 ** 6))
|
||||
)
|
||||
)
|
||||
|
||||
@classmethod
|
||||
def update_from_cloudformation_json(cls, original_resource, new_resource_name, cloudformation_json, region_name):
|
||||
properties = cloudformation_json['Properties']
|
||||
def update_from_cloudformation_json(
|
||||
cls, original_resource, new_resource_name, cloudformation_json, region_name
|
||||
):
|
||||
properties = cloudformation_json["Properties"]
|
||||
|
||||
if original_resource.name != properties['RepositoryName']:
|
||||
if original_resource.name != properties["RepositoryName"]:
|
||||
ecr_backend = ecr_backends[region_name]
|
||||
ecr_backend.delete_cluster(original_resource.arn)
|
||||
return ecr_backend.create_repository(
|
||||
# RepositoryName is optional in CloudFormation, thus create a
|
||||
# random name if necessary
|
||||
repository_name=properties.get(
|
||||
'RepositoryName', 'RepositoryName{0}'.format(int(random() * 10 ** 6))),
|
||||
"RepositoryName",
|
||||
"RepositoryName{0}".format(int(random() * 10 ** 6)),
|
||||
)
|
||||
)
|
||||
else:
|
||||
# no-op when nothing changed between old and new resources
|
||||
|
|
@ -98,8 +105,9 @@ class Repository(BaseObject):
|
|||
|
||||
|
||||
class Image(BaseObject):
|
||||
|
||||
def __init__(self, tag, manifest, repository, digest=None, registry_id=DEFAULT_REGISTRY_ID):
|
||||
def __init__(
|
||||
self, tag, manifest, repository, digest=None, registry_id=DEFAULT_REGISTRY_ID
|
||||
):
|
||||
self.image_tag = tag
|
||||
self.image_tags = [tag] if tag is not None else []
|
||||
self.image_manifest = manifest
|
||||
|
|
@ -110,8 +118,10 @@ class Image(BaseObject):
|
|||
self.image_pushed_at = str(datetime.utcnow().isoformat())
|
||||
|
||||
def _create_digest(self):
|
||||
image_contents = 'docker_image{0}'.format(int(random() * 10 ** 6))
|
||||
self.image_digest = "sha256:%s" % hashlib.sha256(image_contents.encode('utf-8')).hexdigest()
|
||||
image_contents = "docker_image{0}".format(int(random() * 10 ** 6))
|
||||
self.image_digest = (
|
||||
"sha256:%s" % hashlib.sha256(image_contents.encode("utf-8")).hexdigest()
|
||||
)
|
||||
|
||||
def get_image_digest(self):
|
||||
if not self.image_digest:
|
||||
|
|
@ -135,54 +145,61 @@ class Image(BaseObject):
|
|||
@property
|
||||
def response_object(self):
|
||||
response_object = self.gen_response_object()
|
||||
response_object['imageId'] = {}
|
||||
response_object['imageId']['imageTag'] = self.image_tag
|
||||
response_object['imageId']['imageDigest'] = self.get_image_digest()
|
||||
response_object['imageManifest'] = self.image_manifest
|
||||
response_object['repositoryName'] = self.repository
|
||||
response_object['registryId'] = self.registry_id
|
||||
return {k: v for k, v in response_object.items() if v is not None and v != [None]}
|
||||
response_object["imageId"] = {}
|
||||
response_object["imageId"]["imageTag"] = self.image_tag
|
||||
response_object["imageId"]["imageDigest"] = self.get_image_digest()
|
||||
response_object["imageManifest"] = self.image_manifest
|
||||
response_object["repositoryName"] = self.repository
|
||||
response_object["registryId"] = self.registry_id
|
||||
return {
|
||||
k: v for k, v in response_object.items() if v is not None and v != [None]
|
||||
}
|
||||
|
||||
@property
|
||||
def response_list_object(self):
|
||||
response_object = self.gen_response_object()
|
||||
response_object['imageTag'] = self.image_tag
|
||||
response_object['imageDigest'] = "i don't know"
|
||||
return {k: v for k, v in response_object.items() if v is not None and v != [None]}
|
||||
response_object["imageTag"] = self.image_tag
|
||||
response_object["imageDigest"] = "i don't know"
|
||||
return {
|
||||
k: v for k, v in response_object.items() if v is not None and v != [None]
|
||||
}
|
||||
|
||||
@property
|
||||
def response_describe_object(self):
|
||||
response_object = self.gen_response_object()
|
||||
response_object['imageTags'] = self.image_tags
|
||||
response_object['imageDigest'] = self.get_image_digest()
|
||||
response_object['imageManifest'] = self.image_manifest
|
||||
response_object['repositoryName'] = self.repository
|
||||
response_object['registryId'] = self.registry_id
|
||||
response_object['imageSizeInBytes'] = self.image_size_in_bytes
|
||||
response_object['imagePushedAt'] = self.image_pushed_at
|
||||
response_object["imageTags"] = self.image_tags
|
||||
response_object["imageDigest"] = self.get_image_digest()
|
||||
response_object["imageManifest"] = self.image_manifest
|
||||
response_object["repositoryName"] = self.repository
|
||||
response_object["registryId"] = self.registry_id
|
||||
response_object["imageSizeInBytes"] = self.image_size_in_bytes
|
||||
response_object["imagePushedAt"] = self.image_pushed_at
|
||||
return {k: v for k, v in response_object.items() if v is not None and v != []}
|
||||
|
||||
@property
|
||||
def response_batch_get_image(self):
|
||||
response_object = {}
|
||||
response_object['imageId'] = {}
|
||||
response_object['imageId']['imageTag'] = self.image_tag
|
||||
response_object['imageId']['imageDigest'] = self.get_image_digest()
|
||||
response_object['imageManifest'] = self.image_manifest
|
||||
response_object['repositoryName'] = self.repository
|
||||
response_object['registryId'] = self.registry_id
|
||||
return {k: v for k, v in response_object.items() if v is not None and v != [None]}
|
||||
response_object["imageId"] = {}
|
||||
response_object["imageId"]["imageTag"] = self.image_tag
|
||||
response_object["imageId"]["imageDigest"] = self.get_image_digest()
|
||||
response_object["imageManifest"] = self.image_manifest
|
||||
response_object["repositoryName"] = self.repository
|
||||
response_object["registryId"] = self.registry_id
|
||||
return {
|
||||
k: v for k, v in response_object.items() if v is not None and v != [None]
|
||||
}
|
||||
|
||||
@property
|
||||
def response_batch_delete_image(self):
|
||||
response_object = {}
|
||||
response_object['imageDigest'] = self.get_image_digest()
|
||||
response_object['imageTag'] = self.image_tag
|
||||
return {k: v for k, v in response_object.items() if v is not None and v != [None]}
|
||||
response_object["imageDigest"] = self.get_image_digest()
|
||||
response_object["imageTag"] = self.image_tag
|
||||
return {
|
||||
k: v for k, v in response_object.items() if v is not None and v != [None]
|
||||
}
|
||||
|
||||
|
||||
class ECRBackend(BaseBackend):
|
||||
|
||||
def __init__(self):
|
||||
self.repositories = {}
|
||||
|
||||
|
|
@ -193,7 +210,9 @@ class ECRBackend(BaseBackend):
|
|||
if repository_names:
|
||||
for repository_name in repository_names:
|
||||
if repository_name not in self.repositories:
|
||||
raise RepositoryNotFoundException(repository_name, registry_id or DEFAULT_REGISTRY_ID)
|
||||
raise RepositoryNotFoundException(
|
||||
repository_name, registry_id or DEFAULT_REGISTRY_ID
|
||||
)
|
||||
|
||||
repositories = []
|
||||
for repository in self.repositories.values():
|
||||
|
|
@ -218,7 +237,9 @@ class ECRBackend(BaseBackend):
|
|||
if repository_name in self.repositories:
|
||||
return self.repositories.pop(repository_name)
|
||||
else:
|
||||
raise RepositoryNotFoundException(repository_name, registry_id or DEFAULT_REGISTRY_ID)
|
||||
raise RepositoryNotFoundException(
|
||||
repository_name, registry_id or DEFAULT_REGISTRY_ID
|
||||
)
|
||||
|
||||
def list_images(self, repository_name, registry_id=None):
|
||||
"""
|
||||
|
|
@ -235,7 +256,9 @@ class ECRBackend(BaseBackend):
|
|||
found = True
|
||||
|
||||
if not found:
|
||||
raise RepositoryNotFoundException(repository_name, registry_id or DEFAULT_REGISTRY_ID)
|
||||
raise RepositoryNotFoundException(
|
||||
repository_name, registry_id or DEFAULT_REGISTRY_ID
|
||||
)
|
||||
|
||||
images = []
|
||||
for image in repository.images:
|
||||
|
|
@ -247,26 +270,34 @@ class ECRBackend(BaseBackend):
|
|||
if repository_name in self.repositories:
|
||||
repository = self.repositories[repository_name]
|
||||
else:
|
||||
raise RepositoryNotFoundException(repository_name, registry_id or DEFAULT_REGISTRY_ID)
|
||||
raise RepositoryNotFoundException(
|
||||
repository_name, registry_id or DEFAULT_REGISTRY_ID
|
||||
)
|
||||
|
||||
if image_ids:
|
||||
response = set()
|
||||
for image_id in image_ids:
|
||||
found = False
|
||||
for image in repository.images:
|
||||
if (('imageDigest' in image_id and image.get_image_digest() == image_id['imageDigest']) or
|
||||
('imageTag' in image_id and image_id['imageTag'] in image.image_tags)):
|
||||
if (
|
||||
"imageDigest" in image_id
|
||||
and image.get_image_digest() == image_id["imageDigest"]
|
||||
) or (
|
||||
"imageTag" in image_id
|
||||
and image_id["imageTag"] in image.image_tags
|
||||
):
|
||||
found = True
|
||||
response.add(image)
|
||||
if not found:
|
||||
image_id_representation = "{imageDigest:'%s', imageTag:'%s'}" % (
|
||||
image_id.get('imageDigest', 'null'),
|
||||
image_id.get('imageTag', 'null'),
|
||||
image_id.get("imageDigest", "null"),
|
||||
image_id.get("imageTag", "null"),
|
||||
)
|
||||
raise ImageNotFoundException(
|
||||
image_id=image_id_representation,
|
||||
repository_name=repository_name,
|
||||
registry_id=registry_id or DEFAULT_REGISTRY_ID)
|
||||
registry_id=registry_id or DEFAULT_REGISTRY_ID,
|
||||
)
|
||||
|
||||
else:
|
||||
response = []
|
||||
|
|
@ -281,7 +312,12 @@ class ECRBackend(BaseBackend):
|
|||
else:
|
||||
raise Exception("{0} is not a repository".format(repository_name))
|
||||
|
||||
existing_images = list(filter(lambda x: x.response_object['imageManifest'] == image_manifest, repository.images))
|
||||
existing_images = list(
|
||||
filter(
|
||||
lambda x: x.response_object["imageManifest"] == image_manifest,
|
||||
repository.images,
|
||||
)
|
||||
)
|
||||
if not existing_images:
|
||||
# this image is not in ECR yet
|
||||
image = Image(image_tag, image_manifest, repository_name)
|
||||
|
|
@ -292,36 +328,47 @@ class ECRBackend(BaseBackend):
|
|||
existing_images[0].update_tag(image_tag)
|
||||
return existing_images[0]
|
||||
|
||||
def batch_get_image(self, repository_name, registry_id=None, image_ids=None, accepted_media_types=None):
|
||||
def batch_get_image(
|
||||
self,
|
||||
repository_name,
|
||||
registry_id=None,
|
||||
image_ids=None,
|
||||
accepted_media_types=None,
|
||||
):
|
||||
if repository_name in self.repositories:
|
||||
repository = self.repositories[repository_name]
|
||||
else:
|
||||
raise RepositoryNotFoundException(repository_name, registry_id or DEFAULT_REGISTRY_ID)
|
||||
raise RepositoryNotFoundException(
|
||||
repository_name, registry_id or DEFAULT_REGISTRY_ID
|
||||
)
|
||||
|
||||
if not image_ids:
|
||||
raise ParamValidationError(msg='Missing required parameter in input: "imageIds"')
|
||||
raise ParamValidationError(
|
||||
msg='Missing required parameter in input: "imageIds"'
|
||||
)
|
||||
|
||||
response = {
|
||||
'images': [],
|
||||
'failures': [],
|
||||
}
|
||||
response = {"images": [], "failures": []}
|
||||
|
||||
for image_id in image_ids:
|
||||
found = False
|
||||
for image in repository.images:
|
||||
if (('imageDigest' in image_id and image.get_image_digest() == image_id['imageDigest']) or
|
||||
('imageTag' in image_id and image.image_tag == image_id['imageTag'])):
|
||||
if (
|
||||
"imageDigest" in image_id
|
||||
and image.get_image_digest() == image_id["imageDigest"]
|
||||
) or (
|
||||
"imageTag" in image_id and image.image_tag == image_id["imageTag"]
|
||||
):
|
||||
found = True
|
||||
response['images'].append(image.response_batch_get_image)
|
||||
response["images"].append(image.response_batch_get_image)
|
||||
|
||||
if not found:
|
||||
response['failures'].append({
|
||||
'imageId': {
|
||||
'imageTag': image_id.get('imageTag', 'null')
|
||||
},
|
||||
'failureCode': 'ImageNotFound',
|
||||
'failureReason': 'Requested image not found'
|
||||
})
|
||||
response["failures"].append(
|
||||
{
|
||||
"imageId": {"imageTag": image_id.get("imageTag", "null")},
|
||||
"failureCode": "ImageNotFound",
|
||||
"failureReason": "Requested image not found",
|
||||
}
|
||||
)
|
||||
|
||||
return response
|
||||
|
||||
|
|
@ -338,10 +385,7 @@ class ECRBackend(BaseBackend):
|
|||
msg='Missing required parameter in input: "imageIds"'
|
||||
)
|
||||
|
||||
response = {
|
||||
"imageIds": [],
|
||||
"failures": []
|
||||
}
|
||||
response = {"imageIds": [], "failures": []}
|
||||
|
||||
for image_id in image_ids:
|
||||
image_found = False
|
||||
|
|
@ -377,8 +421,8 @@ class ECRBackend(BaseBackend):
|
|||
# Search by matching both digest and tag
|
||||
if "imageDigest" in image_id and "imageTag" in image_id:
|
||||
if (
|
||||
image_id["imageDigest"] == image.get_image_digest() and
|
||||
image_id["imageTag"] in image.image_tags
|
||||
image_id["imageDigest"] == image.get_image_digest()
|
||||
and image_id["imageTag"] in image.image_tags
|
||||
):
|
||||
image_found = True
|
||||
for image_tag in reversed(image.image_tags):
|
||||
|
|
@ -390,7 +434,10 @@ class ECRBackend(BaseBackend):
|
|||
del repository.images[num]
|
||||
|
||||
# Search by matching digest
|
||||
elif "imageDigest" in image_id and image.get_image_digest() == image_id["imageDigest"]:
|
||||
elif (
|
||||
"imageDigest" in image_id
|
||||
and image.get_image_digest() == image_id["imageDigest"]
|
||||
):
|
||||
image_found = True
|
||||
for image_tag in reversed(image.image_tags):
|
||||
repository.images[num].image_tag = image_tag
|
||||
|
|
@ -399,7 +446,9 @@ class ECRBackend(BaseBackend):
|
|||
del repository.images[num]
|
||||
|
||||
# Search by matching tag
|
||||
elif "imageTag" in image_id and image_id["imageTag"] in image.image_tags:
|
||||
elif (
|
||||
"imageTag" in image_id and image_id["imageTag"] in image.image_tags
|
||||
):
|
||||
image_found = True
|
||||
repository.images[num].image_tag = image_id["imageTag"]
|
||||
response["imageIds"].append(image.response_batch_delete_image)
|
||||
|
|
@ -416,10 +465,14 @@ class ECRBackend(BaseBackend):
|
|||
}
|
||||
|
||||
if "imageDigest" in image_id:
|
||||
failure_response["imageId"]["imageDigest"] = image_id.get("imageDigest", "null")
|
||||
failure_response["imageId"]["imageDigest"] = image_id.get(
|
||||
"imageDigest", "null"
|
||||
)
|
||||
|
||||
if "imageTag" in image_id:
|
||||
failure_response["imageId"]["imageTag"] = image_id.get("imageTag", "null")
|
||||
failure_response["imageId"]["imageTag"] = image_id.get(
|
||||
"imageTag", "null"
|
||||
)
|
||||
|
||||
response["failures"].append(failure_response)
|
||||
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue