main.yaml 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530
  1. ---
  2. - name: Ensure that ElasticSearch has nodes to run on
  3. import_role:
  4. name: openshift_control_plane
  5. tasks_from: ensure_nodes_matching_selector.yml
  6. vars:
  7. openshift_master_ensure_nodes_selector: "{{ openshift_logging_es_nodeselector | map_to_pairs }}"
  8. openshift_master_ensure_nodes_service: Elasticsearch
  9. - name: Validate Elasticsearch cluster size
  10. fail: msg="The openshift_logging_es_cluster_size may only be scaled down manually. Please see official documentation on how to do this."
  11. when: openshift_logging_facts.elasticsearch.deploymentconfigs | length > openshift_logging_es_cluster_size|int
  12. - name: Validate Elasticsearch Ops cluster size
  13. fail: msg="The openshift_logging_es_ops_cluster_size may only be scaled down manually. Please see official documentation on how to do this."
  14. when: openshift_logging_facts.elasticsearch_ops.deploymentconfigs | length > openshift_logging_es_ops_cluster_size|int
  15. - fail:
  16. msg: Invalid deployment type, one of ['data-master', 'data-client', 'master', 'client'] allowed
  17. when: not openshift_logging_elasticsearch_deployment_type in __allowed_es_types
  18. - set_fact:
  19. elasticsearch_name: "{{ 'logging-elasticsearch' ~ ( (openshift_logging_elasticsearch_ops_deployment | default(false) | bool) | ternary('-ops', '')) }}"
  20. es_component: "{{ 'es' ~ ( (openshift_logging_elasticsearch_ops_deployment | default(false) | bool) | ternary('-ops', '') ) }}"
  21. - include_tasks: get_es_version.yml
  22. run_once: true
  23. - set_fact:
  24. full_restart_cluster: True
  25. when:
  26. - _es_installed_version is defined
  27. - _es_installed_version.split('.')[0] | int < __es_major_version
  28. - not openshift_logging_elasticsearch_ops_deployment | default(false) | bool
  29. - set_fact:
  30. full_restart_cluster: True
  31. when:
  32. - _es_ops_installed_version is defined
  33. - _es_ops_installed_version.split('.')[0] | int < __es_major_version
  34. - openshift_logging_elasticsearch_ops_deployment | default(false) | bool
  35. # allow passing in a tempdir
  36. - name: Create temp directory for doing work in
  37. command: mktemp -d /tmp/openshift-logging-ansible-XXXXXX
  38. register: mktemp
  39. changed_when: False
  40. - set_fact:
  41. tempdir: "{{ mktemp.stdout }}"
  42. # This may not be necessary in this role
  43. - name: Create templates subdirectory
  44. file:
  45. state: directory
  46. path: "{{ tempdir }}/templates"
  47. mode: 0755
  48. changed_when: False
  49. # we want to make sure we have all the necessary components here
  50. # service account
  51. - name: Create ES service account
  52. oc_serviceaccount:
  53. state: present
  54. name: "aggregated-logging-elasticsearch"
  55. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  56. image_pull_secrets: "{{ openshift_logging_image_pull_secret }}"
  57. when: openshift_logging_image_pull_secret != ''
  58. - name: Create ES service account
  59. oc_serviceaccount:
  60. state: present
  61. name: "aggregated-logging-elasticsearch"
  62. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  63. when:
  64. - openshift_logging_image_pull_secret == ''
  65. # rolebinding reader
  66. - name: Create rolebinding-reader role
  67. oc_clusterrole:
  68. state: present
  69. name: rolebinding-reader
  70. rules:
  71. - apiGroups: [""]
  72. resources: ["clusterrolebindings"]
  73. verbs: ["get"]
  74. # SA roles
  75. - name: Set rolebinding-reader permissions for ES
  76. oc_adm_policy_user:
  77. state: present
  78. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  79. resource_kind: cluster-role
  80. resource_name: rolebinding-reader
  81. user: "system:serviceaccount:{{ openshift_logging_elasticsearch_namespace }}:aggregated-logging-elasticsearch"
  82. - oc_adm_policy_user:
  83. state: present
  84. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  85. resource_kind: cluster-role
  86. resource_name: system:auth-delegator
  87. user: "system:serviceaccount:{{ openshift_logging_elasticsearch_namespace}}:aggregated-logging-elasticsearch"
  88. # logging-metrics-reader role
  89. - template:
  90. src: "logging-metrics-role.j2"
  91. dest: "{{mktemp.stdout}}/templates/logging-metrics-role.yml"
  92. vars:
  93. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  94. - template:
  95. src: "logging-metrics-rolebinding.j2"
  96. dest: "{{mktemp.stdout}}/templates/logging-metrics-rolebinding.yml"
  97. vars:
  98. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  99. role_namespace: "{{ openshift_logging_elasticsearch_prometheus_sa | serviceaccount_namespace(openshift_logging_elasticsearch_namespace) }}"
  100. role_user: "{{ openshift_logging_elasticsearch_prometheus_sa | serviceaccount_name }}"
  101. - name: Create logging-metrics-reader-role
  102. oc_obj:
  103. state: present
  104. name: "prometheus-metrics-viewer"
  105. kind: role
  106. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  107. files:
  108. - "{{mktemp.stdout}}/templates/logging-metrics-role.yml"
  109. delete_after: true
  110. - name: Create logging-metrics-reader-rolebinding
  111. oc_obj:
  112. state: present
  113. name: "prometheus-metrics-viewer"
  114. kind: rolebinding
  115. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  116. files:
  117. - "{{mktemp.stdout}}/templates/logging-metrics-rolebinding.yml"
  118. delete_after: true
  119. - name: Checking for passwd.yml
  120. stat: path="{{ generated_certs_dir }}/passwd.yml"
  121. register: passwd_file
  122. check_mode: no
  123. - when: passwd_file.stat.exists
  124. slurp:
  125. src: "{{ generated_certs_dir }}/passwd.yml"
  126. register: _logging_metrics_proxy_passwd
  127. - when: not passwd_file.stat.exists or openshift_logging_elasticsearch_prometheus_sa not in ( _logging_metrics_proxy_passwd['content'] | b64decode | from_yaml )
  128. template:
  129. src: passwd.j2
  130. dest: "{{ generated_certs_dir }}/passwd.yml"
  131. vars:
  132. logging_user_name: "{{ openshift_logging_elasticsearch_prometheus_sa }}"
  133. logging_user_passwd: "{{ 16 | lib_utils_oo_random_word | b64encode }}"
  134. - slurp:
  135. src: "{{ generated_certs_dir }}/passwd.yml"
  136. register: _logging_metrics_proxy_passwd
  137. # View role and binding
  138. - name: Generate logging-elasticsearch-view-role
  139. template:
  140. src: "rolebinding.j2"
  141. dest: "{{mktemp.stdout}}/logging-elasticsearch-view-role.yaml"
  142. vars:
  143. obj_name: logging-elasticsearch-view-role
  144. roleRef:
  145. name: view
  146. subjects:
  147. - kind: ServiceAccount
  148. name: aggregated-logging-elasticsearch
  149. changed_when: no
  150. - name: Set logging-elasticsearch-view-role role
  151. oc_obj:
  152. state: present
  153. name: "logging-elasticsearch-view-role"
  154. kind: rolebinding
  155. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  156. files:
  157. - "{{ tempdir }}/logging-elasticsearch-view-role.yaml"
  158. delete_after: true
  159. # configmap
  160. - assert:
  161. that:
  162. - openshift_logging_elasticsearch_kibana_index_mode in __kibana_index_modes
  163. msg: "The openshift_logging_elasticsearch_kibana_index_mode '{{ openshift_logging_elasticsearch_kibana_index_mode }}' only supports one of: {{ __kibana_index_modes | join(', ') }}"
  164. - assert:
  165. that:
  166. - "{{ openshift_logging_es_log_appenders | length > 0 }}"
  167. msg: "The openshift_logging_es_log_appenders '{{ openshift_logging_es_log_appenders }}' has an unrecognized option and only supports the following as a list: {{ __es_log_appenders | join(', ') }}"
  168. - template:
  169. src: "elasticsearch.yml.j2"
  170. dest: "{{ tempdir }}/elasticsearch.yml"
  171. vars:
  172. allow_cluster_reader: "{{ openshift_logging_elasticsearch_ops_allow_cluster_reader | lower | default('false') }}"
  173. es_kibana_index_mode: "{{ openshift_logging_elasticsearch_kibana_index_mode | default('unique') }}"
  174. es_unicast_host: "logging-{{ es_component }}-cluster"
  175. changed_when: no
  176. # create diff between current configmap files and our current files
  177. - template:
  178. src: "log4j2.properties.j2"
  179. dest: "{{ tempdir }}/log4j2.properties"
  180. vars:
  181. root_logger: "{{ openshift_logging_es_log_appenders | list }}"
  182. changed_when: no
  183. - include_role:
  184. name: openshift_logging
  185. tasks_from: patch_configmap_files.yaml
  186. vars:
  187. configmap_name: "{{ elasticsearch_name }}"
  188. configmap_namespace: "{{ openshift_logging_namespace }}"
  189. configmap_file_names:
  190. - current_file: "elasticsearch.yml"
  191. new_file: "{{ tempdir }}/elasticsearch.yml"
  192. - current_file: "log4j2.properties"
  193. new_file: "{{ tempdir }}/log4j2.properties"
  194. - slurp:
  195. src: "{{ tempdir }}/elasticsearch.yml"
  196. register: _patched_elasticsearch_config
  197. - copy:
  198. content: "{{ config_source | combine(override_config, recursive=True) | to_nice_yaml(indent=2) }}"
  199. dest: "{{ tempdir }}/elasticsearch.yml"
  200. vars:
  201. config_source: "{{ _patched_elasticsearch_config['content'] | b64decode | from_yaml }}"
  202. override_config: "{{ openshift_logging_es_config | default({}) | from_yaml }}"
  203. - name: Set ES configmap
  204. oc_configmap:
  205. state: present
  206. name: "{{ elasticsearch_name }}"
  207. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  208. from_file:
  209. elasticsearch.yml: "{{ tempdir }}/elasticsearch.yml"
  210. log4j2.properties: "{{ tempdir }}/log4j2.properties"
  211. register: es_config_creation
  212. notify: "restart elasticsearch"
  213. - when: es_config_creation.changed | bool
  214. block:
  215. - set_fact:
  216. _restart_logging_components: "{{ _restart_logging_components | default([]) + [es_component] | unique }}"
  217. - shell: >
  218. {{ openshift_client_binary }} --config={{ openshift.common.config_base }}/master/admin.kubeconfig get dc -l component="{{ es_component }}" -n "{{ openshift_logging_elasticsearch_namespace }}" -o name | cut -d'/' -f2
  219. register: _es_dcs
  220. - set_fact:
  221. _restart_logging_nodes: "{{ _restart_logging_nodes | default([]) + [_es_dcs.stdout] | unique }}"
  222. when: _es_dcs.stdout != ""
  223. # secret
  224. - name: Set ES secret
  225. oc_secret:
  226. state: present
  227. name: "logging-elasticsearch"
  228. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  229. files:
  230. - name: key
  231. path: "{{ generated_certs_dir }}/logging-es.jks"
  232. - name: truststore
  233. path: "{{ generated_certs_dir }}/truststore.jks"
  234. - name: searchguard.key
  235. path: "{{ generated_certs_dir }}/elasticsearch.jks"
  236. - name: searchguard.truststore
  237. path: "{{ generated_certs_dir }}/truststore.jks"
  238. - name: admin-key
  239. path: "{{ generated_certs_dir }}/system.admin.key"
  240. - name: admin-cert
  241. path: "{{ generated_certs_dir }}/system.admin.crt"
  242. - name: admin-ca
  243. path: "{{ generated_certs_dir }}/ca.crt"
  244. - name: admin.jks
  245. path: "{{ generated_certs_dir }}/system.admin.jks"
  246. - name: passwd.yml
  247. path: "{{ generated_certs_dir }}/passwd.yml"
  248. # services
  249. - name: Set logging-{{ es_component }}-cluster service
  250. oc_service:
  251. state: present
  252. name: "logging-{{ es_component }}-cluster"
  253. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  254. clusterip: 'None'
  255. annotations:
  256. service.alpha.kubernetes.io/tolerate-unready-endpoints: 'true'
  257. selector:
  258. component: "{{ es_component }}"
  259. provider: openshift
  260. labels:
  261. logging-infra: 'support'
  262. ports:
  263. - port: 9300
  264. # equivalent to the unready-endpoints annotation
  265. - name: Edit logging-{{ es_component }}-cluster service
  266. oc_edit:
  267. state: present
  268. kind: service
  269. name: "logging-{{ es_component }}-cluster"
  270. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  271. content:
  272. spec.publishNotReadyAddresses: "{{ true | bool }}"
  273. - name: Set logging-{{ es_component }} service
  274. oc_service:
  275. state: present
  276. name: "logging-{{ es_component }}"
  277. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  278. selector:
  279. component: "{{ es_component }}"
  280. provider: openshift
  281. labels:
  282. logging-infra: 'support'
  283. ports:
  284. - port: 9200
  285. targetPort: "restapi"
  286. - name: Set logging-{{ es_component}}-prometheus service
  287. oc_service:
  288. state: present
  289. name: "logging-{{es_component}}-prometheus"
  290. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  291. labels:
  292. logging-infra: 'support'
  293. ports:
  294. - name: proxy
  295. port: 443
  296. targetPort: 4443
  297. selector:
  298. component: "{{ es_component }}"
  299. provider: openshift
  300. - oc_edit:
  301. kind: service
  302. name: "logging-{{es_component}}-prometheus"
  303. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  304. separator: '#'
  305. content:
  306. metadata#annotations#service.alpha.openshift.io/serving-cert-secret-name: "prometheus-tls"
  307. metadata#annotations#prometheus.io/scrape: "true"
  308. metadata#annotations#prometheus.io/scheme: "https"
  309. metadata#annotations#prometheus.io/path: "/_prometheus/metrics"
  310. metadata#annotations#prometheus.io/port: "4443"
  311. - name: Check to see if PVC already exists
  312. oc_obj:
  313. state: list
  314. kind: pvc
  315. name: "{{ openshift_logging_elasticsearch_pvc_name }}"
  316. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  317. register: logging_elasticsearch_pvc
  318. # logging_elasticsearch_pvc.results.results | length > 0 returns a false positive
  319. # so we check for the presence of 'stderr' to determine if the obj exists or not
  320. # the RC for existing and not existing is both 0
  321. - when:
  322. - logging_elasticsearch_pvc.results.stderr is defined
  323. - openshift_logging_elasticsearch_storage_type == "pvc"
  324. block:
  325. # storageclasses are used by default but if static then disable
  326. # storageclasses with the storageClassName set to "" in pvc.j2
  327. - name: Creating ES storage template - static
  328. template:
  329. src: "pvc.j2"
  330. dest: "{{ tempdir }}/templates/logging-es-pvc.yml"
  331. vars:
  332. obj_name: "{{ openshift_logging_elasticsearch_pvc_name }}"
  333. size: "{{ (openshift_logging_elasticsearch_pvc_size | trim | length == 0) | ternary('10Gi', openshift_logging_elasticsearch_pvc_size) }}"
  334. access_modes: "{{ openshift_logging_elasticsearch_pvc_access_modes | list }}"
  335. pv_selector: "{{ openshift_logging_elasticsearch_pvc_pv_selector }}"
  336. storage_class_name: "{{ openshift_logging_elasticsearch_pvc_storage_class_name | default('', true) }}"
  337. when:
  338. - not openshift_logging_elasticsearch_pvc_dynamic | bool
  339. # Storageclasses are used by default if configured
  340. - name: Creating ES storage template - dynamic
  341. template:
  342. src: "pvc.j2"
  343. dest: "{{ tempdir }}/templates/logging-es-pvc.yml"
  344. vars:
  345. obj_name: "{{ openshift_logging_elasticsearch_pvc_name }}"
  346. size: "{{ (openshift_logging_elasticsearch_pvc_size | trim | length == 0) | ternary('10Gi', openshift_logging_elasticsearch_pvc_size) }}"
  347. access_modes: "{{ openshift_logging_elasticsearch_pvc_access_modes | list }}"
  348. pv_selector: "{{ openshift_logging_elasticsearch_pvc_pv_selector }}"
  349. when:
  350. - openshift_logging_elasticsearch_pvc_dynamic | bool
  351. - name: Set ES storage
  352. oc_obj:
  353. state: present
  354. kind: pvc
  355. name: "{{ openshift_logging_elasticsearch_pvc_name }}"
  356. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  357. files:
  358. - "{{ tempdir }}/templates/logging-es-pvc.yml"
  359. delete_after: true
  360. - set_fact:
  361. es_deploy_name: "logging-{{ es_component }}-{{ openshift_logging_elasticsearch_deployment_type }}-{{ 8 | lib_utils_oo_random_word('abcdefghijklmnopqrstuvwxyz0123456789') }}"
  362. when: openshift_logging_elasticsearch_deployment_name == ""
  363. - set_fact:
  364. es_deploy_name: "{{ openshift_logging_elasticsearch_deployment_name }}"
  365. when: openshift_logging_elasticsearch_deployment_name != ""
  366. # DC
  367. - name: Set ES dc templates
  368. template:
  369. src: "es.j2"
  370. dest: "{{ tempdir }}/templates/logging-es-dc.yml"
  371. vars:
  372. es_cluster_name: "{{ es_component }}"
  373. component: "{{ es_component }}"
  374. logging_component: elasticsearch
  375. deploy_name: "{{ es_deploy_name }}"
  376. es_cpu_limit: "{{ openshift_logging_elasticsearch_cpu_limit | default('') }}"
  377. es_cpu_request: "{{ openshift_logging_elasticsearch_cpu_request | min_cpu(openshift_logging_elasticsearch_cpu_limit | default(none)) }}"
  378. es_memory_limit: "{{ openshift_logging_elasticsearch_memory_limit }}"
  379. es_node_selector: "{{ openshift_logging_elasticsearch_nodeselector | default({}) }}"
  380. es_storage_groups: "{{ openshift_logging_elasticsearch_storage_group | default([]) }}"
  381. es_container_security_context: "{{ _es_containers.elasticsearch.securityContext if _es_containers is defined and 'elasticsearch' in _es_containers and 'securityContext' in _es_containers.elasticsearch else None }}"
  382. deploy_type: "{{ openshift_logging_elasticsearch_deployment_type }}"
  383. es_replicas: 1
  384. basic_auth_passwd: "{{ ( _logging_metrics_proxy_passwd['content'] | b64decode | from_yaml )[openshift_logging_elasticsearch_prometheus_sa]['passwd'] | b64decode }}"
  385. es_number_of_shards: "{{ openshift_logging_es_number_of_shards | default(1) }}"
  386. es_number_of_replicas: "{{ openshift_logging_es_number_of_replicas| default(0) }}"
  387. - name: Set ES dc
  388. oc_obj:
  389. state: present
  390. name: "{{ es_deploy_name }}"
  391. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  392. kind: dc
  393. files:
  394. - "{{ tempdir }}/templates/logging-es-dc.yml"
  395. delete_after: true
  396. register: es_dc_creation
  397. notify: "restart elasticsearch"
  398. - set_fact:
  399. _restart_logging_components: "{{ _restart_logging_components | default([]) + [es_component] | unique }}"
  400. _restart_logging_nodes: "{{ _restart_logging_nodes | default([]) + [es_deploy_name] | unique }}"
  401. when: es_dc_creation.changed | bool
  402. - name: Retrieving the cert to use when generating secrets for the {{ es_component }} component
  403. slurp:
  404. src: "{{ generated_certs_dir }}/{{ item.file }}"
  405. register: key_pairs
  406. with_items:
  407. - { name: "ca_file", file: "ca.crt" }
  408. - { name: "es_key", file: "system.logging.es.key" }
  409. - { name: "es_cert", file: "system.logging.es.crt" }
  410. when: openshift_logging_es_allow_external | bool
  411. - set_fact:
  412. es_key: "{{ lookup('file', openshift_logging_es_key) | b64encode }}"
  413. when:
  414. - openshift_logging_es_key | trim | length > 0
  415. - openshift_logging_es_allow_external | bool
  416. changed_when: false
  417. - set_fact:
  418. es_cert: "{{ lookup('file', openshift_logging_es_cert) | b64encode }}"
  419. when:
  420. - openshift_logging_es_cert | trim | length > 0
  421. - openshift_logging_es_allow_external | bool
  422. changed_when: false
  423. - set_fact:
  424. es_ca: "{{ lookup('file', openshift_logging_es_ca_ext) | b64encode }}"
  425. when:
  426. - openshift_logging_es_ca_ext | trim | length > 0
  427. - openshift_logging_es_allow_external | bool
  428. changed_when: false
  429. - set_fact:
  430. es_ca: "{{ key_pairs | entry_from_named_pair('ca_file') }}"
  431. when:
  432. - es_ca is not defined
  433. - openshift_logging_es_allow_external | bool
  434. changed_when: false
  435. - name: Generating Elasticsearch {{ es_component }} route template
  436. template:
  437. src: "route_reencrypt.j2"
  438. dest: "{{mktemp.stdout}}/templates/logging-{{ es_component }}-route.yaml"
  439. vars:
  440. obj_name: "logging-{{ es_component }}"
  441. route_host: "{{ openshift_logging_es_hostname }}"
  442. service_name: "logging-{{ es_component }}"
  443. tls_key: "{{ es_key | default('') | b64decode }}"
  444. tls_cert: "{{ es_cert | default('') | b64decode }}"
  445. tls_ca_cert: "{{ es_ca | b64decode }}"
  446. tls_dest_ca_cert: "{{ key_pairs | entry_from_named_pair('ca_file') | b64decode }}"
  447. edge_term_policy: "{{ openshift_logging_es_edge_term_policy | default('') }}"
  448. labels:
  449. component: support
  450. logging-infra: support
  451. provider: openshift
  452. changed_when: no
  453. when: openshift_logging_es_allow_external | bool
  454. # This currently has an issue if the host name changes
  455. - name: Setting Elasticsearch {{ es_component }} route
  456. oc_obj:
  457. state: present
  458. name: "logging-{{ es_component }}"
  459. namespace: "{{ openshift_logging_elasticsearch_namespace }}"
  460. kind: route
  461. files:
  462. - "{{ tempdir }}/templates/logging-{{ es_component }}-route.yaml"
  463. when: openshift_logging_es_allow_external | bool
  464. ## Placeholder for migration when necessary ##
  465. - name: Delete temp directory
  466. file:
  467. name: "{{ tempdir }}"
  468. state: absent
  469. changed_when: False