mirror of
https://github.com/ansible-collections/community.docker.git
synced 2025-12-17 04:18:42 +00:00
Add docker_image_tag module (#730)
* Add docker_image_tag module. * Add check mode tests. * Improve and test image ID/digest handling. * Adjust more tests.
This commit is contained in:
parent
20e78a92e0
commit
66b341aa9e
@ -66,6 +66,7 @@ If you use the Ansible package and do not update collections independently, use
|
|||||||
- community.docker.docker_image_load: load Docker images from archives
|
- community.docker.docker_image_load: load Docker images from archives
|
||||||
- community.docker.docker_image_pull: pull Docker images from registries
|
- community.docker.docker_image_pull: pull Docker images from registries
|
||||||
- community.docker.docker_image_push: push Docker images to registries
|
- community.docker.docker_image_push: push Docker images to registries
|
||||||
|
- community.docker.docker_image_tag: tag Docker images with new names and/or tags
|
||||||
- community.docker.docker_login: log in and out to/from registries
|
- community.docker.docker_login: log in and out to/from registries
|
||||||
- community.docker.docker_network: manage Docker networks
|
- community.docker.docker_network: manage Docker networks
|
||||||
- community.docker.docker_network_info: retrieve information on Docker networks
|
- community.docker.docker_network_info: retrieve information on Docker networks
|
||||||
|
|||||||
@ -18,6 +18,7 @@ action_groups:
|
|||||||
- docker_image_load
|
- docker_image_load
|
||||||
- docker_image_pull
|
- docker_image_pull
|
||||||
- docker_image_push
|
- docker_image_push
|
||||||
|
- docker_image_tag
|
||||||
- docker_login
|
- docker_login
|
||||||
- docker_network
|
- docker_network
|
||||||
- docker_network_info
|
- docker_network_info
|
||||||
|
|||||||
@ -254,7 +254,7 @@ seealso:
|
|||||||
- module: community.docker.docker_image_info
|
- module: community.docker.docker_image_info
|
||||||
- module: community.docker.docker_image_load
|
- module: community.docker.docker_image_load
|
||||||
- module: community.docker.docker_image_pull
|
- module: community.docker.docker_image_pull
|
||||||
- module: community.docker.docker_image_push
|
- module: community.docker.docker_image_tag
|
||||||
'''
|
'''
|
||||||
|
|
||||||
EXAMPLES = '''
|
EXAMPLES = '''
|
||||||
|
|||||||
@ -46,6 +46,9 @@ requirements:
|
|||||||
|
|
||||||
author:
|
author:
|
||||||
- Felix Fontein (@felixfontein)
|
- Felix Fontein (@felixfontein)
|
||||||
|
|
||||||
|
seealso:
|
||||||
|
- module: community.docker.docker_image_tag
|
||||||
'''
|
'''
|
||||||
|
|
||||||
EXAMPLES = '''
|
EXAMPLES = '''
|
||||||
|
|||||||
272
plugins/modules/docker_image_tag.py
Normal file
272
plugins/modules/docker_image_tag.py
Normal file
@ -0,0 +1,272 @@
|
|||||||
|
#!/usr/bin/python
|
||||||
|
#
|
||||||
|
# Copyright (c) 2023, Felix Fontein <felix@fontein.de>
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
from __future__ import absolute_import, division, print_function
|
||||||
|
__metaclass__ = type
|
||||||
|
|
||||||
|
|
||||||
|
DOCUMENTATION = '''
|
||||||
|
---
|
||||||
|
module: docker_image_tag
|
||||||
|
|
||||||
|
short_description: Tag Docker images with new names and/or tags
|
||||||
|
|
||||||
|
version_added: 3.6.0
|
||||||
|
|
||||||
|
description:
|
||||||
|
- This module allows to tag Docker images with new names and/or tags.
|
||||||
|
|
||||||
|
extends_documentation_fragment:
|
||||||
|
- community.docker.docker.api_documentation
|
||||||
|
- community.docker.attributes
|
||||||
|
- community.docker.attributes.actiongroup_docker
|
||||||
|
|
||||||
|
attributes:
|
||||||
|
check_mode:
|
||||||
|
support: full
|
||||||
|
diff_mode:
|
||||||
|
support: full
|
||||||
|
|
||||||
|
options:
|
||||||
|
name:
|
||||||
|
description:
|
||||||
|
- "Image name. Name format will be one of: C(name), C(repository/name), C(registry_server:port/name).
|
||||||
|
When pushing or pulling an image the name can optionally include the tag by appending C(:tag_name)."
|
||||||
|
- Note that image IDs (hashes) can also be used.
|
||||||
|
type: str
|
||||||
|
required: true
|
||||||
|
tag:
|
||||||
|
description:
|
||||||
|
- Tag for the image name O(name) that is to be tagged.
|
||||||
|
- If O(name)'s format is C(name:tag), then the tag value from O(name) will take precedence.
|
||||||
|
type: str
|
||||||
|
default: latest
|
||||||
|
repository:
|
||||||
|
description:
|
||||||
|
- List of new image names to tag the image as.
|
||||||
|
- Expects format C(repository:tag). If no tag is provided, will use the value of the O(tag) parameter if present, or V(latest).
|
||||||
|
type: list
|
||||||
|
elements: str
|
||||||
|
required: true
|
||||||
|
existing_images:
|
||||||
|
description:
|
||||||
|
- Defines the behavior if the image to be tagged already exists and is another image than the one identified by O(name) and O(tag).
|
||||||
|
- If set to V(keep), the tagged image is kept.
|
||||||
|
- If set to V(overwrite), the tagged image is overwritten by the specified one.
|
||||||
|
type: str
|
||||||
|
choices:
|
||||||
|
- keep
|
||||||
|
- overwrite
|
||||||
|
default: overwrite
|
||||||
|
|
||||||
|
requirements:
|
||||||
|
- "Docker API >= 1.25"
|
||||||
|
|
||||||
|
author:
|
||||||
|
- Felix Fontein (@felixfontein)
|
||||||
|
|
||||||
|
seealso:
|
||||||
|
- module: community.docker.docker_image_push
|
||||||
|
'''
|
||||||
|
|
||||||
|
EXAMPLES = '''
|
||||||
|
- name: Tag Python 3.12 image with two new names
|
||||||
|
community.docker.docker_image:
|
||||||
|
name: python:3.12
|
||||||
|
repository:
|
||||||
|
- python-3:3.12
|
||||||
|
- local-registry:5000/python-3/3.12:latest
|
||||||
|
'''
|
||||||
|
|
||||||
|
RETURN = '''
|
||||||
|
image:
|
||||||
|
description: Image inspection results for the affected image.
|
||||||
|
returned: success
|
||||||
|
type: dict
|
||||||
|
sample: {}
|
||||||
|
tagged_images:
|
||||||
|
description:
|
||||||
|
- A list of images that got tagged.
|
||||||
|
returned: success
|
||||||
|
type: list
|
||||||
|
elements: str
|
||||||
|
sample:
|
||||||
|
- python-3:3.12
|
||||||
|
'''
|
||||||
|
|
||||||
|
import traceback
|
||||||
|
|
||||||
|
from ansible.module_utils.common.text.converters import to_native
|
||||||
|
from ansible.module_utils.common.text.formatters import human_to_bytes
|
||||||
|
|
||||||
|
from ansible_collections.community.docker.plugins.module_utils.common_api import (
|
||||||
|
AnsibleDockerClient,
|
||||||
|
RequestException,
|
||||||
|
)
|
||||||
|
|
||||||
|
from ansible_collections.community.docker.plugins.module_utils.util import (
|
||||||
|
DockerBaseClass,
|
||||||
|
is_image_name_id,
|
||||||
|
is_valid_tag,
|
||||||
|
)
|
||||||
|
|
||||||
|
from ansible_collections.community.docker.plugins.module_utils._api.errors import DockerException
|
||||||
|
from ansible_collections.community.docker.plugins.module_utils._api.utils.utils import (
|
||||||
|
parse_repository_tag,
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def convert_to_bytes(value, module, name, unlimited_value=None):
|
||||||
|
if value is None:
|
||||||
|
return value
|
||||||
|
try:
|
||||||
|
if unlimited_value is not None and value in ('unlimited', str(unlimited_value)):
|
||||||
|
return unlimited_value
|
||||||
|
return human_to_bytes(value)
|
||||||
|
except ValueError as exc:
|
||||||
|
module.fail_json(msg='Failed to convert %s to bytes: %s' % (name, to_native(exc)))
|
||||||
|
|
||||||
|
|
||||||
|
def image_info(name, tag, image):
|
||||||
|
result = dict(name=name, tag=tag)
|
||||||
|
if image:
|
||||||
|
result['id'] = image['Id']
|
||||||
|
else:
|
||||||
|
result['exists'] = False
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
|
class ImageTagger(DockerBaseClass):
|
||||||
|
def __init__(self, client):
|
||||||
|
super(ImageTagger, self).__init__()
|
||||||
|
|
||||||
|
self.client = client
|
||||||
|
parameters = self.client.module.params
|
||||||
|
self.check_mode = self.client.check_mode
|
||||||
|
|
||||||
|
self.name = parameters['name']
|
||||||
|
self.tag = parameters['tag']
|
||||||
|
if not is_valid_tag(self.tag, allow_empty=True):
|
||||||
|
self.fail('"{0}" is not a valid docker tag'.format(self.tag))
|
||||||
|
|
||||||
|
# If name contains a tag, it takes precedence over tag parameter.
|
||||||
|
if not is_image_name_id(self.name):
|
||||||
|
repo, repo_tag = parse_repository_tag(self.name)
|
||||||
|
if repo_tag:
|
||||||
|
self.name = repo
|
||||||
|
self.tag = repo_tag
|
||||||
|
|
||||||
|
self.keep_existing_images = parameters['existing_images'] == 'keep'
|
||||||
|
|
||||||
|
# Make sure names in repository are valid images, and add tag if needed
|
||||||
|
self.repositories = []
|
||||||
|
for i, repository in enumerate(parameters['repository']):
|
||||||
|
if is_image_name_id(repository):
|
||||||
|
self.fail("repository[%d] must not be an image ID; got: %s" % (i + 1, repository))
|
||||||
|
repo, repo_tag = parse_repository_tag(repository)
|
||||||
|
if not repo_tag:
|
||||||
|
repo_tag = parameters['tag']
|
||||||
|
elif not is_valid_tag(repo_tag, allow_empty=False):
|
||||||
|
self.fail("repository[%d] must not have a digest; got: %s" % (i + 1, repository))
|
||||||
|
self.repositories.append((repo, repo_tag))
|
||||||
|
|
||||||
|
def fail(self, msg):
|
||||||
|
self.client.fail(msg)
|
||||||
|
|
||||||
|
def tag_image(self, image, name, tag):
|
||||||
|
tagged_image = self.client.find_image(name=name, tag=tag)
|
||||||
|
if tagged_image:
|
||||||
|
# Idempotency checks
|
||||||
|
if tagged_image['Id'] == image['Id']:
|
||||||
|
return (
|
||||||
|
False,
|
||||||
|
"target image already exists (%s) and is as expected" % tagged_image['Id'],
|
||||||
|
tagged_image,
|
||||||
|
)
|
||||||
|
if self.keep_existing_images:
|
||||||
|
return (
|
||||||
|
False,
|
||||||
|
"target image already exists (%s) and is not as expected, but kept" % tagged_image['Id'],
|
||||||
|
tagged_image,
|
||||||
|
)
|
||||||
|
msg = "target image existed (%s) and was not as expected" % tagged_image['Id']
|
||||||
|
else:
|
||||||
|
msg = "target image did not exist"
|
||||||
|
|
||||||
|
if not self.check_mode:
|
||||||
|
try:
|
||||||
|
params = {
|
||||||
|
'tag': tag,
|
||||||
|
'repo': name,
|
||||||
|
'force': True,
|
||||||
|
}
|
||||||
|
res = self.client._post(self.client._url('/images/{0}/tag', image['Id']), params=params)
|
||||||
|
self.client._raise_for_status(res)
|
||||||
|
if res.status_code != 201:
|
||||||
|
raise Exception("Tag operation failed.")
|
||||||
|
except Exception as exc:
|
||||||
|
self.fail("Error: failed to tag image as %s:%s - %s" % (name, tag, to_native(exc)))
|
||||||
|
|
||||||
|
return True, msg, tagged_image
|
||||||
|
|
||||||
|
def tag_images(self):
|
||||||
|
if is_image_name_id(self.name):
|
||||||
|
image = self.client.find_image_by_id(self.name, accept_missing_image=False)
|
||||||
|
else:
|
||||||
|
image = self.client.find_image(name=self.name, tag=self.tag)
|
||||||
|
if not image:
|
||||||
|
self.fail("Cannot find image %s:%s" % (self.name, self.tag))
|
||||||
|
|
||||||
|
before = []
|
||||||
|
after = []
|
||||||
|
tagged_images = []
|
||||||
|
results = dict(
|
||||||
|
changed=False,
|
||||||
|
actions=[],
|
||||||
|
image=image,
|
||||||
|
tagged_images=tagged_images,
|
||||||
|
diff=dict(before=dict(images=before), after=dict(images=after)),
|
||||||
|
)
|
||||||
|
for repository, tag in self.repositories:
|
||||||
|
tagged, msg, old_image = self.tag_image(image, repository, tag)
|
||||||
|
before.append(image_info(repository, tag, old_image))
|
||||||
|
after.append(image_info(repository, tag, image if tagged else old_image))
|
||||||
|
if tagged:
|
||||||
|
results['changed'] = True
|
||||||
|
results['actions'].append('Tagged image %s as %s:%s: %s' % (image['Id'], repository, tag, msg))
|
||||||
|
tagged_images.append('%s:%s' % (repository, tag))
|
||||||
|
else:
|
||||||
|
results['actions'].append('Not tagged image %s as %s:%s: %s' % (image['Id'], repository, tag, msg))
|
||||||
|
|
||||||
|
return results
|
||||||
|
|
||||||
|
|
||||||
|
def main():
|
||||||
|
argument_spec = dict(
|
||||||
|
name=dict(type='str', required=True),
|
||||||
|
tag=dict(type='str', default='latest'),
|
||||||
|
repository=dict(type='list', elements='str', required=True),
|
||||||
|
existing_images=dict(type='str', choices=['keep', 'overwrite'], default='overwrite'),
|
||||||
|
)
|
||||||
|
|
||||||
|
client = AnsibleDockerClient(
|
||||||
|
argument_spec=argument_spec,
|
||||||
|
supports_check_mode=True,
|
||||||
|
)
|
||||||
|
|
||||||
|
try:
|
||||||
|
results = ImageTagger(client).tag_images()
|
||||||
|
client.module.exit_json(**results)
|
||||||
|
except DockerException as e:
|
||||||
|
client.fail('An unexpected Docker error occurred: {0}'.format(to_native(e)), exception=traceback.format_exc())
|
||||||
|
except RequestException as e:
|
||||||
|
client.fail(
|
||||||
|
'An unexpected requests error occurred when trying to talk to the Docker daemon: {0}'.format(to_native(e)),
|
||||||
|
exception=traceback.format_exc())
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
main()
|
||||||
@ -24,10 +24,8 @@
|
|||||||
- "result.images|length == 0"
|
- "result.images|length == 0"
|
||||||
|
|
||||||
- name: Make sure images are there
|
- name: Make sure images are there
|
||||||
docker_image:
|
docker_image_pull:
|
||||||
name: "{{ item }}"
|
name: "{{ item }}"
|
||||||
source: pull
|
|
||||||
state: present
|
|
||||||
loop:
|
loop:
|
||||||
- "{{ docker_test_image_hello_world }}"
|
- "{{ docker_test_image_hello_world }}"
|
||||||
- "{{ docker_test_image_alpine }}"
|
- "{{ docker_test_image_alpine }}"
|
||||||
|
|||||||
@ -10,9 +10,8 @@
|
|||||||
- "{{ docker_test_image_alpine }}"
|
- "{{ docker_test_image_alpine }}"
|
||||||
|
|
||||||
- name: Make sure images are there
|
- name: Make sure images are there
|
||||||
docker_image:
|
docker_image_pull:
|
||||||
name: "{{ item }}"
|
name: "{{ item }}"
|
||||||
source: pull
|
|
||||||
register: images
|
register: images
|
||||||
loop: "{{ image_names }}"
|
loop: "{{ image_names }}"
|
||||||
|
|
||||||
|
|||||||
@ -29,12 +29,15 @@
|
|||||||
name: "{{ docker_test_image_hello_world }}"
|
name: "{{ docker_test_image_hello_world }}"
|
||||||
diff: true
|
diff: true
|
||||||
|
|
||||||
- name: Push image to test registry
|
- name: Tag image
|
||||||
docker_image:
|
docker_image_tag:
|
||||||
name: "{{ docker_test_image_hello_world }}"
|
name: "{{ docker_test_image_hello_world }}"
|
||||||
repository: "{{ hello_world_image_base }}:latest"
|
repository:
|
||||||
push: true
|
- "{{ hello_world_image_base }}:latest"
|
||||||
source: local
|
|
||||||
|
- name: Push image to test registry
|
||||||
|
docker_image_push:
|
||||||
|
name: "{{ hello_world_image_base }}:latest"
|
||||||
|
|
||||||
- name: Get facts of local image
|
- name: Get facts of local image
|
||||||
docker_image_info:
|
docker_image_info:
|
||||||
|
|||||||
@ -25,14 +25,11 @@
|
|||||||
inames: "{{ inames + [image_name_base ~ ':' ~ image_tag, image_name_base2 ~ ':' ~ image_tag] }}"
|
inames: "{{ inames + [image_name_base ~ ':' ~ image_tag, image_name_base2 ~ ':' ~ image_tag] }}"
|
||||||
|
|
||||||
- name: Tag first image
|
- name: Tag first image
|
||||||
docker_image:
|
docker_image_tag:
|
||||||
name: "{{ docker_test_image_hello_world }}"
|
name: "{{ docker_test_image_hello_world }}"
|
||||||
repository: "{{ item }}"
|
repository:
|
||||||
source: local
|
- "{{ image_name_base }}:{{ image_tag }}"
|
||||||
force_tag: true
|
- "{{ image_name_base2 }}:{{ image_tag }}"
|
||||||
loop:
|
|
||||||
- "{{ image_name_base }}:{{ image_tag }}"
|
|
||||||
- "{{ image_name_base2 }}:{{ image_tag }}"
|
|
||||||
|
|
||||||
- name: Push first image
|
- name: Push first image
|
||||||
docker_image_push:
|
docker_image_push:
|
||||||
@ -45,11 +42,10 @@
|
|||||||
register: push_2
|
register: push_2
|
||||||
|
|
||||||
- name: Tag second image
|
- name: Tag second image
|
||||||
docker_image:
|
docker_image_tag:
|
||||||
name: "{{ docker_test_image_alpine }}"
|
name: "{{ docker_test_image_alpine }}"
|
||||||
repository: "{{ image_name_base }}:{{ image_tag }}"
|
repository:
|
||||||
source: local
|
- "{{ image_name_base }}:{{ image_tag }}"
|
||||||
force_tag: true
|
|
||||||
|
|
||||||
- name: Push second image with same name
|
- name: Push second image with same name
|
||||||
docker_image_push:
|
docker_image_push:
|
||||||
|
|||||||
6
tests/integration/targets/docker_image_tag/aliases
Normal file
6
tests/integration/targets/docker_image_tag/aliases
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
# Copyright (c) Ansible Project
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
azp/4
|
||||||
|
destructive
|
||||||
8
tests/integration/targets/docker_image_tag/meta/main.yml
Normal file
8
tests/integration/targets/docker_image_tag/meta/main.yml
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
---
|
||||||
|
# Copyright (c) Ansible Project
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
dependencies:
|
||||||
|
- setup_docker
|
||||||
|
- setup_docker_python_deps
|
||||||
406
tests/integration/targets/docker_image_tag/tasks/main.yml
Normal file
406
tests/integration/targets/docker_image_tag/tasks/main.yml
Normal file
@ -0,0 +1,406 @@
|
|||||||
|
---
|
||||||
|
# Copyright (c) Ansible Project
|
||||||
|
# GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt)
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
|
||||||
|
####################################################################
|
||||||
|
# WARNING: These are designed specifically for Ansible tests #
|
||||||
|
# and should not be used as examples of how to write Ansible roles #
|
||||||
|
####################################################################
|
||||||
|
|
||||||
|
- block:
|
||||||
|
- name: Pick image prefix
|
||||||
|
set_fact:
|
||||||
|
iname_prefix: "{{ 'ansible-docker-test-%0x' % ((2**32) | random) }}"
|
||||||
|
|
||||||
|
- name: Define image names
|
||||||
|
set_fact:
|
||||||
|
image_1: "{{ docker_test_image_hello_world }}"
|
||||||
|
image_2: "{{ docker_test_image_alpine }}"
|
||||||
|
image_3: "{{ docker_test_image_digest_base }}@sha256:{{ docker_test_image_digest_v1 }}"
|
||||||
|
image_names:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
- "{{ iname_prefix }}-tagged-2:baz"
|
||||||
|
|
||||||
|
- name: Make sure images we work with are there
|
||||||
|
docker_image_pull:
|
||||||
|
name: "{{ item }}"
|
||||||
|
loop:
|
||||||
|
- "{{ image_1 }}"
|
||||||
|
- "{{ image_2 }}"
|
||||||
|
- "{{ image_3 }}"
|
||||||
|
register: pulled_images
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- name: Remove tagged images
|
||||||
|
docker_image:
|
||||||
|
name: "{{ item }}"
|
||||||
|
source: local
|
||||||
|
state: absent
|
||||||
|
loop: "{{ image_names }}"
|
||||||
|
|
||||||
|
- name: Tag image 1 (check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
register: tag_1_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 1
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
register: tag_1
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- name: Fetch image infos
|
||||||
|
docker_image_info:
|
||||||
|
name: "{{ image_names }}"
|
||||||
|
register: info_1
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_1 is changed
|
||||||
|
- tag_1.diff.before.images | length == 2
|
||||||
|
- tag_1.diff.before.images[0] != tag_1.diff.after.images[0]
|
||||||
|
- tag_1.diff.before.images[1] != tag_1.diff.after.images[1]
|
||||||
|
- tag_1.diff.before.images[0].exists is false
|
||||||
|
- tag_1.diff.before.images[1].exists is false
|
||||||
|
- tag_1.diff.after.images[0].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_1.diff.after.images[1].id == pulled_images.results[0].image.Id
|
||||||
|
- info_1.images | length == 2
|
||||||
|
- info_1.images[0].Id == pulled_images.results[0].image.Id
|
||||||
|
- info_1.images[1].Id == pulled_images.results[0].image.Id
|
||||||
|
- tag_1_check == tag_1
|
||||||
|
|
||||||
|
- name: Tag image 1 (idempotent, check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
register: tag_2_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 1 (idempotent)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
register: tag_2
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_2 is not changed
|
||||||
|
- tag_2.diff.before == tag_2.diff.after
|
||||||
|
- tag_2.diff.before.images | length == 2
|
||||||
|
- tag_2_check == tag_2
|
||||||
|
|
||||||
|
- name: Tag image 1 (idempotent, different input format, check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
tag: foo
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1"
|
||||||
|
register: tag_3_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 1 (idempotent, different input format)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
tag: foo
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1"
|
||||||
|
register: tag_3
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_3 is not changed
|
||||||
|
- tag_3.diff.before == tag_3.diff.after
|
||||||
|
- tag_3.diff.before.images | length == 2
|
||||||
|
- tag_3_check == tag_3
|
||||||
|
|
||||||
|
- name: Tag image 1 (one more, check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-2:baz"
|
||||||
|
register: tag_4_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 1 (one more)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_1 }}"
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-2:baz"
|
||||||
|
register: tag_4
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- name: Fetch image infos
|
||||||
|
docker_image_info:
|
||||||
|
name: "{{ image_names }}"
|
||||||
|
register: info_4
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_4 is changed
|
||||||
|
- tag_4.diff.before.images | length == 3
|
||||||
|
- tag_4.diff.before.images[0] == tag_4.diff.after.images[0]
|
||||||
|
- tag_4.diff.before.images[1] == tag_4.diff.after.images[1]
|
||||||
|
- tag_4.diff.before.images[2] != tag_4.diff.after.images[2]
|
||||||
|
- tag_4.diff.before.images[2].exists is false
|
||||||
|
- tag_4.diff.after.images[0].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_4.diff.after.images[1].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_4.diff.after.images[2].id == pulled_images.results[0].image.Id
|
||||||
|
- info_4.images | length == 3
|
||||||
|
- info_4.images[0].Id == pulled_images.results[0].image.Id
|
||||||
|
- info_4.images[1].Id == pulled_images.results[0].image.Id
|
||||||
|
- info_4.images[2].Id == pulled_images.results[0].image.Id
|
||||||
|
- tag_4_check == tag_4
|
||||||
|
|
||||||
|
- name: Tag image 2 (only change missing one, check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: keep
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_5_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 2 (only change missing one)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: keep
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_5
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- name: Fetch image infos
|
||||||
|
docker_image_info:
|
||||||
|
name: "{{ image_names }}"
|
||||||
|
register: info_5
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_5 is changed
|
||||||
|
- tag_5.diff.before.images | length == 3
|
||||||
|
- tag_5.diff.before.images[0] == tag_5.diff.after.images[0]
|
||||||
|
- tag_5.diff.before.images[1] == tag_5.diff.after.images[1]
|
||||||
|
- tag_5.diff.before.images[2] != tag_5.diff.after.images[2]
|
||||||
|
- tag_5.diff.before.images[2].exists is false
|
||||||
|
- tag_5.diff.after.images[0].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_5.diff.after.images[1].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_5.diff.after.images[2].id == pulled_images.results[1].image.Id
|
||||||
|
- info_5.images | length == 4
|
||||||
|
- info_5.images[0].Id == pulled_images.results[0].image.Id
|
||||||
|
- info_5.images[1].Id == pulled_images.results[0].image.Id
|
||||||
|
- info_5.images[2].Id == pulled_images.results[1].image.Id
|
||||||
|
- info_5.images[3].Id == pulled_images.results[0].image.Id
|
||||||
|
- tag_5_check == tag_5
|
||||||
|
|
||||||
|
- name: Tag image 2 (idempotent, check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: keep
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_6_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 2 (idempotent)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: keep
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_6
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_6 is not changed
|
||||||
|
- tag_6.diff.before == tag_6.diff.after
|
||||||
|
- tag_6.diff.before.images | length == 3
|
||||||
|
- tag_6_check == tag_6
|
||||||
|
|
||||||
|
- name: Tag image 2 (only change wrong ones, check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_7_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 2 (only change wrong ones)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_7
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- name: Fetch image infos
|
||||||
|
docker_image_info:
|
||||||
|
name: "{{ image_names }}"
|
||||||
|
register: info_7
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_7 is changed
|
||||||
|
- tag_7.diff.before.images | length == 3
|
||||||
|
- tag_7.diff.before.images[0] != tag_7.diff.after.images[0]
|
||||||
|
- tag_7.diff.before.images[1] != tag_7.diff.after.images[1]
|
||||||
|
- tag_7.diff.before.images[2] == tag_7.diff.after.images[2]
|
||||||
|
- tag_7.diff.before.images[0].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_7.diff.before.images[1].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_7.diff.after.images[0].id == pulled_images.results[1].image.Id
|
||||||
|
- tag_7.diff.after.images[1].id == pulled_images.results[1].image.Id
|
||||||
|
- tag_7.diff.after.images[2].id == pulled_images.results[1].image.Id
|
||||||
|
- info_7.images | length == 4
|
||||||
|
- info_7.images[0].Id == pulled_images.results[1].image.Id
|
||||||
|
- info_7.images[1].Id == pulled_images.results[1].image.Id
|
||||||
|
- info_7.images[2].Id == pulled_images.results[1].image.Id
|
||||||
|
- info_7.images[3].Id == pulled_images.results[0].image.Id
|
||||||
|
- tag_7_check == tag_7
|
||||||
|
|
||||||
|
- name: Tag image 2 (idempotent, check mode)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_8_check
|
||||||
|
diff: true
|
||||||
|
check_mode: true
|
||||||
|
|
||||||
|
- name: Tag image 2 (idempotent)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_2 }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:latest"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
- "{{ iname_prefix }}-tagged-1:bar"
|
||||||
|
register: tag_8
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_8 is not changed
|
||||||
|
- tag_8.diff.before == tag_8.diff.after
|
||||||
|
- tag_8.diff.before.images | length == 3
|
||||||
|
- tag_8_check == tag_8
|
||||||
|
|
||||||
|
- name: Tag image 3 (source image has digest)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_3 }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-2:baz"
|
||||||
|
register: tag_9
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_9 is changed
|
||||||
|
- tag_9.diff.before.images | length == 1
|
||||||
|
- tag_9.diff.before.images[0].id == pulled_images.results[0].image.Id
|
||||||
|
- tag_9.diff.after.images[0].id == pulled_images.results[2].image.Id
|
||||||
|
|
||||||
|
- name: Tag image 3 (source image is ID)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ pulled_images.results[2].image.Id }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-1:foo"
|
||||||
|
register: tag_10
|
||||||
|
diff: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_10 is changed
|
||||||
|
- tag_10.diff.before.images | length == 1
|
||||||
|
- tag_10.diff.before.images[0].id == pulled_images.results[1].image.Id
|
||||||
|
- tag_10.diff.after.images[0].id == pulled_images.results[2].image.Id
|
||||||
|
|
||||||
|
- name: Tag image 3 (fail because of digest)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_3 }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "{{ iname_prefix }}-tagged-2@sha256:{{ docker_test_image_digest_v1 }}"
|
||||||
|
register: tag_11
|
||||||
|
ignore_errors: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_11 is failed
|
||||||
|
- >-
|
||||||
|
tag_11.msg == 'repository[1] must not have a digest; got: ' ~ iname_prefix ~ '-tagged-2@sha256:' ~ docker_test_image_digest_v1
|
||||||
|
|
||||||
|
- name: Tag image 3 (fail because of image ID)
|
||||||
|
docker_image_tag:
|
||||||
|
name: "{{ image_3 }}"
|
||||||
|
existing_images: overwrite
|
||||||
|
repository:
|
||||||
|
- "sha256:{{ docker_test_image_digest_v1 }}"
|
||||||
|
register: tag_12
|
||||||
|
ignore_errors: true
|
||||||
|
|
||||||
|
- assert:
|
||||||
|
that:
|
||||||
|
- tag_12 is failed
|
||||||
|
- >-
|
||||||
|
tag_12.msg == 'repository[1] must not be an image ID; got: sha256:' ~ docker_test_image_digest_v1
|
||||||
|
|
||||||
|
always:
|
||||||
|
- name: Remove tagged images
|
||||||
|
docker_image:
|
||||||
|
name: "{{ item }}"
|
||||||
|
source: local
|
||||||
|
state: absent
|
||||||
|
loop: "{{ image_names }}"
|
||||||
|
|
||||||
|
when: docker_api_version is version('1.25', '>=')
|
||||||
|
|
||||||
|
- fail: msg="Too old docker / docker-py version to run docker_image_info tests!"
|
||||||
|
when: not(docker_api_version is version('1.25', '>=')) and (ansible_distribution != 'CentOS' or ansible_distribution_major_version|int > 6)
|
||||||
Loading…
Reference in New Issue
Block a user