ansible-core 2.18.5rc1__py3-none-any.whl → 2.19.0b2__py3-none-any.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (710) hide show
  1. ansible/_internal/__init__.py +53 -0
  2. ansible/_internal/_ansiballz.py +265 -0
  3. ansible/_internal/_collection_proxy.py +47 -0
  4. ansible/_internal/_datatag/__init__.py +0 -0
  5. ansible/_internal/_datatag/_tags.py +130 -0
  6. ansible/_internal/_datatag/_utils.py +19 -0
  7. ansible/_internal/_datatag/_wrappers.py +33 -0
  8. ansible/_internal/_errors/__init__.py +0 -0
  9. ansible/_internal/_errors/_captured.py +128 -0
  10. ansible/_internal/_errors/_handler.py +91 -0
  11. ansible/_internal/_errors/_utils.py +310 -0
  12. ansible/_internal/_json/__init__.py +203 -0
  13. ansible/_internal/_json/_legacy_encoder.py +34 -0
  14. ansible/_internal/_json/_profiles/__init__.py +0 -0
  15. ansible/_internal/_json/_profiles/_cache_persistence.py +55 -0
  16. ansible/_internal/_json/_profiles/_inventory_legacy.py +40 -0
  17. ansible/_internal/_json/_profiles/_legacy.py +197 -0
  18. ansible/_internal/_locking.py +21 -0
  19. ansible/_internal/_plugins/__init__.py +0 -0
  20. ansible/_internal/_plugins/_cache.py +57 -0
  21. ansible/_internal/_task.py +78 -0
  22. ansible/_internal/_templating/__init__.py +10 -0
  23. ansible/_internal/_templating/_access.py +86 -0
  24. ansible/_internal/_templating/_chain_templar.py +63 -0
  25. ansible/_internal/_templating/_datatag.py +95 -0
  26. ansible/_internal/_templating/_engine.py +588 -0
  27. ansible/_internal/_templating/_errors.py +28 -0
  28. ansible/_internal/_templating/_jinja_bits.py +1066 -0
  29. ansible/_internal/_templating/_jinja_common.py +332 -0
  30. ansible/_internal/_templating/_jinja_patches.py +44 -0
  31. ansible/_internal/_templating/_jinja_plugins.py +345 -0
  32. ansible/_internal/_templating/_lazy_containers.py +633 -0
  33. ansible/_internal/_templating/_marker_behaviors.py +103 -0
  34. ansible/_internal/_templating/_transform.py +63 -0
  35. ansible/_internal/_templating/_utils.py +107 -0
  36. ansible/_internal/_wrapt.py +1052 -0
  37. ansible/_internal/_yaml/__init__.py +0 -0
  38. ansible/_internal/_yaml/_constructor.py +240 -0
  39. ansible/_internal/_yaml/_dumper.py +62 -0
  40. ansible/_internal/_yaml/_errors.py +166 -0
  41. ansible/_internal/_yaml/_loader.py +66 -0
  42. ansible/_internal/ansible_collections/ansible/_protomatter/README.md +11 -0
  43. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/action/debug.py +36 -0
  44. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/apply_trust.py +19 -0
  45. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/dump_object.py +18 -0
  46. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/finalize.py +16 -0
  47. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/origin.py +18 -0
  48. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/python_literal_eval.py +24 -0
  49. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/python_literal_eval.yml +33 -0
  50. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/tag_names.py +16 -0
  51. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/true_type.py +17 -0
  52. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/filter/unmask.py +49 -0
  53. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/lookup/config.py +21 -0
  54. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/lookup/config.yml +2 -0
  55. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/test/tagged.py +15 -0
  56. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/test/tagged.yml +19 -0
  57. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/test/tagged_with.py +18 -0
  58. ansible/_internal/ansible_collections/ansible/_protomatter/plugins/test/tagged_with.yml +19 -0
  59. ansible/cli/__init__.py +159 -89
  60. ansible/cli/_ssh_askpass.py +47 -0
  61. ansible/cli/adhoc.py +14 -7
  62. ansible/cli/arguments/option_helpers.py +154 -7
  63. ansible/cli/config.py +43 -68
  64. ansible/cli/console.py +10 -8
  65. ansible/cli/doc.py +62 -53
  66. ansible/cli/galaxy.py +27 -20
  67. ansible/cli/inventory.py +28 -26
  68. ansible/cli/playbook.py +4 -12
  69. ansible/cli/pull.py +51 -11
  70. ansible/cli/scripts/ansible_connection_cli_stub.py +7 -7
  71. ansible/cli/vault.py +12 -11
  72. ansible/compat/__init__.py +2 -2
  73. ansible/config/base.yml +166 -112
  74. ansible/config/manager.py +52 -49
  75. ansible/constants.py +3 -4
  76. ansible/errors/__init__.py +277 -235
  77. ansible/executor/interpreter_discovery.py +28 -149
  78. ansible/executor/module_common.py +426 -493
  79. ansible/executor/play_iterator.py +22 -27
  80. ansible/executor/playbook_executor.py +11 -11
  81. ansible/executor/powershell/async_watchdog.ps1 +97 -102
  82. ansible/executor/powershell/async_wrapper.ps1 +202 -151
  83. ansible/executor/powershell/become_wrapper.ps1 +89 -144
  84. ansible/executor/powershell/bootstrap_wrapper.ps1 +24 -9
  85. ansible/executor/powershell/coverage_wrapper.ps1 +82 -135
  86. ansible/executor/powershell/exec_wrapper.ps1 +462 -196
  87. ansible/executor/powershell/module_manifest.py +417 -265
  88. ansible/executor/powershell/module_wrapper.ps1 +169 -186
  89. ansible/executor/powershell/psrp_fetch_file.ps1 +41 -0
  90. ansible/executor/powershell/psrp_put_file.ps1 +122 -0
  91. ansible/executor/powershell/winrm_fetch_file.ps1 +46 -0
  92. ansible/executor/powershell/winrm_put_file.ps1 +36 -0
  93. ansible/executor/process/worker.py +161 -96
  94. ansible/executor/stats.py +5 -5
  95. ansible/executor/task_executor.py +268 -258
  96. ansible/executor/task_queue_manager.py +124 -90
  97. ansible/executor/task_result.py +183 -78
  98. ansible/galaxy/__init__.py +2 -2
  99. ansible/galaxy/api.py +22 -18
  100. ansible/galaxy/collection/__init__.py +1 -1
  101. ansible/galaxy/collection/concrete_artifact_manager.py +8 -11
  102. ansible/galaxy/dependency_resolution/dataclasses.py +14 -4
  103. ansible/galaxy/dependency_resolution/providers.py +1 -1
  104. ansible/galaxy/dependency_resolution/reporters.py +81 -0
  105. ansible/galaxy/role.py +4 -8
  106. ansible/galaxy/token.py +28 -21
  107. ansible/inventory/data.py +47 -57
  108. ansible/inventory/group.py +44 -72
  109. ansible/inventory/helpers.py +9 -0
  110. ansible/inventory/host.py +32 -54
  111. ansible/inventory/manager.py +78 -34
  112. ansible/keyword_desc.yml +1 -1
  113. ansible/module_utils/_internal/__init__.py +55 -0
  114. ansible/module_utils/_internal/_ambient_context.py +58 -0
  115. ansible/module_utils/_internal/_ansiballz.py +133 -0
  116. ansible/module_utils/_internal/_concurrent/_daemon_threading.py +1 -0
  117. ansible/module_utils/_internal/_dataclass_annotation_patch.py +64 -0
  118. ansible/module_utils/_internal/_dataclass_validation.py +217 -0
  119. ansible/module_utils/_internal/_datatag/__init__.py +928 -0
  120. ansible/module_utils/_internal/_datatag/_tags.py +38 -0
  121. ansible/module_utils/_internal/_debugging.py +31 -0
  122. ansible/module_utils/_internal/_errors.py +30 -0
  123. ansible/module_utils/_internal/_json/__init__.py +63 -0
  124. ansible/module_utils/_internal/_json/_legacy_encoder.py +26 -0
  125. ansible/module_utils/_internal/_json/_profiles/__init__.py +410 -0
  126. ansible/module_utils/_internal/_json/_profiles/_fallback_to_str.py +73 -0
  127. ansible/module_utils/_internal/_json/_profiles/_module_legacy_c2m.py +31 -0
  128. ansible/module_utils/_internal/_json/_profiles/_module_legacy_m2c.py +35 -0
  129. ansible/module_utils/_internal/_json/_profiles/_module_modern_c2m.py +35 -0
  130. ansible/module_utils/_internal/_json/_profiles/_module_modern_m2c.py +33 -0
  131. ansible/module_utils/_internal/_json/_profiles/_tagless.py +50 -0
  132. ansible/module_utils/_internal/_patches/__init__.py +66 -0
  133. ansible/module_utils/_internal/_patches/_dataclass_annotation_patch.py +55 -0
  134. ansible/module_utils/_internal/_patches/_socket_patch.py +34 -0
  135. ansible/module_utils/_internal/_patches/_sys_intern_patch.py +34 -0
  136. ansible/module_utils/_internal/_plugin_exec_context.py +49 -0
  137. ansible/module_utils/_internal/_testing.py +0 -0
  138. ansible/module_utils/_internal/_traceback.py +89 -0
  139. ansible/module_utils/ansible_release.py +2 -2
  140. ansible/module_utils/api.py +1 -2
  141. ansible/module_utils/basic.py +152 -120
  142. ansible/module_utils/common/_utils.py +24 -28
  143. ansible/module_utils/common/collections.py +1 -2
  144. ansible/module_utils/common/dict_transformations.py +2 -2
  145. ansible/module_utils/common/file.py +2 -2
  146. ansible/module_utils/common/json.py +90 -84
  147. ansible/module_utils/common/locale.py +2 -2
  148. ansible/module_utils/common/messages.py +108 -0
  149. ansible/module_utils/common/parameters.py +27 -24
  150. ansible/module_utils/common/process.py +2 -2
  151. ansible/module_utils/common/respawn.py +41 -19
  152. ansible/module_utils/common/sentinel.py +66 -0
  153. ansible/module_utils/common/sys_info.py +8 -8
  154. ansible/module_utils/common/text/converters.py +16 -37
  155. ansible/module_utils/common/validation.py +35 -24
  156. ansible/module_utils/common/warnings.py +86 -25
  157. ansible/module_utils/common/yaml.py +29 -3
  158. ansible/module_utils/compat/datetime.py +33 -21
  159. ansible/module_utils/compat/paramiko.py +21 -10
  160. ansible/module_utils/compat/typing.py +6 -5
  161. ansible/module_utils/connection.py +2 -2
  162. ansible/module_utils/csharp/Ansible.Basic.cs +14 -11
  163. ansible/module_utils/csharp/Ansible.Become.cs +1 -0
  164. ansible/module_utils/csharp/Ansible._Async.cs +517 -0
  165. ansible/module_utils/datatag.py +46 -0
  166. ansible/module_utils/distro/__init__.py +2 -2
  167. ansible/module_utils/facts/ansible_collector.py +4 -5
  168. ansible/module_utils/facts/collector.py +13 -14
  169. ansible/module_utils/facts/compat.py +4 -4
  170. ansible/module_utils/facts/default_collectors.py +1 -1
  171. ansible/module_utils/facts/hardware/aix.py +34 -0
  172. ansible/module_utils/facts/hardware/base.py +1 -1
  173. ansible/module_utils/facts/hardware/darwin.py +1 -3
  174. ansible/module_utils/facts/hardware/freebsd.py +2 -2
  175. ansible/module_utils/facts/hardware/linux.py +4 -4
  176. ansible/module_utils/facts/namespace.py +1 -1
  177. ansible/module_utils/facts/network/base.py +1 -1
  178. ansible/module_utils/facts/network/fc_wwn.py +1 -2
  179. ansible/module_utils/facts/network/iscsi.py +1 -2
  180. ansible/module_utils/facts/network/nvme.py +1 -2
  181. ansible/module_utils/facts/other/facter.py +1 -2
  182. ansible/module_utils/facts/other/ohai.py +2 -3
  183. ansible/module_utils/facts/system/apparmor.py +1 -2
  184. ansible/module_utils/facts/system/caps.py +1 -1
  185. ansible/module_utils/facts/system/chroot.py +1 -2
  186. ansible/module_utils/facts/system/cmdline.py +1 -2
  187. ansible/module_utils/facts/system/date_time.py +5 -3
  188. ansible/module_utils/facts/system/distribution.py +9 -8
  189. ansible/module_utils/facts/system/dns.py +1 -1
  190. ansible/module_utils/facts/system/env.py +1 -2
  191. ansible/module_utils/facts/system/fips.py +7 -20
  192. ansible/module_utils/facts/system/loadavg.py +1 -2
  193. ansible/module_utils/facts/system/local.py +1 -2
  194. ansible/module_utils/facts/system/lsb.py +1 -2
  195. ansible/module_utils/facts/system/pkg_mgr.py +1 -2
  196. ansible/module_utils/facts/system/platform.py +1 -2
  197. ansible/module_utils/facts/system/python.py +1 -2
  198. ansible/module_utils/facts/system/selinux.py +1 -1
  199. ansible/module_utils/facts/system/service_mgr.py +1 -2
  200. ansible/module_utils/facts/system/ssh_pub_keys.py +1 -1
  201. ansible/module_utils/facts/system/systemd.py +1 -1
  202. ansible/module_utils/facts/system/user.py +1 -2
  203. ansible/module_utils/facts/utils.py +3 -3
  204. ansible/module_utils/facts/virtual/base.py +1 -1
  205. ansible/module_utils/facts/virtual/sunos.py +3 -15
  206. ansible/module_utils/facts/virtual/sysctl.py +3 -16
  207. ansible/module_utils/json_utils.py +2 -2
  208. ansible/module_utils/parsing/convert_bool.py +1 -1
  209. ansible/module_utils/service.py +18 -21
  210. ansible/module_utils/splitter.py +7 -7
  211. ansible/module_utils/testing.py +31 -0
  212. ansible/module_utils/urls.py +60 -31
  213. ansible/modules/add_host.py +4 -4
  214. ansible/modules/apt.py +60 -46
  215. ansible/modules/apt_key.py +19 -12
  216. ansible/modules/apt_repository.py +19 -16
  217. ansible/modules/assemble.py +6 -6
  218. ansible/modules/assert.py +4 -4
  219. ansible/modules/async_status.py +10 -12
  220. ansible/modules/async_wrapper.py +8 -3
  221. ansible/modules/blockinfile.py +6 -7
  222. ansible/modules/command.py +10 -17
  223. ansible/modules/copy.py +57 -144
  224. ansible/modules/cron.py +20 -15
  225. ansible/modules/deb822_repository.py +8 -9
  226. ansible/modules/debconf.py +5 -5
  227. ansible/modules/debug.py +4 -4
  228. ansible/modules/dnf.py +8 -8
  229. ansible/modules/dnf5.py +39 -13
  230. ansible/modules/dpkg_selections.py +4 -4
  231. ansible/modules/expect.py +8 -10
  232. ansible/modules/fail.py +4 -4
  233. ansible/modules/fetch.py +4 -4
  234. ansible/modules/file.py +174 -133
  235. ansible/modules/find.py +19 -17
  236. ansible/modules/gather_facts.py +3 -3
  237. ansible/modules/get_url.py +59 -53
  238. ansible/modules/getent.py +7 -9
  239. ansible/modules/git.py +28 -25
  240. ansible/modules/group.py +6 -6
  241. ansible/modules/group_by.py +4 -4
  242. ansible/modules/hostname.py +13 -29
  243. ansible/modules/import_playbook.py +6 -6
  244. ansible/modules/import_role.py +6 -6
  245. ansible/modules/import_tasks.py +6 -6
  246. ansible/modules/include_role.py +6 -6
  247. ansible/modules/include_tasks.py +6 -6
  248. ansible/modules/include_vars.py +6 -6
  249. ansible/modules/iptables.py +86 -73
  250. ansible/modules/known_hosts.py +10 -10
  251. ansible/modules/lineinfile.py +5 -5
  252. ansible/modules/meta.py +4 -4
  253. ansible/modules/mount_facts.py +2 -2
  254. ansible/modules/package.py +4 -4
  255. ansible/modules/package_facts.py +22 -10
  256. ansible/modules/pause.py +6 -6
  257. ansible/modules/ping.py +6 -6
  258. ansible/modules/pip.py +10 -11
  259. ansible/modules/raw.py +4 -4
  260. ansible/modules/reboot.py +6 -6
  261. ansible/modules/replace.py +9 -13
  262. ansible/modules/rpm_key.py +7 -8
  263. ansible/modules/script.py +4 -4
  264. ansible/modules/service.py +7 -8
  265. ansible/modules/service_facts.py +87 -10
  266. ansible/modules/set_fact.py +5 -5
  267. ansible/modules/set_stats.py +4 -4
  268. ansible/modules/setup.py +2 -2
  269. ansible/modules/shell.py +6 -6
  270. ansible/modules/slurp.py +6 -6
  271. ansible/modules/stat.py +9 -23
  272. ansible/modules/subversion.py +15 -15
  273. ansible/modules/systemd.py +6 -6
  274. ansible/modules/systemd_service.py +6 -6
  275. ansible/modules/sysvinit.py +6 -6
  276. ansible/modules/tempfile.py +5 -6
  277. ansible/modules/template.py +6 -6
  278. ansible/modules/unarchive.py +32 -11
  279. ansible/modules/uri.py +33 -26
  280. ansible/modules/user.py +53 -34
  281. ansible/modules/validate_argument_spec.py +10 -7
  282. ansible/modules/wait_for.py +32 -27
  283. ansible/modules/wait_for_connection.py +6 -6
  284. ansible/modules/yum_repository.py +6 -6
  285. ansible/parsing/ajson.py +14 -32
  286. ansible/parsing/dataloader.py +99 -54
  287. ansible/parsing/mod_args.py +28 -44
  288. ansible/parsing/plugin_docs.py +21 -86
  289. ansible/parsing/quoting.py +1 -1
  290. ansible/parsing/splitter.py +27 -12
  291. ansible/parsing/utils/addresses.py +24 -24
  292. ansible/parsing/utils/jsonify.py +5 -1
  293. ansible/parsing/utils/yaml.py +32 -61
  294. ansible/parsing/vault/__init__.py +319 -87
  295. ansible/parsing/yaml/__init__.py +0 -18
  296. ansible/parsing/yaml/dumper.py +6 -120
  297. ansible/parsing/yaml/loader.py +6 -39
  298. ansible/parsing/yaml/objects.py +43 -335
  299. ansible/playbook/__init__.py +1 -1
  300. ansible/playbook/attribute.py +8 -3
  301. ansible/playbook/base.py +182 -132
  302. ansible/playbook/block.py +26 -24
  303. ansible/playbook/collectionsearch.py +1 -15
  304. ansible/playbook/conditional.py +3 -77
  305. ansible/playbook/handler.py +8 -2
  306. ansible/playbook/helpers.py +41 -53
  307. ansible/playbook/included_file.py +31 -27
  308. ansible/playbook/loop_control.py +2 -2
  309. ansible/playbook/play.py +85 -44
  310. ansible/playbook/play_context.py +12 -17
  311. ansible/playbook/playbook_include.py +14 -15
  312. ansible/playbook/role/__init__.py +24 -26
  313. ansible/playbook/role/definition.py +15 -17
  314. ansible/playbook/role/include.py +2 -4
  315. ansible/playbook/role/metadata.py +10 -11
  316. ansible/playbook/role_include.py +3 -3
  317. ansible/playbook/taggable.py +13 -8
  318. ansible/playbook/task.py +188 -118
  319. ansible/playbook/task_include.py +5 -5
  320. ansible/plugins/__init__.py +68 -21
  321. ansible/plugins/action/__init__.py +209 -176
  322. ansible/plugins/action/add_host.py +1 -1
  323. ansible/plugins/action/assemble.py +1 -1
  324. ansible/plugins/action/assert.py +54 -66
  325. ansible/plugins/action/copy.py +7 -11
  326. ansible/plugins/action/debug.py +37 -31
  327. ansible/plugins/action/dnf.py +3 -4
  328. ansible/plugins/action/fail.py +1 -1
  329. ansible/plugins/action/fetch.py +4 -5
  330. ansible/plugins/action/gather_facts.py +7 -6
  331. ansible/plugins/action/group_by.py +1 -1
  332. ansible/plugins/action/include_vars.py +10 -11
  333. ansible/plugins/action/package.py +3 -6
  334. ansible/plugins/action/pause.py +2 -2
  335. ansible/plugins/action/script.py +15 -8
  336. ansible/plugins/action/service.py +6 -11
  337. ansible/plugins/action/set_fact.py +3 -12
  338. ansible/plugins/action/set_stats.py +3 -8
  339. ansible/plugins/action/template.py +35 -59
  340. ansible/plugins/action/unarchive.py +1 -1
  341. ansible/plugins/action/validate_argument_spec.py +5 -5
  342. ansible/plugins/action/wait_for_connection.py +1 -1
  343. ansible/plugins/become/__init__.py +31 -8
  344. ansible/plugins/become/runas.py +71 -0
  345. ansible/plugins/become/su.py +13 -8
  346. ansible/plugins/become/sudo.py +19 -0
  347. ansible/plugins/cache/__init__.py +35 -44
  348. ansible/plugins/cache/base.py +8 -0
  349. ansible/plugins/cache/jsonfile.py +10 -16
  350. ansible/plugins/cache/memory.py +6 -12
  351. ansible/plugins/callback/__init__.py +284 -179
  352. ansible/plugins/callback/default.py +99 -92
  353. ansible/plugins/callback/junit.py +44 -39
  354. ansible/plugins/callback/minimal.py +28 -25
  355. ansible/plugins/callback/oneline.py +28 -21
  356. ansible/plugins/callback/tree.py +16 -11
  357. ansible/plugins/connection/__init__.py +47 -34
  358. ansible/plugins/connection/local.py +150 -54
  359. ansible/plugins/connection/paramiko_ssh.py +21 -18
  360. ansible/plugins/connection/psrp.py +76 -165
  361. ansible/plugins/connection/ssh.py +301 -78
  362. ansible/plugins/connection/winrm.py +58 -140
  363. ansible/plugins/doc_fragments/action_common_attributes.py +14 -14
  364. ansible/plugins/doc_fragments/action_core.py +6 -6
  365. ansible/plugins/doc_fragments/backup.py +2 -2
  366. ansible/plugins/doc_fragments/checksum_common.py +27 -0
  367. ansible/plugins/doc_fragments/constructed.py +6 -2
  368. ansible/plugins/doc_fragments/decrypt.py +2 -2
  369. ansible/plugins/doc_fragments/default_callback.py +2 -2
  370. ansible/plugins/doc_fragments/files.py +2 -2
  371. ansible/plugins/doc_fragments/inventory_cache.py +2 -2
  372. ansible/plugins/doc_fragments/result_format_callback.py +2 -2
  373. ansible/plugins/doc_fragments/return_common.py +2 -2
  374. ansible/plugins/doc_fragments/template_common.py +4 -4
  375. ansible/plugins/doc_fragments/url.py +17 -1
  376. ansible/plugins/doc_fragments/url_windows.py +2 -2
  377. ansible/plugins/doc_fragments/validate.py +2 -2
  378. ansible/plugins/doc_fragments/vars_plugin_staging.py +2 -2
  379. ansible/plugins/filter/__init__.py +6 -2
  380. ansible/plugins/filter/b64decode.yml +22 -0
  381. ansible/plugins/filter/b64encode.yml +22 -0
  382. ansible/plugins/filter/bool.yml +11 -4
  383. ansible/plugins/filter/core.py +225 -108
  384. ansible/plugins/filter/encryption.py +32 -32
  385. ansible/plugins/filter/flatten.yml +3 -2
  386. ansible/plugins/filter/human_to_bytes.yml +1 -1
  387. ansible/plugins/filter/mathstuff.py +30 -37
  388. ansible/plugins/filter/password_hash.yml +8 -0
  389. ansible/plugins/filter/regex_search.yml +1 -4
  390. ansible/plugins/filter/split.yml +1 -1
  391. ansible/plugins/filter/to_nice_yaml.yml +0 -4
  392. ansible/plugins/filter/to_yaml.yml +0 -4
  393. ansible/plugins/filter/unvault.yml +1 -1
  394. ansible/plugins/filter/urls.py +1 -1
  395. ansible/plugins/filter/urlsplit.py +8 -9
  396. ansible/plugins/filter/vault.yml +14 -9
  397. ansible/plugins/filter/win_basename.yml +6 -1
  398. ansible/plugins/filter/win_dirname.yml +5 -0
  399. ansible/plugins/inventory/__init__.py +97 -77
  400. ansible/plugins/inventory/advanced_host_list.py +7 -5
  401. ansible/plugins/inventory/auto.py +11 -4
  402. ansible/plugins/inventory/constructed.py +21 -24
  403. ansible/plugins/inventory/generator.py +16 -11
  404. ansible/plugins/inventory/host_list.py +7 -5
  405. ansible/plugins/inventory/ini.py +78 -44
  406. ansible/plugins/inventory/script.py +189 -119
  407. ansible/plugins/inventory/toml.py +16 -126
  408. ansible/plugins/inventory/yaml.py +10 -8
  409. ansible/plugins/list.py +3 -3
  410. ansible/plugins/loader.py +197 -82
  411. ansible/plugins/lookup/__init__.py +21 -4
  412. ansible/plugins/lookup/config.py +21 -35
  413. ansible/plugins/lookup/csvfile.py +3 -2
  414. ansible/plugins/lookup/dict.py +1 -6
  415. ansible/plugins/lookup/env.py +12 -9
  416. ansible/plugins/lookup/file.py +5 -8
  417. ansible/plugins/lookup/first_found.py +86 -55
  418. ansible/plugins/lookup/indexed_items.py +1 -10
  419. ansible/plugins/lookup/ini.py +14 -13
  420. ansible/plugins/lookup/items.py +1 -1
  421. ansible/plugins/lookup/lines.py +8 -1
  422. ansible/plugins/lookup/list.py +1 -1
  423. ansible/plugins/lookup/nested.py +2 -18
  424. ansible/plugins/lookup/password.py +5 -5
  425. ansible/plugins/lookup/pipe.py +5 -7
  426. ansible/plugins/lookup/sequence.py +18 -8
  427. ansible/plugins/lookup/subelements.py +1 -4
  428. ansible/plugins/lookup/template.py +42 -36
  429. ansible/plugins/lookup/together.py +0 -12
  430. ansible/plugins/lookup/unvault.py +1 -5
  431. ansible/plugins/lookup/url.py +2 -8
  432. ansible/plugins/lookup/vars.py +16 -24
  433. ansible/plugins/shell/__init__.py +2 -2
  434. ansible/plugins/shell/cmd.py +2 -2
  435. ansible/plugins/shell/powershell.py +39 -22
  436. ansible/plugins/shell/sh.py +3 -2
  437. ansible/plugins/strategy/__init__.py +159 -184
  438. ansible/plugins/strategy/debug.py +2 -2
  439. ansible/plugins/strategy/free.py +16 -31
  440. ansible/plugins/strategy/host_pinned.py +2 -2
  441. ansible/plugins/strategy/linear.py +41 -41
  442. ansible/plugins/terminal/__init__.py +4 -4
  443. ansible/plugins/test/__init__.py +7 -2
  444. ansible/plugins/test/core.py +55 -21
  445. ansible/plugins/test/files.py +1 -1
  446. ansible/plugins/test/mathstuff.py +3 -3
  447. ansible/plugins/test/uri.py +3 -3
  448. ansible/plugins/vars/host_group_vars.py +7 -14
  449. ansible/release.py +2 -2
  450. ansible/template/__init__.py +370 -944
  451. ansible/utils/__init__.py +0 -18
  452. ansible/utils/_ssh_agent.py +657 -0
  453. ansible/utils/collection_loader/__init__.py +52 -5
  454. ansible/utils/collection_loader/_collection_config.py +5 -6
  455. ansible/utils/collection_loader/_collection_finder.py +79 -93
  456. ansible/utils/collection_loader/_collection_meta.py +13 -8
  457. ansible/utils/display.py +433 -63
  458. ansible/utils/encrypt.py +27 -19
  459. ansible/utils/fqcn.py +2 -2
  460. ansible/utils/hashing.py +2 -2
  461. ansible/utils/helpers.py +2 -2
  462. ansible/utils/listify.py +8 -8
  463. ansible/utils/lock.py +2 -2
  464. ansible/utils/path.py +4 -4
  465. ansible/utils/plugin_docs.py +14 -13
  466. ansible/utils/sentinel.py +4 -62
  467. ansible/utils/singleton.py +2 -0
  468. ansible/utils/ssh_functions.py +1 -1
  469. ansible/utils/unsafe_proxy.py +23 -332
  470. ansible/utils/vars.py +51 -8
  471. ansible/utils/version.py +2 -2
  472. ansible/vars/clean.py +5 -5
  473. ansible/vars/hostvars.py +60 -90
  474. ansible/vars/manager.py +206 -282
  475. ansible/vars/reserved.py +8 -9
  476. ansible_core-2.19.0b2.dist-info/BSD-3-Clause.txt +28 -0
  477. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/METADATA +5 -4
  478. ansible_core-2.19.0b2.dist-info/RECORD +1072 -0
  479. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/WHEEL +1 -1
  480. ansible_test/_data/completion/docker.txt +7 -7
  481. ansible_test/_data/completion/remote.txt +6 -6
  482. ansible_test/_data/completion/windows.txt +1 -0
  483. ansible_test/_data/requirements/ansible.txt +2 -2
  484. ansible_test/_data/requirements/sanity.ansible-doc.txt +3 -3
  485. ansible_test/_data/requirements/sanity.changelog.txt +1 -1
  486. ansible_test/_data/requirements/sanity.import.plugin.txt +2 -2
  487. ansible_test/_data/requirements/sanity.pylint.txt +4 -4
  488. ansible_test/_data/requirements/sanity.validate-modules.txt +2 -2
  489. ansible_test/_data/requirements/sanity.yamllint.txt +1 -1
  490. ansible_test/_data/requirements/units.txt +1 -0
  491. ansible_test/_internal/__init__.py +1 -0
  492. ansible_test/_internal/ansible_util.py +2 -0
  493. ansible_test/_internal/become.py +1 -0
  494. ansible_test/_internal/bootstrap.py +1 -0
  495. ansible_test/_internal/cache.py +1 -0
  496. ansible_test/_internal/cgroup.py +1 -0
  497. ansible_test/_internal/ci/__init__.py +1 -0
  498. ansible_test/_internal/ci/azp.py +1 -0
  499. ansible_test/_internal/ci/local.py +1 -0
  500. ansible_test/_internal/classification/__init__.py +1 -0
  501. ansible_test/_internal/classification/common.py +1 -0
  502. ansible_test/_internal/classification/csharp.py +1 -0
  503. ansible_test/_internal/classification/powershell.py +1 -0
  504. ansible_test/_internal/classification/python.py +1 -0
  505. ansible_test/_internal/cli/__init__.py +1 -0
  506. ansible_test/_internal/cli/actions.py +1 -0
  507. ansible_test/_internal/cli/argparsing/__init__.py +1 -0
  508. ansible_test/_internal/cli/argparsing/actions.py +1 -0
  509. ansible_test/_internal/cli/argparsing/argcompletion.py +1 -0
  510. ansible_test/_internal/cli/argparsing/parsers.py +1 -0
  511. ansible_test/_internal/cli/commands/__init__.py +11 -0
  512. ansible_test/_internal/cli/commands/coverage/__init__.py +1 -0
  513. ansible_test/_internal/cli/commands/coverage/analyze/__init__.py +1 -0
  514. ansible_test/_internal/cli/commands/coverage/analyze/targets/__init__.py +1 -0
  515. ansible_test/_internal/cli/commands/coverage/analyze/targets/combine.py +1 -0
  516. ansible_test/_internal/cli/commands/coverage/analyze/targets/expand.py +1 -0
  517. ansible_test/_internal/cli/commands/coverage/analyze/targets/filter.py +1 -0
  518. ansible_test/_internal/cli/commands/coverage/analyze/targets/generate.py +1 -0
  519. ansible_test/_internal/cli/commands/coverage/analyze/targets/missing.py +1 -0
  520. ansible_test/_internal/cli/commands/coverage/combine.py +1 -0
  521. ansible_test/_internal/cli/commands/coverage/erase.py +1 -0
  522. ansible_test/_internal/cli/commands/coverage/html.py +1 -0
  523. ansible_test/_internal/cli/commands/coverage/report.py +1 -0
  524. ansible_test/_internal/cli/commands/coverage/xml.py +1 -0
  525. ansible_test/_internal/cli/commands/env.py +1 -0
  526. ansible_test/_internal/cli/commands/integration/__init__.py +1 -0
  527. ansible_test/_internal/cli/commands/integration/network.py +1 -0
  528. ansible_test/_internal/cli/commands/integration/posix.py +1 -0
  529. ansible_test/_internal/cli/commands/integration/windows.py +1 -0
  530. ansible_test/_internal/cli/commands/sanity.py +9 -0
  531. ansible_test/_internal/cli/commands/shell.py +1 -0
  532. ansible_test/_internal/cli/commands/units.py +1 -0
  533. ansible_test/_internal/cli/compat.py +1 -0
  534. ansible_test/_internal/cli/completers.py +1 -0
  535. ansible_test/_internal/cli/converters.py +1 -0
  536. ansible_test/_internal/cli/environments.py +1 -0
  537. ansible_test/_internal/cli/epilog.py +1 -0
  538. ansible_test/_internal/cli/parsers/__init__.py +1 -0
  539. ansible_test/_internal/cli/parsers/base_argument_parsers.py +1 -0
  540. ansible_test/_internal/cli/parsers/helpers.py +1 -0
  541. ansible_test/_internal/cli/parsers/host_config_parsers.py +1 -0
  542. ansible_test/_internal/cli/parsers/key_value_parsers.py +1 -0
  543. ansible_test/_internal/cli/parsers/value_parsers.py +1 -0
  544. ansible_test/_internal/commands/__init__.py +1 -0
  545. ansible_test/_internal/commands/coverage/__init__.py +2 -1
  546. ansible_test/_internal/commands/coverage/analyze/__init__.py +1 -0
  547. ansible_test/_internal/commands/coverage/analyze/targets/__init__.py +1 -0
  548. ansible_test/_internal/commands/coverage/analyze/targets/combine.py +1 -0
  549. ansible_test/_internal/commands/coverage/analyze/targets/expand.py +1 -0
  550. ansible_test/_internal/commands/coverage/analyze/targets/filter.py +1 -0
  551. ansible_test/_internal/commands/coverage/analyze/targets/generate.py +1 -0
  552. ansible_test/_internal/commands/coverage/analyze/targets/missing.py +1 -0
  553. ansible_test/_internal/commands/coverage/combine.py +2 -1
  554. ansible_test/_internal/commands/coverage/erase.py +1 -0
  555. ansible_test/_internal/commands/coverage/html.py +1 -0
  556. ansible_test/_internal/commands/coverage/report.py +1 -0
  557. ansible_test/_internal/commands/coverage/xml.py +1 -0
  558. ansible_test/_internal/commands/env/__init__.py +2 -0
  559. ansible_test/_internal/commands/integration/__init__.py +4 -0
  560. ansible_test/_internal/commands/integration/cloud/__init__.py +1 -0
  561. ansible_test/_internal/commands/integration/cloud/acme.py +2 -1
  562. ansible_test/_internal/commands/integration/cloud/aws.py +1 -0
  563. ansible_test/_internal/commands/integration/cloud/azure.py +1 -0
  564. ansible_test/_internal/commands/integration/cloud/cs.py +1 -0
  565. ansible_test/_internal/commands/integration/cloud/digitalocean.py +1 -0
  566. ansible_test/_internal/commands/integration/cloud/galaxy.py +3 -2
  567. ansible_test/_internal/commands/integration/cloud/hcloud.py +1 -0
  568. ansible_test/_internal/commands/integration/cloud/httptester.py +2 -1
  569. ansible_test/_internal/commands/integration/cloud/nios.py +2 -1
  570. ansible_test/_internal/commands/integration/cloud/opennebula.py +1 -0
  571. ansible_test/_internal/commands/integration/cloud/openshift.py +1 -0
  572. ansible_test/_internal/commands/integration/cloud/scaleway.py +1 -0
  573. ansible_test/_internal/commands/integration/cloud/vcenter.py +1 -0
  574. ansible_test/_internal/commands/integration/cloud/vultr.py +1 -0
  575. ansible_test/_internal/commands/integration/coverage.py +1 -0
  576. ansible_test/_internal/commands/integration/filters.py +1 -0
  577. ansible_test/_internal/commands/integration/network.py +1 -0
  578. ansible_test/_internal/commands/integration/posix.py +1 -0
  579. ansible_test/_internal/commands/integration/windows.py +1 -0
  580. ansible_test/_internal/commands/sanity/__init__.py +16 -1
  581. ansible_test/_internal/commands/sanity/ansible_doc.py +1 -0
  582. ansible_test/_internal/commands/sanity/bin_symlinks.py +1 -0
  583. ansible_test/_internal/commands/sanity/compile.py +1 -0
  584. ansible_test/_internal/commands/sanity/ignores.py +1 -0
  585. ansible_test/_internal/commands/sanity/import.py +1 -0
  586. ansible_test/_internal/commands/sanity/integration_aliases.py +1 -0
  587. ansible_test/_internal/commands/sanity/pep8.py +1 -0
  588. ansible_test/_internal/commands/sanity/pslint.py +1 -0
  589. ansible_test/_internal/commands/sanity/pylint.py +24 -26
  590. ansible_test/_internal/commands/sanity/shellcheck.py +1 -0
  591. ansible_test/_internal/commands/sanity/validate_modules.py +1 -0
  592. ansible_test/_internal/commands/sanity/yamllint.py +1 -0
  593. ansible_test/_internal/commands/shell/__init__.py +1 -0
  594. ansible_test/_internal/commands/units/__init__.py +1 -0
  595. ansible_test/_internal/compat/__init__.py +1 -0
  596. ansible_test/_internal/compat/packaging.py +1 -0
  597. ansible_test/_internal/compat/yaml.py +1 -0
  598. ansible_test/_internal/completion.py +1 -0
  599. ansible_test/_internal/config.py +2 -0
  600. ansible_test/_internal/connections.py +1 -0
  601. ansible_test/_internal/constants.py +1 -0
  602. ansible_test/_internal/containers.py +1 -0
  603. ansible_test/_internal/content_config.py +1 -0
  604. ansible_test/_internal/core_ci.py +1 -0
  605. ansible_test/_internal/coverage_util.py +11 -10
  606. ansible_test/_internal/data.py +1 -0
  607. ansible_test/_internal/delegation.py +1 -0
  608. ansible_test/_internal/dev/__init__.py +1 -0
  609. ansible_test/_internal/dev/container_probe.py +1 -0
  610. ansible_test/_internal/diff.py +3 -2
  611. ansible_test/_internal/docker_util.py +2 -1
  612. ansible_test/_internal/encoding.py +1 -0
  613. ansible_test/_internal/executor.py +1 -0
  614. ansible_test/_internal/git.py +1 -0
  615. ansible_test/_internal/host_configs.py +1 -0
  616. ansible_test/_internal/host_profiles.py +1 -0
  617. ansible_test/_internal/http.py +1 -0
  618. ansible_test/_internal/init.py +1 -0
  619. ansible_test/_internal/inventory.py +35 -3
  620. ansible_test/_internal/io.py +1 -0
  621. ansible_test/_internal/metadata.py +1 -0
  622. ansible_test/_internal/payload.py +1 -0
  623. ansible_test/_internal/provider/__init__.py +1 -0
  624. ansible_test/_internal/provider/layout/__init__.py +1 -0
  625. ansible_test/_internal/provider/layout/ansible.py +1 -0
  626. ansible_test/_internal/provider/layout/collection.py +1 -0
  627. ansible_test/_internal/provider/layout/unsupported.py +1 -0
  628. ansible_test/_internal/provider/source/__init__.py +1 -0
  629. ansible_test/_internal/provider/source/git.py +1 -0
  630. ansible_test/_internal/provider/source/installed.py +1 -0
  631. ansible_test/_internal/provider/source/unsupported.py +1 -0
  632. ansible_test/_internal/provider/source/unversioned.py +1 -0
  633. ansible_test/_internal/provisioning.py +1 -0
  634. ansible_test/_internal/pypi_proxy.py +6 -5
  635. ansible_test/_internal/python_requirements.py +1 -0
  636. ansible_test/_internal/ssh.py +1 -0
  637. ansible_test/_internal/target.py +1 -0
  638. ansible_test/_internal/test.py +3 -2
  639. ansible_test/_internal/thread.py +1 -0
  640. ansible_test/_internal/timeout.py +1 -0
  641. ansible_test/_internal/util.py +1 -0
  642. ansible_test/_internal/util_common.py +5 -2
  643. ansible_test/_internal/venv.py +1 -0
  644. ansible_test/_util/controller/sanity/code-smell/action-plugin-docs.py +1 -0
  645. ansible_test/_util/controller/sanity/code-smell/changelog/sphinx.py +1 -0
  646. ansible_test/_util/controller/sanity/code-smell/changelog.py +1 -0
  647. ansible_test/_util/controller/sanity/code-smell/empty-init.py +1 -0
  648. ansible_test/_util/controller/sanity/code-smell/line-endings.py +1 -0
  649. ansible_test/_util/controller/sanity/code-smell/no-assert.py +1 -0
  650. ansible_test/_util/controller/sanity/code-smell/no-get-exception.py +1 -0
  651. ansible_test/_util/controller/sanity/code-smell/no-illegal-filenames.py +1 -0
  652. ansible_test/_util/controller/sanity/code-smell/no-smart-quotes.py +1 -0
  653. ansible_test/_util/controller/sanity/code-smell/replace-urlopen.py +1 -0
  654. ansible_test/_util/controller/sanity/code-smell/runtime-metadata.py +28 -1
  655. ansible_test/_util/controller/sanity/code-smell/shebang.py +1 -0
  656. ansible_test/_util/controller/sanity/code-smell/symlinks.py +1 -0
  657. ansible_test/_util/controller/sanity/code-smell/use-argspec-type-path.py +1 -0
  658. ansible_test/_util/controller/sanity/code-smell/use-compat-six.py +1 -0
  659. ansible_test/_util/controller/sanity/integration-aliases/yaml_to_json.py +2 -1
  660. ansible_test/_util/controller/sanity/pep8/current-ignore.txt +4 -0
  661. ansible_test/_util/controller/sanity/pylint/config/ansible-test-target.cfg +7 -5
  662. ansible_test/_util/controller/sanity/pylint/config/ansible-test.cfg +7 -5
  663. ansible_test/_util/controller/sanity/pylint/config/code-smell.cfg +7 -5
  664. ansible_test/_util/controller/sanity/pylint/config/collection.cfg +3 -5
  665. ansible_test/_util/controller/sanity/pylint/config/default.cfg +7 -7
  666. ansible_test/_util/controller/sanity/pylint/plugins/deprecated.py +1 -13
  667. ansible_test/_util/controller/sanity/pylint/plugins/hide_unraisable.py +1 -0
  668. ansible_test/_util/controller/sanity/pylint/plugins/string_format.py +1 -8
  669. ansible_test/_util/controller/sanity/pylint/plugins/unwanted.py +1 -8
  670. ansible_test/_util/controller/sanity/validate-modules/validate_modules/main.py +55 -28
  671. ansible_test/_util/controller/sanity/validate-modules/validate_modules/module_args.py +12 -5
  672. ansible_test/_util/controller/sanity/validate-modules/validate_modules/schema.py +13 -2
  673. ansible_test/_util/controller/sanity/validate-modules/validate_modules/utils.py +1 -0
  674. ansible_test/_util/controller/sanity/yamllint/yamllinter.py +35 -17
  675. ansible_test/_util/controller/tools/collection_detail.py +1 -0
  676. ansible_test/_util/controller/tools/yaml_to_json.py +2 -1
  677. ansible_test/_util/target/pytest/plugins/ansible_forked.py +6 -1
  678. ansible_test/_util/target/pytest/plugins/ansible_pytest_collections.py +2 -1
  679. ansible_test/_util/target/pytest/plugins/ansible_pytest_coverage.py +1 -0
  680. ansible_test/_util/target/sanity/compile/compile.py +1 -0
  681. ansible_test/_util/target/sanity/import/importer.py +15 -16
  682. ansible_test/_util/target/setup/bootstrap.sh +9 -20
  683. ansible_test/_util/target/setup/probe_cgroups.py +1 -0
  684. ansible_test/_util/target/setup/quiet_pip.py +1 -0
  685. ansible_test/_util/target/setup/requirements.py +35 -27
  686. ansible_test/_util/target/tools/virtualenvcheck.py +2 -1
  687. ansible_test/_util/target/tools/yamlcheck.py +2 -1
  688. ansible/compat/selectors.py +0 -32
  689. ansible/errors/yaml_strings.py +0 -138
  690. ansible/executor/action_write_locks.py +0 -44
  691. ansible/executor/discovery/python_target.py +0 -47
  692. ansible/executor/powershell/module_powershell_wrapper.ps1 +0 -86
  693. ansible/executor/powershell/module_script_wrapper.ps1 +0 -22
  694. ansible/module_utils/compat/importlib.py +0 -26
  695. ansible/module_utils/compat/selectors.py +0 -32
  696. ansible/module_utils/pycompat24.py +0 -73
  697. ansible/parsing/yaml/constructor.py +0 -178
  698. ansible/template/native_helpers.py +0 -251
  699. ansible/template/template.py +0 -43
  700. ansible/template/vars.py +0 -77
  701. ansible/utils/native_jinja.py +0 -11
  702. ansible/vars/fact_cache.py +0 -71
  703. ansible_core-2.18.5rc1.dist-info/RECORD +0 -992
  704. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/Apache-License.txt +0 -0
  705. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/COPYING +0 -0
  706. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/MIT-license.txt +0 -0
  707. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/PSF-license.txt +0 -0
  708. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/entry_points.txt +0 -0
  709. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/simplified_bsd.txt +0 -0
  710. {ansible_core-2.18.5rc1.dist-info → ansible_core-2.19.0b2.dist-info}/top_level.txt +0 -0
