Change Docker Stack modules to use common CLI module framework. (#745)

This commit is contained in:
Felix Fontein 2024-01-14 08:54:06 +01:00 committed by GitHub
parent 5adac5216a
commit 1c8272f821
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
8 changed files with 249 additions and 161 deletions

View File

@ -0,0 +1,2 @@
minor_changes:
- "The ``docker_stack*`` modules now use the common CLI-based module code added for the ``docker_image_build`` and ``docker_compose_v2`` modules. This means that the modules now have various more configuration options with respect to talking to the Docker Daemon, and now also are part of the ``community.docker.docker`` and ``docker`` module default groups (https://github.com/ansible-collections/community.docker/pull/745)."

View File

@ -84,7 +84,7 @@ Most plugins and modules can be configured by the following parameters:
Module default group Module default group
.................... ....................
To avoid having to specify common parameters for all the modules in every task, you can use the ``community.docker.docker`` :ref:`module defaults group <module_defaults_groups>`, or its short name ``docker``. Please note that the Docker Swarm stack modules (:ansplugin:`community.docker.docker_stack#module`, :ansplugin:`community.docker.docker_stack_info#module`, and :ansplugin:`community.docker.docker_stack_task_info#module`) are not part of the defaults group. To avoid having to specify common parameters for all the modules in every task, you can use the ``community.docker.docker`` :ref:`module defaults group <module_defaults_groups>`, or its short name ``docker``.
.. note:: .. note::

View File

@ -30,6 +30,9 @@ action_groups:
- docker_plugin - docker_plugin
- docker_prune - docker_prune
- docker_secret - docker_secret
- docker_stack
- docker_stack_info
- docker_stack_task_info
- docker_swarm - docker_swarm
- docker_swarm_info - docker_swarm_info
- docker_swarm_service - docker_swarm_service

View File

@ -18,12 +18,16 @@ description:
- Manage docker stacks using the C(docker stack) command - Manage docker stacks using the C(docker stack) command
on the target node (see examples). on the target node (see examples).
extends_documentation_fragment: extends_documentation_fragment:
- community.docker.docker.cli_documentation
- community.docker.attributes - community.docker.attributes
- community.docker.attributes.actiongroup_docker
attributes: attributes:
check_mode: check_mode:
support: none support: none
diff_mode: diff_mode:
support: none support: none
action_group:
version_added: 3.6.0
options: options:
name: name:
description: description:
@ -80,8 +84,29 @@ options:
- Interval in seconds between consecutive O(absent_retries). - Interval in seconds between consecutive O(absent_retries).
type: int type: int
default: 1 default: 1
docker_cli:
version_added: 3.6.0
docker_host:
version_added: 3.6.0
tls_hostname:
version_added: 3.6.0
api_version:
version_added: 3.6.0
ca_path:
version_added: 3.6.0
client_cert:
version_added: 3.6.0
client_key:
version_added: 3.6.0
tls:
version_added: 3.6.0
validate_certs:
version_added: 3.6.0
cli_context:
version_added: 3.6.0
requirements: requirements:
- Docker CLI tool C(docker)
- jsondiff - jsondiff
- pyyaml - pyyaml
''' '''
@ -128,10 +153,20 @@ EXAMPLES = '''
import json import json
import os
import tempfile import tempfile
import traceback
from ansible.module_utils.six import string_types from ansible.module_utils.six import string_types
from time import sleep from time import sleep
from ansible.module_utils.common.text.converters import to_native
from ansible_collections.community.docker.plugins.module_utils.common_cli import (
AnsibleModuleDockerClient,
DockerException,
)
try: try:
from jsondiff import diff as json_diff from jsondiff import diff as json_diff
HAS_JSONDIFF = True HAS_JSONDIFF = True
@ -144,28 +179,16 @@ try:
except ImportError: except ImportError:
HAS_YAML = False HAS_YAML = False
from ansible.module_utils.basic import AnsibleModule, os
def docker_stack_services(client, stack_name):
def docker_stack_services(module, stack_name): rc, out, err = client.call_cli("stack", "services", stack_name, "--format", "{{.Name}}")
docker_bin = module.get_bin_path('docker', required=True) if to_native(err) == "Nothing found in stack: %s\n" % stack_name:
rc, out, err = module.run_command([docker_bin,
"stack",
"services",
stack_name,
"--format",
"{{.Name}}"])
if err == "Nothing found in stack: %s\n" % stack_name:
return [] return []
return out.strip().split('\n') return to_native(out).strip().split('\n')
def docker_service_inspect(module, service_name): def docker_service_inspect(client, service_name):
docker_bin = module.get_bin_path('docker', required=True) rc, out, err = client.call_cli("service", "inspect", service_name)
rc, out, err = module.run_command([docker_bin,
"service",
"inspect",
service_name])
if rc != 0: if rc != 0:
return None return None
else: else:
@ -173,45 +196,43 @@ def docker_service_inspect(module, service_name):
return ret return ret
def docker_stack_deploy(module, stack_name, compose_files): def docker_stack_deploy(client, stack_name, compose_files):
docker_bin = module.get_bin_path('docker', required=True) command = ["stack", "deploy"]
command = [docker_bin, "stack", "deploy"] if client.module.params["prune"]:
if module.params["prune"]:
command += ["--prune"] command += ["--prune"]
if module.params["with_registry_auth"]: if client.module.params["with_registry_auth"]:
command += ["--with-registry-auth"] command += ["--with-registry-auth"]
if module.params["resolve_image"]: if client.module.params["resolve_image"]:
command += ["--resolve-image", command += ["--resolve-image",
module.params["resolve_image"]] client.module.params["resolve_image"]]
for compose_file in compose_files: for compose_file in compose_files:
command += ["--compose-file", command += ["--compose-file",
compose_file] compose_file]
command += [stack_name] command += [stack_name]
return module.run_command(command) rc, out, err = client.call_cli(*command)
return rc, to_native(out), to_native(err)
def docker_stack_inspect(module, stack_name): def docker_stack_inspect(client, stack_name):
ret = {} ret = {}
for service_name in docker_stack_services(module, stack_name): for service_name in docker_stack_services(client, stack_name):
ret[service_name] = docker_service_inspect(module, service_name) ret[service_name] = docker_service_inspect(client, service_name)
return ret return ret
def docker_stack_rm(module, stack_name, retries, interval): def docker_stack_rm(client, stack_name, retries, interval):
docker_bin = module.get_bin_path('docker', required=True) command = ["stack", "rm", stack_name]
command = [docker_bin, "stack", "rm", stack_name] rc, out, err = client.call_cli(*command)
rc, out, err = module.run_command(command) while to_native(err) != "Nothing found in stack: %s\n" % stack_name and retries > 0:
while err != "Nothing found in stack: %s\n" % stack_name and retries > 0:
sleep(interval) sleep(interval)
retries = retries - 1 retries = retries - 1
rc, out, err = module.run_command(command) rc, out, err = client.call_cli(*command)
return rc, out, err return rc, to_native(out), to_native(err)
def main(): def main():
module = AnsibleModule( client = AnsibleModuleDockerClient(
argument_spec={ argument_spec={
'name': dict(type='str', required=True), 'name': dict(type='str', required=True),
'compose': dict(type='list', elements='raw', default=[]), 'compose': dict(type='list', elements='raw', default=[]),
@ -222,87 +243,97 @@ def main():
'absent_retries': dict(type='int', default=0), 'absent_retries': dict(type='int', default=0),
'absent_retries_interval': dict(type='int', default=1) 'absent_retries_interval': dict(type='int', default=1)
}, },
supports_check_mode=False supports_check_mode=False,
) )
if not HAS_JSONDIFF: if not HAS_JSONDIFF:
return module.fail_json(msg="jsondiff is not installed, try 'pip install jsondiff'") return client.fail("jsondiff is not installed, try 'pip install jsondiff'")
if not HAS_YAML: if not HAS_YAML:
return module.fail_json(msg="yaml is not installed, try 'pip install pyyaml'") return client.fail("yaml is not installed, try 'pip install pyyaml'")
state = module.params['state'] try:
compose = module.params['compose'] state = client.module.params['state']
name = module.params['name'] compose = client.module.params['compose']
absent_retries = module.params['absent_retries'] name = client.module.params['name']
absent_retries_interval = module.params['absent_retries_interval'] absent_retries = client.module.params['absent_retries']
absent_retries_interval = client.module.params['absent_retries_interval']
if state == 'present': if state == 'present':
if not compose: if not compose:
module.fail_json(msg=("compose parameter must be a list " client.fail("compose parameter must be a list containing at least one element")
"containing at least one element"))
compose_files = [] compose_files = []
for i, compose_def in enumerate(compose): for i, compose_def in enumerate(compose):
if isinstance(compose_def, dict): if isinstance(compose_def, dict):
compose_file_fd, compose_file = tempfile.mkstemp() compose_file_fd, compose_file = tempfile.mkstemp()
module.add_cleanup_file(compose_file) client.module.add_cleanup_file(compose_file)
with os.fdopen(compose_file_fd, 'w') as stack_file: with os.fdopen(compose_file_fd, 'w') as stack_file:
compose_files.append(compose_file) compose_files.append(compose_file)
stack_file.write(yaml_dump(compose_def)) stack_file.write(yaml_dump(compose_def))
elif isinstance(compose_def, string_types): elif isinstance(compose_def, string_types):
compose_files.append(compose_def) compose_files.append(compose_def)
else: else:
module.fail_json(msg="compose element '%s' must be a string or a dictionary" % compose_def) client.fail("compose element '%s' must be a string or a dictionary" % compose_def)
before_stack_services = docker_stack_inspect(module, name) before_stack_services = docker_stack_inspect(client, name)
rc, out, err = docker_stack_deploy(module, name, compose_files) rc, out, err = docker_stack_deploy(client, name, compose_files)
after_stack_services = docker_stack_inspect(module, name) after_stack_services = docker_stack_inspect(client, name)
if rc != 0:
module.fail_json(msg="docker stack up deploy command failed",
rc=rc,
stdout=out, stderr=err)
before_after_differences = json_diff(before_stack_services,
after_stack_services)
for k in before_after_differences.keys():
if isinstance(before_after_differences[k], dict):
before_after_differences[k].pop('UpdatedAt', None)
before_after_differences[k].pop('Version', None)
if not list(before_after_differences[k].keys()):
before_after_differences.pop(k)
if not before_after_differences:
module.exit_json(
changed=False,
rc=rc,
stdout=out,
stderr=err)
else:
module.exit_json(
changed=True,
rc=rc,
stdout=out,
stderr=err,
stack_spec_diff=json_diff(before_stack_services,
after_stack_services,
dump=True))
else:
if docker_stack_services(module, name):
rc, out, err = docker_stack_rm(module, name, absent_retries, absent_retries_interval)
if rc != 0: if rc != 0:
module.fail_json(msg="'docker stack down' command failed", client.fail("docker stack up deploy command failed", rc=rc, stdout=out, stderr=err)
rc=rc,
stdout=out, stderr=err) before_after_differences = json_diff(before_stack_services, after_stack_services)
for k in before_after_differences.keys():
if isinstance(before_after_differences[k], dict):
before_after_differences[k].pop('UpdatedAt', None)
before_after_differences[k].pop('Version', None)
if not list(before_after_differences[k].keys()):
before_after_differences.pop(k)
if not before_after_differences:
client.module.exit_json(
changed=False,
rc=rc,
stdout=out,
stderr=err,
)
else: else:
module.exit_json(changed=True, client.module.exit_json(
msg=out, rc=rc, changed=True,
stdout=out, stderr=err) rc=rc,
module.exit_json(changed=False) stdout=out,
stderr=err,
stack_spec_diff=json_diff(
before_stack_services,
after_stack_services,
dump=True,
),
)
else:
if docker_stack_services(client, name):
rc, out, err = docker_stack_rm(client, name, absent_retries, absent_retries_interval)
if rc != 0:
client.module.fail_json(
msg="'docker stack down' command failed",
rc=rc,
stdout=out,
stderr=err,
)
else:
client.module.exit_json(
changed=True,
msg=out,
rc=rc,
stdout=out,
stderr=err,
)
client.module.exit_json(changed=False)
except DockerException as e:
client.fail('An unexpected Docker error occurred: {0}'.format(to_native(e)), exception=traceback.format_exc())
if __name__ == "__main__": if __name__ == "__main__":

View File

@ -17,9 +17,37 @@ short_description: Return information on all docker stacks
description: description:
- Retrieve information on docker stacks using the C(docker stack) command - Retrieve information on docker stacks using the C(docker stack) command
on the target node (see examples). on the target node (see examples).
requirements:
- Docker CLI tool C(docker)
extends_documentation_fragment: extends_documentation_fragment:
- community.docker.docker.cli_documentation
- community.docker.attributes - community.docker.attributes
- community.docker.attributes.actiongroup_docker
- community.docker.attributes.info_module - community.docker.attributes.info_module
attributes:
action_group:
version_added: 3.6.0
options:
docker_cli:
version_added: 3.6.0
docker_host:
version_added: 3.6.0
tls_hostname:
version_added: 3.6.0
api_version:
version_added: 3.6.0
ca_path:
version_added: 3.6.0
client_cert:
version_added: 3.6.0
client_key:
version_added: 3.6.0
tls:
version_added: 3.6.0
validate_certs:
version_added: 3.6.0
cli_context:
version_added: 3.6.0
seealso: seealso:
- module: community.docker.docker_stack_task_info - module: community.docker.docker_stack_task_info
description: >- description: >-
@ -29,8 +57,8 @@ seealso:
RETURN = ''' RETURN = '''
results: results:
description: | description:
List of dictionaries containing the list of stacks on the target node - List of dictionaries containing the list of stacks on the target node
sample: sample:
- {"name":"grafana","namespace":"default","orchestrator":"Kubernetes","services":"2"} - {"name":"grafana","namespace":"default","orchestrator":"Kubernetes","services":"2"}
returned: always returned: always
@ -49,7 +77,14 @@ EXAMPLES = '''
''' '''
import json import json
from ansible.module_utils.basic import AnsibleModule import traceback
from ansible.module_utils.common.text.converters import to_native
from ansible_collections.community.docker.plugins.module_utils.common_cli import (
AnsibleModuleDockerClient,
DockerException,
)
def docker_stack_list(module): def docker_stack_list(module):
@ -61,31 +96,23 @@ def docker_stack_list(module):
def main(): def main():
module = AnsibleModule( client = AnsibleModuleDockerClient(
argument_spec={ argument_spec={
}, },
supports_check_mode=True supports_check_mode=True,
) )
rc, out, err = docker_stack_list(module) try:
rc, ret, stderr = client.call_cli_json_stream('stack', 'ls', '--format={{json .}}', check_rc=True)
if rc != 0: client.module.exit_json(
module.fail_json(msg="Error running docker stack. {0}".format(err), changed=False,
rc=rc, stdout=out, stderr=err) rc=rc,
else: stdout='\n'.join([json.dumps(entry) for entry in ret]),
if out: stderr=to_native(stderr).strip(),
ret = list( results=ret,
json.loads(outitem) )
for outitem in out.splitlines()) except DockerException as e:
client.fail('An unexpected Docker error occurred: {0}'.format(to_native(e)), exception=traceback.format_exc())
else:
ret = []
module.exit_json(changed=False,
rc=rc,
stdout=out,
stderr=err,
results=ret)
if __name__ == "__main__": if __name__ == "__main__":

View File

@ -18,23 +18,50 @@ description:
- Retrieve information on docker stacks tasks using the C(docker stack) command - Retrieve information on docker stacks tasks using the C(docker stack) command
on the target node (see examples). on the target node (see examples).
extends_documentation_fragment: extends_documentation_fragment:
- community.docker.docker.cli_documentation
- community.docker.attributes - community.docker.attributes
- community.docker.attributes.actiongroup_docker
- community.docker.attributes.info_module - community.docker.attributes.info_module
attributes:
action_group:
version_added: 3.6.0
options: options:
name: name:
description: description:
- Stack name. - Stack name.
type: str type: str
required: true required: true
docker_cli:
version_added: 3.6.0
docker_host:
version_added: 3.6.0
tls_hostname:
version_added: 3.6.0
api_version:
version_added: 3.6.0
ca_path:
version_added: 3.6.0
client_cert:
version_added: 3.6.0
client_key:
version_added: 3.6.0
tls:
version_added: 3.6.0
validate_certs:
version_added: 3.6.0
cli_context:
version_added: 3.6.0
requirements:
- Docker CLI tool C(docker)
''' '''
RETURN = ''' RETURN = '''
results: results:
description: | description:
List of dictionaries containing the list of tasks associated - List of dictionaries containing the list of tasks associated
to a stack name. to a stack name.
sample: > sample:
[{"CurrentState":"Running","DesiredState":"Running","Error":"","ID":"7wqv6m02ugkw","Image":"busybox","Name":"test_stack.1","Node":"swarm","Ports":""}] - {"CurrentState":"Running","DesiredState":"Running","Error":"","ID":"7wqv6m02ugkw","Image":"busybox","Name":"test_stack.1","Node":"swarm","Ports":""}
returned: always returned: always
type: list type: list
elements: dict elements: dict
@ -52,7 +79,14 @@ EXAMPLES = '''
''' '''
import json import json
from ansible.module_utils.basic import AnsibleModule import traceback
from ansible.module_utils.common.text.converters import to_native
from ansible_collections.community.docker.plugins.module_utils.common_cli import (
AnsibleModuleDockerClient,
DockerException,
)
def docker_stack_task(module, stack_name): def docker_stack_task(module, stack_name):
@ -64,34 +98,25 @@ def docker_stack_task(module, stack_name):
def main(): def main():
module = AnsibleModule( client = AnsibleModuleDockerClient(
argument_spec={ argument_spec={
'name': dict(type='str', required=True) 'name': dict(type='str', required=True)
}, },
supports_check_mode=True supports_check_mode=True,
) )
name = module.params['name'] try:
name = client.module.params['name']
rc, out, err = docker_stack_task(module, name) rc, ret, stderr = client.call_cli_json_stream('stack', 'ps', name, '--format={{json .}}', check_rc=True)
client.module.exit_json(
if rc != 0: changed=False,
module.fail_json(msg="Error running docker stack. {0}".format(err), rc=rc,
rc=rc, stdout=out, stderr=err) stdout='\n'.join([json.dumps(entry) for entry in ret]),
else: stderr=to_native(stderr).strip(),
if out: results=ret,
ret = list( )
json.loads(outitem) except DockerException as e:
for outitem in out.splitlines()) client.fail('An unexpected Docker error occurred: {0}'.format(to_native(e)), exception=traceback.format_exc())
else:
ret = []
module.exit_json(changed=False,
rc=rc,
stdout=out,
stderr=err,
results=ret)
if __name__ == "__main__": if __name__ == "__main__":

View File

@ -18,7 +18,7 @@
assert: assert:
that: that:
- 'output is failed' - 'output is failed'
- '"Error running docker stack" in output.msg' - '"Error response from daemon: This node is not a swarm manager" in output.msg'
- name: Create a swarm cluster - name: Create a swarm cluster
docker_swarm: docker_swarm:

View File

@ -18,7 +18,7 @@
assert: assert:
that: that:
- 'output is failed' - 'output is failed'
- '"Error running docker stack" in output.msg' - '"Error response from daemon: This node is not a swarm manager" in output.msg'
- name: Create a swarm cluster - name: Create a swarm cluster
docker_swarm: docker_swarm: