mirror of
https://github.com/ansible-collections/community.docker.git
synced 2025-12-16 11:58:43 +00:00
Change Docker Stack modules to use common CLI module framework. (#745)
This commit is contained in:
parent
5adac5216a
commit
1c8272f821
2
changelogs/fragments/745-docker_stack.yml
Normal file
2
changelogs/fragments/745-docker_stack.yml
Normal 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)."
|
||||||
@ -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::
|
||||||
|
|
||||||
|
|||||||
@ -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
|
||||||
|
|||||||
@ -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__":
|
||||||
|
|||||||
@ -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__":
|
||||||
|
|||||||
@ -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__":
|
||||||
|
|||||||
@ -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:
|
||||||
|
|||||||
@ -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:
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user