oc_atomic_container.py 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225
  1. #!/usr/bin/env python
  2. # pylint: disable=missing-docstring
  3. # flake8: noqa: T001
  4. # ___ ___ _ _ ___ ___ _ _____ ___ ___
  5. # / __| __| \| | __| _ \ /_\_ _| __| \
  6. # | (_ | _|| .` | _|| / / _ \| | | _|| |) |
  7. # \___|___|_|\_|___|_|_\/_/_\_\_|_|___|___/_ _____
  8. # | \ / _ \ | \| |/ _ \_ _| | __| \_ _|_ _|
  9. # | |) | (_) | | .` | (_) || | | _|| |) | | | |
  10. # |___/ \___/ |_|\_|\___/ |_| |___|___/___| |_|
  11. #
  12. # Copyright 2016 Red Hat, Inc. and/or its affiliates
  13. # and other contributors as indicated by the @author tags.
  14. #
  15. # Licensed under the Apache License, Version 2.0 (the "License");
  16. # you may not use this file except in compliance with the License.
  17. # You may obtain a copy of the License at
  18. #
  19. # http://www.apache.org/licenses/LICENSE-2.0
  20. #
  21. # Unless required by applicable law or agreed to in writing, software
  22. # distributed under the License is distributed on an "AS IS" BASIS,
  23. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  24. # See the License for the specific language governing permissions and
  25. # limitations under the License.
  26. #
  27. # -*- -*- -*- Begin included fragment: doc/atomic_container -*- -*- -*-
  28. DOCUMENTATION = '''
  29. ---
  30. module: oc_atomic_container
  31. short_description: Manage the container images on the atomic host platform
  32. description:
  33. - Manage the container images on the atomic host platform
  34. - Allows to execute the commands on the container images
  35. requirements:
  36. - atomic
  37. - "python >= 2.6"
  38. options:
  39. name:
  40. description:
  41. - Name of the container
  42. required: True
  43. default: null
  44. image:
  45. description:
  46. - The image to use to install the container
  47. required: True
  48. default: null
  49. state:
  50. description:
  51. - State of the container
  52. required: True
  53. choices: ["latest", "absent", "latest", "rollback"]
  54. default: "latest"
  55. values:
  56. description:
  57. - Values for the installation of the container
  58. required: False
  59. default: None
  60. '''
  61. # -*- -*- -*- End included fragment: doc/atomic_container -*- -*- -*-
  62. # -*- -*- -*- Begin included fragment: ansible/oc_atomic_container.py -*- -*- -*-
  63. # pylint: disable=wrong-import-position,too-many-branches,invalid-name,no-name-in-module, import-error
  64. import json
  65. import os
  66. from distutils.version import StrictVersion
  67. from ansible.module_utils.basic import AnsibleModule
  68. def _install(module, container, image, values_list):
  69. ''' install a container using atomic CLI. values_list is the list of --set arguments.
  70. container is the name given to the container. image is the image to use for the installation. '''
  71. # NOTE: system-package=no is hardcoded. This should be changed to an option in the future.
  72. args = ['atomic', 'install', '--system', '--system-package=no',
  73. '--name=%s' % container] + values_list + [image]
  74. rc, out, err = module.run_command(args, check_rc=False)
  75. if rc != 0:
  76. return rc, out, err, False
  77. else:
  78. changed = "Extracting" in out or "Copying blob" in out
  79. return rc, out, err, changed
  80. def _uninstall(module, name):
  81. ''' uninstall an atomic container by its name. '''
  82. args = ['atomic', 'uninstall', name]
  83. rc, out, err = module.run_command(args, check_rc=False)
  84. return rc, out, err, False
  85. def _ensure_service_file_is_removed(container):
  86. '''atomic install won't overwrite existing service file, so it needs to be removed'''
  87. service_path = '/etc/systemd/system/{}.service'.format(container)
  88. if not os.path.exists(service_path):
  89. return
  90. os.remove(service_path)
  91. def do_install(module, container, image, values_list):
  92. ''' install a container and exit the module. '''
  93. _ensure_service_file_is_removed(container)
  94. rc, out, err, changed = _install(module, container, image, values_list)
  95. if rc != 0:
  96. module.fail_json(rc=rc, msg=err)
  97. else:
  98. module.exit_json(msg=out, changed=changed)
  99. def do_uninstall(module, name):
  100. ''' uninstall a container and exit the module. '''
  101. rc, out, err, changed = _uninstall(module, name)
  102. if rc != 0:
  103. module.fail_json(rc=rc, msg=err)
  104. module.exit_json(msg=out, changed=changed)
  105. def do_update(module, container, old_image, image, values_list):
  106. ''' update a container and exit the module. If the container uses a different
  107. image than the current installed one, then first uninstall the old one '''
  108. # the image we want is different than the installed one
  109. if old_image != image:
  110. rc, out, err, _ = _uninstall(module, container)
  111. if rc != 0:
  112. module.fail_json(rc=rc, msg=err)
  113. return do_install(module, container, image, values_list)
  114. # if the image didn't change, use "atomic containers update"
  115. args = ['atomic', 'containers', 'update'] + values_list + [container]
  116. rc, out, err = module.run_command(args, check_rc=False)
  117. if rc != 0:
  118. module.fail_json(rc=rc, msg=err)
  119. else:
  120. changed = "Extracting" in out or "Copying blob" in out
  121. module.exit_json(msg=out, changed=changed)
  122. def do_rollback(module, name):
  123. ''' move to the previous deployment of the container, if present, and exit the module. '''
  124. args = ['atomic', 'containers', 'rollback', name]
  125. rc, out, err = module.run_command(args, check_rc=False)
  126. if rc != 0:
  127. module.fail_json(rc=rc, msg=err)
  128. else:
  129. changed = "Rolling back" in out
  130. module.exit_json(msg=out, changed=changed)
  131. def core(module):
  132. ''' entrypoint for the module. '''
  133. name = module.params['name']
  134. image = module.params['image']
  135. values = module.params['values']
  136. state = module.params['state']
  137. module.run_command_environ_update = dict(LANG='C', LC_ALL='C', LC_MESSAGES='C')
  138. out = {}
  139. err = {}
  140. rc = 0
  141. values_list = ["--set=%s" % x for x in values] if values else []
  142. args = ['atomic', 'containers', 'list', '--json', '--all', '-f', 'container=%s' % name]
  143. rc, out, err = module.run_command(args, check_rc=False)
  144. if rc != 0:
  145. module.fail_json(rc=rc, msg=err)
  146. return
  147. # NOTE: "or '[]' is a workaround until atomic containers list --json
  148. # provides an empty list when no containers are present.
  149. containers = json.loads(out or '[]')
  150. present = len(containers) > 0
  151. old_image = containers[0]["image_name"] if present else None
  152. if state == 'present' and present:
  153. module.exit_json(msg=out, changed=False)
  154. elif (state in ['latest', 'present']) and not present:
  155. do_install(module, name, image, values_list)
  156. elif state == 'latest':
  157. do_update(module, name, old_image, image, values_list)
  158. elif state == 'absent':
  159. if not present:
  160. module.exit_json(msg="", changed=False)
  161. else:
  162. do_uninstall(module, name)
  163. elif state == 'rollback':
  164. do_rollback(module, name)
  165. def main():
  166. module = AnsibleModule(
  167. argument_spec=dict(
  168. name=dict(default=None, required=True),
  169. image=dict(default=None, required=True),
  170. state=dict(default='latest', choices=['present', 'absent', 'latest', 'rollback']),
  171. values=dict(type='list', default=[]),
  172. ),
  173. )
  174. # Verify that the platform supports atomic command
  175. rc, version_out, err = module.run_command('rpm -q --queryformat "%{VERSION}\n" atomic', check_rc=False)
  176. if rc != 0:
  177. module.fail_json(msg="Error in running atomic command", err=err)
  178. # This module requires atomic version 1.17.2 or later
  179. atomic_version = StrictVersion(version_out.replace('\n', ''))
  180. if atomic_version < StrictVersion('1.17.2'):
  181. module.fail_json(
  182. msg="atomic version 1.17.2+ is required",
  183. err=str(atomic_version))
  184. try:
  185. core(module)
  186. except Exception as e: # pylint: disable=broad-except
  187. module.fail_json(msg=str(e))
  188. if __name__ == '__main__':
  189. main()
  190. # -*- -*- -*- End included fragment: ansible/oc_atomic_container.py -*- -*- -*-