@@ -0,0 +1,53 @@
1
+ from __future__ import annotations
2
+
3
+ import importlib
4
+ import typing as t
5
+
6
+ from ansible.module_utils import _internal
7
+ from ansible.module_utils._internal._json import _profiles
8
+
9
+
10
+ def get_controller_serialize_map() -> dict[type, t.Callable]:
11
+ """
12
+ Injected into module_utils code to augment serialization maps with controller-only types.
13
+ This implementation replaces the no-op version in module_utils._internal in controller contexts.
14
+ """
15
+ from ansible._internal._templating import _lazy_containers
16
+ from ansible.parsing.vault import EncryptedString
17
+
18
+ return {
19
+ _lazy_containers._AnsibleLazyTemplateDict: _profiles._JSONSerializationProfile.discard_tags,
20
+ _lazy_containers._AnsibleLazyTemplateList: _profiles._JSONSerializationProfile.discard_tags,
21
+ EncryptedString: str, # preserves tags since this is an intance of EncryptedString; if tags should be discarded from str, another entry will handle it
22
+ }
23
+
24
+
25
+ def import_controller_module(module_name: str, /) -> t.Any:
26
+ """
27
+ Injected into module_utils code to import and return the specified module.
28
+ This implementation replaces the no-op version in module_utils._internal in controller contexts.
29
+ """
30
+ return importlib.import_module(module_name)
31
+
32
+
33
+ _T = t.TypeVar('_T')
34
+
35
+
36
+ def experimental(obj: _T) -> _T:
37
+ """
38
+ Decorator for experimental types and methods outside the `_internal` package which accept or expose internal types.
39
+ As with internal APIs, these are subject to change at any time without notice.
40
+ """
41
+ return obj
42
+
43
+
44
+ def setup() -> None:
45
+ """No-op function to ensure that side-effect only imports of this module are not flagged/removed as 'unused'."""
46
+
47
+
48
+ # DTFIX-RELEASE: this is really fragile- disordered/incorrect imports (among other things) can mess it up. Consider a hosting-env-managed context
49
+ # with an enum with at least Controller/Target/Unknown values, and possibly using lazy-init module shims or some other mechanism to allow controller-side
50
+ # notification/augmentation of this kind of metadata.
51
+ _internal.get_controller_serialize_map = get_controller_serialize_map
52
+ _internal.import_controller_module = import_controller_module
53
+ _internal.is_controller = True
@@ -0,0 +1,265 @@
1
+ # shebang placeholder
2
+
3
+ from __future__ import annotations
4
+
5
+ import datetime
6
+
7
+ # For test-module.py script to tell this is a ANSIBALLZ_WRAPPER
8
+ _ANSIBALLZ_WRAPPER = True
9
+
10
+ # This code is part of Ansible, but is an independent component.
11
+ # The code in this particular templatable string, and this templatable string
12
+ # only, is BSD licensed. Modules which end up using this snippet, which is
13
+ # dynamically combined together by Ansible still belong to the author of the
14
+ # module, and they may assign their own license to the complete work.
15
+ #
16
+ # Copyright (c), James Cammarata, 2016
17
+ # Copyright (c), Toshio Kuratomi, 2016
18
+ #
19
+ # Redistribution and use in source and binary forms, with or without modification,
20
+ # are permitted provided that the following conditions are met:
21
+ #
22
+ # * Redistributions of source code must retain the above copyright
23
+ # notice, this list of conditions and the following disclaimer.
24
+ # * Redistributions in binary form must reproduce the above copyright notice,
25
+ # this list of conditions and the following disclaimer in the documentation
26
+ # and/or other materials provided with the distribution.
27
+ #
28
+ # THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
29
+ # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
30
+ # WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
31
+ # IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
32
+ # INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
33
+ # PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
34
+ # INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
35
+ # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE
36
+ # USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
37
+
38
+
39
+ def _ansiballz_main(
40
+ zipdata: str,
41
+ ansible_module: str,
42
+ module_fqn: str,
43
+ params: str,
44
+ profile: str,
45
+ plugin_info_dict: dict[str, object],
46
+ date_time: datetime.datetime,
47
+ coverage_config: str | None,
48
+ coverage_output: str | None,
49
+ rlimit_nofile: int,
50
+ ) -> None:
51
+ import os
52
+ import os.path
53
+
54
+ # Access to the working directory is required by Python when using pipelining, as well as for the coverage module.
55
+ # Some platforms, such as macOS, may not allow querying the working directory when using become to drop privileges.
56
+ try:
57
+ os.getcwd()
58
+ except OSError:
59
+ try:
60
+ os.chdir(os.path.expanduser('~'))
61
+ except OSError:
62
+ os.chdir('/')
63
+
64
+ if rlimit_nofile:
65
+ import resource
66
+
67
+ existing_soft, existing_hard = resource.getrlimit(resource.RLIMIT_NOFILE)
68
+
69
+ # adjust soft limit subject to existing hard limit
70
+ requested_soft = min(existing_hard, rlimit_nofile)
71
+
72
+ if requested_soft != existing_soft:
73
+ try:
74
+ resource.setrlimit(resource.RLIMIT_NOFILE, (requested_soft, existing_hard))
75
+ except ValueError:
76
+ # some platforms (eg macOS) lie about their hard limit
77
+ pass
78
+
79
+ import sys
80
+ import __main__
81
+
82
+ # For some distros and python versions we pick up this script in the temporary
83
+ # directory. This leads to problems when the ansible module masks a python
84
+ # library that another import needs. We have not figured out what about the
85
+ # specific distros and python versions causes this to behave differently.
86
+ #
87
+ # Tested distros:
88
+ # Fedora23 with python3.4 Works
89
+ # Ubuntu15.10 with python2.7 Works
90
+ # Ubuntu15.10 with python3.4 Fails without this
91
+ # Ubuntu16.04.1 with python3.5 Fails without this
92
+ # To test on another platform:
93
+ # * use the copy module (since this shadows the stdlib copy module)
94
+ # * Turn off pipelining
95
+ # * Make sure that the destination file does not exist
96
+ # * ansible ubuntu16-test -m copy -a 'src=/etc/motd dest=/var/tmp/m'
97
+ # This will traceback in shutil. Looking at the complete traceback will show
98
+ # that shutil is importing copy which finds the ansible module instead of the
99
+ # stdlib module
100
+ scriptdir = None
101
+ try:
102
+ scriptdir = os.path.dirname(os.path.realpath(__main__.__file__))
103
+ except (AttributeError, OSError):
104
+ # Some platforms don't set __file__ when reading from stdin
105
+ # OSX raises OSError if using abspath() in a directory we don't have
106
+ # permission to read (realpath calls abspath)
107
+ pass
108
+
109
+ # Strip cwd from sys.path to avoid potential permissions issues
110
+ excludes = {'', '.', scriptdir}
111
+ sys.path = [p for p in sys.path if p not in excludes]
112
+
113
+ import base64
114
+ import shutil
115
+ import tempfile
116
+ import zipfile
117
+
118
+ def invoke_module(modlib_path: str, json_params: bytes) -> None:
119
+ # When installed via setuptools (including python setup.py install),
120
+ # ansible may be installed with an easy-install.pth file. That file
121
+ # may load the system-wide install of ansible rather than the one in
122
+ # the module. sitecustomize is the only way to override that setting.
123
+ z = zipfile.ZipFile(modlib_path, mode='a')
124
+
125
+ # py3: modlib_path will be text, py2: it's bytes. Need bytes at the end
126
+ sitecustomize = u'import sys\\nsys.path.insert(0,"%s")\\n' % modlib_path
127
+ sitecustomize = sitecustomize.encode('utf-8')
128
+ # Use a ZipInfo to work around zipfile limitation on hosts with
129
+ # clocks set to a pre-1980 year (for instance, Raspberry Pi)
130
+ zinfo = zipfile.ZipInfo()
131
+ zinfo.filename = 'sitecustomize.py'
132
+ zinfo.date_time = date_time.utctimetuple()[:6]
133
+ z.writestr(zinfo, sitecustomize)
134
+ z.close()
135
+
136
+ # Put the zipped up module_utils we got from the controller first in the python path so that we
137
+ # can monkeypatch the right basic
138
+ sys.path.insert(0, modlib_path)
139
+
140
+ from ansible.module_utils._internal._ansiballz import run_module
141
+
142
+ run_module(
143
+ json_params=json_params,
144
+ profile=profile,
145
+ plugin_info_dict=plugin_info_dict,
146
+ module_fqn=module_fqn,
147
+ modlib_path=modlib_path,
148
+ coverage_config=coverage_config,
149
+ coverage_output=coverage_output,
150
+ )
151
+
152
+ def debug(command: str, modlib_path: str, json_params: bytes) -> None:
153
+ # The code here normally doesn't run. It's only used for debugging on the
154
+ # remote machine.
155
+ #
156
+ # The subcommands in this function make it easier to debug ansiballz
157
+ # modules. Here's the basic steps:
158
+ #
159
+ # Run ansible with the environment variable: ANSIBLE_KEEP_REMOTE_FILES=1 and -vvv
160
+ # to save the module file remotely::
161
+ # $ ANSIBLE_KEEP_REMOTE_FILES=1 ansible host1 -m ping -a 'data=october' -vvv
162
+ #
163
+ # Part of the verbose output will tell you where on the remote machine the
164
+ # module was written to::
165
+ # [...]
166
+ # <host1> SSH: EXEC ssh -C -q -o ControlMaster=auto -o ControlPersist=60s -o KbdInteractiveAuthentication=no -o
167
+ # PreferredAuthentications=gssapi-with-mic,gssapi-keyex,hostbased,publickey -o PasswordAuthentication=no -o ConnectTimeout=10 -o
168
+ # ControlPath=/home/badger/.ansible/cp/ansible-ssh-%h-%p-%r -tt rhel7 '/bin/sh -c '"'"'LANG=en_US.UTF-8 LC_ALL=en_US.UTF-8
169
+ # LC_MESSAGES=en_US.UTF-8 /usr/bin/python /home/badger/.ansible/tmp/ansible-tmp-1461173013.93-9076457629738/ping'"'"''
170
+ # [...]
171
+ #
172
+ # Login to the remote machine and run the module file via from the previous
173
+ # step with the explode subcommand to extract the module payload into
174
+ # source files::
175
+ # $ ssh host1
176
+ # $ /usr/bin/python /home/badger/.ansible/tmp/ansible-tmp-1461173013.93-9076457629738/ping explode
177
+ # Module expanded into:
178
+ # /home/badger/.ansible/tmp/ansible-tmp-1461173408.08-279692652635227/ansible
179
+ #
180
+ # You can now edit the source files to instrument the code or experiment with
181
+ # different parameter values. When you're ready to run the code you've modified
182
+ # (instead of the code from the actual zipped module), use the execute subcommand like this::
183
+ # $ /usr/bin/python /home/badger/.ansible/tmp/ansible-tmp-1461173013.93-9076457629738/ping execute
184
+
185
+ # Okay to use __file__ here because we're running from a kept file
186
+ basedir = os.path.join(os.path.abspath(os.path.dirname(__file__)), 'debug_dir')
187
+ args_path = os.path.join(basedir, 'args')
188
+
189
+ if command == 'explode':
190
+ # transform the ZIPDATA into an exploded directory of code and then
191
+ # print the path to the code. This is an easy way for people to look
192
+ # at the code on the remote machine for debugging it in that
193
+ # environment
194
+ z = zipfile.ZipFile(modlib_path)
195
+ for filename in z.namelist():
196
+ if filename.startswith('/'):
197
+ raise Exception('Something wrong with this module zip file: should not contain absolute paths')
198
+
199
+ dest_filename = os.path.join(basedir, filename)
200
+ if dest_filename.endswith(os.path.sep) and not os.path.exists(dest_filename):
201
+ os.makedirs(dest_filename)
202
+ else:
203
+ directory = os.path.dirname(dest_filename)
204
+ if not os.path.exists(directory):
205
+ os.makedirs(directory)
206
+ with open(dest_filename, 'wb') as writer:
207
+ writer.write(z.read(filename))
208
+
209
+ # write the args file
210
+ with open(args_path, 'wb') as writer:
211
+ writer.write(json_params)
212
+
213
+ print('Module expanded into:')
214
+ print(basedir)
215
+
216
+ elif command == 'execute':
217
+ # Execute the exploded code instead of executing the module from the
218
+ # embedded ZIPDATA. This allows people to easily run their modified
219
+ # code on the remote machine to see how changes will affect it.
220
+
221
+ # Set pythonpath to the debug dir
222
+ sys.path.insert(0, basedir)
223
+
224
+ # read in the args file which the user may have modified
225
+ with open(args_path, 'rb') as reader:
226
+ json_params = reader.read()
227
+
228
+ from ansible.module_utils._internal._ansiballz import run_module
229
+
230
+ run_module(
231
+ json_params=json_params,
232
+ profile=profile,
233
+ plugin_info_dict=plugin_info_dict,
234
+ module_fqn=module_fqn,
235
+ modlib_path=modlib_path,
236
+ )
237
+
238
+ else:
239
+ print('WARNING: Unknown debug command. Doing nothing.')
240
+
241
+ #
242
+ # See comments in the debug() method for information on debugging
243
+ #
244
+
245
+ encoded_params = params.encode()
246
+
247
+ # There's a race condition with the controller removing the
248
+ # remote_tmpdir and this module executing under async. So we cannot
249
+ # store this in remote_tmpdir (use system tempdir instead)
250
+ # Only need to use [ansible_module]_payload_ in the temp_path until we move to zipimport
251
+ # (this helps ansible-test produce coverage stats)
252
+ temp_path = tempfile.mkdtemp(prefix='ansible_' + ansible_module + '_payload_')
253
+
254
+ try:
255
+ zipped_mod = os.path.join(temp_path, 'ansible_' + ansible_module + '_payload.zip')
256
+
257
+ with open(zipped_mod, 'wb') as modlib:
258
+ modlib.write(base64.b64decode(zipdata))
259
+
260
+ if len(sys.argv) == 2:
261
+ debug(sys.argv[1], zipped_mod, encoded_params)
262
+ else:
263
+ invoke_module(zipped_mod, encoded_params)
264
+ finally:
265
+ shutil.rmtree(temp_path, ignore_errors=True)
@@ -0,0 +1,47 @@
1
+ from __future__ import annotations as _annotations
2
+
3
+ import collections.abc as _c
4
+ import typing as _t
5
+
6
+ _T_co = _t.TypeVar('_T_co', covariant=True)
7
+
8
+
9
+ class SequenceProxy(_c.Sequence[_T_co]):
10
+ """A read-only sequence proxy."""
11
+
12
+ # DTFIX-RELEASE: needs unit test coverage
13
+
14
+ __slots__ = ('__value',)
15
+
16
+ def __init__(self, value: _c.Sequence[_T_co]) -> None:
17
+ self.__value = value
18
+
19
+ @_t.overload
20
+ def __getitem__(self, index: int) -> _T_co: ...
21
+
22
+ @_t.overload
23
+ def __getitem__(self, index: slice) -> _c.Sequence[_T_co]: ...
24
+
25
+ def __getitem__(self, index: int | slice) -> _T_co | _c.Sequence[_T_co]:
26
+ if isinstance(index, slice):
27
+ return self.__class__(self.__value[index])
28
+
29
+ return self.__value[index]
30
+
31
+ def __len__(self) -> int:
32
+ return len(self.__value)
33
+
34
+ def __contains__(self, item: object) -> bool:
35
+ return item in self.__value
36
+
37
+ def __iter__(self) -> _t.Iterator[_T_co]:
38
+ yield from self.__value
39
+
40
+ def __reversed__(self) -> _c.Iterator[_T_co]:
41
+ return reversed(self.__value)
42
+
43
+ def index(self, *args) -> int:
44
+ return self.__value.index(*args)
45
+
46
+ def count(self, value: object) -> int:
47
+ return self.__value.count(value)
File without changes
@@ -0,0 +1,130 @@
1
+ from __future__ import annotations
2
+
3
+ import dataclasses
4
+ import os
5
+ import types
6
+ import typing as t
7
+
8
+ from ansible.module_utils._internal._datatag import _tag_dataclass_kwargs, AnsibleDatatagBase, AnsibleSingletonTagBase
9
+
10
+
11
+ @dataclasses.dataclass(**_tag_dataclass_kwargs)
12
+ class Origin(AnsibleDatatagBase):
13
+ """
14
+ A tag that stores origin metadata for a tagged value, intended for forensic/diagnostic use.
15
+ Origin metadata should not be used to make runtime decisions, as it is not guaranteed to be present or accurate.
16
+ Setting both `path` and `line_num` can result in diagnostic display of referenced file contents.
17
+ Either `path` or `description` must be present.
18
+ """
19
+
20
+ path: str | None = None
21
+ """The path from which the tagged content originated."""
22
+ description: str | None = None
23
+ """A description of the origin, for display to users."""
24
+ line_num: int | None = None
25
+ """An optional line number, starting at 1."""
26
+ col_num: int | None = None
27
+ """An optional column number, starting at 1."""
28
+
29
+ UNKNOWN: t.ClassVar[t.Self]
30
+
31
+ @classmethod
32
+ def get_or_create_tag(cls, value: t.Any, path: str | os.PathLike | None) -> Origin:
33
+ """Return the tag from the given value, creating a tag from the provided path if no tag was found."""
34
+ if not (origin := cls.get_tag(value)):
35
+ if path:
36
+ origin = Origin(path=str(path)) # convert tagged strings and path-like values to a native str
37
+ else:
38
+ origin = Origin.UNKNOWN
39
+
40
+ return origin
41
+
42
+ def replace(
43
+ self,
44
+ path: str | types.EllipsisType = ...,
45
+ description: str | types.EllipsisType = ...,
46
+ line_num: int | None | types.EllipsisType = ...,
47
+ col_num: int | None | types.EllipsisType = ...,
48
+ ) -> t.Self:
49
+ """Return a new origin based on an existing one, with the given fields replaced."""
50
+ return dataclasses.replace(
51
+ self,
52
+ **{
53
+ key: value
54
+ for key, value in dict(
55
+ path=path,
56
+ description=description,
57
+ line_num=line_num,
58
+ col_num=col_num,
59
+ ).items()
60
+ if value is not ...
61
+ }, # type: ignore[arg-type]
62
+ )
63
+
64
+ def _post_validate(self) -> None:
65
+ if self.path:
66
+ if not self.path.startswith('/'):
67
+ raise RuntimeError('The `src` field must be an absolute path.')
68
+ elif not self.description:
69
+ raise RuntimeError('The `src` or `description` field must be specified.')
70
+
71
+ def __str__(self) -> str:
72
+ """Renders the origin in the form of path:line_num:col_num, omitting missing/invalid elements from the right."""
73
+ if self.path:
74
+ value = self.path
75
+ else:
76
+ value = self.description
77
+
78
+ if self.line_num and self.line_num > 0:
79
+ value += f':{self.line_num}'
80
+
81
+ if self.col_num and self.col_num > 0:
82
+ value += f':{self.col_num}'
83
+
84
+ if self.path and self.description:
85
+ value += f' ({self.description})'
86
+
87
+ return value
88
+
89
+
90
+ Origin.UNKNOWN = Origin(description='<unknown>')
91
+
92
+
93
+ @dataclasses.dataclass(**_tag_dataclass_kwargs)
94
+ class VaultedValue(AnsibleDatatagBase):
95
+ """Tag for vault-encrypted strings that carries the original ciphertext for round-tripping."""
96
+
97
+ ciphertext: str
98
+
99
+ def _get_tag_to_propagate(self, src: t.Any, value: object, *, value_type: t.Optional[type] = None) -> t.Self | None:
100
+ # Since VaultedValue stores the encrypted representation of the value on which it is tagged,
101
+ # it is incorrect to propagate the tag to a value which is not equal to the original.
102
+ # If the tag were copied to another value and subsequently serialized as the original encrypted value,
103
+ # the result would then differ from the value on which the tag was applied.
104
+
105
+ # Comparisons which can trigger an exception are indicative of a bug and should not be handled here.
106
+ # For example:
107
+ # * When `src` is an undecryptable `EncryptedString` -- it is not valid to apply this tag to that type.
108
+ # * When `value` is a `Marker` -- this requires a templating, but vaulted values do not support templating.
109
+
110
+ if src == value: # assume the tag was correctly applied to src
111
+ return self # same plaintext value, tag propagation with same ciphertext is safe
112
+
113
+ return self.get_tag(value) # different value, preserve the existing tag, if any
114
+
115
+
116
+ @dataclasses.dataclass(**_tag_dataclass_kwargs)
117
+ class TrustedAsTemplate(AnsibleSingletonTagBase):
118
+ """
119
+ Indicates the tagged string is trusted to parse and render as a template.
120
+ Do *NOT* apply this tag to data from untrusted sources, as this would allow code injection during templating.
121
+ """
122
+
123
+
124
+ @dataclasses.dataclass(**_tag_dataclass_kwargs)
125
+ class SourceWasEncrypted(AnsibleSingletonTagBase):
126
+ """
127
+ For internal use only.
128
+ Indicates the tagged value was sourced from an encrypted file.
129
+ Currently applied only by DataLoader.get_text_file_contents() and by extension DataLoader.load_from_file().
130
+ """
@@ -0,0 +1,19 @@
1
+ from __future__ import annotations
2
+
3
+ from ansible.module_utils._internal._datatag import AnsibleTagHelper
4
+
5
+
6
+ def str_problematic_strip(value: str) -> str:
7
+ """
8
+ Return a copy of `value` with leading and trailing whitespace removed.
9
+ Used where `str.strip` is needed, but tags must be preserved *AND* the stripping behavior likely shouldn't exist.
10
+ If the stripping behavior is non-problematic, use `AnsibleTagHelper.tag_copy` around `str.strip` instead.
11
+ """
12
+ if (stripped_value := value.strip()) == value:
13
+ return value
14
+
15
+ # FUTURE: consider deprecating some/all usages of this method; they generally imply a code smell or pattern we shouldn't be supporting
16
+
17
+ stripped_value = AnsibleTagHelper.tag_copy(value, stripped_value)
18
+
19
+ return stripped_value
@@ -0,0 +1,33 @@
1
+ from __future__ import annotations
2
+
3
+ import io
4
+ import typing as _t
5
+
6
+ from .._wrapt import ObjectProxy
7
+ from ...module_utils._internal import _datatag
8
+
9
+
10
+ class TaggedStreamWrapper(ObjectProxy):
11
+ """
12
+ Janky proxy around IOBase to allow streams to carry tags and support basic interrogation by the tagging API.
13
+ Most tagging operations will have undefined behavior for this type.
14
+ """
15
+
16
+ _self__ansible_tags_mapping: _datatag._AnsibleTagsMapping
17
+
18
+ def __init__(self, stream: io.IOBase, tags: _datatag.AnsibleDatatagBase | _t.Iterable[_datatag.AnsibleDatatagBase]) -> None:
19
+ super().__init__(stream)
20
+
21
+ tag_list: list[_datatag.AnsibleDatatagBase]
22
+
23
+ # noinspection PyProtectedMember
24
+ if type(tags) in _datatag._known_tag_types:
25
+ tag_list = [tags] # type: ignore[list-item]
26
+ else:
27
+ tag_list = list(tags) # type: ignore[arg-type]
28
+
29
+ self._self__ansible_tags_mapping = _datatag._AnsibleTagsMapping((type(tag), tag) for tag in tag_list)
30
+
31
+ @property
32
+ def _ansible_tags_mapping(self) -> _datatag._AnsibleTagsMapping:
33
+ return self._self__ansible_tags_mapping
File without changes
@@ -0,0 +1,128 @@
1
+ from __future__ import annotations
2
+
3
+ import dataclasses
4
+ import typing as t
5
+
6
+ from ansible.errors import AnsibleRuntimeError
7
+ from ansible.module_utils.common.messages import ErrorSummary, Detail, _dataclass_kwargs
8
+
9
+
10
+ class AnsibleCapturedError(AnsibleRuntimeError):
11
+ """An exception representing error detail captured in another context where the error detail must be serialized to be preserved."""
12
+
13
+ context: t.ClassVar[str]
14
+
15
+ def __init__(
16
+ self,
17
+ *,
18
+ obj: t.Any = None,
19
+ error_summary: ErrorSummary,
20
+ ) -> None:
21
+ super().__init__(
22
+ obj=obj,
23
+ )
24
+
25
+ self._error_summary = error_summary
26
+
27
+ @property
28
+ def error_summary(self) -> ErrorSummary:
29
+ return self._error_summary
30
+
31
+
32
+ class AnsibleResultCapturedError(AnsibleCapturedError):
33
+ """An exception representing error detail captured in a foreign context where an action/module result dictionary is involved."""
34
+
35
+ def __init__(self, error_summary: ErrorSummary, result: dict[str, t.Any]) -> None:
36
+ super().__init__(error_summary=error_summary)
37
+
38
+ self._result = result
39
+
40
+ @classmethod
41
+ def maybe_raise_on_result(cls, result: dict[str, t.Any]) -> None:
42
+ """Normalize the result and raise an exception if the result indicated failure."""
43
+ if error_summary := cls.normalize_result_exception(result):
44
+ raise error_summary.error_type(error_summary, result)
45
+
46
+ @classmethod
47
+ def find_first_remoted_error(cls, exception: BaseException) -> t.Self | None:
48
+ """Find the first captured module error in the cause chain, starting with the given exception, returning None if not found."""
49
+ while exception:
50
+ if isinstance(exception, cls):
51
+ return exception
52
+
53
+ exception = exception.__cause__
54
+
55
+ return None
56
+
57
+ @classmethod
58
+ def normalize_result_exception(cls, result: dict[str, t.Any]) -> CapturedErrorSummary | None:
59
+ """
60
+ Normalize the result `exception`, if any, to be a `CapturedErrorSummary` instance.
61
+ If a new `CapturedErrorSummary` was created, the `error_type` will be `cls`.
62
+ The `exception` key will be removed if falsey.
63
+ A `CapturedErrorSummary` instance will be returned if `failed` is truthy.
64
+ """
65
+ if type(cls) is AnsibleResultCapturedError: # pylint: disable=unidiomatic-typecheck
66
+ raise TypeError('The normalize_result_exception method cannot be called on the AnsibleCapturedError base type, use a derived type.')
67
+
68
+ if not isinstance(result, dict):
69
+ raise TypeError(f'Malformed result. Received {type(result)} instead of {dict}.')
70
+
71
+ failed = result.get('failed') # DTFIX-FUTURE: warn if failed is present and not a bool, or exception is present without failed being True
72
+ exception = result.pop('exception', None)
73
+
74
+ if not failed and not exception:
75
+ return None
76
+
77
+ if isinstance(exception, CapturedErrorSummary):
78
+ error_summary = exception
79
+ elif isinstance(exception, ErrorSummary):
80
+ error_summary = CapturedErrorSummary(
81
+ details=exception.details,
82
+ formatted_traceback=cls._normalize_traceback(exception.formatted_traceback),
83
+ error_type=cls,
84
+ )
85
+ else:
86
+ # translate non-ErrorDetail errors
87
+ error_summary = CapturedErrorSummary(
88
+ details=(Detail(msg=str(result.get('msg', 'Unknown error.'))),),
89
+ formatted_traceback=cls._normalize_traceback(exception),
90
+ error_type=cls,
91
+ )
92
+
93
+ result.update(exception=error_summary)
94
+
95
+ return error_summary if failed else None # even though error detail was normalized, only return it if the result indicated failure
96
+
97
+ @classmethod
98
+ def _normalize_traceback(cls, value: object | None) -> str | None:
99
+ """Normalize the provided traceback value, returning None if it is falsey."""
100
+ if not value:
101
+ return None
102
+
103
+ value = str(value).rstrip()
104
+
105
+ if not value:
106
+ return None
107
+
108
+ return value + '\n'
109
+
110
+
111
+ class AnsibleActionCapturedError(AnsibleResultCapturedError):
112
+ """An exception representing error detail sourced directly by an action in its result dictionary."""
113
+
114
+ _default_message = 'Action failed.'
115
+ context = 'action'
116
+
117
+
118
+ class AnsibleModuleCapturedError(AnsibleResultCapturedError):
119
+ """An exception representing error detail captured in a module context and returned from an action's result dictionary."""
120
+
121
+ _default_message = 'Module failed.'
122
+ context = 'target'
123
+
124
+
125
+ @dataclasses.dataclass(**_dataclass_kwargs)
126
+ class CapturedErrorSummary(ErrorSummary):
127
+ # DTFIX-RELEASE: where to put this, name, etc. since it shows up in results, it's not exactly private (and contains a type ref to an internal type)
128
+ error_type: type[AnsibleResultCapturedError] | None = None