mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-25 13:34:01 -07:00 
			
		
		
		
	Compare commits
	
		
			82 commits
		
	
	
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | e978fd4d61 | ||
|  | 6fc8492ecf | ||
|  | 95beb452a8 | ||
|  | c10e9e2650 | ||
|  | ac35bf4acb | ||
|  | 50b9855ace | ||
|  | 2ab26db197 | ||
|  | 5fcf5d0c8b | ||
|  | 0f0ad6b6d1 | ||
|  | 95f3109ddc | ||
|  | 6037c5d1e6 | ||
|  | a70d9773dd | ||
|  | bc50b48205 | ||
|  | 02e6a8608f | ||
|  | 82f4b51873 | ||
|  | 589e8fd5e1 | ||
|  | 58f74b96ef | ||
|  | 1489c080a7 | ||
|  | 6f845f61f0 | ||
|  | c17f5ff3e8 | ||
|  | ff21afb227 | ||
|  | c1d6e5c3c2 | ||
|  | 377b5d4ccd | ||
|  | f3f7b2776f | ||
|  | df8bfad9b9 | ||
|  | 8a231e4b36 | ||
|  | 671f850069 | ||
|  | 2fa36592e4 | ||
|  | 51d704bfe3 | ||
|  | 2b0e335752 | ||
|  | cc28cde3a2 | ||
|  | 2d616bf4d1 | ||
|  | 25d9ab8dcd | ||
|  | 9abda18071 | ||
|  | 406fa12142 | ||
|  | caaebb38e7 | ||
|  | 2bc74f4f04 | ||
|  | e1e89f7735 | ||
|  | efedd0d6e2 | ||
|  | 8079aea1ee | ||
|  | ee7fdf5f8c | ||
|  | ced1baad63 | ||
|  | a0d4ee4fc1 | ||
|  | d930c8d877 | ||
|  | 352e91a389 | ||
|  | 4b7554445b | ||
|  | 3a456a645d | ||
|  | 6f4580ebd9 | ||
|  | 8d83557e52 | ||
|  | 5ebd980e26 | ||
|  | 17447d2a84 | ||
|  | ffee01cd9c | ||
|  | 38b4e316ae | ||
|  | b52a6f3611 | ||
|  | 2435fb3f30 | ||
|  | d6d9f84b0a | ||
|  | 4b04e3cc32 | ||
|  | c681249364 | ||
|  | 57a4195b0d | ||
|  | 41a23f093d | ||
|  | 0bd085714f | ||
|  | a4be229f67 | ||
|  | 9c4487ebc5 | ||
|  | 09ea441316 | ||
|  | fef6abc8c8 | ||
|  | 618e567377 | ||
|  | 246abffce5 | ||
|  | 076ebb4b2d | ||
|  | 4948b521a3 | ||
|  | e9ec26ff1b | ||
|  | 72d4476813 | ||
|  | e96bfd07b4 | ||
|  | c6d0419460 | ||
|  | 081b4068a0 | ||
|  | 8fba9ca751 | ||
|  | fad4c2d956 | ||
|  | 6065dd0f18 | ||
|  | a411ff5ea8 | ||
|  | 42b245eabf | ||
|  | 9a676bb88f | ||
|  | cd26aec2f3 | ||
|  | e9327a0464 | 
					 281 changed files with 3702 additions and 1052 deletions
				
			
		|  | @ -29,6 +29,7 @@ schedules: | ||||||
|     always: true |     always: true | ||||||
|     branches: |     branches: | ||||||
|       include: |       include: | ||||||
|  |         - stable-6 | ||||||
|         - stable-5 |         - stable-5 | ||||||
|   - cron: 0 11 * * 0 |   - cron: 0 11 * * 0 | ||||||
|     displayName: Weekly (old stable branches) |     displayName: Weekly (old stable branches) | ||||||
|  | @ -188,6 +189,24 @@ stages: | ||||||
|             - test: 3.5 |             - test: 3.5 | ||||||
| 
 | 
 | ||||||
| ## Remote | ## Remote | ||||||
|  |   - stage: Remote_devel_extra_vms | ||||||
|  |     displayName: Remote devel extra VMs | ||||||
|  |     dependsOn: [] | ||||||
|  |     jobs: | ||||||
|  |       - template: templates/matrix.yml | ||||||
|  |         parameters: | ||||||
|  |           testFormat: devel/{0} | ||||||
|  |           targets: | ||||||
|  |             - name: Alpine 3.16 | ||||||
|  |               test: alpine/3.16 | ||||||
|  |             # - name: Fedora 36 | ||||||
|  |             #   test: fedora/36 | ||||||
|  |             # - name: Ubuntu 20.04 | ||||||
|  |             #   test: ubuntu/20.04 | ||||||
|  |             - name: Ubuntu 22.04 | ||||||
|  |               test: ubuntu/22.04 | ||||||
|  |           groups: | ||||||
|  |             - vm | ||||||
|   - stage: Remote_devel |   - stage: Remote_devel | ||||||
|     displayName: Remote devel |     displayName: Remote devel | ||||||
|     dependsOn: [] |     dependsOn: [] | ||||||
|  | @ -202,8 +221,6 @@ stages: | ||||||
|               test: rhel/7.9 |               test: rhel/7.9 | ||||||
|             - name: RHEL 9.0 |             - name: RHEL 9.0 | ||||||
|               test: rhel/9.0 |               test: rhel/9.0 | ||||||
|             - name: FreeBSD 12.3 |  | ||||||
|               test: freebsd/12.3 |  | ||||||
|             - name: FreeBSD 13.1 |             - name: FreeBSD 13.1 | ||||||
|               test: freebsd/13.1 |               test: freebsd/13.1 | ||||||
|           groups: |           groups: | ||||||
|  | @ -220,8 +237,8 @@ stages: | ||||||
|           targets: |           targets: | ||||||
|             - name: RHEL 9.0 |             - name: RHEL 9.0 | ||||||
|               test: rhel/9.0 |               test: rhel/9.0 | ||||||
|             - name: FreeBSD 13.1 |             - name: FreeBSD 12.3 | ||||||
|               test: freebsd/13.1 |               test: freebsd/12.3 | ||||||
|           groups: |           groups: | ||||||
|             - 1 |             - 1 | ||||||
|             - 2 |             - 2 | ||||||
|  | @ -385,7 +402,7 @@ stages: | ||||||
|             - name: ArchLinux |             - name: ArchLinux | ||||||
|               test: archlinux/3.10 |               test: archlinux/3.10 | ||||||
|             - name: CentOS Stream 8 |             - name: CentOS Stream 8 | ||||||
|               test: centos-stream8/3.8 |               test: centos-stream8/3.9 | ||||||
|           groups: |           groups: | ||||||
|             - 1 |             - 1 | ||||||
|             - 2 |             - 2 | ||||||
|  | @ -458,6 +475,7 @@ stages: | ||||||
|       - Units_2_12 |       - Units_2_12 | ||||||
|       - Units_2_13 |       - Units_2_13 | ||||||
|       - Units_2_14 |       - Units_2_14 | ||||||
|  |       - Remote_devel_extra_vms | ||||||
|       - Remote_devel |       - Remote_devel | ||||||
|       - Remote_2_11 |       - Remote_2_11 | ||||||
|       - Remote_2_12 |       - Remote_2_12 | ||||||
|  |  | ||||||
							
								
								
									
										22
									
								
								.github/BOTMETA.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										22
									
								
								.github/BOTMETA.yml
									
										
									
									
										vendored
									
									
								
							|  | @ -265,6 +265,8 @@ files: | ||||||
|     maintainers: delineaKrehl tylerezimmerman |     maintainers: delineaKrehl tylerezimmerman | ||||||
|   $module_utils/: |   $module_utils/: | ||||||
|     labels: module_utils |     labels: module_utils | ||||||
|  |   $module_utils/deps.py: | ||||||
|  |     maintainers: russoz | ||||||
|   $module_utils/gconftool2.py: |   $module_utils/gconftool2.py: | ||||||
|     labels: gconftool2 |     labels: gconftool2 | ||||||
|     maintainers: russoz |     maintainers: russoz | ||||||
|  | @ -279,9 +281,14 @@ files: | ||||||
|     maintainers: $team_huawei |     maintainers: $team_huawei | ||||||
|   $module_utils/identity/keycloak/keycloak.py: |   $module_utils/identity/keycloak/keycloak.py: | ||||||
|     maintainers: $team_keycloak |     maintainers: $team_keycloak | ||||||
|  |   $module_utils/identity/keycloak/keycloak_clientsecret.py: | ||||||
|  |     maintainers: $team_keycloak fynncfchen johncant | ||||||
|   $module_utils/ipa.py: |   $module_utils/ipa.py: | ||||||
|     labels: ipa |     labels: ipa | ||||||
|     maintainers: $team_ipa |     maintainers: $team_ipa | ||||||
|  |   $module_utils/jenkins.py: | ||||||
|  |     labels: jenkins | ||||||
|  |     maintainers: russoz | ||||||
|   $module_utils/manageiq.py: |   $module_utils/manageiq.py: | ||||||
|     labels: manageiq |     labels: manageiq | ||||||
|     maintainers: $team_manageiq |     maintainers: $team_manageiq | ||||||
|  | @ -302,6 +309,9 @@ files: | ||||||
|   $module_utils/pipx.py: |   $module_utils/pipx.py: | ||||||
|     labels: pipx |     labels: pipx | ||||||
|     maintainers: russoz |     maintainers: russoz | ||||||
|  |   $module_utils/puppet.py: | ||||||
|  |     labels: puppet | ||||||
|  |     maintainers: russoz | ||||||
|   $module_utils/pure.py: |   $module_utils/pure.py: | ||||||
|     labels: pure pure_storage |     labels: pure pure_storage | ||||||
|     maintainers: $team_purestorage |     maintainers: $team_purestorage | ||||||
|  | @ -313,6 +323,8 @@ files: | ||||||
|   $module_utils/scaleway.py: |   $module_utils/scaleway.py: | ||||||
|     labels: cloud scaleway |     labels: cloud scaleway | ||||||
|     maintainers: $team_scaleway |     maintainers: $team_scaleway | ||||||
|  |   $module_utils/ssh.py: | ||||||
|  |     maintainers: russoz | ||||||
|   $module_utils/storage/hpe3par/hpe3par.py: |   $module_utils/storage/hpe3par/hpe3par.py: | ||||||
|     maintainers: farhan7500 gautamphegde |     maintainers: farhan7500 gautamphegde | ||||||
|   $module_utils/utm_utils.py: |   $module_utils/utm_utils.py: | ||||||
|  | @ -665,6 +677,10 @@ files: | ||||||
|     maintainers: Gaetan2907 |     maintainers: Gaetan2907 | ||||||
|   $modules/keycloak_clientscope.py: |   $modules/keycloak_clientscope.py: | ||||||
|     maintainers: Gaetan2907 |     maintainers: Gaetan2907 | ||||||
|  |   $modules/keycloak_clientsecret_info.py: | ||||||
|  |     maintainers: fynncfchen johncant | ||||||
|  |   $modules/keycloak_clientsecret_regenerate.py: | ||||||
|  |     maintainers: fynncfchen johncant | ||||||
|   $modules/keycloak_group.py: |   $modules/keycloak_group.py: | ||||||
|     maintainers: adamgoossens |     maintainers: adamgoossens | ||||||
|   $modules/keycloak_identity_provider.py: |   $modules/keycloak_identity_provider.py: | ||||||
|  | @ -822,7 +838,8 @@ files: | ||||||
|     maintainers: marc-sensenich |     maintainers: marc-sensenich | ||||||
|   $modules/ohai.py: |   $modules/ohai.py: | ||||||
|     labels: ohai |     labels: ohai | ||||||
|     maintainers: $team_ansible_core mpdehaan |     maintainers: $team_ansible_core | ||||||
|  |     ignore: mpdehaan | ||||||
|   $modules/omapi_host.py: |   $modules/omapi_host.py: | ||||||
|     maintainers: amasolov nerzhul |     maintainers: amasolov nerzhul | ||||||
|   $modules/one_: |   $modules/one_: | ||||||
|  | @ -1072,7 +1089,8 @@ files: | ||||||
|   $modules/sapcar_extract.py: |   $modules/sapcar_extract.py: | ||||||
|     maintainers: RainerLeber |     maintainers: RainerLeber | ||||||
|   $modules/say.py: |   $modules/say.py: | ||||||
|     maintainers: $team_ansible_core mpdehaan |     maintainers: $team_ansible_core | ||||||
|  |     ignore: mpdehaan | ||||||
|   $modules/scaleway_: |   $modules/scaleway_: | ||||||
|     maintainers: $team_scaleway |     maintainers: $team_scaleway | ||||||
|   $modules/scaleway_compute_private_network.py: |   $modules/scaleway_compute_private_network.py: | ||||||
|  |  | ||||||
							
								
								
									
										93
									
								
								.github/workflows/docs-pr.yml
									
										
									
									
										vendored
									
									
								
							
							
						
						
									
										93
									
								
								.github/workflows/docs-pr.yml
									
										
									
									
										vendored
									
									
								
							|  | @ -1,93 +0,0 @@ | ||||||
| --- |  | ||||||
| # Copyright (c) Ansible Project |  | ||||||
| # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) |  | ||||||
| # SPDX-License-Identifier: GPL-3.0-or-later |  | ||||||
| 
 |  | ||||||
| name: Collection Docs |  | ||||||
| concurrency: |  | ||||||
|   group: docs-${{ github.head_ref }} |  | ||||||
|   cancel-in-progress: true |  | ||||||
| on: |  | ||||||
|   pull_request_target: |  | ||||||
|     types: [opened, synchronize, reopened, closed] |  | ||||||
|     paths-ignore: |  | ||||||
|       - '.azure-pipelines/**' |  | ||||||
|       - 'changelogs/**' |  | ||||||
|       - 'meta/**' |  | ||||||
|       - 'tests/**' |  | ||||||
| 
 |  | ||||||
| jobs: |  | ||||||
|   build-docs: |  | ||||||
|     permissions: |  | ||||||
|       contents: read |  | ||||||
|     name: Build Ansible Docs |  | ||||||
|     uses: ansible-community/github-docs-build/.github/workflows/_shared-docs-build-pr.yml@main |  | ||||||
|     with: |  | ||||||
|       init-fail-on-error: true |  | ||||||
|       provide-link-targets: | |  | ||||||
|         ansible_collections.ansible.builtin.dict2items_filter |  | ||||||
|         ansible_collections.ansible.builtin.items_lookup |  | ||||||
|         ansible_collections.ansible.builtin.path_join_filter |  | ||||||
|         ansible_collections.community.kubevirt.kubevirt_cdi_upload_module |  | ||||||
|         ansible_collections.community.kubevirt.kubevirt_inventory |  | ||||||
|         ansible_collections.community.kubevirt.kubevirt_preset_module |  | ||||||
|         ansible_collections.community.kubevirt.kubevirt_pvc_module |  | ||||||
|         ansible_collections.community.kubevirt.kubevirt_rs_module |  | ||||||
|         ansible_collections.community.kubevirt.kubevirt_template_module |  | ||||||
|         ansible_collections.community.kubevirt.kubevirt_vm_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_a_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_aaaa_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_cname_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_dns_view_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_fixed_address_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_host_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_lookup_lookup |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_member_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_mx_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_naptr_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_network_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_network_view_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_next_ip_lookup |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_next_network_lookup |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_nsgroup_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_ptr_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_srv_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_txt_record_module |  | ||||||
|         ansible_collections.infoblox.nios_modules.nios_zone_module |  | ||||||
| 
 |  | ||||||
|   comment: |  | ||||||
|     permissions: |  | ||||||
|       pull-requests: write |  | ||||||
|     runs-on: ubuntu-latest |  | ||||||
|     needs: build-docs |  | ||||||
|     name: PR comments |  | ||||||
|     steps: |  | ||||||
|       - name: PR comment |  | ||||||
|         uses: ansible-community/github-docs-build/actions/ansible-docs-build-comment@main |  | ||||||
|         with: |  | ||||||
|           body-includes: '## Docs Build' |  | ||||||
|           reactions: heart |  | ||||||
|           action: ${{ needs.build-docs.outputs.changed != 'true' && 'remove' || '' }} |  | ||||||
|           on-closed-body: | |  | ||||||
|             ## Docs Build 📝 |  | ||||||
| 
 |  | ||||||
|             This PR is closed and any previously published docsite has been unpublished. |  | ||||||
|           on-merged-body: | |  | ||||||
|             ## Docs Build 📝 |  | ||||||
| 
 |  | ||||||
|             Thank you for contribution!✨ |  | ||||||
| 
 |  | ||||||
|             This PR has been merged and your docs changes will be incorporated when they are next published. |  | ||||||
|           body: | |  | ||||||
|             ## Docs Build 📝 |  | ||||||
| 
 |  | ||||||
|             Thank you for contribution!✨ |  | ||||||
| 
 |  | ||||||
|             The docsite for **this PR** is available for download as an artifact from this run: |  | ||||||
|             ${{ needs.build-docs.outputs.artifact-url }} |  | ||||||
| 
 |  | ||||||
|             File changes: |  | ||||||
| 
 |  | ||||||
|             ${{ needs.build-docs.outputs.diff-files-rendered }} |  | ||||||
| 
 |  | ||||||
|             ${{ needs.build-docs.outputs.diff-rendered }} |  | ||||||
							
								
								
									
										185
									
								
								CHANGELOG.rst
									
										
									
									
									
								
							
							
						
						
									
										185
									
								
								CHANGELOG.rst
									
										
									
									
									
								
							|  | @ -6,13 +6,149 @@ Community General Release Notes | ||||||
| 
 | 
 | ||||||
| This changelog describes changes after version 5.0.0. | This changelog describes changes after version 5.0.0. | ||||||
| 
 | 
 | ||||||
| v6.0.0-a1 | v6.2.0 | ||||||
| ========= | ====== | ||||||
| 
 | 
 | ||||||
| Release Summary | Release Summary | ||||||
| --------------- | --------------- | ||||||
| 
 | 
 | ||||||
| This is a pre-release for the upcoming 6.0.0 major release. The main objective of this pre-release is to make it possible to test the large stuctural changes by flattening the directory structure. See the corresponding entry in the changelog for details. | Regular bugfix and feature release. | ||||||
|  | 
 | ||||||
|  | Minor Changes | ||||||
|  | ------------- | ||||||
|  | 
 | ||||||
|  | - opkg - allow installing a package in a certain version (https://github.com/ansible-collections/community.general/pull/5688). | ||||||
|  | - proxmox - added new module parameter ``tags`` for use with PVE 7+ (https://github.com/ansible-collections/community.general/pull/5714). | ||||||
|  | - puppet - refactored module to use ``CmdRunner`` for executing ``puppet`` (https://github.com/ansible-collections/community.general/pull/5612). | ||||||
|  | - redhat_subscription - add a ``server_proxy_scheme`` parameter to configure the scheme for the proxy server (https://github.com/ansible-collections/community.general/pull/5662). | ||||||
|  | - ssh_config - refactor code to module util to fix sanity check (https://github.com/ansible-collections/community.general/pull/5720). | ||||||
|  | - sudoers - adds ``host`` parameter for setting hostname restrictions in sudoers rules (https://github.com/ansible-collections/community.general/issues/5702). | ||||||
|  | 
 | ||||||
|  | Deprecated Features | ||||||
|  | ------------------- | ||||||
|  | 
 | ||||||
|  | - manageiq_policies - deprecate ``state=list`` in favour of using ``community.general.manageiq_policies_info`` (https://github.com/ansible-collections/community.general/pull/5721). | ||||||
|  | - rax - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_cbs - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_cbs_attachments - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_cdb - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_cdb_database - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_cdb_user - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_clb - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_clb_nodes - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_clb_ssl - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_dns - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_dns_record - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_facts - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_files - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_files_objects - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_identity - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_keypair - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_meta - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_mon_alarm - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_mon_check - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_mon_entity - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_mon_notification - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_mon_notification_plan - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_network - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_queue - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_scaling_group - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | - rax_scaling_policy - module relies on deprecates library ``pyrax``. Unless maintainers step up to work on the module, it will be marked as deprecated in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  | 
 | ||||||
|  | Bugfixes | ||||||
|  | -------- | ||||||
|  | 
 | ||||||
|  | - ansible_galaxy_install - set default to raise exception if command's return code is different from zero (https://github.com/ansible-collections/community.general/pull/5680). | ||||||
|  | - ansible_galaxy_install - try ``C.UTF-8`` and then fall back to ``en_US.UTF-8`` before failing (https://github.com/ansible-collections/community.general/pull/5680). | ||||||
|  | - gitlab_group_variables - fix dropping variables accidentally when GitLab introduced new properties (https://github.com/ansible-collections/community.general/pull/5667). | ||||||
|  | - gitlab_project_variables - fix dropping variables accidentally when GitLab introduced new properties (https://github.com/ansible-collections/community.general/pull/5667). | ||||||
|  | - lxc_container - fix the arguments of the lxc command which broke the creation and cloning of containers (https://github.com/ansible-collections/community.general/issues/5578). | ||||||
|  | - opkg - fix issue that ``force=reinstall`` would not reinstall an existing package (https://github.com/ansible-collections/community.general/pull/5705). | ||||||
|  | - proxmox_disk - fixed possible issues with redundant ``vmid`` parameter (https://github.com/ansible-collections/community.general/issues/5492, https://github.com/ansible-collections/community.general/pull/5672). | ||||||
|  | - proxmox_nic - fixed possible issues with redundant ``vmid`` parameter (https://github.com/ansible-collections/community.general/issues/5492, https://github.com/ansible-collections/community.general/pull/5672). | ||||||
|  | - unixy callback plugin - fix typo introduced when updating to use Ansible's configuration manager for handling options (https://github.com/ansible-collections/community.general/issues/5600). | ||||||
|  | 
 | ||||||
|  | v6.1.0 | ||||||
|  | ====== | ||||||
|  | 
 | ||||||
|  | Release Summary | ||||||
|  | --------------- | ||||||
|  | 
 | ||||||
|  | Regular bugfix and feature release. | ||||||
|  | 
 | ||||||
|  | Minor Changes | ||||||
|  | ------------- | ||||||
|  | 
 | ||||||
|  | - cmd_runner module utils - ``cmd_runner_fmt.as_bool()`` can now take an extra parameter to format when value is false (https://github.com/ansible-collections/community.general/pull/5647). | ||||||
|  | - gconftool2 - refactor using ``ModuleHelper`` and ``CmdRunner`` (https://github.com/ansible-collections/community.general/pull/5545). | ||||||
|  | - java_certs - add more detailed error output when extracting certificate from PKCS12 fails (https://github.com/ansible-collections/community.general/pull/5550). | ||||||
|  | - jenkins_plugin - refactor code to module util to fix sanity check (https://github.com/ansible-collections/community.general/pull/5565). | ||||||
|  | - lxd_project - refactored code out to module utils to clear sanity check (https://github.com/ansible-collections/community.general/pull/5549). | ||||||
|  | - nmap inventory plugin - add new options ``udp_scan``, ``icmp_timestamp``, and ``dns_resolve`` for different types of scans (https://github.com/ansible-collections/community.general/pull/5566). | ||||||
|  | - rax_scaling_group - refactored out code to the ``rax`` module utils to clear the sanity check (https://github.com/ansible-collections/community.general/pull/5563). | ||||||
|  | - redfish_command - add ``PerformRequestedOperations`` command to perform any operations necessary to continue the update flow (https://github.com/ansible-collections/community.general/issues/4276). | ||||||
|  | - redfish_command - add ``update_apply_time`` to ``SimpleUpdate`` command (https://github.com/ansible-collections/community.general/issues/3910). | ||||||
|  | - redfish_command - add ``update_status`` to output of ``SimpleUpdate`` command to allow a user monitor the update in progress (https://github.com/ansible-collections/community.general/issues/4276). | ||||||
|  | - redfish_info - add ``GetUpdateStatus`` command to check the progress of a previous update request (https://github.com/ansible-collections/community.general/issues/4276). | ||||||
|  | - redfish_utils module utils - added PUT (``put_request()``) functionality (https://github.com/ansible-collections/community.general/pull/5490). | ||||||
|  | - slack - add option ``prepend_hash`` which allows to control whether a ``#`` is prepended to ``channel_id``. The current behavior (value ``auto``) is to prepend ``#`` unless some specific prefixes are found. That list of prefixes is incomplete, and there does not seem to exist a documented condition on when exactly ``#`` must not be prepended. We recommend to explicitly set ``prepend_hash=always`` or ``prepend_hash=never`` to avoid any ambiguity (https://github.com/ansible-collections/community.general/pull/5629). | ||||||
|  | - spotinst_aws_elastigroup - add ``elements`` attribute when missing in ``list`` parameters (https://github.com/ansible-collections/community.general/pull/5553). | ||||||
|  | - ssh_config - add ``host_key_algorithms`` option (https://github.com/ansible-collections/community.general/pull/5605). | ||||||
|  | - udm_share - added ``elements`` attribute to ``list`` type parameters (https://github.com/ansible-collections/community.general/pull/5557). | ||||||
|  | - udm_user - add ``elements`` attribute when missing in ``list`` parameters (https://github.com/ansible-collections/community.general/pull/5559). | ||||||
|  | 
 | ||||||
|  | Deprecated Features | ||||||
|  | ------------------- | ||||||
|  | 
 | ||||||
|  | - The ``sap`` modules ``sapcar_extract``, ``sap_task_list_execute``, and ``hana_query``, will be removed from this collection in community.general 7.0.0 and replaced with redirects to ``community.sap_libs``. If you want to continue using these modules, make sure to also install ``community.sap_libs`` (it is part of the Ansible package) (https://github.com/ansible-collections/community.general/pull/5614). | ||||||
|  | 
 | ||||||
|  | Bugfixes | ||||||
|  | -------- | ||||||
|  | 
 | ||||||
|  | - chroot connection plugin - add ``inventory_hostname`` to vars under ``remote_addr``. This is needed for compatibility with ansible-core 2.13 (https://github.com/ansible-collections/community.general/pull/5570). | ||||||
|  | - cmd_runner module utils - fixed bug when handling default cases in ``cmd_runner_fmt.as_map()`` (https://github.com/ansible-collections/community.general/pull/5538). | ||||||
|  | - cmd_runner module utils - formatting arguments ``cmd_runner_fmt.as_fixed()`` was expecting an non-existing argument (https://github.com/ansible-collections/community.general/pull/5538). | ||||||
|  | - keycloak_client_rolemapping - calculate ``proposed`` and ``after`` return values properly (https://github.com/ansible-collections/community.general/pull/5619). | ||||||
|  | - keycloak_client_rolemapping - remove only listed mappings with ``state=absent`` (https://github.com/ansible-collections/community.general/pull/5619). | ||||||
|  | - proxmox inventory plugin - fix bug while templating when using templates for the ``url``, ``user``, ``password``, ``token_id``, or ``token_secret`` options (https://github.com/ansible-collections/community.general/pull/5640). | ||||||
|  | - proxmox inventory plugin - handle tags delimited by semicolon instead of comma, which happens from Proxmox 7.3 on (https://github.com/ansible-collections/community.general/pull/5602). | ||||||
|  | - redhat_subscription - do not ignore ``consumer_name`` and other variables if ``activationkey`` is specified (https://github.com/ansible-collections/community.general/issues/3486, https://github.com/ansible-collections/community.general/pull/5627). | ||||||
|  | - redhat_subscription - do not pass arguments to ``subscription-manager register`` for things already configured; now a specified ``rhsm_baseurl`` is properly set for subscription-manager (https://github.com/ansible-collections/community.general/pull/5583). | ||||||
|  | - unixy callback plugin - fix plugin to work with ansible-core 2.14 by using Ansible's configuration manager for handling options (https://github.com/ansible-collections/community.general/issues/5600). | ||||||
|  | - vdo - now uses ``yaml.safe_load()`` to parse command output instead of the deprecated ``yaml.load()`` which is potentially unsafe. Using ``yaml.load()`` without explicitely setting a ``Loader=`` is also an error in pyYAML 6.0 (https://github.com/ansible-collections/community.general/pull/5632). | ||||||
|  | - vmadm - fix for index out of range error in ``get_vm_uuid`` (https://github.com/ansible-collections/community.general/pull/5628). | ||||||
|  | 
 | ||||||
|  | New Modules | ||||||
|  | ----------- | ||||||
|  | 
 | ||||||
|  | - gitlab_project_badge - Manage project badges on GitLab Server | ||||||
|  | - keycloak_clientsecret_info - Retrieve client secret via Keycloak API | ||||||
|  | - keycloak_clientsecret_regenerate - Regenerate Keycloak client secret via Keycloak API | ||||||
|  | 
 | ||||||
|  | v6.0.1 | ||||||
|  | ====== | ||||||
|  | 
 | ||||||
|  | Release Summary | ||||||
|  | --------------- | ||||||
|  | 
 | ||||||
|  | Bugfix release for Ansible 7.0.0. | ||||||
|  | 
 | ||||||
|  | Bugfixes | ||||||
|  | -------- | ||||||
|  | 
 | ||||||
|  | - dependent lookup plugin - avoid warning on deprecated parameter for ``Templar.template()`` (https://github.com/ansible-collections/community.general/pull/5543). | ||||||
|  | - jenkins_build - fix the logical flaw when deleting a Jenkins build (https://github.com/ansible-collections/community.general/pull/5514). | ||||||
|  | - one_vm - avoid splitting labels that are ``None`` (https://github.com/ansible-collections/community.general/pull/5489). | ||||||
|  | - onepassword_raw - add missing parameter to plugin documentation (https://github.com/ansible-collections/community.general/issues/5506). | ||||||
|  | - proxmox_disk - avoid duplicate ``vmid`` reference (https://github.com/ansible-collections/community.general/issues/5492, https://github.com/ansible-collections/community.general/pull/5493). | ||||||
|  | 
 | ||||||
|  | v6.0.0 | ||||||
|  | ====== | ||||||
|  | 
 | ||||||
|  | Release Summary | ||||||
|  | --------------- | ||||||
|  | 
 | ||||||
|  | New major release of community.general with lots of bugfixes, new features, some removed deprecated features, and some other breaking changes. Please check the coresponding sections of the changelog for more details. | ||||||
| 
 | 
 | ||||||
| Major Changes | Major Changes | ||||||
| ------------- | ------------- | ||||||
|  | @ -33,6 +169,7 @@ Minor Changes | ||||||
| - alternatives - add ``state=absent`` to be able to remove an alternative (https://github.com/ansible-collections/community.general/pull/4654). | - alternatives - add ``state=absent`` to be able to remove an alternative (https://github.com/ansible-collections/community.general/pull/4654). | ||||||
| - alternatives - add ``subcommands`` parameter (https://github.com/ansible-collections/community.general/pull/4654). | - alternatives - add ``subcommands`` parameter (https://github.com/ansible-collections/community.general/pull/4654). | ||||||
| - ansible_galaxy_install - minor refactoring using latest ``ModuleHelper`` updates (https://github.com/ansible-collections/community.general/pull/4752). | - ansible_galaxy_install - minor refactoring using latest ``ModuleHelper`` updates (https://github.com/ansible-collections/community.general/pull/4752). | ||||||
|  | - ansible_galaxy_install - refactored module to use ``CmdRunner`` to execute ``ansible-galaxy`` (https://github.com/ansible-collections/community.general/pull/5477). | ||||||
| - apk - add ``world`` parameter for supporting a custom world file (https://github.com/ansible-collections/community.general/pull/4976). | - apk - add ``world`` parameter for supporting a custom world file (https://github.com/ansible-collections/community.general/pull/4976). | ||||||
| - bitwarden lookup plugin - add option ``search`` to search for other attributes than name (https://github.com/ansible-collections/community.general/pull/5297). | - bitwarden lookup plugin - add option ``search`` to search for other attributes than name (https://github.com/ansible-collections/community.general/pull/5297). | ||||||
| - cartesian lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | - cartesian lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | ||||||
|  | @ -41,6 +178,7 @@ Minor Changes | ||||||
| - consul - adds ``ttl`` parameter for session  (https://github.com/ansible-collections/community.general/pull/4996). | - consul - adds ``ttl`` parameter for session  (https://github.com/ansible-collections/community.general/pull/4996). | ||||||
| - consul - minor refactoring (https://github.com/ansible-collections/community.general/pull/5367). | - consul - minor refactoring (https://github.com/ansible-collections/community.general/pull/5367). | ||||||
| - consul_session - adds ``token`` parameter for session (https://github.com/ansible-collections/community.general/pull/5193). | - consul_session - adds ``token`` parameter for session (https://github.com/ansible-collections/community.general/pull/5193). | ||||||
|  | - cpanm - refactored module to use ``CmdRunner`` to execute ``cpanm`` (https://github.com/ansible-collections/community.general/pull/5485). | ||||||
| - cpanm - using ``do_raise()`` to raise exceptions in ``ModuleHelper`` derived modules (https://github.com/ansible-collections/community.general/pull/4674). | - cpanm - using ``do_raise()`` to raise exceptions in ``ModuleHelper`` derived modules (https://github.com/ansible-collections/community.general/pull/4674). | ||||||
| - credstash lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | - credstash lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | ||||||
| - dependent lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | - dependent lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | ||||||
|  | @ -65,9 +203,11 @@ Minor Changes | ||||||
| - gitlab_user - minor refactor when checking for installed dependency (https://github.com/ansible-collections/community.general/pull/5259). | - gitlab_user - minor refactor when checking for installed dependency (https://github.com/ansible-collections/community.general/pull/5259). | ||||||
| - hiera lookup plugin - start using Ansible's configuration manager to parse options. The Hiera executable and config file can now also be passed as lookup parameters (https://github.com/ansible-collections/community.general/pull/5440). | - hiera lookup plugin - start using Ansible's configuration manager to parse options. The Hiera executable and config file can now also be passed as lookup parameters (https://github.com/ansible-collections/community.general/pull/5440). | ||||||
| - homebrew, homebrew_tap - added Homebrew on Linux path to defaults (https://github.com/ansible-collections/community.general/pull/5241). | - homebrew, homebrew_tap - added Homebrew on Linux path to defaults (https://github.com/ansible-collections/community.general/pull/5241). | ||||||
|  | - hponcfg - refactored module to use ``CmdRunner`` to execute ``hponcfg`` (https://github.com/ansible-collections/community.general/pull/5483). | ||||||
| - keycloak_* modules - add ``http_agent`` parameter with default value ``Ansible`` (https://github.com/ansible-collections/community.general/issues/5023). | - keycloak_* modules - add ``http_agent`` parameter with default value ``Ansible`` (https://github.com/ansible-collections/community.general/issues/5023). | ||||||
| - keyring lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | - keyring lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | ||||||
| - lastpass - use config manager for handling plugin options (https://github.com/ansible-collections/community.general/pull/5022). | - lastpass - use config manager for handling plugin options (https://github.com/ansible-collections/community.general/pull/5022). | ||||||
|  | - ldap_attrs - allow for DNs to have ``{x}`` prefix on first RDN (https://github.com/ansible-collections/community.general/issues/977, https://github.com/ansible-collections/community.general/pull/5450). | ||||||
| - linode inventory plugin - simplify option handling (https://github.com/ansible-collections/community.general/pull/5438). | - linode inventory plugin - simplify option handling (https://github.com/ansible-collections/community.general/pull/5438). | ||||||
| - listen_ports_facts - add new ``include_non_listening`` option which adds ``-a`` option to ``netstat`` and ``ss``. This shows both listening and non-listening (for TCP this means established connections) sockets, and returns ``state`` and ``foreign_address`` (https://github.com/ansible-collections/community.general/issues/4762, https://github.com/ansible-collections/community.general/pull/4953). | - listen_ports_facts - add new ``include_non_listening`` option which adds ``-a`` option to ``netstat`` and ``ss``. This shows both listening and non-listening (for TCP this means established connections) sockets, and returns ``state`` and ``foreign_address`` (https://github.com/ansible-collections/community.general/issues/4762, https://github.com/ansible-collections/community.general/pull/4953). | ||||||
| - lmdb_kv lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | - lmdb_kv lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | ||||||
|  | @ -76,6 +216,7 @@ Minor Changes | ||||||
| - machinectl become plugin - combine the success command when building the become command to be consistent with other become plugins (https://github.com/ansible-collections/community.general/pull/5287). | - machinectl become plugin - combine the success command when building the become command to be consistent with other become plugins (https://github.com/ansible-collections/community.general/pull/5287). | ||||||
| - manifold lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | - manifold lookup plugin - start using Ansible's configuration manager to parse options (https://github.com/ansible-collections/community.general/pull/5440). | ||||||
| - maven_artifact - add a new ``unredirected_headers`` option that can be used with ansible-core 2.12 and above. The default value is to not use ``Authorization`` and ``Cookie`` headers on redirects for security reasons. With ansible-core 2.11, all headers are still passed on for redirects (https://github.com/ansible-collections/community.general/pull/4812). | - maven_artifact - add a new ``unredirected_headers`` option that can be used with ansible-core 2.12 and above. The default value is to not use ``Authorization`` and ``Cookie`` headers on redirects for security reasons. With ansible-core 2.11, all headers are still passed on for redirects (https://github.com/ansible-collections/community.general/pull/4812). | ||||||
|  | - mksysb - refactored module to use ``CmdRunner`` to execute ``mksysb`` (https://github.com/ansible-collections/community.general/pull/5484). | ||||||
| - mksysb - using ``do_raise()`` to raise exceptions in ``ModuleHelper`` derived modules (https://github.com/ansible-collections/community.general/pull/4674). | - mksysb - using ``do_raise()`` to raise exceptions in ``ModuleHelper`` derived modules (https://github.com/ansible-collections/community.general/pull/4674). | ||||||
| - nagios - minor refactoring on parameter validation for different actions (https://github.com/ansible-collections/community.general/pull/5239). | - nagios - minor refactoring on parameter validation for different actions (https://github.com/ansible-collections/community.general/pull/5239). | ||||||
| - netcup_dnsapi - add ``timeout`` parameter (https://github.com/ansible-collections/community.general/pull/5301). | - netcup_dnsapi - add ``timeout`` parameter (https://github.com/ansible-collections/community.general/pull/5301). | ||||||
|  | @ -83,6 +224,7 @@ Minor Changes | ||||||
| - nmcli - add bond option ``xmit_hash_policy`` to bond options (https://github.com/ansible-collections/community.general/issues/5148). | - nmcli - add bond option ``xmit_hash_policy`` to bond options (https://github.com/ansible-collections/community.general/issues/5148). | ||||||
| - nmcli - adds ``vpn`` type and parameter for supporting VPN with service type L2TP and PPTP (https://github.com/ansible-collections/community.general/pull/4746). | - nmcli - adds ``vpn`` type and parameter for supporting VPN with service type L2TP and PPTP (https://github.com/ansible-collections/community.general/pull/4746). | ||||||
| - nmcli - honor IP options for VPNs (https://github.com/ansible-collections/community.general/pull/5228). | - nmcli - honor IP options for VPNs (https://github.com/ansible-collections/community.general/pull/5228). | ||||||
|  | - onepassword - support version 2 of the OnePassword CLI (https://github.com/ansible-collections/community.general/pull/4728) | ||||||
| - opentelemetry callback plugin - allow configuring opentelementry callback via config file (https://github.com/ansible-collections/community.general/pull/4916). | - opentelemetry callback plugin - allow configuring opentelementry callback via config file (https://github.com/ansible-collections/community.general/pull/4916). | ||||||
| - opentelemetry callback plugin - send logs. This can be disabled by setting ``disable_logs=false`` (https://github.com/ansible-collections/community.general/pull/4175). | - opentelemetry callback plugin - send logs. This can be disabled by setting ``disable_logs=false`` (https://github.com/ansible-collections/community.general/pull/4175). | ||||||
| - pacman - added parameters ``reason`` and ``reason_for`` to set/change the install reason of packages (https://github.com/ansible-collections/community.general/pull/4956). | - pacman - added parameters ``reason`` and ``reason_for`` to set/change the install reason of packages (https://github.com/ansible-collections/community.general/pull/4956). | ||||||
|  | @ -125,6 +267,7 @@ Breaking Changes / Porting Guide | ||||||
| -------------------------------- | -------------------------------- | ||||||
| 
 | 
 | ||||||
| - newrelic_deployment - ``revision`` is required for v2 API (https://github.com/ansible-collections/community.general/pull/5341). | - newrelic_deployment - ``revision`` is required for v2 API (https://github.com/ansible-collections/community.general/pull/5341). | ||||||
|  | - scaleway_container_registry_info - no longer replace ``secret_environment_variables`` in the output by ``SENSITIVE_VALUE`` (https://github.com/ansible-collections/community.general/pull/5497). | ||||||
| 
 | 
 | ||||||
| Deprecated Features | Deprecated Features | ||||||
| ------------------- | ------------------- | ||||||
|  | @ -179,8 +322,10 @@ Bugfixes | ||||||
| - filesystem - improve error messages when output cannot be parsed by including newlines in escaped form (https://github.com/ansible-collections/community.general/pull/4700). | - filesystem - improve error messages when output cannot be parsed by including newlines in escaped form (https://github.com/ansible-collections/community.general/pull/4700). | ||||||
| - funcd connection plugin - fix signature of ``exec_command`` (https://github.com/ansible-collections/community.general/pull/5111). | - funcd connection plugin - fix signature of ``exec_command`` (https://github.com/ansible-collections/community.general/pull/5111). | ||||||
| - ini_file - minor refactor fixing a python lint error (https://github.com/ansible-collections/community.general/pull/5307). | - ini_file - minor refactor fixing a python lint error (https://github.com/ansible-collections/community.general/pull/5307). | ||||||
|  | - iso_create - the module somtimes failed to add folders for Joliet and UDF formats (https://github.com/ansible-collections/community.general/issues/5275). | ||||||
| - keycloak_realm - fix default groups and roles (https://github.com/ansible-collections/community.general/issues/4241). | - keycloak_realm - fix default groups and roles (https://github.com/ansible-collections/community.general/issues/4241). | ||||||
| - keyring_info - fix the result from the keyring library never getting returned (https://github.com/ansible-collections/community.general/pull/4964). | - keyring_info - fix the result from the keyring library never getting returned (https://github.com/ansible-collections/community.general/pull/4964). | ||||||
|  | - ldap_attrs - fix bug which caused a ``Bad search filter`` error. The error was occuring when the ldap attribute value contained special characters such as ``(`` or ``*`` (https://github.com/ansible-collections/community.general/issues/5434, https://github.com/ansible-collections/community.general/pull/5435). | ||||||
| - ldap_attrs - fix ordering issue by ignoring the ``{x}`` prefix on attribute values (https://github.com/ansible-collections/community.general/issues/977, https://github.com/ansible-collections/community.general/pull/5385). | - ldap_attrs - fix ordering issue by ignoring the ``{x}`` prefix on attribute values (https://github.com/ansible-collections/community.general/issues/977, https://github.com/ansible-collections/community.general/pull/5385). | ||||||
| - listen_ports_facts - removed leftover ``EnvironmentError`` . The ``else`` clause had a wrong indentation. The check is now handled in the ``split_pid_name`` function (https://github.com/ansible-collections/community.general/pull/5202). | - listen_ports_facts - removed leftover ``EnvironmentError`` . The ``else`` clause had a wrong indentation. The check is now handled in the ``split_pid_name`` function (https://github.com/ansible-collections/community.general/pull/5202). | ||||||
| - locale_gen - fix support for Ubuntu (https://github.com/ansible-collections/community.general/issues/5281). | - locale_gen - fix support for Ubuntu (https://github.com/ansible-collections/community.general/issues/5281). | ||||||
|  | @ -220,6 +365,7 @@ Bugfixes | ||||||
| - redis* modules - fix call to ``module.fail_json`` when failing because of missing Python libraries (https://github.com/ansible-collections/community.general/pull/4733). | - redis* modules - fix call to ``module.fail_json`` when failing because of missing Python libraries (https://github.com/ansible-collections/community.general/pull/4733). | ||||||
| - slack - fix incorrect channel prefix ``#`` caused by incomplete pattern detection by adding ``G0`` and ``GF`` as channel ID patterns (https://github.com/ansible-collections/community.general/pull/5019). | - slack - fix incorrect channel prefix ``#`` caused by incomplete pattern detection by adding ``G0`` and ``GF`` as channel ID patterns (https://github.com/ansible-collections/community.general/pull/5019). | ||||||
| - slack - fix message update for channels which start with ``CP``. When ``message-id`` was passed it failed for channels which started with ``CP`` because the ``#`` symbol was added before the ``channel_id`` (https://github.com/ansible-collections/community.general/pull/5249). | - slack - fix message update for channels which start with ``CP``. When ``message-id`` was passed it failed for channels which started with ``CP`` because the ``#`` symbol was added before the ``channel_id`` (https://github.com/ansible-collections/community.general/pull/5249). | ||||||
|  | - snap - allow values in the ``options`` parameter to contain whitespaces (https://github.com/ansible-collections/community.general/pull/5475). | ||||||
| - sudoers - ensure sudoers config files are created with the permissions requested by sudoers (0440) (https://github.com/ansible-collections/community.general/pull/4814). | - sudoers - ensure sudoers config files are created with the permissions requested by sudoers (0440) (https://github.com/ansible-collections/community.general/pull/4814). | ||||||
| - sudoers - fix incorrect handling of ``state: absent`` (https://github.com/ansible-collections/community.general/issues/4852). | - sudoers - fix incorrect handling of ``state: absent`` (https://github.com/ansible-collections/community.general/issues/4852). | ||||||
| - tss lookup plugin - adding support for updated Delinea library (https://github.com/DelineaXPM/python-tss-sdk/issues/9, https://github.com/ansible-collections/community.general/pull/5151). | - tss lookup plugin - adding support for updated Delinea library (https://github.com/DelineaXPM/python-tss-sdk/issues/9, https://github.com/ansible-collections/community.general/pull/5151). | ||||||
|  | @ -229,8 +375,41 @@ Bugfixes | ||||||
| - xfconf - fix setting of boolean values (https://github.com/ansible-collections/community.general/issues/4999, https://github.com/ansible-collections/community.general/pull/5007). | - xfconf - fix setting of boolean values (https://github.com/ansible-collections/community.general/issues/4999, https://github.com/ansible-collections/community.general/pull/5007). | ||||||
| - zfs - fix wrong quoting of properties (https://github.com/ansible-collections/community.general/issues/4707, https://github.com/ansible-collections/community.general/pull/4726). | - zfs - fix wrong quoting of properties (https://github.com/ansible-collections/community.general/issues/4707, https://github.com/ansible-collections/community.general/pull/4726). | ||||||
| 
 | 
 | ||||||
|  | New Plugins | ||||||
|  | ----------- | ||||||
|  | 
 | ||||||
|  | Filter | ||||||
|  | ~~~~~~ | ||||||
|  | 
 | ||||||
|  | - counter - Counts hashable elements in a sequence | ||||||
|  | 
 | ||||||
|  | Lookup | ||||||
|  | ~~~~~~ | ||||||
|  | 
 | ||||||
|  | - bitwarden - Retrieve secrets from Bitwarden | ||||||
|  | 
 | ||||||
| New Modules | New Modules | ||||||
| ----------- | ----------- | ||||||
| 
 | 
 | ||||||
|  | - gconftool2_info - Retrieve GConf configurations | ||||||
|  | - iso_customize - Add/remove/change files in ISO file | ||||||
|  | - keycloak_user_rolemapping - Allows administration of Keycloak user_rolemapping with the Keycloak API | ||||||
|  | - keyring - Set or delete a passphrase using the Operating System's native keyring | ||||||
|  | - keyring_info - Get a passphrase using the Operating System's native keyring | ||||||
|  | - manageiq_policies_info - Listing of resource policy_profiles in ManageIQ | ||||||
|  | - manageiq_tags_info - Retrieve resource tags in ManageIQ | ||||||
|  | - pipx_info - Rretrieves information about applications installed with pipx | ||||||
|  | - proxmox_disk - Management of a disk of a Qemu(KVM) VM in a Proxmox VE cluster. | ||||||
|  | - scaleway_compute_private_network - Scaleway compute - private network management | ||||||
|  | - scaleway_container - Scaleway Container management | ||||||
|  | - scaleway_container_info - Retrieve information on Scaleway Container | ||||||
|  | - scaleway_container_namespace - Scaleway Container namespace management | ||||||
|  | - scaleway_container_namespace_info - Retrieve information on Scaleway Container namespace | ||||||
|  | - scaleway_container_registry - Scaleway Container registry management module | ||||||
|  | - scaleway_container_registry_info - Scaleway Container registry info module | ||||||
|  | - scaleway_function - Scaleway Function management | ||||||
|  | - scaleway_function_info - Retrieve information on Scaleway Function | ||||||
| - scaleway_function_namespace - Scaleway Function namespace management | - scaleway_function_namespace - Scaleway Function namespace management | ||||||
| - scaleway_function_namespace_info - Retrieve information on Scaleway Function namespace | - scaleway_function_namespace_info - Retrieve information on Scaleway Function namespace | ||||||
|  | - wdc_redfish_command - Manages WDC UltraStar Data102 Out-Of-Band controllers using Redfish APIs | ||||||
|  | - wdc_redfish_info - Manages WDC UltraStar Data102 Out-Of-Band controllers using Redfish APIs | ||||||
|  |  | ||||||
|  | @ -31,7 +31,7 @@ Also, consider taking up a valuable, reviewed, but abandoned pull request which | ||||||
| * Try committing your changes with an informative but short commit message. | * Try committing your changes with an informative but short commit message. | ||||||
| * Do not squash your commits and force-push to your branch if not needed. Reviews of your pull request are much easier with individual commits to comprehend the pull request history. All commits of your pull request branch will be squashed into one commit by GitHub upon merge. | * Do not squash your commits and force-push to your branch if not needed. Reviews of your pull request are much easier with individual commits to comprehend the pull request history. All commits of your pull request branch will be squashed into one commit by GitHub upon merge. | ||||||
| * Do not add merge commits to your PR. The bot will complain and you will have to rebase ([instructions for rebasing](https://docs.ansible.com/ansible/latest/dev_guide/developing_rebasing.html)) to remove them before your PR can be merged. To avoid that git automatically does merges during pulls, you can configure it to do rebases instead by running `git config pull.rebase true` inside the repository checkout. | * Do not add merge commits to your PR. The bot will complain and you will have to rebase ([instructions for rebasing](https://docs.ansible.com/ansible/latest/dev_guide/developing_rebasing.html)) to remove them before your PR can be merged. To avoid that git automatically does merges during pulls, you can configure it to do rebases instead by running `git config pull.rebase true` inside the repository checkout. | ||||||
| * Make sure your PR includes a [changelog fragment](https://docs.ansible.com/ansible/devel/community/development_process.html#changelogs-how-to). (You must not include a fragment for new modules or new plugins, except for test and filter plugins. Also you shouldn't include one for docs-only changes. If you're not sure, simply don't include one, we'll tell you whether one is needed or not :) ) | * Make sure your PR includes a [changelog fragment](https://docs.ansible.com/ansible/devel/community/development_process.html#creating-changelog-fragments). (You must not include a fragment for new modules or new plugins, except for test and filter plugins. Also you shouldn't include one for docs-only changes. If you're not sure, simply don't include one, we'll tell you whether one is needed or not :) ) | ||||||
| * Avoid reformatting unrelated parts of the codebase in your PR. These types of changes will likely be requested for reversion, create additional work for reviewers, and may cause approval to be delayed. | * Avoid reformatting unrelated parts of the codebase in your PR. These types of changes will likely be requested for reversion, create additional work for reviewers, and may cause approval to be delayed. | ||||||
| 
 | 
 | ||||||
| You can also read [our Quick-start development guide](https://github.com/ansible/community-docs/blob/main/create_pr_quick_start_guide.rst). | You can also read [our Quick-start development guide](https://github.com/ansible/community-docs/blob/main/create_pr_quick_start_guide.rst). | ||||||
|  |  | ||||||
							
								
								
									
										16
									
								
								README.md
									
										
									
									
									
								
							
							
						
						
									
										16
									
								
								README.md
									
										
									
									
									
								
							|  | @ -6,7 +6,7 @@ SPDX-License-Identifier: GPL-3.0-or-later | ||||||
| 
 | 
 | ||||||
| # Community General Collection | # Community General Collection | ||||||
| 
 | 
 | ||||||
| [](https://dev.azure.com/ansible/community.general/_build?definitionId=31) | [](https://dev.azure.com/ansible/community.general/_build?definitionId=31) | ||||||
| [](https://codecov.io/gh/ansible-collections/community.general) | [](https://codecov.io/gh/ansible-collections/community.general) | ||||||
| 
 | 
 | ||||||
| This repository contains the `community.general` Ansible Collection. The collection is a part of the Ansible package and includes many modules and plugins supported by Ansible community which are not part of more specialized community collections. | This repository contains the `community.general` Ansible Collection. The collection is a part of the Ansible package and includes many modules and plugins supported by Ansible community which are not part of more specialized community collections. | ||||||
|  | @ -72,13 +72,13 @@ We are actively accepting new contributors. | ||||||
| 
 | 
 | ||||||
| All types of contributions are very welcome. | All types of contributions are very welcome. | ||||||
| 
 | 
 | ||||||
| You don't know how to start? Refer to our [contribution guide](https://github.com/ansible-collections/community.general/blob/main/CONTRIBUTING.md)! | You don't know how to start? Refer to our [contribution guide](https://github.com/ansible-collections/community.general/blob/stable-6/CONTRIBUTING.md)! | ||||||
| 
 | 
 | ||||||
| The current maintainers are listed in the [commit-rights.md](https://github.com/ansible-collections/community.general/blob/main/commit-rights.md#people) file. If you have questions or need help, feel free to mention them in the proposals. | The current maintainers are listed in the [commit-rights.md](https://github.com/ansible-collections/community.general/blob/stable-6/commit-rights.md#people) file. If you have questions or need help, feel free to mention them in the proposals. | ||||||
| 
 | 
 | ||||||
| You can find more information in the [developer guide for collections](https://docs.ansible.com/ansible/devel/dev_guide/developing_collections.html#contributing-to-collections), and in the [Ansible Community Guide](https://docs.ansible.com/ansible/latest/community/index.html). | You can find more information in the [developer guide for collections](https://docs.ansible.com/ansible/devel/dev_guide/developing_collections.html#contributing-to-collections), and in the [Ansible Community Guide](https://docs.ansible.com/ansible/latest/community/index.html). | ||||||
| 
 | 
 | ||||||
| Also for some notes specific to this collection see [our CONTRIBUTING documentation](https://github.com/ansible-collections/community.general/blob/main/CONTRIBUTING.md). | Also for some notes specific to this collection see [our CONTRIBUTING documentation](https://github.com/ansible-collections/community.general/blob/stable-6/CONTRIBUTING.md). | ||||||
| 
 | 
 | ||||||
| ### Running tests | ### Running tests | ||||||
| 
 | 
 | ||||||
|  | @ -88,7 +88,7 @@ See [here](https://docs.ansible.com/ansible/devel/dev_guide/developing_collectio | ||||||
| 
 | 
 | ||||||
| To learn how to maintain / become a maintainer of this collection, refer to: | To learn how to maintain / become a maintainer of this collection, refer to: | ||||||
| 
 | 
 | ||||||
| * [Committer guidelines](https://github.com/ansible-collections/community.general/blob/main/commit-rights.md). | * [Committer guidelines](https://github.com/ansible-collections/community.general/blob/stable-6/commit-rights.md). | ||||||
| * [Maintainer guidelines](https://github.com/ansible/community-docs/blob/main/maintaining.rst). | * [Maintainer guidelines](https://github.com/ansible/community-docs/blob/main/maintaining.rst). | ||||||
| 
 | 
 | ||||||
| It is necessary for maintainers of this collection to be subscribed to: | It is necessary for maintainers of this collection to be subscribed to: | ||||||
|  | @ -116,7 +116,7 @@ See the [Releasing guidelines](https://github.com/ansible/community-docs/blob/ma | ||||||
| 
 | 
 | ||||||
| ## Release notes | ## Release notes | ||||||
| 
 | 
 | ||||||
| See the [changelog](https://github.com/ansible-collections/community.general/blob/main/CHANGELOG.rst). | See the [changelog](https://github.com/ansible-collections/community.general/blob/stable-6/CHANGELOG.rst). | ||||||
| 
 | 
 | ||||||
| ## Roadmap | ## Roadmap | ||||||
| 
 | 
 | ||||||
|  | @ -135,8 +135,8 @@ See [this issue](https://github.com/ansible-collections/community.general/issues | ||||||
| 
 | 
 | ||||||
| This collection is primarily licensed and distributed as a whole under the GNU General Public License v3.0 or later. | This collection is primarily licensed and distributed as a whole under the GNU General Public License v3.0 or later. | ||||||
| 
 | 
 | ||||||
| See [LICENSES/GPL-3.0-or-later.txt](https://github.com/ansible-collections/community.general/blob/main/COPYING) for the full text. | See [LICENSES/GPL-3.0-or-later.txt](https://github.com/ansible-collections/community.general/blob/stable-6/COPYING) for the full text. | ||||||
| 
 | 
 | ||||||
| Parts of the collection are licensed under the [BSD 2-Clause license](https://github.com/ansible-collections/community.general/blob/main/LICENSES/BSD-2-Clause.txt), the [MIT license](https://github.com/ansible-collections/community.general/blob/main/LICENSES/MIT.txt), and the [PSF 2.0 license](https://github.com/ansible-collections/community.general/blob/main/LICENSES/PSF-2.0.txt). | Parts of the collection are licensed under the [BSD 2-Clause license](https://github.com/ansible-collections/community.general/blob/stable-6/LICENSES/BSD-2-Clause.txt), the [MIT license](https://github.com/ansible-collections/community.general/blob/stable-6/LICENSES/MIT.txt), and the [PSF 2.0 license](https://github.com/ansible-collections/community.general/blob/stable-6/LICENSES/PSF-2.0.txt). | ||||||
| 
 | 
 | ||||||
| All files have a machine readable `SDPX-License-Identifier:` comment denoting its respective license(s) or an equivalent entry in an accompanying `.license` file. Only changelog fragments (which will not be part of a release) are covered by a blanket statement in `.reuse/dep5`. This conforms to the [REUSE specification](https://reuse.software/spec/). | All files have a machine readable `SDPX-License-Identifier:` comment denoting its respective license(s) or an equivalent entry in an accompanying `.license` file. Only changelog fragments (which will not be part of a release) are covered by a blanket statement in `.reuse/dep5`. This conforms to the [REUSE specification](https://reuse.software/spec/). | ||||||
|  |  | ||||||
|  | @ -1,5 +1,62 @@ | ||||||
| ancestor: 5.0.0 | ancestor: 5.0.0 | ||||||
| releases: | releases: | ||||||
|  |   6.0.0: | ||||||
|  |     changes: | ||||||
|  |       breaking_changes: | ||||||
|  |       - scaleway_container_registry_info - no longer replace ``secret_environment_variables`` | ||||||
|  |         in the output by ``SENSITIVE_VALUE`` (https://github.com/ansible-collections/community.general/pull/5497). | ||||||
|  |       bugfixes: | ||||||
|  |       - iso_create - the module somtimes failed to add folders for Joliet and UDF | ||||||
|  |         formats (https://github.com/ansible-collections/community.general/issues/5275). | ||||||
|  |       - ldap_attrs - fix bug which caused a ``Bad search filter`` error. The error | ||||||
|  |         was occuring when the ldap attribute value contained special characters such | ||||||
|  |         as ``(`` or ``*`` (https://github.com/ansible-collections/community.general/issues/5434, | ||||||
|  |         https://github.com/ansible-collections/community.general/pull/5435). | ||||||
|  |       - snap - allow values in the ``options`` parameter to contain whitespaces (https://github.com/ansible-collections/community.general/pull/5475). | ||||||
|  |       minor_changes: | ||||||
|  |       - ansible_galaxy_install - refactored module to use ``CmdRunner`` to execute | ||||||
|  |         ``ansible-galaxy`` (https://github.com/ansible-collections/community.general/pull/5477). | ||||||
|  |       - cpanm - refactored module to use ``CmdRunner`` to execute ``cpanm`` (https://github.com/ansible-collections/community.general/pull/5485). | ||||||
|  |       - hponcfg - refactored module to use ``CmdRunner`` to execute ``hponcfg`` (https://github.com/ansible-collections/community.general/pull/5483). | ||||||
|  |       - ldap_attrs - allow for DNs to have ``{x}`` prefix on first RDN (https://github.com/ansible-collections/community.general/issues/977, | ||||||
|  |         https://github.com/ansible-collections/community.general/pull/5450). | ||||||
|  |       - mksysb - refactored module to use ``CmdRunner`` to execute ``mksysb`` (https://github.com/ansible-collections/community.general/pull/5484). | ||||||
|  |       - onepassword - support version 2 of the OnePassword CLI (https://github.com/ansible-collections/community.general/pull/4728) | ||||||
|  |       release_summary: New major release of community.general with lots of bugfixes, | ||||||
|  |         new features, some removed deprecated features, and some other breaking changes. | ||||||
|  |         Please check the coresponding sections of the changelog for more details. | ||||||
|  |     fragments: | ||||||
|  |     - 4728-onepassword-v2.yml | ||||||
|  |     - 5435-escape-ldap-param.yml | ||||||
|  |     - 5450-allow-for-xordered-dns.yaml | ||||||
|  |     - 5468-iso-create-not-add-folders.yml | ||||||
|  |     - 5475-snap-option-value-whitespace.yml | ||||||
|  |     - 5477-ansible-galaxy-install-cmd-runner.yml | ||||||
|  |     - 5483-hponcfg-cmd-runner.yml | ||||||
|  |     - 5484-mksysb-cmd-runner.yml | ||||||
|  |     - 5485-cpanm-cmd-runner.yml | ||||||
|  |     - 5497-scaleway-filtering.yml | ||||||
|  |     - 6.0.0.yml | ||||||
|  |     modules: | ||||||
|  |     - description: Scaleway Container management | ||||||
|  |       name: scaleway_container | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Retrieve information on Scaleway Container | ||||||
|  |       name: scaleway_container_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Scaleway Container namespace management | ||||||
|  |       name: scaleway_container_namespace | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Retrieve information on Scaleway Container namespace | ||||||
|  |       name: scaleway_container_namespace_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Scaleway Function management | ||||||
|  |       name: scaleway_function | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Retrieve information on Scaleway Function | ||||||
|  |       name: scaleway_function_info | ||||||
|  |       namespace: '' | ||||||
|  |     release_date: '2022-11-07' | ||||||
|   6.0.0-a1: |   6.0.0-a1: | ||||||
|     changes: |     changes: | ||||||
|       breaking_changes: |       breaking_changes: | ||||||
|  | @ -508,10 +565,321 @@ releases: | ||||||
|     - simplified-bsd-license.yml |     - simplified-bsd-license.yml | ||||||
|     - unflatmap.yml |     - unflatmap.yml | ||||||
|     modules: |     modules: | ||||||
|  |     - description: Retrieve GConf configurations | ||||||
|  |       name: gconftool2_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Add/remove/change files in ISO file | ||||||
|  |       name: iso_customize | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Allows administration of Keycloak user_rolemapping with the Keycloak | ||||||
|  |         API | ||||||
|  |       name: keycloak_user_rolemapping | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Set or delete a passphrase using the Operating System's native | ||||||
|  |         keyring | ||||||
|  |       name: keyring | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Get a passphrase using the Operating System's native keyring | ||||||
|  |       name: keyring_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Listing of resource policy_profiles in ManageIQ | ||||||
|  |       name: manageiq_policies_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Retrieve resource tags in ManageIQ | ||||||
|  |       name: manageiq_tags_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Rretrieves information about applications installed with pipx | ||||||
|  |       name: pipx_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Management of a disk of a Qemu(KVM) VM in a Proxmox VE cluster. | ||||||
|  |       name: proxmox_disk | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Scaleway compute - private network management | ||||||
|  |       name: scaleway_compute_private_network | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Scaleway Container registry management module | ||||||
|  |       name: scaleway_container_registry | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Scaleway Container registry info module | ||||||
|  |       name: scaleway_container_registry_info | ||||||
|  |       namespace: '' | ||||||
|     - description: Scaleway Function namespace management |     - description: Scaleway Function namespace management | ||||||
|       name: scaleway_function_namespace |       name: scaleway_function_namespace | ||||||
|       namespace: '' |       namespace: '' | ||||||
|     - description: Retrieve information on Scaleway Function namespace |     - description: Retrieve information on Scaleway Function namespace | ||||||
|       name: scaleway_function_namespace_info |       name: scaleway_function_namespace_info | ||||||
|       namespace: '' |       namespace: '' | ||||||
|  |     - description: Manages WDC UltraStar Data102 Out-Of-Band controllers using Redfish | ||||||
|  |         APIs | ||||||
|  |       name: wdc_redfish_command | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Manages WDC UltraStar Data102 Out-Of-Band controllers using Redfish | ||||||
|  |         APIs | ||||||
|  |       name: wdc_redfish_info | ||||||
|  |       namespace: '' | ||||||
|  |     plugins: | ||||||
|  |       filter: | ||||||
|  |       - description: Counts hashable elements in a sequence | ||||||
|  |         name: counter | ||||||
|  |         namespace: null | ||||||
|  |       lookup: | ||||||
|  |       - description: Retrieve secrets from Bitwarden | ||||||
|  |         name: bitwarden | ||||||
|  |         namespace: null | ||||||
|     release_date: '2022-11-02' |     release_date: '2022-11-02' | ||||||
|  |   6.0.1: | ||||||
|  |     changes: | ||||||
|  |       bugfixes: | ||||||
|  |       - dependent lookup plugin - avoid warning on deprecated parameter for ``Templar.template()`` | ||||||
|  |         (https://github.com/ansible-collections/community.general/pull/5543). | ||||||
|  |       - jenkins_build - fix the logical flaw when deleting a Jenkins build (https://github.com/ansible-collections/community.general/pull/5514). | ||||||
|  |       - one_vm - avoid splitting labels that are ``None`` (https://github.com/ansible-collections/community.general/pull/5489). | ||||||
|  |       - onepassword_raw - add missing parameter to plugin documentation (https://github.com/ansible-collections/community.general/issues/5506). | ||||||
|  |       - proxmox_disk - avoid duplicate ``vmid`` reference (https://github.com/ansible-collections/community.general/issues/5492, | ||||||
|  |         https://github.com/ansible-collections/community.general/pull/5493). | ||||||
|  |       release_summary: Bugfix release for Ansible 7.0.0. | ||||||
|  |     fragments: | ||||||
|  |     - 5489-nonetype-in-get-vm-by-label.yml | ||||||
|  |     - 5493-proxmox.yml | ||||||
|  |     - 5506-onepassword_raw-missing-param.yml | ||||||
|  |     - 5514-fix-logical-flaw-when-deleting-jenkins-build.yml | ||||||
|  |     - 5543-dependent-template.yml | ||||||
|  |     - 6.0.1.yml | ||||||
|  |     release_date: '2022-11-15' | ||||||
|  |   6.1.0: | ||||||
|  |     changes: | ||||||
|  |       bugfixes: | ||||||
|  |       - chroot connection plugin - add ``inventory_hostname`` to vars under ``remote_addr``. | ||||||
|  |         This is needed for compatibility with ansible-core 2.13 (https://github.com/ansible-collections/community.general/pull/5570). | ||||||
|  |       - cmd_runner module utils - fixed bug when handling default cases in ``cmd_runner_fmt.as_map()`` | ||||||
|  |         (https://github.com/ansible-collections/community.general/pull/5538). | ||||||
|  |       - cmd_runner module utils - formatting arguments ``cmd_runner_fmt.as_fixed()`` | ||||||
|  |         was expecting an non-existing argument (https://github.com/ansible-collections/community.general/pull/5538). | ||||||
|  |       - keycloak_client_rolemapping - calculate ``proposed`` and ``after`` return | ||||||
|  |         values properly (https://github.com/ansible-collections/community.general/pull/5619). | ||||||
|  |       - keycloak_client_rolemapping - remove only listed mappings with ``state=absent`` | ||||||
|  |         (https://github.com/ansible-collections/community.general/pull/5619). | ||||||
|  |       - proxmox inventory plugin - fix bug while templating when using templates for | ||||||
|  |         the ``url``, ``user``, ``password``, ``token_id``, or ``token_secret`` options | ||||||
|  |         (https://github.com/ansible-collections/community.general/pull/5640). | ||||||
|  |       - proxmox inventory plugin - handle tags delimited by semicolon instead of comma, | ||||||
|  |         which happens from Proxmox 7.3 on (https://github.com/ansible-collections/community.general/pull/5602). | ||||||
|  |       - redhat_subscription - do not ignore ``consumer_name`` and other variables | ||||||
|  |         if ``activationkey`` is specified (https://github.com/ansible-collections/community.general/issues/3486, | ||||||
|  |         https://github.com/ansible-collections/community.general/pull/5627). | ||||||
|  |       - redhat_subscription - do not pass arguments to ``subscription-manager register`` | ||||||
|  |         for things already configured; now a specified ``rhsm_baseurl`` is properly | ||||||
|  |         set for subscription-manager (https://github.com/ansible-collections/community.general/pull/5583). | ||||||
|  |       - unixy callback plugin - fix plugin to work with ansible-core 2.14 by using | ||||||
|  |         Ansible's configuration manager for handling options (https://github.com/ansible-collections/community.general/issues/5600). | ||||||
|  |       - vdo - now uses ``yaml.safe_load()`` to parse command output instead of the | ||||||
|  |         deprecated ``yaml.load()`` which is potentially unsafe. Using ``yaml.load()`` | ||||||
|  |         without explicitely setting a ``Loader=`` is also an error in pyYAML 6.0 (https://github.com/ansible-collections/community.general/pull/5632). | ||||||
|  |       - vmadm - fix for index out of range error in ``get_vm_uuid`` (https://github.com/ansible-collections/community.general/pull/5628). | ||||||
|  |       deprecated_features: | ||||||
|  |       - The ``sap`` modules ``sapcar_extract``, ``sap_task_list_execute``, and ``hana_query``, | ||||||
|  |         will be removed from this collection in community.general 7.0.0 and replaced | ||||||
|  |         with redirects to ``community.sap_libs``. If you want to continue using these | ||||||
|  |         modules, make sure to also install ``community.sap_libs`` (it is part of the | ||||||
|  |         Ansible package) (https://github.com/ansible-collections/community.general/pull/5614). | ||||||
|  |       minor_changes: | ||||||
|  |       - cmd_runner module utils - ``cmd_runner_fmt.as_bool()`` can now take an extra | ||||||
|  |         parameter to format when value is false (https://github.com/ansible-collections/community.general/pull/5647). | ||||||
|  |       - gconftool2 - refactor using ``ModuleHelper`` and ``CmdRunner`` (https://github.com/ansible-collections/community.general/pull/5545). | ||||||
|  |       - java_certs - add more detailed error output when extracting certificate from | ||||||
|  |         PKCS12 fails (https://github.com/ansible-collections/community.general/pull/5550). | ||||||
|  |       - jenkins_plugin - refactor code to module util to fix sanity check (https://github.com/ansible-collections/community.general/pull/5565). | ||||||
|  |       - lxd_project - refactored code out to module utils to clear sanity check (https://github.com/ansible-collections/community.general/pull/5549). | ||||||
|  |       - nmap inventory plugin - add new options ``udp_scan``, ``icmp_timestamp``, | ||||||
|  |         and ``dns_resolve`` for different types of scans (https://github.com/ansible-collections/community.general/pull/5566). | ||||||
|  |       - rax_scaling_group - refactored out code to the ``rax`` module utils to clear | ||||||
|  |         the sanity check (https://github.com/ansible-collections/community.general/pull/5563). | ||||||
|  |       - redfish_command - add ``PerformRequestedOperations`` command to perform any | ||||||
|  |         operations necessary to continue the update flow (https://github.com/ansible-collections/community.general/issues/4276). | ||||||
|  |       - redfish_command - add ``update_apply_time`` to ``SimpleUpdate`` command (https://github.com/ansible-collections/community.general/issues/3910). | ||||||
|  |       - redfish_command - add ``update_status`` to output of ``SimpleUpdate`` command | ||||||
|  |         to allow a user monitor the update in progress (https://github.com/ansible-collections/community.general/issues/4276). | ||||||
|  |       - redfish_info - add ``GetUpdateStatus`` command to check the progress of a | ||||||
|  |         previous update request (https://github.com/ansible-collections/community.general/issues/4276). | ||||||
|  |       - redfish_utils module utils - added PUT (``put_request()``) functionality (https://github.com/ansible-collections/community.general/pull/5490). | ||||||
|  |       - slack - add option ``prepend_hash`` which allows to control whether a ``#`` | ||||||
|  |         is prepended to ``channel_id``. The current behavior (value ``auto``) is to | ||||||
|  |         prepend ``#`` unless some specific prefixes are found. That list of prefixes | ||||||
|  |         is incomplete, and there does not seem to exist a documented condition on | ||||||
|  |         when exactly ``#`` must not be prepended. We recommend to explicitly set ``prepend_hash=always`` | ||||||
|  |         or ``prepend_hash=never`` to avoid any ambiguity (https://github.com/ansible-collections/community.general/pull/5629). | ||||||
|  |       - spotinst_aws_elastigroup - add ``elements`` attribute when missing in ``list`` | ||||||
|  |         parameters (https://github.com/ansible-collections/community.general/pull/5553). | ||||||
|  |       - ssh_config - add ``host_key_algorithms`` option (https://github.com/ansible-collections/community.general/pull/5605). | ||||||
|  |       - udm_share - added ``elements`` attribute to ``list`` type parameters (https://github.com/ansible-collections/community.general/pull/5557). | ||||||
|  |       - udm_user - add ``elements`` attribute when missing in ``list`` parameters | ||||||
|  |         (https://github.com/ansible-collections/community.general/pull/5559). | ||||||
|  |       release_summary: Regular bugfix and feature release. | ||||||
|  |     fragments: | ||||||
|  |     - 3910-redfish-add-operation-apply-time-to-simple-update.yml | ||||||
|  |     - 4276-redfish-command-updates-for-full-simple-update-workflow.yml | ||||||
|  |     - 5490-adding-put-functionality.yml | ||||||
|  |     - 5538-cmd-runner-as-fixed.yml | ||||||
|  |     - 5545-gconftool-cmd-runner.yml | ||||||
|  |     - 5549-lxd-project-sanity.yml | ||||||
|  |     - 5550-java_certs-not-enough-info-on-error.yml | ||||||
|  |     - 5553-spotinst-aws-elasticgroup-sanity.yml | ||||||
|  |     - 5557-udm-share-sanity.yml | ||||||
|  |     - 5559-udm-user-sanity.yml | ||||||
|  |     - 5563-rax-scaling-group-sanity.yml | ||||||
|  |     - 5565-jenkins-plugin-sanity.yml | ||||||
|  |     - 5566-additional-flags-nmap.yml | ||||||
|  |     - 5570-chroot-plugin-fix-default-inventory_hostname.yml | ||||||
|  |     - 5583-redhat_subscription-subscribe-parameters.yaml | ||||||
|  |     - 5601-unixy-callback-use-config-manager.yml | ||||||
|  |     - 5602-proxmox-tags.yml | ||||||
|  |     - 5605-ssh-config-add-host-key-algorithms.yaml | ||||||
|  |     - 5619-keycloak-improvements.yml | ||||||
|  |     - 5627-redhat_subscription-subscribe-parameters-2.yaml | ||||||
|  |     - 5628-fix-vmadm-off-by-one.yml | ||||||
|  |     - 5629-add-prepend-hash-option-for-channel-id.yml | ||||||
|  |     - 5632-vdo-Use-yaml-safe-load-instead-of-yaml-load.yml | ||||||
|  |     - 5640-fix-typo-proxmox-inventory.yml | ||||||
|  |     - 5647-cmd-runner-as-bool-false.yml | ||||||
|  |     - 6.1.0.yml | ||||||
|  |     - sap-removal.yml | ||||||
|  |     modules: | ||||||
|  |     - description: Manage project badges on GitLab Server | ||||||
|  |       name: gitlab_project_badge | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Retrieve client secret via Keycloak API | ||||||
|  |       name: keycloak_clientsecret_info | ||||||
|  |       namespace: '' | ||||||
|  |     - description: Regenerate Keycloak client secret via Keycloak API | ||||||
|  |       name: keycloak_clientsecret_regenerate | ||||||
|  |       namespace: '' | ||||||
|  |     release_date: '2022-12-06' | ||||||
|  |   6.2.0: | ||||||
|  |     changes: | ||||||
|  |       bugfixes: | ||||||
|  |       - ansible_galaxy_install - set default to raise exception if command's return | ||||||
|  |         code is different from zero (https://github.com/ansible-collections/community.general/pull/5680). | ||||||
|  |       - ansible_galaxy_install - try ``C.UTF-8`` and then fall back to ``en_US.UTF-8`` | ||||||
|  |         before failing (https://github.com/ansible-collections/community.general/pull/5680). | ||||||
|  |       - gitlab_group_variables - fix dropping variables accidentally when GitLab introduced | ||||||
|  |         new properties (https://github.com/ansible-collections/community.general/pull/5667). | ||||||
|  |       - gitlab_project_variables - fix dropping variables accidentally when GitLab | ||||||
|  |         introduced new properties (https://github.com/ansible-collections/community.general/pull/5667). | ||||||
|  |       - lxc_container - fix the arguments of the lxc command which broke the creation | ||||||
|  |         and cloning of containers (https://github.com/ansible-collections/community.general/issues/5578). | ||||||
|  |       - opkg - fix issue that ``force=reinstall`` would not reinstall an existing | ||||||
|  |         package (https://github.com/ansible-collections/community.general/pull/5705). | ||||||
|  |       - proxmox_disk - fixed possible issues with redundant ``vmid`` parameter (https://github.com/ansible-collections/community.general/issues/5492, | ||||||
|  |         https://github.com/ansible-collections/community.general/pull/5672). | ||||||
|  |       - proxmox_nic - fixed possible issues with redundant ``vmid`` parameter (https://github.com/ansible-collections/community.general/issues/5492, | ||||||
|  |         https://github.com/ansible-collections/community.general/pull/5672). | ||||||
|  |       - unixy callback plugin - fix typo introduced when updating to use Ansible's | ||||||
|  |         configuration manager for handling options (https://github.com/ansible-collections/community.general/issues/5600). | ||||||
|  |       deprecated_features: | ||||||
|  |       - manageiq_policies - deprecate ``state=list`` in favour of using ``community.general.manageiq_policies_info`` | ||||||
|  |         (https://github.com/ansible-collections/community.general/pull/5721). | ||||||
|  |       - rax - module relies on deprecates library ``pyrax``. Unless maintainers step | ||||||
|  |         up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_cbs - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_cbs_attachments - module relies on deprecates library ``pyrax``. Unless | ||||||
|  |         maintainers step up to work on the module, it will be marked as deprecated | ||||||
|  |         in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_cdb - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_cdb_database - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_cdb_user - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_clb - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_clb_nodes - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_clb_ssl - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_dns - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_dns_record - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_facts - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_files - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_files_objects - module relies on deprecates library ``pyrax``. Unless | ||||||
|  |         maintainers step up to work on the module, it will be marked as deprecated | ||||||
|  |         in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_identity - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_keypair - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_meta - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_mon_alarm - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_mon_check - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_mon_entity - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_mon_notification - module relies on deprecates library ``pyrax``. Unless | ||||||
|  |         maintainers step up to work on the module, it will be marked as deprecated | ||||||
|  |         in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_mon_notification_plan - module relies on deprecates library ``pyrax``. | ||||||
|  |         Unless maintainers step up to work on the module, it will be marked as deprecated | ||||||
|  |         in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_network - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_queue - module relies on deprecates library ``pyrax``. Unless maintainers | ||||||
|  |         step up to work on the module, it will be marked as deprecated in community.general | ||||||
|  |         7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_scaling_group - module relies on deprecates library ``pyrax``. Unless | ||||||
|  |         maintainers step up to work on the module, it will be marked as deprecated | ||||||
|  |         in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       - rax_scaling_policy - module relies on deprecates library ``pyrax``. Unless | ||||||
|  |         maintainers step up to work on the module, it will be marked as deprecated | ||||||
|  |         in community.general 7.0.0 and removed in version 9.0.0 (https://github.com/ansible-collections/community.general/pull/5733). | ||||||
|  |       minor_changes: | ||||||
|  |       - opkg - allow installing a package in a certain version (https://github.com/ansible-collections/community.general/pull/5688). | ||||||
|  |       - proxmox - added new module parameter ``tags`` for use with PVE 7+ (https://github.com/ansible-collections/community.general/pull/5714). | ||||||
|  |       - puppet - refactored module to use ``CmdRunner`` for executing ``puppet`` (https://github.com/ansible-collections/community.general/pull/5612). | ||||||
|  |       - redhat_subscription - add a ``server_proxy_scheme`` parameter to configure | ||||||
|  |         the scheme for the proxy server (https://github.com/ansible-collections/community.general/pull/5662). | ||||||
|  |       - ssh_config - refactor code to module util to fix sanity check (https://github.com/ansible-collections/community.general/pull/5720). | ||||||
|  |       - sudoers - adds ``host`` parameter for setting hostname restrictions in sudoers | ||||||
|  |         rules (https://github.com/ansible-collections/community.general/issues/5702). | ||||||
|  |       release_summary: Regular bugfix and feature release. | ||||||
|  |     fragments: | ||||||
|  |     - 5612-puppet-cmd-runner.yml | ||||||
|  |     - 5659-fix-lxc_container-command.yml | ||||||
|  |     - 5662-redhat_subscription-server_proxy_scheme.yaml | ||||||
|  |     - 5666-gitlab-variables.yml | ||||||
|  |     - 5672-proxmox.yml | ||||||
|  |     - 5680-ansible_galaxy_install-fx-locale.yaml | ||||||
|  |     - 5688-opkg-module-install-certain-version.yml | ||||||
|  |     - 5703-sudoers-host-support.yml | ||||||
|  |     - 5705-opkg-fix-force-reinstall.yml | ||||||
|  |     - 5714-proxmox-lxc-tag-support.yml | ||||||
|  |     - 5720-ssh_config-plugin-sanity.yml | ||||||
|  |     - 5721-manageiq-policies-deprecate-list-state.yaml | ||||||
|  |     - 5733-rax-deprecation-notice.yml | ||||||
|  |     - 5744-unixy-callback-fix-config-manager-typo.yml | ||||||
|  |     - 6.2.0.yml | ||||||
|  |     release_date: '2023-01-04' | ||||||
|  |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| minor_changes: |  | ||||||
|    - onepassword - support version 2 of the OnePassword CLI (https://github.com/ansible-collections/community.general/pull/4728) |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| bugfixes: |  | ||||||
|   - ldap_attrs - fix bug which caused a ``Bad search filter`` error. The error was occuring when the ldap attribute value contained special characters such as ``(`` or ``*`` (https://github.com/ansible-collections/community.general/issues/5434, https://github.com/ansible-collections/community.general/pull/5435). |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| minor_changes: |  | ||||||
|   - ldap_attrs - allow for DNs to have ``{x}`` prefix on first RDN (https://github.com/ansible-collections/community.general/issues/977, https://github.com/ansible-collections/community.general/pull/5450). |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| bugfixes: |  | ||||||
|   - iso_create - the module somtimes failed to add folders for Joliet and UDF formats (https://github.com/ansible-collections/community.general/issues/5275). |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| bugfixes: |  | ||||||
|   - snap - allow values in the ``options`` parameter to contain whitespaces (https://github.com/ansible-collections/community.general/pull/5475). |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| minor_changes: |  | ||||||
|   - ansible_galaxy_install - refactored module to use ``CmdRunner`` to execute ``ansible-galaxy`` (https://github.com/ansible-collections/community.general/pull/5477). |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| minor_changes: |  | ||||||
|   - hponcfg - refactored module to use ``CmdRunner`` to execute ``hponcfg`` (https://github.com/ansible-collections/community.general/pull/5483). |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| minor_changes: |  | ||||||
|   - mksysb - refactored module to use ``CmdRunner`` to execute ``mksysb`` (https://github.com/ansible-collections/community.general/pull/5484). |  | ||||||
|  | @ -1,2 +0,0 @@ | ||||||
| minor_changes: |  | ||||||
|   - cpanm - refactored module to use ``CmdRunner`` to execute ``cpanm`` (https://github.com/ansible-collections/community.general/pull/5485). |  | ||||||
|  | @ -1,3 +0,0 @@ | ||||||
| release_summary: >- |  | ||||||
|   New major release of community.general with lots of bugfixes, new features, some removed deprecated features, and some other breaking changes. |  | ||||||
|   Please check the coresponding sections of the changelog for more details. |  | ||||||
|  | @ -5,7 +5,7 @@ | ||||||
| 
 | 
 | ||||||
| namespace: community | namespace: community | ||||||
| name: general | name: general | ||||||
| version: 6.0.0 | version: 6.2.0 | ||||||
| readme: README.md | readme: README.md | ||||||
| authors: | authors: | ||||||
|   - Ansible (https://github.com/ansible) |   - Ansible (https://github.com/ansible) | ||||||
|  |  | ||||||
|  | @ -63,7 +63,7 @@ class CallbackModule(CallbackModule_default): | ||||||
| 
 | 
 | ||||||
|     def _preprocess_result(self, result): |     def _preprocess_result(self, result): | ||||||
|         self.delegated_vars = result._result.get('_ansible_delegated_vars', None) |         self.delegated_vars = result._result.get('_ansible_delegated_vars', None) | ||||||
|         self._handle_exception(result._result, use_stderr=self.display_failed_stderr) |         self._handle_exception(result._result, use_stderr=self.get_option('display_failed_stderr')) | ||||||
|         self._handle_warnings(result._result) |         self._handle_warnings(result._result) | ||||||
| 
 | 
 | ||||||
|     def _process_result_output(self, result, msg): |     def _process_result_output(self, result, msg): | ||||||
|  | @ -109,7 +109,7 @@ class CallbackModule(CallbackModule_default): | ||||||
|         self._display.display(msg) |         self._display.display(msg) | ||||||
| 
 | 
 | ||||||
|     def v2_runner_on_skipped(self, result, ignore_errors=False): |     def v2_runner_on_skipped(self, result, ignore_errors=False): | ||||||
|         if self.display_skipped_hosts: |         if self.get_option('display_skipped_hosts'): | ||||||
|             self._preprocess_result(result) |             self._preprocess_result(result) | ||||||
|             display_color = C.COLOR_SKIP |             display_color = C.COLOR_SKIP | ||||||
|             msg = "skipped" |             msg = "skipped" | ||||||
|  | @ -128,7 +128,7 @@ class CallbackModule(CallbackModule_default): | ||||||
|             msg += " | item: %s" % (item_value,) |             msg += " | item: %s" % (item_value,) | ||||||
| 
 | 
 | ||||||
|         task_result = self._process_result_output(result, msg) |         task_result = self._process_result_output(result, msg) | ||||||
|         self._display.display("  " + task_result, display_color, stderr=self.display_failed_stderr) |         self._display.display("  " + task_result, display_color, stderr=self.get_option('display_failed_stderr')) | ||||||
| 
 | 
 | ||||||
|     def v2_runner_on_ok(self, result, msg="ok", display_color=C.COLOR_OK): |     def v2_runner_on_ok(self, result, msg="ok", display_color=C.COLOR_OK): | ||||||
|         self._preprocess_result(result) |         self._preprocess_result(result) | ||||||
|  | @ -142,7 +142,7 @@ class CallbackModule(CallbackModule_default): | ||||||
|             display_color = C.COLOR_CHANGED |             display_color = C.COLOR_CHANGED | ||||||
|             task_result = self._process_result_output(result, msg) |             task_result = self._process_result_output(result, msg) | ||||||
|             self._display.display("  " + task_result, display_color) |             self._display.display("  " + task_result, display_color) | ||||||
|         elif self.display_ok_hosts: |         elif self.get_option('display_ok_hosts'): | ||||||
|             task_result = self._process_result_output(result, msg) |             task_result = self._process_result_output(result, msg) | ||||||
|             self._display.display("  " + task_result, display_color) |             self._display.display("  " + task_result, display_color) | ||||||
| 
 | 
 | ||||||
|  | @ -162,7 +162,7 @@ class CallbackModule(CallbackModule_default): | ||||||
|         display_color = C.COLOR_UNREACHABLE |         display_color = C.COLOR_UNREACHABLE | ||||||
|         task_result = self._process_result_output(result, msg) |         task_result = self._process_result_output(result, msg) | ||||||
| 
 | 
 | ||||||
|         self._display.display("  " + task_result, display_color, stderr=self.display_failed_stderr) |         self._display.display("  " + task_result, display_color, stderr=self.get_option('display_failed_stderr')) | ||||||
| 
 | 
 | ||||||
|     def v2_on_file_diff(self, result): |     def v2_on_file_diff(self, result): | ||||||
|         if result._task.loop and 'results' in result._result: |         if result._task.loop and 'results' in result._result: | ||||||
|  | @ -205,7 +205,7 @@ class CallbackModule(CallbackModule_default): | ||||||
|                 colorize(u'ignored', t['ignored'], None)), |                 colorize(u'ignored', t['ignored'], None)), | ||||||
|                 log_only=True |                 log_only=True | ||||||
|             ) |             ) | ||||||
|         if stats.custom and self.show_custom_stats: |         if stats.custom and self.get_option('show_custom_stats'): | ||||||
|             self._display.banner("CUSTOM STATS: ") |             self._display.banner("CUSTOM STATS: ") | ||||||
|             # per host |             # per host | ||||||
|             # TODO: come up with 'pretty format' |             # TODO: come up with 'pretty format' | ||||||
|  |  | ||||||
|  | @ -22,6 +22,7 @@ DOCUMENTATION = ''' | ||||||
|             - The path of the chroot you want to access. |             - The path of the chroot you want to access. | ||||||
|         default: inventory_hostname |         default: inventory_hostname | ||||||
|         vars: |         vars: | ||||||
|  |             - name: inventory_hostname | ||||||
|             - name: ansible_host |             - name: ansible_host | ||||||
|       executable: |       executable: | ||||||
|         description: |         description: | ||||||
|  |  | ||||||
|  | @ -26,6 +26,7 @@ DOCUMENTATION = ''' | ||||||
|       description: |       description: | ||||||
|         - The correct parser for the input data. |         - The correct parser for the input data. | ||||||
|         - For example C(ifconfig). |         - For example C(ifconfig). | ||||||
|  |         - "Note: use underscores instead of dashes (if any) in the parser module name." | ||||||
|         - See U(https://github.com/kellyjonbrazil/jc#parsers) for the latest list of parsers. |         - See U(https://github.com/kellyjonbrazil/jc#parsers) for the latest list of parsers. | ||||||
|       type: string |       type: string | ||||||
|       required: true |       required: true | ||||||
|  |  | ||||||
|  | @ -55,6 +55,11 @@ DOCUMENTATION = r''' | ||||||
|             type: str |             type: str | ||||||
|             default: none |             default: none | ||||||
|             choices: [ 'STOPPED', 'STARTING', 'RUNNING', 'none' ] |             choices: [ 'STOPPED', 'STARTING', 'RUNNING', 'none' ] | ||||||
|  |         project: | ||||||
|  |             description: Filter the instance according to the given project. | ||||||
|  |             type: str | ||||||
|  |             default: default | ||||||
|  |             version_added: 6.2.0 | ||||||
|         type_filter: |         type_filter: | ||||||
|             description: |             description: | ||||||
|             - Filter the instances by type C(virtual-machine), C(container) or C(both). |             - Filter the instances by type C(virtual-machine), C(container) or C(both). | ||||||
|  | @ -140,6 +145,9 @@ groupby: | ||||||
|   vlan666: |   vlan666: | ||||||
|     type: vlanid |     type: vlanid | ||||||
|     attribute: 666 |     attribute: 666 | ||||||
|  |   projectInternals: | ||||||
|  |     type: project | ||||||
|  |     attribute: internals | ||||||
| ''' | ''' | ||||||
| 
 | 
 | ||||||
| import binascii | import binascii | ||||||
|  | @ -153,6 +161,7 @@ from ansible.module_utils.common.text.converters import to_native, to_text | ||||||
| from ansible.module_utils.common.dict_transformations import dict_merge | from ansible.module_utils.common.dict_transformations import dict_merge | ||||||
| from ansible.module_utils.six import raise_from | from ansible.module_utils.six import raise_from | ||||||
| from ansible.errors import AnsibleError, AnsibleParserError | from ansible.errors import AnsibleError, AnsibleParserError | ||||||
|  | from ansible.module_utils.six.moves.urllib.parse import urlencode | ||||||
| from ansible_collections.community.general.plugins.module_utils.lxd import LXDClient, LXDClientException | from ansible_collections.community.general.plugins.module_utils.lxd import LXDClient, LXDClientException | ||||||
| 
 | 
 | ||||||
| try: | try: | ||||||
|  | @ -330,7 +339,15 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|         #        "status_code": 200, |         #        "status_code": 200, | ||||||
|         #        "type": "sync" |         #        "type": "sync" | ||||||
|         #      } |         #      } | ||||||
|         instances = self.socket.do('GET', '/1.0/instances') |         url = '/1.0/instances' | ||||||
|  |         if self.project: | ||||||
|  |             url = url + '?{0}'.format(urlencode(dict(project=self.project))) | ||||||
|  | 
 | ||||||
|  |         instances = self.socket.do('GET', url) | ||||||
|  | 
 | ||||||
|  |         if self.project: | ||||||
|  |             return [m.split('/')[3].split('?')[0] for m in instances['metadata']] | ||||||
|  | 
 | ||||||
|         return [m.split('/')[3] for m in instances['metadata']] |         return [m.split('/')[3] for m in instances['metadata']] | ||||||
| 
 | 
 | ||||||
|     def _get_config(self, branch, name): |     def _get_config(self, branch, name): | ||||||
|  | @ -351,9 +368,11 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|             dict(config): Config of the instance""" |             dict(config): Config of the instance""" | ||||||
|         config = {} |         config = {} | ||||||
|         if isinstance(branch, (tuple, list)): |         if isinstance(branch, (tuple, list)): | ||||||
|             config[name] = {branch[1]: self.socket.do('GET', '/1.0/{0}/{1}/{2}'.format(to_native(branch[0]), to_native(name), to_native(branch[1])))} |             config[name] = {branch[1]: self.socket.do( | ||||||
|  |                 'GET', '/1.0/{0}/{1}/{2}?{3}'.format(to_native(branch[0]), to_native(name), to_native(branch[1]), urlencode(dict(project=self.project))))} | ||||||
|         else: |         else: | ||||||
|             config[name] = {branch: self.socket.do('GET', '/1.0/{0}/{1}'.format(to_native(branch), to_native(name)))} |             config[name] = {branch: self.socket.do( | ||||||
|  |                 'GET', '/1.0/{0}/{1}?{2}'.format(to_native(branch), to_native(name), urlencode(dict(project=self.project))))} | ||||||
|         return config |         return config | ||||||
| 
 | 
 | ||||||
|     def get_instance_data(self, names): |     def get_instance_data(self, names): | ||||||
|  | @ -583,6 +602,8 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|             self._set_data_entry(instance_name, 'network_interfaces', self.extract_network_information_from_instance_config(instance_name)) |             self._set_data_entry(instance_name, 'network_interfaces', self.extract_network_information_from_instance_config(instance_name)) | ||||||
|             self._set_data_entry(instance_name, 'preferred_interface', self.get_prefered_instance_network_interface(instance_name)) |             self._set_data_entry(instance_name, 'preferred_interface', self.get_prefered_instance_network_interface(instance_name)) | ||||||
|             self._set_data_entry(instance_name, 'vlan_ids', self.get_instance_vlans(instance_name)) |             self._set_data_entry(instance_name, 'vlan_ids', self.get_instance_vlans(instance_name)) | ||||||
|  |             self._set_data_entry(instance_name, 'project', self._get_data_entry( | ||||||
|  |                 'instances/{0}/instances/metadata/project'.format(instance_name))) | ||||||
| 
 | 
 | ||||||
|     def build_inventory_network(self, instance_name): |     def build_inventory_network(self, instance_name): | ||||||
|         """Add the network interfaces of the instance to the inventory |         """Add the network interfaces of the instance to the inventory | ||||||
|  | @ -686,6 +707,8 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|             # add VLAN_ID information |             # add VLAN_ID information | ||||||
|             if self._get_data_entry('inventory/{0}/vlan_ids'.format(instance_name)): |             if self._get_data_entry('inventory/{0}/vlan_ids'.format(instance_name)): | ||||||
|                 self.inventory.set_variable(instance_name, 'ansible_lxd_vlan_ids', self._get_data_entry('inventory/{0}/vlan_ids'.format(instance_name))) |                 self.inventory.set_variable(instance_name, 'ansible_lxd_vlan_ids', self._get_data_entry('inventory/{0}/vlan_ids'.format(instance_name))) | ||||||
|  |             # add project | ||||||
|  |             self.inventory.set_variable(instance_name, 'ansible_lxd_project', self._get_data_entry('inventory/{0}/project'.format(instance_name))) | ||||||
| 
 | 
 | ||||||
|     def build_inventory_groups_location(self, group_name): |     def build_inventory_groups_location(self, group_name): | ||||||
|         """create group by attribute: location |         """create group by attribute: location | ||||||
|  | @ -761,6 +784,28 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|                             # Ignore invalid IP addresses returned by lxd |                             # Ignore invalid IP addresses returned by lxd | ||||||
|                             pass |                             pass | ||||||
| 
 | 
 | ||||||
|  |     def build_inventory_groups_project(self, group_name): | ||||||
|  |         """create group by attribute: project | ||||||
|  | 
 | ||||||
|  |         Args: | ||||||
|  |             str(group_name): Group name | ||||||
|  |         Kwargs: | ||||||
|  |             None | ||||||
|  |         Raises: | ||||||
|  |             None | ||||||
|  |         Returns: | ||||||
|  |             None""" | ||||||
|  |         # maybe we just want to expand one group | ||||||
|  |         if group_name not in self.inventory.groups: | ||||||
|  |             self.inventory.add_group(group_name) | ||||||
|  | 
 | ||||||
|  |         gen_instances = [ | ||||||
|  |             instance_name for instance_name in self.inventory.hosts | ||||||
|  |             if 'ansible_lxd_project' in self.inventory.get_host(instance_name).get_vars()] | ||||||
|  |         for instance_name in gen_instances: | ||||||
|  |             if self.groupby[group_name].get('attribute').lower() == self.inventory.get_host(instance_name).get_vars().get('ansible_lxd_project'): | ||||||
|  |                 self.inventory.add_child(group_name, instance_name) | ||||||
|  | 
 | ||||||
|     def build_inventory_groups_os(self, group_name): |     def build_inventory_groups_os(self, group_name): | ||||||
|         """create group by attribute: os |         """create group by attribute: os | ||||||
| 
 | 
 | ||||||
|  | @ -899,6 +944,7 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|                 * 'profile' |                 * 'profile' | ||||||
|                 * 'vlanid' |                 * 'vlanid' | ||||||
|                 * 'type' |                 * 'type' | ||||||
|  |                 * 'project' | ||||||
| 
 | 
 | ||||||
|             Args: |             Args: | ||||||
|                 str(group_name): Group name |                 str(group_name): Group name | ||||||
|  | @ -926,6 +972,8 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|                 self.build_inventory_groups_vlanid(group_name) |                 self.build_inventory_groups_vlanid(group_name) | ||||||
|             elif self.groupby[group_name].get('type') == 'type': |             elif self.groupby[group_name].get('type') == 'type': | ||||||
|                 self.build_inventory_groups_type(group_name) |                 self.build_inventory_groups_type(group_name) | ||||||
|  |             elif self.groupby[group_name].get('type') == 'project': | ||||||
|  |                 self.build_inventory_groups_project(group_name) | ||||||
|             else: |             else: | ||||||
|                 raise AnsibleParserError('Unknown group type: {0}'.format(to_native(group_name))) |                 raise AnsibleParserError('Unknown group type: {0}'.format(to_native(group_name))) | ||||||
| 
 | 
 | ||||||
|  | @ -1032,6 +1080,7 @@ class InventoryModule(BaseInventoryPlugin): | ||||||
|         try: |         try: | ||||||
|             self.client_key = self.get_option('client_key') |             self.client_key = self.get_option('client_key') | ||||||
|             self.client_cert = self.get_option('client_cert') |             self.client_cert = self.get_option('client_cert') | ||||||
|  |             self.project = self.get_option('project') | ||||||
|             self.debug = self.DEBUG |             self.debug = self.DEBUG | ||||||
|             self.data = {}  # store for inventory-data |             self.data = {}  # store for inventory-data | ||||||
|             self.groupby = self.get_option('groupby') |             self.groupby = self.get_option('groupby') | ||||||
|  |  | ||||||
|  | @ -46,6 +46,25 @@ DOCUMENTATION = ''' | ||||||
|             description: use IPv6 type addresses |             description: use IPv6 type addresses | ||||||
|             type: boolean |             type: boolean | ||||||
|             default: true |             default: true | ||||||
|  |         udp_scan: | ||||||
|  |             description: | ||||||
|  |                 - Scan via UDP. | ||||||
|  |                 - Depending on your system you might need I(sudo=true) for this to work. | ||||||
|  |             type: boolean | ||||||
|  |             default: false | ||||||
|  |             version_added: 6.1.0 | ||||||
|  |         icmp_timestamp: | ||||||
|  |             description: | ||||||
|  |                 - Scan via ICMP Timestamp (C(-PP)). | ||||||
|  |                 - Depending on your system you might need I(sudo=true) for this to work. | ||||||
|  |             type: boolean | ||||||
|  |             default: false | ||||||
|  |             version_added: 6.1.0 | ||||||
|  |         dns_resolve: | ||||||
|  |             description: Whether to always (C(true)) or never (C(false)) do DNS resolution. | ||||||
|  |             type: boolean | ||||||
|  |             default: false | ||||||
|  |             version_added: 6.1.0 | ||||||
|     notes: |     notes: | ||||||
|         - At least one of ipv4 or ipv6 is required to be True, both can be True, but they cannot both be False. |         - At least one of ipv4 or ipv6 is required to be True, both can be True, but they cannot both be False. | ||||||
|         - 'TODO: add OS fingerprinting' |         - 'TODO: add OS fingerprinting' | ||||||
|  | @ -166,6 +185,15 @@ class InventoryModule(BaseInventoryPlugin, Constructable, Cacheable): | ||||||
|                 cmd.append('--exclude') |                 cmd.append('--exclude') | ||||||
|                 cmd.append(','.join(self._options['exclude'])) |                 cmd.append(','.join(self._options['exclude'])) | ||||||
| 
 | 
 | ||||||
|  |             if self._options['dns_resolve']: | ||||||
|  |                 cmd.append('-n') | ||||||
|  | 
 | ||||||
|  |             if self._options['udp_scan']: | ||||||
|  |                 cmd.append('-sU') | ||||||
|  | 
 | ||||||
|  |             if self._options['icmp_timestamp']: | ||||||
|  |                 cmd.append('-PP') | ||||||
|  | 
 | ||||||
|             cmd.append(self._options['address']) |             cmd.append(self._options['address']) | ||||||
|             try: |             try: | ||||||
|                 # execute |                 # execute | ||||||
|  |  | ||||||
|  | @ -408,7 +408,7 @@ class InventoryModule(BaseInventoryPlugin, Constructable, Cacheable): | ||||||
|                     stripped_value = value.strip() |                     stripped_value = value.strip() | ||||||
|                     if stripped_value: |                     if stripped_value: | ||||||
|                         parsed_key = key + "_parsed" |                         parsed_key = key + "_parsed" | ||||||
|                         properties[parsed_key] = [tag.strip() for tag in stripped_value.split(",")] |                         properties[parsed_key] = [tag.strip() for tag in stripped_value.replace(',', ';').split(";")] | ||||||
| 
 | 
 | ||||||
|                 # The first field in the agent string tells you whether the agent is enabled |                 # The first field in the agent string tells you whether the agent is enabled | ||||||
|                 # the rest of the comma separated string is extra config for the agent. |                 # the rest of the comma separated string is extra config for the agent. | ||||||
|  | @ -615,7 +615,7 @@ class InventoryModule(BaseInventoryPlugin, Constructable, Cacheable): | ||||||
|         for o in ('url', 'user', 'password', 'token_id', 'token_secret'): |         for o in ('url', 'user', 'password', 'token_id', 'token_secret'): | ||||||
|             v = self.get_option(o) |             v = self.get_option(o) | ||||||
|             if self.templar.is_template(v): |             if self.templar.is_template(v): | ||||||
|                 v = self.templar.template(v, disable_looups=False) |                 v = self.templar.template(v, disable_lookups=False) | ||||||
|             setattr(self, 'proxmox_%s' % o, v) |             setattr(self, 'proxmox_%s' % o, v) | ||||||
| 
 | 
 | ||||||
|         # some more cleanup and validation |         # some more cleanup and validation | ||||||
|  |  | ||||||
|  | @ -125,8 +125,16 @@ from ansible.errors import AnsibleLookupError | ||||||
| from ansible.module_utils.common._collections_compat import Mapping, Sequence | from ansible.module_utils.common._collections_compat import Mapping, Sequence | ||||||
| from ansible.module_utils.six import string_types | from ansible.module_utils.six import string_types | ||||||
| from ansible.plugins.lookup import LookupBase | from ansible.plugins.lookup import LookupBase | ||||||
|  | from ansible.release import __version__ as ansible_version | ||||||
| from ansible.template import Templar | from ansible.template import Templar | ||||||
| 
 | 
 | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.version import LooseVersion | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # Whether Templar has a cache, which can be controlled by Templar.template()'s cache option. | ||||||
|  | # The cache was removed for ansible-core 2.14 (https://github.com/ansible/ansible/pull/78419) | ||||||
|  | _TEMPLAR_HAS_TEMPLATE_CACHE = LooseVersion(ansible_version) < LooseVersion('2.14.0') | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| class LookupModule(LookupBase): | class LookupModule(LookupBase): | ||||||
|     def __evaluate(self, expression, templar, variables): |     def __evaluate(self, expression, templar, variables): | ||||||
|  | @ -136,7 +144,10 @@ class LookupModule(LookupBase): | ||||||
|         ``variables`` are the variables to use. |         ``variables`` are the variables to use. | ||||||
|         """ |         """ | ||||||
|         templar.available_variables = variables or {} |         templar.available_variables = variables or {} | ||||||
|         return templar.template("{0}{1}{2}".format("{{", expression, "}}"), cache=False) |         expression = "{0}{1}{2}".format("{{", expression, "}}") | ||||||
|  |         if _TEMPLAR_HAS_TEMPLATE_CACHE: | ||||||
|  |             return templar.template(expression, cache=False) | ||||||
|  |         return templar.template(expression) | ||||||
| 
 | 
 | ||||||
|     def __process(self, result, terms, index, current, templar, variables): |     def __process(self, result, terms, index, current, templar, variables): | ||||||
|         """Fills ``result`` list with evaluated items. |         """Fills ``result`` list with evaluated items. | ||||||
|  |  | ||||||
|  | @ -32,7 +32,7 @@ DOCUMENTATION = ''' | ||||||
|       section: |       section: | ||||||
|         description: Item section containing the field to retrieve (case-insensitive). If absent will return first match from any section. |         description: Item section containing the field to retrieve (case-insensitive). If absent will return first match from any section. | ||||||
|       domain: |       domain: | ||||||
|         description: Domain of 1Password. Default is U(1password.com). |         description: Domain of 1Password. | ||||||
|         version_added: 3.2.0 |         version_added: 3.2.0 | ||||||
|         default: '1password.com' |         default: '1password.com' | ||||||
|         type: str |         type: str | ||||||
|  |  | ||||||
|  | @ -30,6 +30,11 @@ DOCUMENTATION = ''' | ||||||
|         description: Item section containing the field to retrieve (case-insensitive). If absent will return first match from any section. |         description: Item section containing the field to retrieve (case-insensitive). If absent will return first match from any section. | ||||||
|       subdomain: |       subdomain: | ||||||
|         description: The 1Password subdomain to authenticate against. |         description: The 1Password subdomain to authenticate against. | ||||||
|  |       domain: | ||||||
|  |         description: Domain of 1Password. | ||||||
|  |         version_added: 6.0.0 | ||||||
|  |         default: '1password.com' | ||||||
|  |         type: str | ||||||
|       username: |       username: | ||||||
|         description: The username used to sign in. |         description: The username used to sign in. | ||||||
|       secret_key: |       secret_key: | ||||||
|  |  | ||||||
|  | @ -88,9 +88,10 @@ class FormatError(CmdRunnerException): | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class _ArgFormat(object): | class _ArgFormat(object): | ||||||
|     def __init__(self, func, ignore_none=None): |     def __init__(self, func, ignore_none=None, ignore_missing_value=False): | ||||||
|         self.func = func |         self.func = func | ||||||
|         self.ignore_none = ignore_none |         self.ignore_none = ignore_none | ||||||
|  |         self.ignore_missing_value = ignore_missing_value | ||||||
| 
 | 
 | ||||||
|     def __call__(self, value, ctx_ignore_none): |     def __call__(self, value, ctx_ignore_none): | ||||||
|         ignore_none = self.ignore_none if self.ignore_none is not None else ctx_ignore_none |         ignore_none = self.ignore_none if self.ignore_none is not None else ctx_ignore_none | ||||||
|  | @ -102,8 +103,13 @@ class _ArgFormat(object): | ||||||
| 
 | 
 | ||||||
| class _Format(object): | class _Format(object): | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def as_bool(args): |     def as_bool(args_true, args_false=None, ignore_none=None): | ||||||
|         return _ArgFormat(lambda value: _ensure_list(args) if value else []) |         if args_false is not None: | ||||||
|  |             if ignore_none is None: | ||||||
|  |                 ignore_none = False | ||||||
|  |         else: | ||||||
|  |             args_false = [] | ||||||
|  |         return _ArgFormat(lambda value: _ensure_list(args_true) if value else _ensure_list(args_false), ignore_none=ignore_none) | ||||||
| 
 | 
 | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def as_bool_not(args): |     def as_bool_not(args): | ||||||
|  | @ -127,7 +133,7 @@ class _Format(object): | ||||||
| 
 | 
 | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def as_fixed(args): |     def as_fixed(args): | ||||||
|         return _ArgFormat(lambda value: _ensure_list(args), ignore_none=False) |         return _ArgFormat(lambda value: _ensure_list(args), ignore_none=False, ignore_missing_value=True) | ||||||
| 
 | 
 | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def as_func(func, ignore_none=None): |     def as_func(func, ignore_none=None): | ||||||
|  | @ -135,14 +141,15 @@ class _Format(object): | ||||||
| 
 | 
 | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def as_map(_map, default=None, ignore_none=None): |     def as_map(_map, default=None, ignore_none=None): | ||||||
|  |         if default is None: | ||||||
|  |             default = [] | ||||||
|         return _ArgFormat(lambda value: _ensure_list(_map.get(value, default)), ignore_none=ignore_none) |         return _ArgFormat(lambda value: _ensure_list(_map.get(value, default)), ignore_none=ignore_none) | ||||||
| 
 | 
 | ||||||
|     @staticmethod |     @staticmethod | ||||||
|     def as_default_type(_type, arg="", ignore_none=None): |     def as_default_type(_type, arg="", ignore_none=None): | ||||||
|         fmt = _Format |         fmt = _Format | ||||||
|         if _type == "dict": |         if _type == "dict": | ||||||
|             return fmt.as_func(lambda d: ["--{0}={1}".format(*a) for a in iteritems(d)], |             return fmt.as_func(lambda d: ["--{0}={1}".format(*a) for a in iteritems(d)], ignore_none=ignore_none) | ||||||
|                                ignore_none=ignore_none) |  | ||||||
|         if _type == "list": |         if _type == "list": | ||||||
|             return fmt.as_func(lambda value: ["--{0}".format(x) for x in value], ignore_none=ignore_none) |             return fmt.as_func(lambda value: ["--{0}".format(x) for x in value], ignore_none=ignore_none) | ||||||
|         if _type == "bool": |         if _type == "bool": | ||||||
|  | @ -261,10 +268,13 @@ class _CmdRunnerContext(object): | ||||||
|         for arg_name in self.args_order: |         for arg_name in self.args_order: | ||||||
|             value = None |             value = None | ||||||
|             try: |             try: | ||||||
|  |                 if arg_name in named_args: | ||||||
|                     value = named_args[arg_name] |                     value = named_args[arg_name] | ||||||
|                 self.cmd.extend(runner.arg_formats[arg_name](value, ctx_ignore_none=self.ignore_value_none)) |                 elif not runner.arg_formats[arg_name].ignore_missing_value: | ||||||
|             except KeyError: |  | ||||||
|                     raise MissingArgumentValue(self.args_order, arg_name) |                     raise MissingArgumentValue(self.args_order, arg_name) | ||||||
|  |                 self.cmd.extend(runner.arg_formats[arg_name](value, ctx_ignore_none=self.ignore_value_none)) | ||||||
|  |             except MissingArgumentValue: | ||||||
|  |                 raise | ||||||
|             except Exception as e: |             except Exception as e: | ||||||
|                 raise FormatError(arg_name, value, runner.arg_formats[arg_name], e) |                 raise FormatError(arg_name, value, runner.arg_formats[arg_name], e) | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										90
									
								
								plugins/module_utils/deps.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										90
									
								
								plugins/module_utils/deps.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,90 @@ | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # (c) 2022, Alexei Znamensky <russoz@gmail.com> | ||||||
|  | # Copyright (c) 2022, Ansible Project | ||||||
|  | # Simplified BSD License (see LICENSES/BSD-2-Clause.txt or https://opensource.org/licenses/BSD-2-Clause) | ||||||
|  | # SPDX-License-Identifier: BSD-2-Clause | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | import traceback | ||||||
|  | from contextlib import contextmanager | ||||||
|  | 
 | ||||||
|  | from ansible.module_utils.common.text.converters import to_native | ||||||
|  | from ansible.module_utils.basic import missing_required_lib | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | _deps = dict() | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | class _Dependency(object): | ||||||
|  |     _states = ["pending", "failure", "success"] | ||||||
|  | 
 | ||||||
|  |     def __init__(self, name, reason=None, url=None, msg=None): | ||||||
|  |         self.name = name | ||||||
|  |         self.reason = reason | ||||||
|  |         self.url = url | ||||||
|  |         self.msg = msg | ||||||
|  | 
 | ||||||
|  |         self.state = 0 | ||||||
|  |         self.trace = None | ||||||
|  |         self.exc = None | ||||||
|  | 
 | ||||||
|  |     def succeed(self): | ||||||
|  |         self.state = 2 | ||||||
|  | 
 | ||||||
|  |     def fail(self, exc, trace): | ||||||
|  |         self.state = 1 | ||||||
|  |         self.exc = exc | ||||||
|  |         self.trace = trace | ||||||
|  | 
 | ||||||
|  |     @property | ||||||
|  |     def message(self): | ||||||
|  |         if self.msg: | ||||||
|  |             return to_native(self.msg) | ||||||
|  |         else: | ||||||
|  |             return missing_required_lib(self.name, reason=self.reason, url=self.url) | ||||||
|  | 
 | ||||||
|  |     @property | ||||||
|  |     def failed(self): | ||||||
|  |         return self.state == 1 | ||||||
|  | 
 | ||||||
|  |     def verify(self, module): | ||||||
|  |         if self.failed: | ||||||
|  |             module.fail_json(msg=self.message, exception=self.trace) | ||||||
|  | 
 | ||||||
|  |     def __str__(self): | ||||||
|  |         return "<dependency: {0} [{1}]>".format(self.name, self._states[self.state]) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | @contextmanager | ||||||
|  | def declare(name, *args, **kwargs): | ||||||
|  |     dep = _Dependency(name, *args, **kwargs) | ||||||
|  |     try: | ||||||
|  |         yield dep | ||||||
|  |     except Exception as e: | ||||||
|  |         dep.fail(e, traceback.format_exc()) | ||||||
|  |     else: | ||||||
|  |         dep.succeed() | ||||||
|  |     finally: | ||||||
|  |         _deps[name] = dep | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def validate(module, spec=None): | ||||||
|  |     dep_names = sorted(_deps) | ||||||
|  | 
 | ||||||
|  |     if spec is not None: | ||||||
|  |         if spec.startswith("-"): | ||||||
|  |             spec_split = spec[1:].split(":") | ||||||
|  |             for d in spec_split: | ||||||
|  |                 dep_names.remove(d) | ||||||
|  |         else: | ||||||
|  |             spec_split = spec[1:].split(":") | ||||||
|  |             dep_names = [] | ||||||
|  |             for d in spec_split: | ||||||
|  |                 _deps[d]  # ensure it exists | ||||||
|  |                 dep_names.append(d) | ||||||
|  | 
 | ||||||
|  |     for dep in dep_names: | ||||||
|  |         _deps[dep].verify(module) | ||||||
|  | @ -6,7 +6,14 @@ | ||||||
| from __future__ import absolute_import, division, print_function | from __future__ import absolute_import, division, print_function | ||||||
| __metaclass__ = type | __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| from ansible_collections.community.general.plugins.module_utils.cmd_runner import CmdRunner, cmd_runner_fmt as fmt | from ansible_collections.community.general.plugins.module_utils.cmd_runner import CmdRunner, cmd_runner_fmt | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | _state_map = { | ||||||
|  |     "present": "--set", | ||||||
|  |     "absent": "--unset", | ||||||
|  |     "get": "--get", | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def gconftool2_runner(module, **kwargs): | def gconftool2_runner(module, **kwargs): | ||||||
|  | @ -14,14 +21,12 @@ def gconftool2_runner(module, **kwargs): | ||||||
|         module, |         module, | ||||||
|         command='gconftool-2', |         command='gconftool-2', | ||||||
|         arg_formats=dict( |         arg_formats=dict( | ||||||
|             key=fmt.as_list(), |             state=cmd_runner_fmt.as_map(_state_map), | ||||||
|             value_type=fmt.as_opt_val("--type"), |             key=cmd_runner_fmt.as_list(), | ||||||
|             value=fmt.as_list(), |             value_type=cmd_runner_fmt.as_opt_val("--type"), | ||||||
|             direct=fmt.as_bool("--direct"), |             value=cmd_runner_fmt.as_list(), | ||||||
|             config_source=fmt.as_opt_val("--config-source"), |             direct=cmd_runner_fmt.as_bool("--direct"), | ||||||
|             get=fmt.as_bool("--get"), |             config_source=cmd_runner_fmt.as_opt_val("--config-source"), | ||||||
|             set_arg=fmt.as_bool("--set"), |  | ||||||
|             unset=fmt.as_bool("--unset"), |  | ||||||
|         ), |         ), | ||||||
|         **kwargs |         **kwargs | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  | @ -110,3 +110,14 @@ def gitlab_authentication(module): | ||||||
|             GitLab remove Session API now that private tokens are removed from user API endpoints since version 10.2." % to_native(e)) |             GitLab remove Session API now that private tokens are removed from user API endpoints since version 10.2." % to_native(e)) | ||||||
| 
 | 
 | ||||||
|     return gitlab_instance |     return gitlab_instance | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def filter_returned_variables(gitlab_variables): | ||||||
|  |     # pop properties we don't know | ||||||
|  |     existing_variables = [dict(x.attributes) for x in gitlab_variables] | ||||||
|  |     KNOWN = ['key', 'value', 'masked', 'protected', 'variable_type', 'environment_scope'] | ||||||
|  |     for item in existing_variables: | ||||||
|  |         for key in list(item.keys()): | ||||||
|  |             if key not in KNOWN: | ||||||
|  |                 item.pop(key) | ||||||
|  |     return existing_variables | ||||||
|  |  | ||||||
|  | @ -58,6 +58,8 @@ URL_CLIENT_USER_ROLEMAPPINGS = "{url}/admin/realms/{realm}/users/{id}/role-mappi | ||||||
| URL_CLIENT_USER_ROLEMAPPINGS_AVAILABLE = "{url}/admin/realms/{realm}/users/{id}/role-mappings/clients/{client}/available" | URL_CLIENT_USER_ROLEMAPPINGS_AVAILABLE = "{url}/admin/realms/{realm}/users/{id}/role-mappings/clients/{client}/available" | ||||||
| URL_CLIENT_USER_ROLEMAPPINGS_COMPOSITE = "{url}/admin/realms/{realm}/users/{id}/role-mappings/clients/{client}/composite" | URL_CLIENT_USER_ROLEMAPPINGS_COMPOSITE = "{url}/admin/realms/{realm}/users/{id}/role-mappings/clients/{client}/composite" | ||||||
| 
 | 
 | ||||||
|  | URL_CLIENTSECRET = "{url}/admin/realms/{realm}/clients/{id}/client-secret" | ||||||
|  | 
 | ||||||
| URL_AUTHENTICATION_FLOWS = "{url}/admin/realms/{realm}/authentication/flows" | URL_AUTHENTICATION_FLOWS = "{url}/admin/realms/{realm}/authentication/flows" | ||||||
| URL_AUTHENTICATION_FLOW = "{url}/admin/realms/{realm}/authentication/flows/{id}" | URL_AUTHENTICATION_FLOW = "{url}/admin/realms/{realm}/authentication/flows/{id}" | ||||||
| URL_AUTHENTICATION_FLOW_COPY = "{url}/admin/realms/{realm}/authentication/flows/{copyfrom}/copy" | URL_AUTHENTICATION_FLOW_COPY = "{url}/admin/realms/{realm}/authentication/flows/{copyfrom}/copy" | ||||||
|  | @ -606,7 +608,7 @@ class KeycloakAPI(object): | ||||||
|         """ |         """ | ||||||
|         available_rolemappings_url = URL_CLIENT_GROUP_ROLEMAPPINGS.format(url=self.baseurl, realm=realm, id=gid, client=cid) |         available_rolemappings_url = URL_CLIENT_GROUP_ROLEMAPPINGS.format(url=self.baseurl, realm=realm, id=gid, client=cid) | ||||||
|         try: |         try: | ||||||
|             open_url(available_rolemappings_url, method="DELETE", http_agent=self.http_agent, headers=self.restheaders, |             open_url(available_rolemappings_url, method="DELETE", http_agent=self.http_agent, headers=self.restheaders, data=json.dumps(role_rep), | ||||||
|                      validate_certs=self.validate_certs, timeout=self.connection_timeout) |                      validate_certs=self.validate_certs, timeout=self.connection_timeout) | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             self.module.fail_json(msg="Could not delete available rolemappings for client %s in group %s, realm %s: %s" |             self.module.fail_json(msg="Could not delete available rolemappings for client %s in group %s, realm %s: %s" | ||||||
|  | @ -1160,6 +1162,52 @@ class KeycloakAPI(object): | ||||||
|             self.module.fail_json(msg='Could not update protocolmappers for clientscope %s in realm %s: %s' |             self.module.fail_json(msg='Could not update protocolmappers for clientscope %s in realm %s: %s' | ||||||
|                                       % (mapper_rep, realm, str(e))) |                                       % (mapper_rep, realm, str(e))) | ||||||
| 
 | 
 | ||||||
|  |     def create_clientsecret(self, id, realm="master"): | ||||||
|  |         """ Generate a new client secret by id | ||||||
|  | 
 | ||||||
|  |         :param id: id (not clientId) of client to be queried | ||||||
|  |         :param realm: client from this realm | ||||||
|  |         :return: dict of credential representation | ||||||
|  |         """ | ||||||
|  |         clientsecret_url = URL_CLIENTSECRET.format(url=self.baseurl, realm=realm, id=id) | ||||||
|  | 
 | ||||||
|  |         try: | ||||||
|  |             return json.loads(to_native(open_url(clientsecret_url, method='POST', headers=self.restheaders, timeout=self.connection_timeout, | ||||||
|  |                                                  validate_certs=self.validate_certs).read())) | ||||||
|  | 
 | ||||||
|  |         except HTTPError as e: | ||||||
|  |             if e.code == 404: | ||||||
|  |                 return None | ||||||
|  |             else: | ||||||
|  |                 self.module.fail_json(msg='Could not obtain clientsecret of client %s for realm %s: %s' | ||||||
|  |                                           % (id, realm, str(e))) | ||||||
|  |         except Exception as e: | ||||||
|  |             self.module.fail_json(msg='Could not obtain clientsecret of client %s for realm %s: %s' | ||||||
|  |                                       % (id, realm, str(e))) | ||||||
|  | 
 | ||||||
|  |     def get_clientsecret(self, id, realm="master"): | ||||||
|  |         """ Obtain client secret by id | ||||||
|  | 
 | ||||||
|  |         :param id: id (not clientId) of client to be queried | ||||||
|  |         :param realm: client from this realm | ||||||
|  |         :return: dict of credential representation | ||||||
|  |         """ | ||||||
|  |         clientsecret_url = URL_CLIENTSECRET.format(url=self.baseurl, realm=realm, id=id) | ||||||
|  | 
 | ||||||
|  |         try: | ||||||
|  |             return json.loads(to_native(open_url(clientsecret_url, method='GET', headers=self.restheaders, timeout=self.connection_timeout, | ||||||
|  |                                                  validate_certs=self.validate_certs).read())) | ||||||
|  | 
 | ||||||
|  |         except HTTPError as e: | ||||||
|  |             if e.code == 404: | ||||||
|  |                 return None | ||||||
|  |             else: | ||||||
|  |                 self.module.fail_json(msg='Could not obtain clientsecret of client %s for realm %s: %s' | ||||||
|  |                                           % (id, realm, str(e))) | ||||||
|  |         except Exception as e: | ||||||
|  |             self.module.fail_json(msg='Could not obtain clientsecret of client %s for realm %s: %s' | ||||||
|  |                                       % (id, realm, str(e))) | ||||||
|  | 
 | ||||||
|     def get_groups(self, realm="master"): |     def get_groups(self, realm="master"): | ||||||
|         """ Fetch the name and ID of all groups on the Keycloak server. |         """ Fetch the name and ID of all groups on the Keycloak server. | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -0,0 +1,77 @@ | ||||||
|  | #!/usr/bin/env python | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # Copyright (c) 2022, John Cant <a.johncant@gmail.com> | ||||||
|  | # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | 
 | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | from ansible.module_utils.basic import AnsibleModule | ||||||
|  | 
 | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.identity.keycloak.keycloak import \ | ||||||
|  |     keycloak_argument_spec | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def keycloak_clientsecret_module(): | ||||||
|  |     """ | ||||||
|  |     Returns an AnsibleModule definition for modules that interact with a client | ||||||
|  |     secret. | ||||||
|  | 
 | ||||||
|  |     :return: argument_spec dict | ||||||
|  |     """ | ||||||
|  |     argument_spec = keycloak_argument_spec() | ||||||
|  | 
 | ||||||
|  |     meta_args = dict( | ||||||
|  |         realm=dict(default='master'), | ||||||
|  |         id=dict(type='str'), | ||||||
|  |         client_id=dict(type='str', aliases=['clientId']), | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     argument_spec.update(meta_args) | ||||||
|  | 
 | ||||||
|  |     module = AnsibleModule( | ||||||
|  |         argument_spec=argument_spec, | ||||||
|  |         supports_check_mode=True, | ||||||
|  |         required_one_of=([['id', 'client_id'], | ||||||
|  |                          ['token', 'auth_realm', 'auth_username', 'auth_password']]), | ||||||
|  |         required_together=([['auth_realm', 'auth_username', 'auth_password']]), | ||||||
|  |         mutually_exclusive=[ | ||||||
|  |             ['token', 'auth_realm'], | ||||||
|  |             ['token', 'auth_username'], | ||||||
|  |             ['token', 'auth_password'] | ||||||
|  |         ]) | ||||||
|  | 
 | ||||||
|  |     return module | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def keycloak_clientsecret_module_resolve_params(module, kc): | ||||||
|  |     """ | ||||||
|  |     Given an AnsibleModule definition for keycloak_clientsecret_*, and a | ||||||
|  |     KeycloakAPI client, resolve the params needed to interact with the Keycloak | ||||||
|  |     client secret, looking up the client by clientId if necessary via an API | ||||||
|  |     call. | ||||||
|  | 
 | ||||||
|  |     :return: tuple of id, realm | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|  |     realm = module.params.get('realm') | ||||||
|  |     id = module.params.get('id') | ||||||
|  |     client_id = module.params.get('client_id') | ||||||
|  | 
 | ||||||
|  |     # only lookup the client_id if id isn't provided. | ||||||
|  |     # in the case that both are provided, prefer the ID, since it's one | ||||||
|  |     # less lookup. | ||||||
|  |     if id is None: | ||||||
|  |         # Due to the required_one_of spec, client_id is guaranteed to not be None | ||||||
|  |         client = kc.get_client_by_clientid(client_id, realm=realm) | ||||||
|  | 
 | ||||||
|  |         if client is None: | ||||||
|  |             module.fail_json( | ||||||
|  |                 msg='Client does not exist {client_id}'.format(client_id=client_id) | ||||||
|  |             ) | ||||||
|  | 
 | ||||||
|  |         id = client['id'] | ||||||
|  | 
 | ||||||
|  |     return id, realm | ||||||
							
								
								
									
										35
									
								
								plugins/module_utils/jenkins.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										35
									
								
								plugins/module_utils/jenkins.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,35 @@ | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | 
 | ||||||
|  | # Copyright (c) 2022, Alexei Znamensky <russoz@gmail.com> | ||||||
|  | # | ||||||
|  | # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | import os | ||||||
|  | import time | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def download_updates_file(updates_expiration): | ||||||
|  |     updates_filename = 'jenkins-plugin-cache.json' | ||||||
|  |     updates_dir = os.path.expanduser('~/.ansible/tmp') | ||||||
|  |     updates_file = os.path.join(updates_dir, updates_filename) | ||||||
|  |     download_updates = True | ||||||
|  | 
 | ||||||
|  |     # Make sure the destination directory exists | ||||||
|  |     if not os.path.isdir(updates_dir): | ||||||
|  |         os.makedirs(updates_dir, 0o700) | ||||||
|  | 
 | ||||||
|  |     # Check if we need to download new updates file | ||||||
|  |     if os.path.isfile(updates_file): | ||||||
|  |         # Get timestamp when the file was changed last time | ||||||
|  |         ts_file = os.stat(updates_file).st_mtime | ||||||
|  |         ts_now = time.time() | ||||||
|  | 
 | ||||||
|  |         if ts_now - ts_file < updates_expiration: | ||||||
|  |             download_updates = False | ||||||
|  | 
 | ||||||
|  |     return updates_file, download_updates | ||||||
|  | @ -8,8 +8,10 @@ from __future__ import (absolute_import, division, print_function) | ||||||
| __metaclass__ = type | __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | import os | ||||||
| import socket | import socket | ||||||
| import ssl | import ssl | ||||||
|  | import json | ||||||
| 
 | 
 | ||||||
| from ansible.module_utils.urls import generic_urlparse | from ansible.module_utils.urls import generic_urlparse | ||||||
| from ansible.module_utils.six.moves.urllib.parse import urlparse | from ansible.module_utils.six.moves.urllib.parse import urlparse | ||||||
|  | @ -20,8 +22,6 @@ from ansible.module_utils.common.text.converters import to_text | ||||||
| HTTPConnection = http_client.HTTPConnection | HTTPConnection = http_client.HTTPConnection | ||||||
| HTTPSConnection = http_client.HTTPSConnection | HTTPSConnection = http_client.HTTPSConnection | ||||||
| 
 | 
 | ||||||
| import json |  | ||||||
| 
 |  | ||||||
| 
 | 
 | ||||||
| class UnixHTTPConnection(HTTPConnection): | class UnixHTTPConnection(HTTPConnection): | ||||||
|     def __init__(self, path): |     def __init__(self, path): | ||||||
|  | @ -124,3 +124,11 @@ class LXDClient(object): | ||||||
|         if err is None: |         if err is None: | ||||||
|             err = resp_json.get('error', None) |             err = resp_json.get('error', None) | ||||||
|         return err |         return err | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def default_key_file(): | ||||||
|  |     return os.path.expanduser('~/.config/lxc/client.key') | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def default_cert_file(): | ||||||
|  |     return os.path.expanduser('~/.config/lxc/client.crt') | ||||||
|  |  | ||||||
							
								
								
									
										114
									
								
								plugins/module_utils/puppet.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										114
									
								
								plugins/module_utils/puppet.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,114 @@ | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # Copyright (c) 2022, Alexei Znamensky <russoz@gmail.com> | ||||||
|  | # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | import os | ||||||
|  | 
 | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.cmd_runner import CmdRunner, cmd_runner_fmt | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | _PUPPET_PATH_PREFIX = ["/opt/puppetlabs/bin"] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def get_facter_dir(): | ||||||
|  |     if os.getuid() == 0: | ||||||
|  |         return '/etc/facter/facts.d' | ||||||
|  |     else: | ||||||
|  |         return os.path.expanduser('~/.facter/facts.d') | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def _puppet_cmd(module): | ||||||
|  |     return module.get_bin_path("puppet", False, _PUPPET_PATH_PREFIX) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | # If the `timeout` CLI command feature is removed, | ||||||
|  | # Then we could add this as a fixed param to `puppet_runner` | ||||||
|  | def ensure_agent_enabled(module): | ||||||
|  |     runner = CmdRunner( | ||||||
|  |         module, | ||||||
|  |         command="puppet", | ||||||
|  |         path_prefix=_PUPPET_PATH_PREFIX, | ||||||
|  |         arg_formats=dict( | ||||||
|  |             _agent_disabled=cmd_runner_fmt.as_fixed(['config', 'print', 'agent_disabled_lockfile']), | ||||||
|  |         ), | ||||||
|  |         check_rc=False, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     rc, stdout, stderr = runner("_agent_disabled").run() | ||||||
|  |     if os.path.exists(stdout.strip()): | ||||||
|  |         module.fail_json( | ||||||
|  |             msg="Puppet agent is administratively disabled.", | ||||||
|  |             disabled=True) | ||||||
|  |     elif rc != 0: | ||||||
|  |         module.fail_json( | ||||||
|  |             msg="Puppet agent state could not be determined.") | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def puppet_runner(module): | ||||||
|  | 
 | ||||||
|  |     # Keeping backward compatibility, allow for running with the `timeout` CLI command. | ||||||
|  |     # If this can be replaced with ansible `timeout` parameter in playbook, | ||||||
|  |     # then this function could be removed. | ||||||
|  |     def _prepare_base_cmd(): | ||||||
|  |         _tout_cmd = module.get_bin_path("timeout", False) | ||||||
|  |         if _tout_cmd: | ||||||
|  |             cmd = ["timeout", "-s", "9", module.params["timeout"], _puppet_cmd(module)] | ||||||
|  |         else: | ||||||
|  |             cmd = ["puppet"] | ||||||
|  |         return cmd | ||||||
|  | 
 | ||||||
|  |     def noop_func(v): | ||||||
|  |         _noop = cmd_runner_fmt.as_map({ | ||||||
|  |             True: "--noop", | ||||||
|  |             False: "--no-noop", | ||||||
|  |         }) | ||||||
|  |         return _noop(module.check_mode or v) | ||||||
|  | 
 | ||||||
|  |     _logdest_map = { | ||||||
|  |         "syslog": ["--logdest", "syslog"], | ||||||
|  |         "all": ["--logdest", "syslog", "--logdest", "console"], | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     @cmd_runner_fmt.unpack_args | ||||||
|  |     def execute_func(execute, manifest): | ||||||
|  |         if execute: | ||||||
|  |             return ["--execute", execute] | ||||||
|  |         else: | ||||||
|  |             return [manifest] | ||||||
|  | 
 | ||||||
|  |     runner = CmdRunner( | ||||||
|  |         module, | ||||||
|  |         command=_prepare_base_cmd(), | ||||||
|  |         path_prefix=_PUPPET_PATH_PREFIX, | ||||||
|  |         arg_formats=dict( | ||||||
|  |             _agent_fixed=cmd_runner_fmt.as_fixed([ | ||||||
|  |                 "agent", "--onetime", "--no-daemonize", "--no-usecacheonfailure", | ||||||
|  |                 "--no-splay", "--detailed-exitcodes", "--verbose", "--color", "0", | ||||||
|  |             ]), | ||||||
|  |             _apply_fixed=cmd_runner_fmt.as_fixed(["apply", "--detailed-exitcodes"]), | ||||||
|  |             puppetmaster=cmd_runner_fmt.as_opt_val("--server"), | ||||||
|  |             show_diff=cmd_runner_fmt.as_bool("--show-diff"), | ||||||
|  |             confdir=cmd_runner_fmt.as_opt_val("--confdir"), | ||||||
|  |             environment=cmd_runner_fmt.as_opt_val("--environment"), | ||||||
|  |             tags=cmd_runner_fmt.as_func(lambda v: ["--tags", ",".join(v)]), | ||||||
|  |             certname=cmd_runner_fmt.as_opt_eq_val("--certname"), | ||||||
|  |             noop=cmd_runner_fmt.as_func(noop_func), | ||||||
|  |             use_srv_records=cmd_runner_fmt.as_map({ | ||||||
|  |                 True: "--usr_srv_records", | ||||||
|  |                 False: "--no-usr_srv_records", | ||||||
|  |             }), | ||||||
|  |             logdest=cmd_runner_fmt.as_map(_logdest_map, default=[]), | ||||||
|  |             modulepath=cmd_runner_fmt.as_opt_eq_val("--modulepath"), | ||||||
|  |             _execute=cmd_runner_fmt.as_func(execute_func), | ||||||
|  |             summarize=cmd_runner_fmt.as_bool("--summarize"), | ||||||
|  |             debug=cmd_runner_fmt.as_bool("--debug"), | ||||||
|  |             verbose=cmd_runner_fmt.as_bool("--verbose"), | ||||||
|  |         ), | ||||||
|  |         check_rc=False, | ||||||
|  |     ) | ||||||
|  |     return runner | ||||||
|  | @ -314,3 +314,21 @@ def setup_rax_module(module, rax_module, region_required=True): | ||||||
|                          (region, ','.join(rax_module.regions))) |                          (region, ','.join(rax_module.regions))) | ||||||
| 
 | 
 | ||||||
|     return rax_module |     return rax_module | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def rax_scaling_group_personality_file(module, files): | ||||||
|  |     if not files: | ||||||
|  |         return [] | ||||||
|  | 
 | ||||||
|  |     results = [] | ||||||
|  |     for rpath, lpath in files.items(): | ||||||
|  |         lpath = os.path.expanduser(lpath) | ||||||
|  |         try: | ||||||
|  |             with open(lpath, 'r') as f: | ||||||
|  |                 results.append({ | ||||||
|  |                     'path': rpath, | ||||||
|  |                     'contents': f.read(), | ||||||
|  |                 }) | ||||||
|  |         except Exception as e: | ||||||
|  |             module.fail_json(msg='Failed to load %s: %s' % (lpath, str(e))) | ||||||
|  |     return results | ||||||
|  |  | ||||||
|  | @ -19,6 +19,8 @@ POST_HEADERS = {'content-type': 'application/json', 'accept': 'application/json' | ||||||
|                 'OData-Version': '4.0'} |                 'OData-Version': '4.0'} | ||||||
| PATCH_HEADERS = {'content-type': 'application/json', 'accept': 'application/json', | PATCH_HEADERS = {'content-type': 'application/json', 'accept': 'application/json', | ||||||
|                  'OData-Version': '4.0'} |                  'OData-Version': '4.0'} | ||||||
|  | PUT_HEADERS = {'content-type': 'application/json', 'accept': 'application/json', | ||||||
|  |                'OData-Version': '4.0'} | ||||||
| DELETE_HEADERS = {'accept': 'application/json', 'OData-Version': '4.0'} | DELETE_HEADERS = {'accept': 'application/json', 'OData-Version': '4.0'} | ||||||
| 
 | 
 | ||||||
| FAIL_MSG = 'Issuing a data modification command without specifying the '\ | FAIL_MSG = 'Issuing a data modification command without specifying the '\ | ||||||
|  | @ -143,7 +145,7 @@ class RedfishUtils(object): | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             return {'ret': False, |             return {'ret': False, | ||||||
|                     'msg': "Failed GET request to '%s': '%s'" % (uri, to_text(e))} |                     'msg': "Failed GET request to '%s': '%s'" % (uri, to_text(e))} | ||||||
|         return {'ret': True, 'data': data, 'headers': headers} |         return {'ret': True, 'data': data, 'headers': headers, 'resp': resp} | ||||||
| 
 | 
 | ||||||
|     def post_request(self, uri, pyld): |     def post_request(self, uri, pyld): | ||||||
|         req_headers = dict(POST_HEADERS) |         req_headers = dict(POST_HEADERS) | ||||||
|  | @ -155,6 +157,11 @@ class RedfishUtils(object): | ||||||
|                             force_basic_auth=basic_auth, validate_certs=False, |                             force_basic_auth=basic_auth, validate_certs=False, | ||||||
|                             follow_redirects='all', |                             follow_redirects='all', | ||||||
|                             use_proxy=True, timeout=self.timeout) |                             use_proxy=True, timeout=self.timeout) | ||||||
|  |             try: | ||||||
|  |                 data = json.loads(to_native(resp.read())) | ||||||
|  |             except Exception as e: | ||||||
|  |                 # No response data; this is okay in many cases | ||||||
|  |                 data = None | ||||||
|             headers = dict((k.lower(), v) for (k, v) in resp.info().items()) |             headers = dict((k.lower(), v) for (k, v) in resp.info().items()) | ||||||
|         except HTTPError as e: |         except HTTPError as e: | ||||||
|             msg = self._get_extended_message(e) |             msg = self._get_extended_message(e) | ||||||
|  | @ -169,7 +176,7 @@ class RedfishUtils(object): | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             return {'ret': False, |             return {'ret': False, | ||||||
|                     'msg': "Failed POST request to '%s': '%s'" % (uri, to_text(e))} |                     'msg': "Failed POST request to '%s': '%s'" % (uri, to_text(e))} | ||||||
|         return {'ret': True, 'headers': headers, 'resp': resp} |         return {'ret': True, 'data': data, 'headers': headers, 'resp': resp} | ||||||
| 
 | 
 | ||||||
|     def patch_request(self, uri, pyld, check_pyld=False): |     def patch_request(self, uri, pyld, check_pyld=False): | ||||||
|         req_headers = dict(PATCH_HEADERS) |         req_headers = dict(PATCH_HEADERS) | ||||||
|  | @ -219,6 +226,41 @@ class RedfishUtils(object): | ||||||
|                     'msg': "Failed PATCH request to '%s': '%s'" % (uri, to_text(e))} |                     'msg': "Failed PATCH request to '%s': '%s'" % (uri, to_text(e))} | ||||||
|         return {'ret': True, 'changed': True, 'resp': resp, 'msg': 'Modified %s' % uri} |         return {'ret': True, 'changed': True, 'resp': resp, 'msg': 'Modified %s' % uri} | ||||||
| 
 | 
 | ||||||
|  |     def put_request(self, uri, pyld): | ||||||
|  |         req_headers = dict(PUT_HEADERS) | ||||||
|  |         r = self.get_request(uri) | ||||||
|  |         if r['ret']: | ||||||
|  |             # Get etag from etag header or @odata.etag property | ||||||
|  |             etag = r['headers'].get('etag') | ||||||
|  |             if not etag: | ||||||
|  |                 etag = r['data'].get('@odata.etag') | ||||||
|  |             if etag: | ||||||
|  |                 if self.strip_etag_quotes: | ||||||
|  |                     etag = etag.strip('"') | ||||||
|  |                 req_headers['If-Match'] = etag | ||||||
|  |         username, password, basic_auth = self._auth_params(req_headers) | ||||||
|  |         try: | ||||||
|  |             resp = open_url(uri, data=json.dumps(pyld), | ||||||
|  |                             headers=req_headers, method="PUT", | ||||||
|  |                             url_username=username, url_password=password, | ||||||
|  |                             force_basic_auth=basic_auth, validate_certs=False, | ||||||
|  |                             follow_redirects='all', | ||||||
|  |                             use_proxy=True, timeout=self.timeout) | ||||||
|  |         except HTTPError as e: | ||||||
|  |             msg = self._get_extended_message(e) | ||||||
|  |             return {'ret': False, | ||||||
|  |                     'msg': "HTTP Error %s on PUT request to '%s', extended message: '%s'" | ||||||
|  |                            % (e.code, uri, msg), | ||||||
|  |                     'status': e.code} | ||||||
|  |         except URLError as e: | ||||||
|  |             return {'ret': False, 'msg': "URL Error on PUT request to '%s': '%s'" | ||||||
|  |                                          % (uri, e.reason)} | ||||||
|  |         # Almost all errors should be caught above, but just in case | ||||||
|  |         except Exception as e: | ||||||
|  |             return {'ret': False, | ||||||
|  |                     'msg': "Failed PUT request to '%s': '%s'" % (uri, to_text(e))} | ||||||
|  |         return {'ret': True, 'resp': resp} | ||||||
|  | 
 | ||||||
|     def delete_request(self, uri, pyld=None): |     def delete_request(self, uri, pyld=None): | ||||||
|         req_headers = dict(DELETE_HEADERS) |         req_headers = dict(DELETE_HEADERS) | ||||||
|         username, password, basic_auth = self._auth_params(req_headers) |         username, password, basic_auth = self._auth_params(req_headers) | ||||||
|  | @ -1384,11 +1426,82 @@ class RedfishUtils(object): | ||||||
|         else: |         else: | ||||||
|             return self._software_inventory(self.software_uri) |             return self._software_inventory(self.software_uri) | ||||||
| 
 | 
 | ||||||
|  |     def _operation_results(self, response, data, handle=None): | ||||||
|  |         """ | ||||||
|  |         Builds the results for an operation from task, job, or action response. | ||||||
|  | 
 | ||||||
|  |         :param response: HTTP response object | ||||||
|  |         :param data: HTTP response data | ||||||
|  |         :param handle: The task or job handle that was last used | ||||||
|  |         :return: dict containing operation results | ||||||
|  |         """ | ||||||
|  | 
 | ||||||
|  |         operation_results = {'status': None, 'messages': [], 'handle': None, 'ret': True, | ||||||
|  |                              'resets_requested': []} | ||||||
|  | 
 | ||||||
|  |         if response.status == 204: | ||||||
|  |             # No content; successful, but nothing to return | ||||||
|  |             # Use the Redfish "Completed" enum from TaskState for the operation status | ||||||
|  |             operation_results['status'] = 'Completed' | ||||||
|  |         else: | ||||||
|  |             # Parse the response body for details | ||||||
|  | 
 | ||||||
|  |             # Determine the next handle, if any | ||||||
|  |             operation_results['handle'] = handle | ||||||
|  |             if response.status == 202: | ||||||
|  |                 # Task generated; get the task monitor URI | ||||||
|  |                 operation_results['handle'] = response.getheader('Location', handle) | ||||||
|  | 
 | ||||||
|  |             # Pull out the status and messages based on the body format | ||||||
|  |             if data is not None: | ||||||
|  |                 response_type = data.get('@odata.type', '') | ||||||
|  |                 if response_type.startswith('#Task.') or response_type.startswith('#Job.'): | ||||||
|  |                     # Task and Job have similar enough structures to treat the same | ||||||
|  |                     operation_results['status'] = data.get('TaskState', data.get('JobState')) | ||||||
|  |                     operation_results['messages'] = data.get('Messages', []) | ||||||
|  |                 else: | ||||||
|  |                     # Error response body, which is a bit of a misnomer since it's used in successful action responses | ||||||
|  |                     operation_results['status'] = 'Completed' | ||||||
|  |                     if response.status >= 400: | ||||||
|  |                         operation_results['status'] = 'Exception' | ||||||
|  |                     operation_results['messages'] = data.get('error', {}).get('@Message.ExtendedInfo', []) | ||||||
|  |             else: | ||||||
|  |                 # No response body (or malformed); build based on status code | ||||||
|  |                 operation_results['status'] = 'Completed' | ||||||
|  |                 if response.status == 202: | ||||||
|  |                     operation_results['status'] = 'New' | ||||||
|  |                 elif response.status >= 400: | ||||||
|  |                     operation_results['status'] = 'Exception' | ||||||
|  | 
 | ||||||
|  |             # Clear out the handle if the operation is complete | ||||||
|  |             if operation_results['status'] in ['Completed', 'Cancelled', 'Exception', 'Killed']: | ||||||
|  |                 operation_results['handle'] = None | ||||||
|  | 
 | ||||||
|  |             # Scan the messages to see if next steps are needed | ||||||
|  |             for message in operation_results['messages']: | ||||||
|  |                 message_id = message['MessageId'] | ||||||
|  | 
 | ||||||
|  |                 if message_id.startswith('Update.1.') and message_id.endswith('.OperationTransitionedToJob'): | ||||||
|  |                     # Operation rerouted to a job; update the status and handle | ||||||
|  |                     operation_results['status'] = 'New' | ||||||
|  |                     operation_results['handle'] = message['MessageArgs'][0] | ||||||
|  |                     operation_results['resets_requested'] = [] | ||||||
|  |                     # No need to process other messages in this case | ||||||
|  |                     break | ||||||
|  | 
 | ||||||
|  |                 if message_id.startswith('Base.1.') and message_id.endswith('.ResetRequired'): | ||||||
|  |                     # A reset to some device is needed to continue the update | ||||||
|  |                     reset = {'uri': message['MessageArgs'][0], 'type': message['MessageArgs'][1]} | ||||||
|  |                     operation_results['resets_requested'].append(reset) | ||||||
|  | 
 | ||||||
|  |         return operation_results | ||||||
|  | 
 | ||||||
|     def simple_update(self, update_opts): |     def simple_update(self, update_opts): | ||||||
|         image_uri = update_opts.get('update_image_uri') |         image_uri = update_opts.get('update_image_uri') | ||||||
|         protocol = update_opts.get('update_protocol') |         protocol = update_opts.get('update_protocol') | ||||||
|         targets = update_opts.get('update_targets') |         targets = update_opts.get('update_targets') | ||||||
|         creds = update_opts.get('update_creds') |         creds = update_opts.get('update_creds') | ||||||
|  |         apply_time = update_opts.get('update_apply_time') | ||||||
| 
 | 
 | ||||||
|         if not image_uri: |         if not image_uri: | ||||||
|             return {'ret': False, 'msg': |             return {'ret': False, 'msg': | ||||||
|  | @ -1439,11 +1552,65 @@ class RedfishUtils(object): | ||||||
|                 payload["Username"] = creds.get('username') |                 payload["Username"] = creds.get('username') | ||||||
|             if creds.get('password'): |             if creds.get('password'): | ||||||
|                 payload["Password"] = creds.get('password') |                 payload["Password"] = creds.get('password') | ||||||
|  |         if apply_time: | ||||||
|  |             payload["@Redfish.OperationApplyTime"] = apply_time | ||||||
|         response = self.post_request(self.root_uri + update_uri, payload) |         response = self.post_request(self.root_uri + update_uri, payload) | ||||||
|         if response['ret'] is False: |         if response['ret'] is False: | ||||||
|             return response |             return response | ||||||
|         return {'ret': True, 'changed': True, |         return {'ret': True, 'changed': True, | ||||||
|                 'msg': "SimpleUpdate requested"} |                 'msg': "SimpleUpdate requested", | ||||||
|  |                 'update_status': self._operation_results(response['resp'], response['data'])} | ||||||
|  | 
 | ||||||
|  |     def get_update_status(self, update_handle): | ||||||
|  |         """ | ||||||
|  |         Gets the status of an update operation. | ||||||
|  | 
 | ||||||
|  |         :param handle: The task or job handle tracking the update | ||||||
|  |         :return: dict containing the response of the update status | ||||||
|  |         """ | ||||||
|  | 
 | ||||||
|  |         if not update_handle: | ||||||
|  |             return {'ret': False, 'msg': 'Must provide a handle tracking the update.'} | ||||||
|  | 
 | ||||||
|  |         # Get the task or job tracking the update | ||||||
|  |         response = self.get_request(self.root_uri + update_handle) | ||||||
|  |         if response['ret'] is False: | ||||||
|  |             return response | ||||||
|  | 
 | ||||||
|  |         # Inspect the response to build the update status | ||||||
|  |         return self._operation_results(response['resp'], response['data'], update_handle) | ||||||
|  | 
 | ||||||
|  |     def perform_requested_update_operations(self, update_handle): | ||||||
|  |         """ | ||||||
|  |         Performs requested operations to allow the update to continue. | ||||||
|  | 
 | ||||||
|  |         :param handle: The task or job handle tracking the update | ||||||
|  |         :return: dict containing the result of the operations | ||||||
|  |         """ | ||||||
|  | 
 | ||||||
|  |         # Get the current update status | ||||||
|  |         update_status = self.get_update_status(update_handle) | ||||||
|  |         if update_status['ret'] is False: | ||||||
|  |             return update_status | ||||||
|  | 
 | ||||||
|  |         changed = False | ||||||
|  | 
 | ||||||
|  |         # Perform any requested updates | ||||||
|  |         for reset in update_status['resets_requested']: | ||||||
|  |             resp = self.post_request(self.root_uri + reset['uri'], {'ResetType': reset['type']}) | ||||||
|  |             if resp['ret'] is False: | ||||||
|  |                 # Override the 'changed' indicator since other resets may have | ||||||
|  |                 # been successful | ||||||
|  |                 resp['changed'] = changed | ||||||
|  |                 return resp | ||||||
|  |             changed = True | ||||||
|  | 
 | ||||||
|  |         msg = 'No operations required for the update' | ||||||
|  |         if changed: | ||||||
|  |             # Will need to consider finetuning this message if the scope of the | ||||||
|  |             # requested operations grow over time | ||||||
|  |             msg = 'One or more components reset to continue the update' | ||||||
|  |         return {'ret': True, 'changed': changed, 'msg': msg} | ||||||
| 
 | 
 | ||||||
|     def get_bios_attributes(self, systems_uri): |     def get_bios_attributes(self, systems_uri): | ||||||
|         result = {} |         result = {} | ||||||
|  |  | ||||||
							
								
								
									
										21
									
								
								plugins/module_utils/ssh.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										21
									
								
								plugins/module_utils/ssh.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,21 @@ | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | # Copyright (c) 2015, Björn Andersson | ||||||
|  | # Copyright (c) 2021, Ansible Project | ||||||
|  | # Copyright (c) 2021, Abhijeet Kasurde <akasurde@redhat.com> | ||||||
|  | # Copyright (c) 2022, Alexei Znamensky <russoz@gmail.com> | ||||||
|  | # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | import os | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def determine_config_file(user, config_file): | ||||||
|  |     if user: | ||||||
|  |         config_file = os.path.join(os.path.expanduser('~%s' % user), '.ssh', 'config') | ||||||
|  |     elif config_file is None: | ||||||
|  |         config_file = '/etc/ssh/ssh_config' | ||||||
|  |     return config_file | ||||||
|  | @ -27,7 +27,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ali_instance | module: ali_instance | ||||||
| short_description: Create, Start, Stop, Restart or Terminate an Instance in ECS. Add or Remove Instance to/from a Security Group. | short_description: Create, Start, Stop, Restart or Terminate an Instance in ECS; Add or Remove Instance to/from a Security Group | ||||||
| description: | description: | ||||||
|     - Create, start, stop, restart, modify or terminate ecs instances. |     - Create, start, stop, restart, modify or terminate ecs instances. | ||||||
|     - Add or remove ecs instances to/from security group. |     - Add or remove ecs instances to/from security group. | ||||||
|  |  | ||||||
|  | @ -27,7 +27,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ali_instance_info | module: ali_instance_info | ||||||
| short_description: Gather information on instances of Alibaba Cloud ECS. | short_description: Gather information on instances of Alibaba Cloud ECS | ||||||
| description: | description: | ||||||
|      - This module fetches data from the Open API in Alicloud. |      - This module fetches data from the Open API in Alicloud. | ||||||
|        The module must be called from within the ECS instance itself. |        The module must be called from within the ECS instance itself. | ||||||
|  |  | ||||||
|  | @ -20,6 +20,10 @@ notes: | ||||||
|   - > |   - > | ||||||
|     B(Ansible 2.9/2.10): The C(ansible-galaxy) command changed significantly between Ansible 2.9 and |     B(Ansible 2.9/2.10): The C(ansible-galaxy) command changed significantly between Ansible 2.9 and | ||||||
|     ansible-base 2.10 (later ansible-core 2.11). See comments in the parameters. |     ansible-base 2.10 (later ansible-core 2.11). See comments in the parameters. | ||||||
|  |   - > | ||||||
|  |     The module will try and run using the C(C.UTF-8) locale. | ||||||
|  |     If that fails, it will try C(en_US.UTF-8). | ||||||
|  |     If that one also fails, the module will fail. | ||||||
| requirements: | requirements: | ||||||
|   - Ansible 2.9, ansible-base 2.10, or ansible-core 2.11 or newer |   - Ansible 2.9, ansible-base 2.10, or ansible-core 2.11 or newer | ||||||
| options: | options: | ||||||
|  | @ -185,7 +189,7 @@ RETURN = """ | ||||||
| import re | import re | ||||||
| 
 | 
 | ||||||
| from ansible_collections.community.general.plugins.module_utils.cmd_runner import CmdRunner, cmd_runner_fmt as fmt | from ansible_collections.community.general.plugins.module_utils.cmd_runner import CmdRunner, cmd_runner_fmt as fmt | ||||||
| from ansible_collections.community.general.plugins.module_utils.module_helper import ModuleHelper | from ansible_collections.community.general.plugins.module_utils.module_helper import ModuleHelper, ModuleHelperException | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class AnsibleGalaxyInstall(ModuleHelper): | class AnsibleGalaxyInstall(ModuleHelper): | ||||||
|  | @ -226,11 +230,17 @@ class AnsibleGalaxyInstall(ModuleHelper): | ||||||
|         version=fmt.as_bool("--version"), |         version=fmt.as_bool("--version"), | ||||||
|         name=fmt.as_list(), |         name=fmt.as_list(), | ||||||
|     ) |     ) | ||||||
|     force_lang = "en_US.UTF-8" | 
 | ||||||
|     check_rc = True |     def _make_runner(self, lang): | ||||||
|  |         return CmdRunner(self.module, command=self.command, arg_formats=self.command_args_formats, force_lang=lang, check_rc=True) | ||||||
| 
 | 
 | ||||||
|     def _get_ansible_galaxy_version(self): |     def _get_ansible_galaxy_version(self): | ||||||
|  |         class UnsupportedLocale(ModuleHelperException): | ||||||
|  |             pass | ||||||
|  | 
 | ||||||
|         def process(rc, out, err): |         def process(rc, out, err): | ||||||
|  |             if (rc != 0 and "unsupported locale setting" in err) or (rc == 0 and "cannot change locale" in err): | ||||||
|  |                 raise UnsupportedLocale(msg=err) | ||||||
|             line = out.splitlines()[0] |             line = out.splitlines()[0] | ||||||
|             match = self._RE_GALAXY_VERSION.match(line) |             match = self._RE_GALAXY_VERSION.match(line) | ||||||
|             if not match: |             if not match: | ||||||
|  | @ -239,12 +249,18 @@ class AnsibleGalaxyInstall(ModuleHelper): | ||||||
|             version = tuple(int(x) for x in version.split('.')[:3]) |             version = tuple(int(x) for x in version.split('.')[:3]) | ||||||
|             return version |             return version | ||||||
| 
 | 
 | ||||||
|         with self.runner("version", check_rc=True, output_process=process) as ctx: |         try: | ||||||
|             return ctx.run(version=True) |             runner = self._make_runner("C.UTF-8") | ||||||
|  |             with runner("version", check_rc=False, output_process=process) as ctx: | ||||||
|  |                 return runner, ctx.run(version=True) | ||||||
|  |         except UnsupportedLocale as e: | ||||||
|  |             runner = self._make_runner("en_US.UTF-8") | ||||||
|  |             with runner("version", check_rc=True, output_process=process) as ctx: | ||||||
|  |                 return runner, ctx.run(version=True) | ||||||
| 
 | 
 | ||||||
|     def __init_module__(self): |     def __init_module__(self): | ||||||
|         self.runner = CmdRunner(self.module, command=self.command, arg_formats=self.command_args_formats, force_lang=self.force_lang) |         # self.runner = CmdRunner(self.module, command=self.command, arg_formats=self.command_args_formats, force_lang=self.force_lang) | ||||||
|         self.ansible_version = self._get_ansible_galaxy_version() |         self.runner, self.ansible_version = self._get_ansible_galaxy_version() | ||||||
|         if self.ansible_version < (2, 11) and not self.vars.ack_min_ansiblecore211: |         if self.ansible_version < (2, 11) and not self.vars.ack_min_ansiblecore211: | ||||||
|             self.module.deprecate( |             self.module.deprecate( | ||||||
|                 "Support for Ansible 2.9 and ansible-base 2.10 is being deprecated. " |                 "Support for Ansible 2.9 and ansible-base 2.10 is being deprecated. " | ||||||
|  | @ -339,11 +355,12 @@ class AnsibleGalaxyInstall(ModuleHelper): | ||||||
|             self._setup210plus() |             self._setup210plus() | ||||||
|         with self.runner("type galaxy_cmd force no_deps dest requirements_file name", output_process=process) as ctx: |         with self.runner("type galaxy_cmd force no_deps dest requirements_file name", output_process=process) as ctx: | ||||||
|             ctx.run(galaxy_cmd="install") |             ctx.run(galaxy_cmd="install") | ||||||
|  |             if self.verbosity > 2: | ||||||
|  |                 self.vars.set("run_info", ctx.run_info) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def main(): | def main(): | ||||||
|     galaxy = AnsibleGalaxyInstall() |     AnsibleGalaxyInstall.execute() | ||||||
|     galaxy.run() |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|  |  | ||||||
|  | @ -16,7 +16,7 @@ author: | ||||||
|     - Christian Berendt (@berendt) |     - Christian Berendt (@berendt) | ||||||
|     - Ralf Hertel (@n0trax) |     - Ralf Hertel (@n0trax) | ||||||
|     - Robin Roth (@robinro) |     - Robin Roth (@robinro) | ||||||
| short_description: Enables/disables a module of the Apache2 webserver. | short_description: Enables/disables a module of the Apache2 webserver | ||||||
| description: | description: | ||||||
|    - Enables or disables a specified module of the Apache2 webserver. |    - Enables or disables a specified module of the Apache2 webserver. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -14,7 +14,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: apt_rpm | module: apt_rpm | ||||||
| short_description: apt_rpm package manager | short_description: APT-RPM package manager | ||||||
| description: | description: | ||||||
|   - Manages packages with I(apt-rpm). Both low-level (I(rpm)) and high-level (I(apt-get)) package manager binaries required. |   - Manages packages with I(apt-rpm). Both low-level (I(rpm)) and high-level (I(apt-get)) package manager binaries required. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = r''' | DOCUMENTATION = r''' | ||||||
| --- | --- | ||||||
| module: beadm | module: beadm | ||||||
| short_description: Manage ZFS boot environments on FreeBSD/Solaris/illumos systems. | short_description: Manage ZFS boot environments on FreeBSD/Solaris/illumos systems | ||||||
| description: | description: | ||||||
|     - Create, delete or activate ZFS boot environments. |     - Create, delete or activate ZFS boot environments. | ||||||
|     - Mount and unmount ZFS boot environments. |     - Mount and unmount ZFS boot environments. | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: circonus_annotation | module: circonus_annotation | ||||||
| short_description: create an annotation in circonus | short_description: Create an annotation in circonus | ||||||
| description: | description: | ||||||
|     - Create an annotation event with a given category, title and description. Optionally start, end or durations can be provided |     - Create an annotation event with a given category, title and description. Optionally start, end or durations can be provided | ||||||
| author: "Nick Harring (@NickatEpic)" | author: "Nick Harring (@NickatEpic)" | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_aa_policy | module: clc_aa_policy | ||||||
| short_description: Create or Delete Anti Affinity Policies at CenturyLink Cloud. | short_description: Create or Delete Anti Affinity Policies at CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to Create or Delete Anti Affinity Policies at CenturyLink Cloud. |   - An Ansible module to Create or Delete Anti Affinity Policies at CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_alert_policy | module: clc_alert_policy | ||||||
| short_description: Create or Delete Alert Policies at CenturyLink Cloud. | short_description: Create or Delete Alert Policies at CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to Create or Delete Alert Policies at CenturyLink Cloud. |   - An Ansible module to Create or Delete Alert Policies at CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_blueprint_package | module: clc_blueprint_package | ||||||
| short_description: deploys a blue print package on a set of servers in CenturyLink Cloud. | short_description: Deploys a blue print package on a set of servers in CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to deploy blue print package on a set of servers in CenturyLink Cloud. |   - An Ansible module to deploy blue print package on a set of servers in CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_loadbalancer | module: clc_loadbalancer | ||||||
| short_description: Create, Delete shared loadbalancers in CenturyLink Cloud. | short_description: Create, Delete shared loadbalancers in CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to Create, Delete shared loadbalancers in CenturyLink Cloud. |   - An Ansible module to Create, Delete shared loadbalancers in CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_modify_server | module: clc_modify_server | ||||||
| short_description: modify servers in CenturyLink Cloud. | short_description: Modify servers in CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to modify servers in CenturyLink Cloud. |   - An Ansible module to modify servers in CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_publicip | module: clc_publicip | ||||||
| short_description: Add and Delete public ips on servers in CenturyLink Cloud. | short_description: Add and Delete public ips on servers in CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to add or delete public ip addresses on an existing server or servers in CenturyLink Cloud. |   - An Ansible module to add or delete public ip addresses on an existing server or servers in CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_server | module: clc_server | ||||||
| short_description: Create, Delete, Start and Stop servers in CenturyLink Cloud. | short_description: Create, Delete, Start and Stop servers in CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to Create, Delete, Start and Stop servers in CenturyLink Cloud. |   - An Ansible module to Create, Delete, Start and Stop servers in CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: clc_server_snapshot | module: clc_server_snapshot | ||||||
| short_description: Create, Delete and Restore server snapshots in CenturyLink Cloud. | short_description: Create, Delete and Restore server snapshots in CenturyLink Cloud | ||||||
| description: | description: | ||||||
|   - An Ansible module to Create, Delete and Restore server snapshots in CenturyLink Cloud. |   - An Ansible module to Create, Delete and Restore server snapshots in CenturyLink Cloud. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: cloud_init_data_facts | module: cloud_init_data_facts | ||||||
| short_description: Retrieve facts of cloud-init. | short_description: Retrieve facts of cloud-init | ||||||
| description: | description: | ||||||
|   - Gathers facts by reading the status.json and result.json of cloud-init. |   - Gathers facts by reading the status.json and result.json of cloud-init. | ||||||
| author: René Moser (@resmo) | author: René Moser (@resmo) | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: consul | module: consul | ||||||
| short_description: "Add, modify & delete services within a consul cluster." | short_description: Add, modify & delete services within a consul cluster | ||||||
| description: | description: | ||||||
|  - Registers services and checks for an agent with a consul cluster. |  - Registers services and checks for an agent with a consul cluster. | ||||||
|    A service is some process running on the agent node that should be advertised by |    A service is some process running on the agent node that should be advertised by | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: cpanm | module: cpanm | ||||||
| short_description: Manages Perl library dependencies. | short_description: Manages Perl library dependencies | ||||||
| description: | description: | ||||||
|   - Manage Perl library dependencies using cpanminus. |   - Manage Perl library dependencies using cpanminus. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -15,7 +15,7 @@ DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: deploy_helper | module: deploy_helper | ||||||
| author: "Ramon de la Fuente (@ramondelafuente)" | author: "Ramon de la Fuente (@ramondelafuente)" | ||||||
| short_description: Manages some of the steps common in deploying projects. | short_description: Manages some of the steps common in deploying projects | ||||||
| description: | description: | ||||||
|   - The Deploy Helper manages some of the steps common in deploying software. |   - The Deploy Helper manages some of the steps common in deploying software. | ||||||
|     It creates a folder structure, manages a symlink for the current release |     It creates a folder structure, manages a symlink for the current release | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: dimensiondata_vlan | module: dimensiondata_vlan | ||||||
| short_description: Manage a VLAN in a Cloud Control network domain. | short_description: Manage a VLAN in a Cloud Control network domain | ||||||
| extends_documentation_fragment: | extends_documentation_fragment: | ||||||
| - community.general.dimensiondata | - community.general.dimensiondata | ||||||
| - community.general.dimensiondata_wait | - community.general.dimensiondata_wait | ||||||
|  |  | ||||||
|  | @ -230,18 +230,11 @@ dnsimple_record_info: | ||||||
|         type: str |         type: str | ||||||
| ''' | ''' | ||||||
| 
 | 
 | ||||||
| import traceback |  | ||||||
| from ansible.module_utils.basic import AnsibleModule | from ansible.module_utils.basic import AnsibleModule | ||||||
| from ansible.module_utils.basic import missing_required_lib | from ansible_collections.community.general.plugins.module_utils import deps | ||||||
| 
 | 
 | ||||||
| try: | with deps.declare("requests"): | ||||||
|     from requests import Request, Session |     from requests import Request, Session | ||||||
| except ImportError: |  | ||||||
|     HAS_REQUESTS = False |  | ||||||
|     REQUESTS_IMPORT_ERROR = traceback.format_exc() |  | ||||||
| else: |  | ||||||
|     HAS_REQUESTS = True |  | ||||||
|     REQUESTS_IMPORT_ERROR = None |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def build_url(account, key, is_sandbox): | def build_url(account, key, is_sandbox): | ||||||
|  | @ -310,10 +303,7 @@ def main(): | ||||||
|                     params['api_key'], |                     params['api_key'], | ||||||
|                     params['sandbox']) |                     params['sandbox']) | ||||||
| 
 | 
 | ||||||
|     if not HAS_REQUESTS: |     deps.validate(module) | ||||||
|         module.exit_json( |  | ||||||
|             msg=missing_required_lib('requests'), |  | ||||||
|             exception=REQUESTS_IMPORT_ERROR) |  | ||||||
| 
 | 
 | ||||||
|     # At minimum we need account and key |     # At minimum we need account and key | ||||||
|     if params['account_id'] and params['api_key']: |     if params['account_id'] and params['api_key']: | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: dnsmadeeasy | module: dnsmadeeasy | ||||||
| short_description: Interface with dnsmadeeasy.com (a DNS hosting service). | short_description: Interface with dnsmadeeasy.com (a DNS hosting service) | ||||||
| description: | description: | ||||||
|    - > |    - > | ||||||
|      Manages DNS records via the v2 REST API of the DNS Made Easy service.  It handles records only; there is no manipulation of domains or |      Manages DNS records via the v2 REST API of the DNS Made Easy service.  It handles records only; there is no manipulation of domains or | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: etcd3 | module: etcd3 | ||||||
| short_description: "Set or delete key value pairs from an etcd3 cluster" | short_description: Set or delete key value pairs from an etcd3 cluster | ||||||
| requirements: | requirements: | ||||||
|   - etcd3 |   - etcd3 | ||||||
| description: | description: | ||||||
|  |  | ||||||
|  | @ -83,75 +83,17 @@ RETURN = ''' | ||||||
| ... | ... | ||||||
| ''' | ''' | ||||||
| 
 | 
 | ||||||
| from ansible.module_utils.basic import AnsibleModule | from ansible_collections.community.general.plugins.module_utils.module_helper import StateModuleHelper | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.gconftool2 import gconftool2_runner | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| class GConf2Preference(object): | class GConftool(StateModuleHelper): | ||||||
|     def __init__(self, ansible, key, value_type, value, |     change_params = 'value', | ||||||
|                  direct=False, config_source=""): |     diff_params = 'value', | ||||||
|         self.ansible = ansible |     output_params = ('key', 'value_type') | ||||||
|         self.key = key |     facts_params = ('key', 'value_type') | ||||||
|         self.value_type = value_type |     facts_name = 'gconftool2' | ||||||
|         self.value = value |     module = dict( | ||||||
|         self.config_source = config_source |  | ||||||
|         self.direct = direct |  | ||||||
| 
 |  | ||||||
|     def value_already_set(self): |  | ||||||
|         return False |  | ||||||
| 
 |  | ||||||
|     def call(self, call_type, fail_onerr=True): |  | ||||||
|         """ Helper function to perform gconftool-2 operations """ |  | ||||||
|         config_source = [] |  | ||||||
|         direct = [] |  | ||||||
|         changed = False |  | ||||||
|         out = '' |  | ||||||
| 
 |  | ||||||
|         # If the configuration source is different from the default, create |  | ||||||
|         # the argument |  | ||||||
|         if self.config_source is not None and len(self.config_source) > 0: |  | ||||||
|             config_source = ["--config-source", self.config_source] |  | ||||||
| 
 |  | ||||||
|         # If direct is true, create the argument |  | ||||||
|         if self.direct: |  | ||||||
|             direct = ["--direct"] |  | ||||||
| 
 |  | ||||||
|         # Execute the call |  | ||||||
|         cmd = ["gconftool-2"] |  | ||||||
|         try: |  | ||||||
|             # If the call is "get", then we don't need as many parameters and |  | ||||||
|             # we can ignore some |  | ||||||
|             if call_type == 'get': |  | ||||||
|                 self.ansible.deprecate( |  | ||||||
|                     msg="State 'get' is deprecated. Please use the module community.general.gconftool2_info instead", |  | ||||||
|                     version="8.0.0", collection_name="community.general" |  | ||||||
|                 ) |  | ||||||
|                 cmd.extend(["--get", self.key]) |  | ||||||
|             # Otherwise, we will use all relevant parameters |  | ||||||
|             elif call_type == 'set': |  | ||||||
|                 cmd.extend(direct) |  | ||||||
|                 cmd.extend(config_source) |  | ||||||
|                 cmd.extend(["--type", self.value_type, "--{3}".format(call_type), self.key, self.value]) |  | ||||||
|             elif call_type == 'unset': |  | ||||||
|                 cmd.extend(["--unset", self.key]) |  | ||||||
| 
 |  | ||||||
|             # Start external command |  | ||||||
|             rc, out, err = self.ansible.run_command(cmd) |  | ||||||
| 
 |  | ||||||
|             if err and fail_onerr: |  | ||||||
|                 self.ansible.fail_json(msg='gconftool-2 failed with ' |  | ||||||
|                                            'error: %s' % (str(err))) |  | ||||||
|             else: |  | ||||||
|                 changed = True |  | ||||||
| 
 |  | ||||||
|         except OSError as exception: |  | ||||||
|             self.ansible.fail_json(msg='gconftool-2 failed with exception: ' |  | ||||||
|                                        '%s' % exception) |  | ||||||
|         return changed, out.rstrip() |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def main(): |  | ||||||
|     # Setup the Ansible module |  | ||||||
|     module = AnsibleModule( |  | ||||||
|         argument_spec=dict( |         argument_spec=dict( | ||||||
|             key=dict(type='str', required=True, no_log=False), |             key=dict(type='str', required=True, no_log=False), | ||||||
|             value_type=dict(type='str', choices=['bool', 'float', 'int', 'string']), |             value_type=dict(type='str', choices=['bool', 'float', 'int', 'string']), | ||||||
|  | @ -160,75 +102,54 @@ def main(): | ||||||
|             direct=dict(type='bool', default=False), |             direct=dict(type='bool', default=False), | ||||||
|             config_source=dict(type='str'), |             config_source=dict(type='str'), | ||||||
|         ), |         ), | ||||||
|         supports_check_mode=True |         required_if=[ | ||||||
|  |             ('state', 'present', ['value', 'value_type']), | ||||||
|  |             ('state', 'absent', ['value']), | ||||||
|  |             ('direct', True, ['config_source']), | ||||||
|  |         ], | ||||||
|  |         supports_check_mode=True, | ||||||
|     ) |     ) | ||||||
| 
 | 
 | ||||||
|     state_values = {"present": "set", "absent": "unset", "get": "get"} |     def __init_module__(self): | ||||||
|  |         self.runner = gconftool2_runner(self.module, check_rc=True) | ||||||
|  |         if self.vars.state != "get": | ||||||
|  |             if not self.vars.direct and self.vars.config_source is not None: | ||||||
|  |                 self.module.fail_json(msg='If the "config_source" is specified then "direct" must be "true"') | ||||||
| 
 | 
 | ||||||
|     # Assign module values to dictionary values |         self.vars.set('previous_value', self._get(), fact=True) | ||||||
|     key = module.params['key'] |         self.vars.set('value_type', self.vars.value_type) | ||||||
|     value_type = module.params['value_type'] |         self.vars.set_meta('value', initial_value=self.vars.previous_value) | ||||||
|     if module.params['value'].lower() == "true": |         self.vars.set('playbook_value', self.vars.value, fact=True) | ||||||
|         value = "true" |  | ||||||
|     elif module.params['value'] == "false": |  | ||||||
|         value = "false" |  | ||||||
|     else: |  | ||||||
|         value = module.params['value'] |  | ||||||
| 
 | 
 | ||||||
|     state = state_values[module.params['state']] |     def _make_process(self, fail_on_err): | ||||||
|     direct = module.params['direct'] |         def process(rc, out, err): | ||||||
|     config_source = module.params['config_source'] |             if err and fail_on_err: | ||||||
|  |                 self.ansible.fail_json(msg='gconftool-2 failed with error: %s' % (str(err))) | ||||||
|  |             self.vars.value = out.rstrip() | ||||||
|  |             return self.vars.value | ||||||
|  |         return process | ||||||
| 
 | 
 | ||||||
|     # Initialize some variables for later |     def _get(self): | ||||||
|     change = False |         return self.runner("state key", output_process=self._make_process(False)).run(state="get") | ||||||
|     new_value = '' |  | ||||||
| 
 | 
 | ||||||
|     if state != "get": |     def state_get(self): | ||||||
|         if value is None or value == "": |         self.deprecate( | ||||||
|             module.fail_json(msg='State %s requires "value" to be set' |             msg="State 'get' is deprecated. Please use the module community.general.gconftool2_info instead", | ||||||
|                                  % str(state)) |             version="8.0.0", collection_name="community.general" | ||||||
|         elif value_type is None or value_type == "": |         ) | ||||||
|             module.fail_json(msg='State %s requires "value_type" to be set' |  | ||||||
|                                  % str(state)) |  | ||||||
| 
 | 
 | ||||||
|         if direct and config_source is None: |     def state_absent(self): | ||||||
|             module.fail_json(msg='If "direct" is "true" then the ' + |         with self.runner("state key", output_process=self._make_process(False)) as ctx: | ||||||
|                                  '"config_source" must be specified') |             ctx.run() | ||||||
|         elif not direct and config_source is not None: |         self.vars.set('new_value', None, fact=True) | ||||||
|             module.fail_json(msg='If the "config_source" is specified ' + |  | ||||||
|                                  'then "direct" must be "true"') |  | ||||||
| 
 | 
 | ||||||
|     # Create a gconf2 preference |     def state_present(self): | ||||||
|     gconf_pref = GConf2Preference(module, key, value_type, |         with self.runner("direct config_source value_type state key value", output_process=self._make_process(True)) as ctx: | ||||||
|                                   value, direct, config_source) |             self.vars.set('new_value', ctx.run(), fact=True) | ||||||
|     # Now we get the current value, if not found don't fail |  | ||||||
|     dummy, current_value = gconf_pref.call("get", fail_onerr=False) |  | ||||||
| 
 | 
 | ||||||
|     # Check if the current value equals the value we want to set.  If not, make |  | ||||||
|     # a change |  | ||||||
|     if current_value != value: |  | ||||||
|         # If check mode, we know a change would have occurred. |  | ||||||
|         if module.check_mode: |  | ||||||
|             # So we will set the change to True |  | ||||||
|             change = True |  | ||||||
|             # And set the new_value to the value that would have been set |  | ||||||
|             new_value = value |  | ||||||
|         # If not check mode make the change. |  | ||||||
|         else: |  | ||||||
|             change, new_value = gconf_pref.call(state) |  | ||||||
|     # If the value we want to set is the same as the current_value, we will |  | ||||||
|     # set the new_value to the current_value for reporting |  | ||||||
|     else: |  | ||||||
|         new_value = current_value |  | ||||||
| 
 | 
 | ||||||
|     facts = dict(gconftool2={'changed': change, | def main(): | ||||||
|                              'key': key, |     GConftool.execute() | ||||||
|                              'value_type': value_type, |  | ||||||
|                              'new_value': new_value, |  | ||||||
|                              'previous_value': current_value, |  | ||||||
|                              'playbook_value': module.params['value']}) |  | ||||||
| 
 |  | ||||||
|     module.exit_json(changed=change, ansible_facts=facts) |  | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| if __name__ == '__main__': | if __name__ == '__main__': | ||||||
|  |  | ||||||
|  | @ -65,8 +65,8 @@ class GConftoolInfo(ModuleHelper): | ||||||
|         self.runner = gconftool2_runner(self.module, check_rc=True) |         self.runner = gconftool2_runner(self.module, check_rc=True) | ||||||
| 
 | 
 | ||||||
|     def __run__(self): |     def __run__(self): | ||||||
|         with self.runner.context(args_order=["get", "key"]) as ctx: |         with self.runner.context(args_order=["state", "key"]) as ctx: | ||||||
|             rc, out, err = ctx.run(get=True) |             rc, out, err = ctx.run(state="get") | ||||||
|             self.vars.value = None if err and not out else out.rstrip() |             self.vars.value = None if err and not out else out.rstrip() | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: github_deploy_key | module: github_deploy_key | ||||||
| author: "Ali (@bincyber)" | author: "Ali (@bincyber)" | ||||||
| short_description: Manages deploy keys for GitHub repositories. | short_description: Manages deploy keys for GitHub repositories | ||||||
| description: | description: | ||||||
|   - "Adds or removes deploy keys for GitHub repositories. Supports authentication using username and password, |   - "Adds or removes deploy keys for GitHub repositories. Supports authentication using username and password, | ||||||
|   username and password and 2-factor authentication code (OTP), OAuth2 token, or personal access token. Admin |   username and password and 2-factor authentication code (OTP), OAuth2 token, or personal access token. Admin | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: github_issue | module: github_issue | ||||||
| short_description: View GitHub issue. | short_description: View GitHub issue | ||||||
| description: | description: | ||||||
|     - View GitHub issue for a given repository and organization. |     - View GitHub issue for a given repository and organization. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: github_key | module: github_key | ||||||
| short_description: Manage GitHub access keys. | short_description: Manage GitHub access keys | ||||||
| description: | description: | ||||||
|     - Creates, removes, or updates GitHub access keys. |     - Creates, removes, or updates GitHub access keys. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -108,17 +108,8 @@ EXAMPLES = ''' | ||||||
| ''' | ''' | ||||||
| 
 | 
 | ||||||
| RETURN = ''' | RETURN = ''' | ||||||
| create_release: | tag: | ||||||
|     description: |     description: Version of the created/latest release. | ||||||
|     - Version of the created release |  | ||||||
|     - "For Ansible version 2.5 and later, if specified release version already exists, then State is unchanged" |  | ||||||
|     - "For Ansible versions prior to 2.5, if specified release version already exists, then State is skipped" |  | ||||||
|     type: str |  | ||||||
|     returned: success |  | ||||||
|     sample: 1.1.0 |  | ||||||
| 
 |  | ||||||
| latest_release: |  | ||||||
|     description: Version of the latest release |  | ||||||
|     type: str |     type: str | ||||||
|     returned: success |     returned: success | ||||||
|     sample: 1.1.0 |     sample: 1.1.0 | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: gitlab_deploy_key | module: gitlab_deploy_key | ||||||
| short_description: Manages GitLab project deploy keys. | short_description: Manages GitLab project deploy keys | ||||||
| description: | description: | ||||||
|   - Adds, updates and removes project deploy keys |   - Adds, updates and removes project deploy keys | ||||||
| author: | author: | ||||||
|  |  | ||||||
|  | @ -165,7 +165,7 @@ from ansible.module_utils.six import string_types | ||||||
| from ansible.module_utils.six import integer_types | from ansible.module_utils.six import integer_types | ||||||
| 
 | 
 | ||||||
| from ansible_collections.community.general.plugins.module_utils.gitlab import ( | from ansible_collections.community.general.plugins.module_utils.gitlab import ( | ||||||
|     auth_argument_spec, gitlab_authentication, ensure_gitlab_package |     auth_argument_spec, gitlab_authentication, ensure_gitlab_package, filter_returned_variables | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -296,11 +296,7 @@ def native_python_main(this_gitlab, purge, requested_variables, state, module): | ||||||
|     before = [x.attributes for x in gitlab_keys] |     before = [x.attributes for x in gitlab_keys] | ||||||
| 
 | 
 | ||||||
|     gitlab_keys = this_gitlab.list_all_group_variables() |     gitlab_keys = this_gitlab.list_all_group_variables() | ||||||
|     existing_variables = [x.attributes for x in gitlab_keys] |     existing_variables = filter_returned_variables(gitlab_keys) | ||||||
| 
 |  | ||||||
|     # preprocessing:filter out and enrich before compare |  | ||||||
|     for item in existing_variables: |  | ||||||
|         item.pop('group_id') |  | ||||||
| 
 | 
 | ||||||
|     for item in requested_variables: |     for item in requested_variables: | ||||||
|         item['key'] = item.pop('name') |         item['key'] = item.pop('name') | ||||||
|  | @ -331,9 +327,7 @@ def native_python_main(this_gitlab, purge, requested_variables, state, module): | ||||||
|         if purge: |         if purge: | ||||||
|             # refetch and filter |             # refetch and filter | ||||||
|             gitlab_keys = this_gitlab.list_all_group_variables() |             gitlab_keys = this_gitlab.list_all_group_variables() | ||||||
|             existing_variables = [x.attributes for x in gitlab_keys] |             existing_variables = filter_returned_variables(gitlab_keys) | ||||||
|             for item in existing_variables: |  | ||||||
|                 item.pop('group_id') |  | ||||||
| 
 | 
 | ||||||
|             remove = [x for x in existing_variables if x not in requested_variables] |             remove = [x for x in existing_variables if x not in requested_variables] | ||||||
|             for item in remove: |             for item in remove: | ||||||
|  |  | ||||||
|  | @ -14,7 +14,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: gitlab_hook | module: gitlab_hook | ||||||
| short_description: Manages GitLab project hooks. | short_description: Manages GitLab project hooks | ||||||
| description: | description: | ||||||
|   - Adds, updates and removes project hook |   - Adds, updates and removes project hook | ||||||
| author: | author: | ||||||
|  |  | ||||||
							
								
								
									
										216
									
								
								plugins/modules/gitlab_project_badge.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										216
									
								
								plugins/modules/gitlab_project_badge.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,216 @@ | ||||||
|  | #!/usr/bin/python | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | 
 | ||||||
|  | # Copyright (c) 2022, Guillaume MARTINEZ (lunik@tiwabbit.fr) | ||||||
|  | # GNU General Public License v3.0+ (see LICENSES/GPL-3.0-or-later.txt or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | DOCUMENTATION = r''' | ||||||
|  | --- | ||||||
|  | module: gitlab_project_badge | ||||||
|  | short_description: Manage project badges on GitLab Server | ||||||
|  | version_added: 6.1.0 | ||||||
|  | description: | ||||||
|  |   - This module allows to add and remove badges to/from a project. | ||||||
|  | author: Guillaume MARTINEZ (@Lunik) | ||||||
|  | requirements: | ||||||
|  |   - C(owner) or C(maintainer) rights to project on the GitLab server | ||||||
|  | extends_documentation_fragment: | ||||||
|  |   - community.general.auth_basic | ||||||
|  |   - community.general.gitlab | ||||||
|  |   - community.general.attributes | ||||||
|  | 
 | ||||||
|  | attributes: | ||||||
|  |   check_mode: | ||||||
|  |     support: full | ||||||
|  |   diff_mode: | ||||||
|  |     support: none | ||||||
|  | 
 | ||||||
|  | options: | ||||||
|  |   project: | ||||||
|  |     description: | ||||||
|  |       - The name (or full path) of the GitLab project the badge is added to/removed from. | ||||||
|  |     required: true | ||||||
|  |     type: str | ||||||
|  | 
 | ||||||
|  |   state: | ||||||
|  |     description: | ||||||
|  |       - State of the badge in the project. | ||||||
|  |       - On C(present), it adds a badge to a GitLab project. | ||||||
|  |       - On C(absent), it removes a badge from a GitLab project. | ||||||
|  |     choices: ['present', 'absent'] | ||||||
|  |     default: 'present' | ||||||
|  |     type: str | ||||||
|  | 
 | ||||||
|  |   link_url: | ||||||
|  |     description: | ||||||
|  |       - The URL associated with the badge. | ||||||
|  |     required: true | ||||||
|  |     type: str | ||||||
|  | 
 | ||||||
|  |   image_url: | ||||||
|  |     description: | ||||||
|  |       - The image URL of the badge. | ||||||
|  |       - A badge is identified by this URL. | ||||||
|  |     required: true | ||||||
|  |     type: str | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | EXAMPLES = r''' | ||||||
|  | - name: Add a badge to a GitLab Project | ||||||
|  |   community.general.gitlab_project_badge: | ||||||
|  |     api_url: 'https://example.gitlab.com' | ||||||
|  |     api_token: 'Your-Private-Token' | ||||||
|  |     project: projectname | ||||||
|  |     state: present | ||||||
|  |     link_url: 'https://example.gitlab.com/%{project_path}' | ||||||
|  |     image_url: 'https://example.gitlab.com/%{project_path}/badges/%{default_branch}/pipeline.svg' | ||||||
|  | 
 | ||||||
|  | - name: Remove a badge from a GitLab Project | ||||||
|  |   community.general.gitlab_project_badge: | ||||||
|  |     api_url: 'https://example.gitlab.com' | ||||||
|  |     api_token: 'Your-Private-Token' | ||||||
|  |     project: projectname | ||||||
|  |     state: absent | ||||||
|  |     link_url: 'https://example.gitlab.com/%{project_path}' | ||||||
|  |     image_url: 'https://example.gitlab.com/%{project_path}/badges/%{default_branch}/pipeline.svg' | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | RETURN = ''' | ||||||
|  | badge: | ||||||
|  |   description: The badge information. | ||||||
|  |   returned: when I(state=present) | ||||||
|  |   type: dict | ||||||
|  |   sample: | ||||||
|  |     id: 1 | ||||||
|  |     link_url: 'http://example.com/ci_status.svg?project=%{project_path}&ref=%{default_branch}' | ||||||
|  |     image_url: 'https://shields.io/my/badge' | ||||||
|  |     rendered_link_url: 'http://example.com/ci_status.svg?project=example-org/example-project&ref=master' | ||||||
|  |     rendered_image_url: 'https://shields.io/my/badge' | ||||||
|  |     kind: project | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | from ansible.module_utils.api import basic_auth_argument_spec | ||||||
|  | from ansible.module_utils.basic import AnsibleModule | ||||||
|  | 
 | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.gitlab import ( | ||||||
|  |     auth_argument_spec, gitlab_authentication, find_project, ensure_gitlab_package | ||||||
|  | ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def present_strategy(module, gl, project, wished_badge): | ||||||
|  |     changed = False | ||||||
|  | 
 | ||||||
|  |     existing_badge = None | ||||||
|  |     for badge in project.badges.list(iterator=True): | ||||||
|  |         if badge.image_url == wished_badge["image_url"]: | ||||||
|  |             existing_badge = badge | ||||||
|  |             break | ||||||
|  | 
 | ||||||
|  |     if not existing_badge: | ||||||
|  |         changed = True | ||||||
|  |         if module.check_mode: | ||||||
|  |             return changed, {"status": "A project badge would be created."} | ||||||
|  | 
 | ||||||
|  |         badge = project.badges.create(wished_badge) | ||||||
|  |         return changed, badge.attributes | ||||||
|  | 
 | ||||||
|  |     if existing_badge.link_url != wished_badge["link_url"]: | ||||||
|  |         changed = True | ||||||
|  |         existing_badge.link_url = wished_badge["link_url"] | ||||||
|  | 
 | ||||||
|  |     if changed: | ||||||
|  |         if module.check_mode: | ||||||
|  |             return changed, {"status": "Project badge attributes would be changed."} | ||||||
|  | 
 | ||||||
|  |         existing_badge.save() | ||||||
|  | 
 | ||||||
|  |     return changed, existing_badge.attributes | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def absent_strategy(module, gl, project, wished_badge): | ||||||
|  |     changed = False | ||||||
|  | 
 | ||||||
|  |     existing_badge = None | ||||||
|  |     for badge in project.badges.list(iterator=True): | ||||||
|  |         if badge.image_url == wished_badge["image_url"]: | ||||||
|  |             existing_badge = badge | ||||||
|  |             break | ||||||
|  | 
 | ||||||
|  |     if not existing_badge: | ||||||
|  |         return changed, None | ||||||
|  | 
 | ||||||
|  |     changed = True | ||||||
|  |     if module.check_mode: | ||||||
|  |         return changed, {"status": "Project badge would be destroyed."} | ||||||
|  | 
 | ||||||
|  |     existing_badge.delete() | ||||||
|  | 
 | ||||||
|  |     return changed, None | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | state_strategy = { | ||||||
|  |     "present": present_strategy, | ||||||
|  |     "absent": absent_strategy | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def core(module): | ||||||
|  |     ensure_gitlab_package(module) | ||||||
|  | 
 | ||||||
|  |     gitlab_project = module.params['project'] | ||||||
|  |     state = module.params['state'] | ||||||
|  | 
 | ||||||
|  |     gl = gitlab_authentication(module) | ||||||
|  | 
 | ||||||
|  |     project = find_project(gl, gitlab_project) | ||||||
|  |     # project doesn't exist | ||||||
|  |     if not project: | ||||||
|  |         module.fail_json(msg="project '%s' not found." % gitlab_project) | ||||||
|  | 
 | ||||||
|  |     wished_badge = { | ||||||
|  |         "link_url": module.params["link_url"], | ||||||
|  |         "image_url": module.params["image_url"], | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     changed, summary = state_strategy[state](module=module, gl=gl, project=project, wished_badge=wished_badge) | ||||||
|  | 
 | ||||||
|  |     module.exit_json(changed=changed, badge=summary) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def main(): | ||||||
|  |     argument_spec = basic_auth_argument_spec() | ||||||
|  |     argument_spec.update(auth_argument_spec()) | ||||||
|  |     argument_spec.update(dict( | ||||||
|  |         project=dict(type='str', required=True), | ||||||
|  |         state=dict(type='str', default='present', choices=['present', 'absent']), | ||||||
|  |         link_url=dict(type='str', required=True), | ||||||
|  |         image_url=dict(type='str', required=True), | ||||||
|  |     )) | ||||||
|  | 
 | ||||||
|  |     module = AnsibleModule( | ||||||
|  |         argument_spec=argument_spec, | ||||||
|  |         mutually_exclusive=[ | ||||||
|  |             ['api_username', 'api_token'], | ||||||
|  |             ['api_username', 'api_oauth_token'], | ||||||
|  |             ['api_username', 'api_job_token'], | ||||||
|  |             ['api_token', 'api_oauth_token'], | ||||||
|  |             ['api_token', 'api_job_token'], | ||||||
|  |         ], | ||||||
|  |         required_together=[ | ||||||
|  |             ['api_username', 'api_password'], | ||||||
|  |         ], | ||||||
|  |         required_one_of=[ | ||||||
|  |             ['api_username', 'api_token', 'api_oauth_token', 'api_job_token'], | ||||||
|  |         ], | ||||||
|  |         supports_check_mode=True, | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     core(module) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     main() | ||||||
|  | @ -189,7 +189,7 @@ except Exception: | ||||||
|     HAS_GITLAB_PACKAGE = False |     HAS_GITLAB_PACKAGE = False | ||||||
| 
 | 
 | ||||||
| from ansible_collections.community.general.plugins.module_utils.gitlab import ( | from ansible_collections.community.general.plugins.module_utils.gitlab import ( | ||||||
|     auth_argument_spec, gitlab_authentication, ensure_gitlab_package |     auth_argument_spec, gitlab_authentication, ensure_gitlab_package, filter_returned_variables | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -255,9 +255,11 @@ class GitlabProjectVariables(object): | ||||||
|             return True |             return True | ||||||
| 
 | 
 | ||||||
|         var = { |         var = { | ||||||
|             "key": var_obj.get('key'), "value": var_obj.get('value'), |             "key": var_obj.get('key'), | ||||||
|             "masked": var_obj.get('masked'), "protected": var_obj.get('protected'), |             "value": var_obj.get('value'), | ||||||
|             "variable_type": var_obj.get('variable_type') |             "masked": var_obj.get('masked'), | ||||||
|  |             "protected": var_obj.get('protected'), | ||||||
|  |             "variable_type": var_obj.get('variable_type'), | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if var_obj.get('environment_scope') is not None: |         if var_obj.get('environment_scope') is not None: | ||||||
|  | @ -319,12 +321,9 @@ def native_python_main(this_gitlab, purge, requested_variables, state, module): | ||||||
|     before = [x.attributes for x in gitlab_keys] |     before = [x.attributes for x in gitlab_keys] | ||||||
| 
 | 
 | ||||||
|     gitlab_keys = this_gitlab.list_all_project_variables() |     gitlab_keys = this_gitlab.list_all_project_variables() | ||||||
|     existing_variables = [x.attributes for x in gitlab_keys] |     existing_variables = filter_returned_variables(gitlab_keys) | ||||||
| 
 |  | ||||||
|     # preprocessing:filter out and enrich before compare |  | ||||||
|     for item in existing_variables: |  | ||||||
|         item.pop('project_id') |  | ||||||
| 
 | 
 | ||||||
|  |     # filter out and enrich before compare | ||||||
|     for item in requested_variables: |     for item in requested_variables: | ||||||
|         item['key'] = item.pop('name') |         item['key'] = item.pop('name') | ||||||
|         item['value'] = str(item.get('value')) |         item['value'] = str(item.get('value')) | ||||||
|  | @ -354,9 +353,7 @@ def native_python_main(this_gitlab, purge, requested_variables, state, module): | ||||||
|         if purge: |         if purge: | ||||||
|             # refetch and filter |             # refetch and filter | ||||||
|             gitlab_keys = this_gitlab.list_all_project_variables() |             gitlab_keys = this_gitlab.list_all_project_variables() | ||||||
|             existing_variables = [x.attributes for x in gitlab_keys] |             existing_variables = filter_returned_variables(gitlab_keys) | ||||||
|             for item in existing_variables: |  | ||||||
|                 item.pop('project_id') |  | ||||||
| 
 | 
 | ||||||
|             remove = [x for x in existing_variables if x not in requested_variables] |             remove = [x for x in existing_variables if x not in requested_variables] | ||||||
|             for item in remove: |             for item in remove: | ||||||
|  | @ -409,7 +406,7 @@ def main(): | ||||||
|             masked=dict(type='bool', default=False), |             masked=dict(type='bool', default=False), | ||||||
|             protected=dict(type='bool', default=False), |             protected=dict(type='bool', default=False), | ||||||
|             environment_scope=dict(type='str', default='*'), |             environment_scope=dict(type='str', default='*'), | ||||||
|             variable_type=dict(type='str', default='env_var', choices=["env_var", "file"]) |             variable_type=dict(type='str', default='env_var', choices=["env_var", "file"]), | ||||||
|         )), |         )), | ||||||
|         state=dict(type='str', default="present", choices=["absent", "present"]), |         state=dict(type='str', default="present", choices=["absent", "present"]), | ||||||
|     ) |     ) | ||||||
|  |  | ||||||
|  | @ -9,7 +9,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: gitlab_protected_branch | module: gitlab_protected_branch | ||||||
| short_description: (un)Marking existing branches for protection | short_description: Manage protection of existing branches | ||||||
| version_added: 3.4.0 | version_added: 3.4.0 | ||||||
| description: | description: | ||||||
|   - (un)Marking existing branches for protection. |   - (un)Marking existing branches for protection. | ||||||
|  | @ -25,7 +25,7 @@ extends_documentation_fragment: | ||||||
| options: | options: | ||||||
|   state: |   state: | ||||||
|     description: |     description: | ||||||
|       - Create or delete proteced branch. |       - Create or delete protected branch. | ||||||
|     default: present |     default: present | ||||||
|     type: str |     type: str | ||||||
|     choices: ["present", "absent"] |     choices: ["present", "absent"] | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: gitlab_runner | module: gitlab_runner | ||||||
| short_description: Create, modify and delete GitLab Runners. | short_description: Create, modify and delete GitLab Runners | ||||||
| description: | description: | ||||||
|   - Register, update and delete runners with the GitLab API. |   - Register, update and delete runners with the GitLab API. | ||||||
|   - All operations are performed using the GitLab API v4. |   - All operations are performed using the GitLab API v4. | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: gunicorn | module: gunicorn | ||||||
| short_description: Run gunicorn with various settings. | short_description: Run gunicorn with various settings | ||||||
| description: | description: | ||||||
|      - Starts gunicorn with the parameters specified. Common settings for gunicorn |      - Starts gunicorn with the parameters specified. Common settings for gunicorn | ||||||
|        configuration are supported. For additional configuration use a config file |        configuration are supported. For additional configuration use a config file | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: heroku_collaborator | module: heroku_collaborator | ||||||
| short_description: "Add or delete app collaborators on Heroku" | short_description: Add or delete app collaborators on Heroku | ||||||
| description: | description: | ||||||
|   - Manages collaborators for Heroku apps. |   - Manages collaborators for Heroku apps. | ||||||
|   - If set to C(present) and heroku user is already collaborator, then do nothing. |   - If set to C(present) and heroku user is already collaborator, then do nothing. | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: hipchat | module: hipchat | ||||||
| short_description: Send a message to Hipchat. | short_description: Send a message to Hipchat | ||||||
| description: | description: | ||||||
|    - Send a message to a Hipchat room, with options to control the formatting. |    - Send a message to a Hipchat room, with options to control the formatting. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -19,7 +19,7 @@ module: homebrew_tap | ||||||
| author: | author: | ||||||
|     - "Indrajit Raychaudhuri (@indrajitr)" |     - "Indrajit Raychaudhuri (@indrajitr)" | ||||||
|     - "Daniel Jaouen (@danieljaouen)" |     - "Daniel Jaouen (@danieljaouen)" | ||||||
| short_description: Tap a Homebrew repository. | short_description: Tap a Homebrew repository | ||||||
| description: | description: | ||||||
|     - Tap external Homebrew repositories. |     - Tap external Homebrew repositories. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| module: htpasswd | module: htpasswd | ||||||
| short_description: manage user files for basic authentication | short_description: Manage user files for basic authentication | ||||||
| description: | description: | ||||||
|   - Add and remove username/password entries in a password file using htpasswd. |   - Add and remove username/password entries in a password file using htpasswd. | ||||||
|   - This is used by web servers such as Apache and Nginx for basic authentication. |   - This is used by web servers such as Apache and Nginx for basic authentication. | ||||||
|  | @ -41,9 +41,12 @@ options: | ||||||
|     description: |     description: | ||||||
|       - Encryption scheme to be used.  As well as the four choices listed |       - Encryption scheme to be used.  As well as the four choices listed | ||||||
|         here, you can also use any other hash supported by passlib, such as |         here, you can also use any other hash supported by passlib, such as | ||||||
|         md5_crypt and sha256_crypt, which are linux passwd hashes.  If you |         C(portable_apache22) and C(host_apache24); or C(md5_crypt) and C(sha256_crypt), | ||||||
|         do so the password file will not be compatible with Apache or Nginx |         which are Linux passwd hashes.  Only some schemes in addition to | ||||||
|       - 'Some of the available choices might be: C(apr_md5_crypt), C(des_crypt), C(ldap_sha1), C(plaintext)' |         the four choices below will be compatible with Apache or Nginx, and | ||||||
|  |         supported schemes depend on passlib version and its dependencies. | ||||||
|  |       - See U(https://passlib.readthedocs.io/en/stable/lib/passlib.apache.html#passlib.apache.HtpasswdFile) parameter C(default_scheme). | ||||||
|  |       - 'Some of the available choices might be: C(apr_md5_crypt), C(des_crypt), C(ldap_sha1), C(plaintext).' | ||||||
|   state: |   state: | ||||||
|     type: str |     type: str | ||||||
|     required: false |     required: false | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ibm_sa_host | module: ibm_sa_host | ||||||
| short_description: Adds hosts to or removes them from IBM Spectrum Accelerate Family storage systems. | short_description: Adds hosts to or removes them from IBM Spectrum Accelerate Family storage systems | ||||||
| 
 | 
 | ||||||
| description: | description: | ||||||
|     - "This module adds hosts to or removes them from IBM Spectrum Accelerate Family storage systems." |     - "This module adds hosts to or removes them from IBM Spectrum Accelerate Family storage systems." | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ibm_sa_host_ports | module: ibm_sa_host_ports | ||||||
| short_description: Add host ports on IBM Spectrum Accelerate Family storage systems. | short_description: Add host ports on IBM Spectrum Accelerate Family storage systems | ||||||
| 
 | 
 | ||||||
| description: | description: | ||||||
|     - "This module adds ports to or removes them from the hosts |     - "This module adds ports to or removes them from the hosts | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ibm_sa_pool | module: ibm_sa_pool | ||||||
| short_description: Handles pools on IBM Spectrum Accelerate Family storage systems. | short_description: Handles pools on IBM Spectrum Accelerate Family storage systems | ||||||
| 
 | 
 | ||||||
| description: | description: | ||||||
|     - "This module creates or deletes pools to be used on IBM Spectrum Accelerate Family storage systems" |     - "This module creates or deletes pools to be used on IBM Spectrum Accelerate Family storage systems" | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ibm_sa_vol | module: ibm_sa_vol | ||||||
| short_description: Handle volumes on IBM Spectrum Accelerate Family storage systems. | short_description: Handle volumes on IBM Spectrum Accelerate Family storage systems | ||||||
| 
 | 
 | ||||||
| description: | description: | ||||||
|     - "This module creates or deletes volumes to be used on IBM Spectrum Accelerate Family storage systems." |     - "This module creates or deletes volumes to be used on IBM Spectrum Accelerate Family storage systems." | ||||||
|  |  | ||||||
|  | @ -13,7 +13,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ibm_sa_vol_map | module: ibm_sa_vol_map | ||||||
| short_description: Handles volume mapping on IBM Spectrum Accelerate Family storage systems. | short_description: Handles volume mapping on IBM Spectrum Accelerate Family storage systems | ||||||
| 
 | 
 | ||||||
| description: | description: | ||||||
|     - "This module maps volumes to or unmaps them from the hosts on |     - "This module maps volumes to or unmaps them from the hosts on | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ DOCUMENTATION = r''' | ||||||
| --- | --- | ||||||
| module: ipa_subca | module: ipa_subca | ||||||
| author: Abhijeet Kasurde (@Akasurde) | author: Abhijeet Kasurde (@Akasurde) | ||||||
| short_description: Manage FreeIPA Lightweight Sub Certificate Authorities. | short_description: Manage FreeIPA Lightweight Sub Certificate Authorities | ||||||
| description: | description: | ||||||
| - Add, modify, enable, disable and delete an IPA Lightweight Sub Certificate Authorities using IPA API. | - Add, modify, enable, disable and delete an IPA Lightweight Sub Certificate Authorities using IPA API. | ||||||
| options: | options: | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ipinfoio_facts | module: ipinfoio_facts | ||||||
| short_description: "Retrieve IP geolocation facts of a host's IP address" | short_description: Retrieve IP geolocation facts of a host's IP address | ||||||
| description: | description: | ||||||
|   - "Gather IP geolocation facts of a host's IP address using ipinfo.io API" |   - "Gather IP geolocation facts of a host's IP address using ipinfo.io API" | ||||||
| author: "Aleksei Kostiuk (@akostyuk)" | author: "Aleksei Kostiuk (@akostyuk)" | ||||||
|  |  | ||||||
|  | @ -97,19 +97,14 @@ dest_iso: | ||||||
| ''' | ''' | ||||||
| 
 | 
 | ||||||
| import os | import os | ||||||
| import traceback |  | ||||||
| 
 | 
 | ||||||
| PYCDLIB_IMP_ERR = None | from ansible_collections.community.general.plugins.module_utils import deps | ||||||
| try: | from ansible.module_utils.basic import AnsibleModule | ||||||
|     import pycdlib |  | ||||||
|     HAS_PYCDLIB = True |  | ||||||
| except ImportError: |  | ||||||
|     PYCDLIB_IMP_ERR = traceback.format_exc() |  | ||||||
|     HAS_PYCDLIB = False |  | ||||||
| 
 |  | ||||||
| from ansible.module_utils.basic import AnsibleModule, missing_required_lib |  | ||||||
| from ansible.module_utils.common.text.converters import to_native | from ansible.module_utils.common.text.converters import to_native | ||||||
| 
 | 
 | ||||||
|  | with deps.declare("pycdlib"): | ||||||
|  |     import pycdlib | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| # The upper dir exist, we only add subdirectoy | # The upper dir exist, we only add subdirectoy | ||||||
| def iso_add_dir(module, opened_iso, iso_type, dir_path): | def iso_add_dir(module, opened_iso, iso_type, dir_path): | ||||||
|  | @ -306,9 +301,7 @@ def main(): | ||||||
|         required_one_of=[('delete_files', 'add_files'), ], |         required_one_of=[('delete_files', 'add_files'), ], | ||||||
|         supports_check_mode=True, |         supports_check_mode=True, | ||||||
|     ) |     ) | ||||||
|     if not HAS_PYCDLIB: |     deps.validate(module) | ||||||
|         module.fail_json( |  | ||||||
|             missing_required_lib('pycdlib'), exception=PYCDLIB_IMP_ERR) |  | ||||||
| 
 | 
 | ||||||
|     src_iso = module.params['src_iso'] |     src_iso = module.params['src_iso'] | ||||||
|     if not os.path.exists(src_iso): |     if not os.path.exists(src_iso): | ||||||
|  |  | ||||||
|  | @ -281,7 +281,8 @@ def _export_public_cert_from_pkcs12(module, executable, pkcs_file, alias, passwo | ||||||
|     (export_rc, export_stdout, export_err) = module.run_command(export_cmd, data=password, check_rc=False) |     (export_rc, export_stdout, export_err) = module.run_command(export_cmd, data=password, check_rc=False) | ||||||
| 
 | 
 | ||||||
|     if export_rc != 0: |     if export_rc != 0: | ||||||
|         module.fail_json(msg="Internal module failure, cannot extract public certificate from pkcs12, error: %s" % export_stdout, |         module.fail_json(msg="Internal module failure, cannot extract public certificate from PKCS12, message: %s" % export_stdout, | ||||||
|  |                          stderr=export_err, | ||||||
|                          rc=export_rc) |                          rc=export_rc) | ||||||
| 
 | 
 | ||||||
|     with open(dest, 'w') as f: |     with open(dest, 'w') as f: | ||||||
|  |  | ||||||
|  | @ -183,7 +183,10 @@ class JenkinsBuild: | ||||||
|         try: |         try: | ||||||
|             response = self.server.get_build_info(self.name, self.build_number) |             response = self.server.get_build_info(self.name, self.build_number) | ||||||
|             return response |             return response | ||||||
| 
 |         except jenkins.JenkinsException as e: | ||||||
|  |             response = {} | ||||||
|  |             response["result"] = "ABSENT" | ||||||
|  |             return response | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             self.module.fail_json(msg='Unable to fetch build information, %s' % to_native(e), |             self.module.fail_json(msg='Unable to fetch build information, %s' % to_native(e), | ||||||
|                                   exception=traceback.format_exc()) |                                   exception=traceback.format_exc()) | ||||||
|  | @ -231,7 +234,10 @@ class JenkinsBuild: | ||||||
|             if self.state == "stopped" and build_status['result'] == "ABORTED": |             if self.state == "stopped" and build_status['result'] == "ABORTED": | ||||||
|                 result['changed'] = True |                 result['changed'] = True | ||||||
|                 result['build_info'] = build_status |                 result['build_info'] = build_status | ||||||
|             elif build_status['result'] == "SUCCESS": |             elif self.state == "absent" and build_status['result'] == "ABSENT": | ||||||
|  |                 result['changed'] = True | ||||||
|  |                 result['build_info'] = build_status | ||||||
|  |             elif self.state != "absent" and build_status['result'] == "SUCCESS": | ||||||
|                 result['changed'] = True |                 result['changed'] = True | ||||||
|                 result['build_info'] = build_status |                 result['build_info'] = build_status | ||||||
|             else: |             else: | ||||||
|  |  | ||||||
|  | @ -290,12 +290,6 @@ state: | ||||||
|     sample: "present" |     sample: "present" | ||||||
| ''' | ''' | ||||||
| 
 | 
 | ||||||
| from ansible.module_utils.basic import AnsibleModule, to_bytes |  | ||||||
| from ansible.module_utils.six.moves import http_cookiejar as cookiejar |  | ||||||
| from ansible.module_utils.six.moves.urllib.parse import urlencode |  | ||||||
| from ansible.module_utils.urls import fetch_url, url_argument_spec |  | ||||||
| from ansible.module_utils.six import text_type, binary_type |  | ||||||
| from ansible.module_utils.common.text.converters import to_native |  | ||||||
| import hashlib | import hashlib | ||||||
| import io | import io | ||||||
| import json | import json | ||||||
|  | @ -303,6 +297,15 @@ import os | ||||||
| import tempfile | import tempfile | ||||||
| import time | import time | ||||||
| 
 | 
 | ||||||
|  | from ansible.module_utils.basic import AnsibleModule, to_bytes | ||||||
|  | from ansible.module_utils.six.moves import http_cookiejar as cookiejar | ||||||
|  | from ansible.module_utils.six.moves.urllib.parse import urlencode | ||||||
|  | from ansible.module_utils.urls import fetch_url, url_argument_spec | ||||||
|  | from ansible.module_utils.six import text_type, binary_type | ||||||
|  | from ansible.module_utils.common.text.converters import to_native | ||||||
|  | 
 | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.jenkins import download_updates_file | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| class FailedInstallingWithPluginManager(Exception): | class FailedInstallingWithPluginManager(Exception): | ||||||
|     pass |     pass | ||||||
|  | @ -605,21 +608,12 @@ class JenkinsPlugin(object): | ||||||
|         return urls |         return urls | ||||||
| 
 | 
 | ||||||
|     def _download_updates(self): |     def _download_updates(self): | ||||||
|         updates_filename = 'jenkins-plugin-cache.json' |         try: | ||||||
|         updates_dir = os.path.expanduser('~/.ansible/tmp') |             updates_file, download_updates = download_updates_file(self.params['updates_expiration']) | ||||||
|         updates_file = "%s/%s" % (updates_dir, updates_filename) |         except OSError as e: | ||||||
|         download_updates = True |             self.module.fail_json( | ||||||
| 
 |                 msg="Cannot create temporal directory.", | ||||||
|         # Check if we need to download new updates file |                 details=to_native(e)) | ||||||
|         if os.path.isfile(updates_file): |  | ||||||
|             # Get timestamp when the file was changed last time |  | ||||||
|             ts_file = os.stat(updates_file).st_mtime |  | ||||||
|             ts_now = time.time() |  | ||||||
| 
 |  | ||||||
|             if ts_now - ts_file < self.params['updates_expiration']: |  | ||||||
|                 download_updates = False |  | ||||||
| 
 |  | ||||||
|         updates_file_orig = updates_file |  | ||||||
| 
 | 
 | ||||||
|         # Download the updates file if needed |         # Download the updates file if needed | ||||||
|         if download_updates: |         if download_updates: | ||||||
|  | @ -632,56 +626,39 @@ class JenkinsPlugin(object): | ||||||
|                 msg_exception="Updates download failed.") |                 msg_exception="Updates download failed.") | ||||||
| 
 | 
 | ||||||
|             # Write the updates file |             # Write the updates file | ||||||
|             update_fd, updates_file = tempfile.mkstemp() |             tmp_update_fd, tmp_updates_file = tempfile.mkstemp() | ||||||
|             os.write(update_fd, r.read()) |             os.write(tmp_update_fd, r.read()) | ||||||
| 
 | 
 | ||||||
|             try: |             try: | ||||||
|                 os.close(update_fd) |                 os.close(tmp_update_fd) | ||||||
|             except IOError as e: |             except IOError as e: | ||||||
|                 self.module.fail_json( |                 self.module.fail_json( | ||||||
|                     msg="Cannot close the tmp updates file %s." % updates_file, |                     msg="Cannot close the tmp updates file %s." % tmp_updates_file, | ||||||
|                     details=to_native(e)) |                     details=to_native(e)) | ||||||
| 
 | 
 | ||||||
|         # Open the updates file |         # Open the updates file | ||||||
|         try: |         try: | ||||||
|             f = io.open(updates_file, encoding='utf-8') |             f = io.open(tmp_updates_file, encoding='utf-8') | ||||||
|  | 
 | ||||||
|  |             # Read only the second line | ||||||
|  |             dummy = f.readline() | ||||||
|  |             data = json.loads(f.readline()) | ||||||
|         except IOError as e: |         except IOError as e: | ||||||
|             self.module.fail_json( |             self.module.fail_json( | ||||||
|                 msg="Cannot open temporal updates file.", |                 msg="Cannot open temporal updates file.", | ||||||
|                 details=to_native(e)) |                 details=to_native(e)) | ||||||
| 
 |  | ||||||
|         i = 0 |  | ||||||
|         for line in f: |  | ||||||
|             # Read only the second line |  | ||||||
|             if i == 1: |  | ||||||
|                 try: |  | ||||||
|                     data = json.loads(line) |  | ||||||
|         except Exception as e: |         except Exception as e: | ||||||
|             self.module.fail_json( |             self.module.fail_json( | ||||||
|                 msg="Cannot load JSON data from the tmp updates file.", |                 msg="Cannot load JSON data from the tmp updates file.", | ||||||
|                 details=to_native(e)) |                 details=to_native(e)) | ||||||
| 
 | 
 | ||||||
|                 break |  | ||||||
| 
 |  | ||||||
|             i += 1 |  | ||||||
| 
 |  | ||||||
|         # Move the updates file to the right place if we could read it |         # Move the updates file to the right place if we could read it | ||||||
|         if download_updates: |         if download_updates: | ||||||
|             # Make sure the destination directory exists |             self.module.atomic_move(tmp_updates_file, updates_file) | ||||||
|             if not os.path.isdir(updates_dir): |  | ||||||
|                 try: |  | ||||||
|                     os.makedirs(updates_dir, int('0700', 8)) |  | ||||||
|                 except OSError as e: |  | ||||||
|                     self.module.fail_json( |  | ||||||
|                         msg="Cannot create temporal directory.", |  | ||||||
|                         details=to_native(e)) |  | ||||||
| 
 |  | ||||||
|             self.module.atomic_move(updates_file, updates_file_orig) |  | ||||||
| 
 | 
 | ||||||
|         # Check if we have the plugin data available |         # Check if we have the plugin data available | ||||||
|         if 'plugins' not in data or self.params['name'] not in data['plugins']: |         if not data.get('plugins', {}).get(self.params['name']): | ||||||
|             self.module.fail_json( |             self.module.fail_json(msg="Cannot find plugin data in the updates file.") | ||||||
|                 msg="Cannot find plugin data in the updates file.") |  | ||||||
| 
 | 
 | ||||||
|         return data['plugins'][self.params['name']] |         return data['plugins'][self.params['name']] | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -16,7 +16,7 @@ __metaclass__ = type | ||||||
| 
 | 
 | ||||||
| DOCUMENTATION = r""" | DOCUMENTATION = r""" | ||||||
| module: jira | module: jira | ||||||
| short_description: create and modify issues in a JIRA instance | short_description: Create and modify issues in a JIRA instance | ||||||
| description: | description: | ||||||
|   - Create and modify issues in a JIRA instance. |   - Create and modify issues in a JIRA instance. | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -295,7 +295,7 @@ def main(): | ||||||
|     assigned_roles_before = kc.get_client_group_composite_rolemappings(gid, cid, realm=realm) |     assigned_roles_before = kc.get_client_group_composite_rolemappings(gid, cid, realm=realm) | ||||||
| 
 | 
 | ||||||
|     result['existing'] = assigned_roles_before |     result['existing'] = assigned_roles_before | ||||||
|     result['proposed'] = roles |     result['proposed'] = list(assigned_roles_before) if assigned_roles_before else [] | ||||||
| 
 | 
 | ||||||
|     update_roles = [] |     update_roles = [] | ||||||
|     for role_index, role in enumerate(roles, start=0): |     for role_index, role in enumerate(roles, start=0): | ||||||
|  | @ -307,6 +307,7 @@ def main(): | ||||||
|                         'id': role['id'], |                         'id': role['id'], | ||||||
|                         'name': role['name'], |                         'name': role['name'], | ||||||
|                     }) |                     }) | ||||||
|  |                     result['proposed'].append(available_role) | ||||||
|         # Fetch roles to remove if state absent |         # Fetch roles to remove if state absent | ||||||
|         else: |         else: | ||||||
|             for assigned_role in assigned_roles_before: |             for assigned_role in assigned_roles_before: | ||||||
|  | @ -315,13 +316,15 @@ def main(): | ||||||
|                         'id': role['id'], |                         'id': role['id'], | ||||||
|                         'name': role['name'], |                         'name': role['name'], | ||||||
|                     }) |                     }) | ||||||
|  |                     if assigned_role in result['proposed']:  # Handle double removal | ||||||
|  |                         result['proposed'].remove(assigned_role) | ||||||
| 
 | 
 | ||||||
|     if len(update_roles): |     if len(update_roles): | ||||||
|         if state == 'present': |         if state == 'present': | ||||||
|             # Assign roles |             # Assign roles | ||||||
|             result['changed'] = True |             result['changed'] = True | ||||||
|             if module._diff: |             if module._diff: | ||||||
|                 result['diff'] = dict(before=assigned_roles_before, after=update_roles) |                 result['diff'] = dict(before=assigned_roles_before, after=result['proposed']) | ||||||
|             if module.check_mode: |             if module.check_mode: | ||||||
|                 module.exit_json(**result) |                 module.exit_json(**result) | ||||||
|             kc.add_group_rolemapping(gid, cid, update_roles, realm=realm) |             kc.add_group_rolemapping(gid, cid, update_roles, realm=realm) | ||||||
|  | @ -333,7 +336,7 @@ def main(): | ||||||
|             # Remove mapping of role |             # Remove mapping of role | ||||||
|             result['changed'] = True |             result['changed'] = True | ||||||
|             if module._diff: |             if module._diff: | ||||||
|                 result['diff'] = dict(before=assigned_roles_before, after=update_roles) |                 result['diff'] = dict(before=assigned_roles_before, after=result['proposed']) | ||||||
|             if module.check_mode: |             if module.check_mode: | ||||||
|                 module.exit_json(**result) |                 module.exit_json(**result) | ||||||
|             kc.delete_group_rolemapping(gid, cid, update_roles, realm=realm) |             kc.delete_group_rolemapping(gid, cid, update_roles, realm=realm) | ||||||
|  | @ -344,7 +347,7 @@ def main(): | ||||||
|     # Do nothing |     # Do nothing | ||||||
|     else: |     else: | ||||||
|         result['changed'] = False |         result['changed'] = False | ||||||
|         result['msg'] = 'Nothing to do, roles %s are correctly mapped with group %s.' % (roles, group_name) |         result['msg'] = 'Nothing to do, roles %s are %s with group %s.' % (roles, 'mapped' if state == 'present' else 'not mapped', group_name) | ||||||
|         module.exit_json(**result) |         module.exit_json(**result) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										161
									
								
								plugins/modules/keycloak_clientsecret_info.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										161
									
								
								plugins/modules/keycloak_clientsecret_info.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,161 @@ | ||||||
|  | #!/usr/bin/python | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | 
 | ||||||
|  | # Copyright (c) 2022, Fynn Chen <ethan.cfchen@gmail.com> | ||||||
|  | # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | 
 | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | DOCUMENTATION = ''' | ||||||
|  | --- | ||||||
|  | module: keycloak_clientsecret_info | ||||||
|  | 
 | ||||||
|  | short_description: Retrieve client secret via Keycloak API | ||||||
|  | 
 | ||||||
|  | version_added: 6.1.0 | ||||||
|  | 
 | ||||||
|  | description: | ||||||
|  |   - This module allows you to get a Keycloak client secret via the Keycloak | ||||||
|  |     REST API. It requires access to the REST API via OpenID Connect; the user | ||||||
|  |     connecting and the client being used must have the requisite access rights. | ||||||
|  |     In a default Keycloak installation, admin-cli and an admin user would work, | ||||||
|  |     as would a separate client definition with the scope tailored to your needs | ||||||
|  |     and a user having the expected roles. | ||||||
|  | 
 | ||||||
|  |   - When retrieving a new client secret, where possible provide the client's | ||||||
|  |     I(id) (not I(client_id)) to the module. This removes a lookup to the API to | ||||||
|  |     translate the I(client_id) into the client ID. | ||||||
|  | 
 | ||||||
|  |   - "Note that this module returns the client secret. To avoid this showing up in the logs, | ||||||
|  |      please add C(no_log: true) to the task." | ||||||
|  | 
 | ||||||
|  | options: | ||||||
|  |   realm: | ||||||
|  |     type: str | ||||||
|  |     description: | ||||||
|  |       - They Keycloak realm under which this client resides. | ||||||
|  |     default: 'master' | ||||||
|  | 
 | ||||||
|  |   id: | ||||||
|  |     description: | ||||||
|  |       - The unique identifier for this client. | ||||||
|  |       - This parameter is not required for getting or generating a client secret but | ||||||
|  |         providing it will reduce the number of API calls required. | ||||||
|  |     type: str | ||||||
|  | 
 | ||||||
|  |   client_id: | ||||||
|  |     description: | ||||||
|  |       - The I(client_id) of the client. Passing this instead of I(id) results in an | ||||||
|  |         extra API call. | ||||||
|  |     aliases: | ||||||
|  |       - clientId | ||||||
|  |     type: str | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | extends_documentation_fragment: | ||||||
|  |   - community.general.keycloak | ||||||
|  |   - community.general.attributes | ||||||
|  |   - community.general.attributes.info_module | ||||||
|  | 
 | ||||||
|  | author: | ||||||
|  |   - Fynn Chen (@fynncfchen) | ||||||
|  |   - John Cant (@johncant) | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | EXAMPLES = ''' | ||||||
|  | - name: Get a Keycloak client secret, authentication with credentials | ||||||
|  |   community.general.keycloak_clientsecret_info: | ||||||
|  |     id: '9d59aa76-2755-48c6-b1af-beb70a82c3cd' | ||||||
|  |     realm: MyCustomRealm | ||||||
|  |     auth_client_id: admin-cli | ||||||
|  |     auth_keycloak_url: https://auth.example.com/auth | ||||||
|  |     auth_realm: master | ||||||
|  |     auth_username: USERNAME | ||||||
|  |     auth_password: PASSWORD | ||||||
|  |   delegate_to: localhost | ||||||
|  |   no_log: true | ||||||
|  | 
 | ||||||
|  | - name: Get a new Keycloak client secret, authentication with token | ||||||
|  |   community.general.keycloak_clientsecret_info: | ||||||
|  |     id: '9d59aa76-2755-48c6-b1af-beb70a82c3cd' | ||||||
|  |     realm: MyCustomRealm | ||||||
|  |     auth_client_id: admin-cli | ||||||
|  |     auth_keycloak_url: https://auth.example.com/auth | ||||||
|  |     token: TOKEN | ||||||
|  |   delegate_to: localhost | ||||||
|  |   no_log: true | ||||||
|  | 
 | ||||||
|  | - name: Get a new Keycloak client secret, passing client_id instead of id | ||||||
|  |   community.general.keycloak_clientsecret_info: | ||||||
|  |     client_id: 'myClientId' | ||||||
|  |     realm: MyCustomRealm | ||||||
|  |     auth_client_id: admin-cli | ||||||
|  |     auth_keycloak_url: https://auth.example.com/auth | ||||||
|  |     token: TOKEN | ||||||
|  |   delegate_to: localhost | ||||||
|  |   no_log: true | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | RETURN = ''' | ||||||
|  | msg: | ||||||
|  |   description: Textual description of whether we succeeded or failed | ||||||
|  |   returned: always | ||||||
|  |   type: str | ||||||
|  | 
 | ||||||
|  | clientsecret_info: | ||||||
|  |   description: Representation of the client secret | ||||||
|  |   returned: on success | ||||||
|  |   type: complex | ||||||
|  |   contains: | ||||||
|  |     type: | ||||||
|  |       description: Credential type. | ||||||
|  |       type: str | ||||||
|  |       returned: always | ||||||
|  |       sample: secret | ||||||
|  |     value: | ||||||
|  |       description: Client secret. | ||||||
|  |       type: str | ||||||
|  |       returned: always | ||||||
|  |       sample: cUGnX1EIeTtPPAkcyGMv0ncyqDPu68P1 | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.identity.keycloak.keycloak import ( | ||||||
|  |     KeycloakAPI, KeycloakError, get_token) | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.identity.keycloak.keycloak_clientsecret import ( | ||||||
|  |     keycloak_clientsecret_module, keycloak_clientsecret_module_resolve_params) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def main(): | ||||||
|  |     """ | ||||||
|  |     Module keycloak_clientsecret_info | ||||||
|  | 
 | ||||||
|  |     :return: | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|  |     module = keycloak_clientsecret_module() | ||||||
|  | 
 | ||||||
|  |     # Obtain access token, initialize API | ||||||
|  |     try: | ||||||
|  |         connection_header = get_token(module.params) | ||||||
|  |     except KeycloakError as e: | ||||||
|  |         module.fail_json(msg=str(e)) | ||||||
|  | 
 | ||||||
|  |     kc = KeycloakAPI(module, connection_header) | ||||||
|  | 
 | ||||||
|  |     id, realm = keycloak_clientsecret_module_resolve_params(module, kc) | ||||||
|  | 
 | ||||||
|  |     clientsecret = kc.get_clientsecret(id=id, realm=realm) | ||||||
|  | 
 | ||||||
|  |     result = { | ||||||
|  |         'clientsecret_info': clientsecret, | ||||||
|  |         'msg': 'Get client secret successful for ID {id}'.format(id=id) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     module.exit_json(**result) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     main() | ||||||
							
								
								
									
										167
									
								
								plugins/modules/keycloak_clientsecret_regenerate.py
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										167
									
								
								plugins/modules/keycloak_clientsecret_regenerate.py
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,167 @@ | ||||||
|  | #!/usr/bin/python | ||||||
|  | # -*- coding: utf-8 -*- | ||||||
|  | 
 | ||||||
|  | # Copyright (c) 2022, Fynn Chen <ethan.cfchen@gmail.com> | ||||||
|  | # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) | ||||||
|  | # SPDX-License-Identifier: GPL-3.0-or-later | ||||||
|  | 
 | ||||||
|  | from __future__ import absolute_import, division, print_function | ||||||
|  | 
 | ||||||
|  | __metaclass__ = type | ||||||
|  | 
 | ||||||
|  | DOCUMENTATION = ''' | ||||||
|  | --- | ||||||
|  | module: keycloak_clientsecret_regenerate | ||||||
|  | 
 | ||||||
|  | short_description: Regenerate Keycloak client secret via Keycloak API | ||||||
|  | 
 | ||||||
|  | version_added: 6.1.0 | ||||||
|  | 
 | ||||||
|  | description: | ||||||
|  |   - This module allows you to regenerate a Keycloak client secret via the | ||||||
|  |     Keycloak REST API. It requires access to the REST API via OpenID Connect; | ||||||
|  |     the user connecting and the client being used must have the requisite access | ||||||
|  |     rights. In a default Keycloak installation, admin-cli and an admin user | ||||||
|  |     would work, as would a separate client definition with the scope tailored to | ||||||
|  |     your needs and a user having the expected roles. | ||||||
|  | 
 | ||||||
|  |   - When regenerating a client secret, where possible provide the client's id | ||||||
|  |     (not client_id) to the module. This removes a lookup to the API to | ||||||
|  |     translate the client_id into the client ID. | ||||||
|  | 
 | ||||||
|  |   - "Note that this module returns the client secret. To avoid this showing up in the logs, | ||||||
|  |      please add C(no_log: true) to the task." | ||||||
|  | 
 | ||||||
|  | options: | ||||||
|  |   realm: | ||||||
|  |     type: str | ||||||
|  |     description: | ||||||
|  |       - They Keycloak realm under which this client resides. | ||||||
|  |     default: 'master' | ||||||
|  | 
 | ||||||
|  |   id: | ||||||
|  |     description: | ||||||
|  |       - The unique identifier for this client. | ||||||
|  |       - This parameter is not required for getting or generating a client secret but | ||||||
|  |         providing it will reduce the number of API calls required. | ||||||
|  |     type: str | ||||||
|  | 
 | ||||||
|  |   client_id: | ||||||
|  |     description: | ||||||
|  |       - The client_id of the client. Passing this instead of id results in an | ||||||
|  |         extra API call. | ||||||
|  |     aliases: | ||||||
|  |       - clientId | ||||||
|  |     type: str | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | extends_documentation_fragment: | ||||||
|  |   - community.general.keycloak | ||||||
|  | 
 | ||||||
|  | author: | ||||||
|  |   - Fynn Chen (@fynncfchen) | ||||||
|  |   - John Cant (@johncant) | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | EXAMPLES = ''' | ||||||
|  | - name: Regenerate a Keycloak client secret, authentication with credentials | ||||||
|  |   community.general.keycloak_clientsecret_regenerate: | ||||||
|  |     id: '9d59aa76-2755-48c6-b1af-beb70a82c3cd' | ||||||
|  |     realm: MyCustomRealm | ||||||
|  |     auth_client_id: admin-cli | ||||||
|  |     auth_keycloak_url: https://auth.example.com/auth | ||||||
|  |     auth_realm: master | ||||||
|  |     auth_username: USERNAME | ||||||
|  |     auth_password: PASSWORD | ||||||
|  |   delegate_to: localhost | ||||||
|  |   no_log: true | ||||||
|  | 
 | ||||||
|  | - name: Regenerate a Keycloak client secret, authentication with token | ||||||
|  |   community.general.keycloak_clientsecret_regenerate: | ||||||
|  |     id: '9d59aa76-2755-48c6-b1af-beb70a82c3cd' | ||||||
|  |     realm: MyCustomRealm | ||||||
|  |     auth_client_id: admin-cli | ||||||
|  |     auth_keycloak_url: https://auth.example.com/auth | ||||||
|  |     token: TOKEN | ||||||
|  |   delegate_to: localhost | ||||||
|  |   no_log: true | ||||||
|  | 
 | ||||||
|  | - name: Regenerate a Keycloak client secret, passing client_id instead of id | ||||||
|  |   community.general.keycloak_clientsecret_info: | ||||||
|  |     client_id: 'myClientId' | ||||||
|  |     realm: MyCustomRealm | ||||||
|  |     auth_client_id: admin-cli | ||||||
|  |     auth_keycloak_url: https://auth.example.com/auth | ||||||
|  |     token: TOKEN | ||||||
|  |   delegate_to: localhost | ||||||
|  |   no_log: true | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | RETURN = ''' | ||||||
|  | msg: | ||||||
|  |   description: Message as to what action was taken. | ||||||
|  |   returned: always | ||||||
|  |   type: str | ||||||
|  | 
 | ||||||
|  | end_state: | ||||||
|  |   description: Representation of the client credential after module execution | ||||||
|  |   returned: on success | ||||||
|  |   type: complex | ||||||
|  |   contains: | ||||||
|  |     type: | ||||||
|  |       description: Credential type. | ||||||
|  |       type: str | ||||||
|  |       returned: always | ||||||
|  |       sample: secret | ||||||
|  |     value: | ||||||
|  |       description: Client secret. | ||||||
|  |       type: str | ||||||
|  |       returned: always | ||||||
|  |       sample: cUGnX1EIeTtPPAkcyGMv0ncyqDPu68P1 | ||||||
|  | 
 | ||||||
|  | ''' | ||||||
|  | 
 | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.identity.keycloak.keycloak import ( | ||||||
|  |     KeycloakAPI, KeycloakError, get_token) | ||||||
|  | from ansible_collections.community.general.plugins.module_utils.identity.keycloak.keycloak_clientsecret import ( | ||||||
|  |     keycloak_clientsecret_module, keycloak_clientsecret_module_resolve_params) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | def main(): | ||||||
|  |     """ | ||||||
|  |     Module keycloak_clientsecret_regenerate | ||||||
|  | 
 | ||||||
|  |     :return: | ||||||
|  |     """ | ||||||
|  | 
 | ||||||
|  |     module = keycloak_clientsecret_module() | ||||||
|  | 
 | ||||||
|  |     # Obtain access token, initialize API | ||||||
|  |     try: | ||||||
|  |         connection_header = get_token(module.params) | ||||||
|  |     except KeycloakError as e: | ||||||
|  |         module.fail_json(msg=str(e)) | ||||||
|  | 
 | ||||||
|  |     kc = KeycloakAPI(module, connection_header) | ||||||
|  | 
 | ||||||
|  |     id, realm = keycloak_clientsecret_module_resolve_params(module, kc) | ||||||
|  | 
 | ||||||
|  |     if module.check_mode: | ||||||
|  |         dummy_result = { | ||||||
|  |             "msg": 'No action taken while in check mode', | ||||||
|  |             "end_state": {'type': 'secret', 'value': 'X' * 32} | ||||||
|  |         } | ||||||
|  |         module.exit_json(**dummy_result) | ||||||
|  | 
 | ||||||
|  |     # Create new secret | ||||||
|  |     clientsecret = kc.create_clientsecret(id=id, realm=realm) | ||||||
|  | 
 | ||||||
|  |     result = { | ||||||
|  |         "msg": 'New client secret has been generated for ID {id}'.format(id=id), | ||||||
|  |         "end_state": clientsecret | ||||||
|  |     } | ||||||
|  |     module.exit_json(**result) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | if __name__ == '__main__': | ||||||
|  |     main() | ||||||
|  | @ -519,6 +519,7 @@ EXAMPLES = ''' | ||||||
|     auth_username: USERNAME |     auth_username: USERNAME | ||||||
|     auth_password: PASSWORD |     auth_password: PASSWORD | ||||||
|     id: realm |     id: realm | ||||||
|  |     realm: realm | ||||||
|     state: present |     state: present | ||||||
| 
 | 
 | ||||||
| - name: Delete a Keycloak realm | - name: Delete a Keycloak realm | ||||||
|  |  | ||||||
|  | @ -14,7 +14,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ldap_entry | module: ldap_entry | ||||||
| short_description: Add or remove LDAP entries. | short_description: Add or remove LDAP entries | ||||||
| description: | description: | ||||||
|   - Add or remove LDAP entries. This module only asserts the existence or |   - Add or remove LDAP entries. This module only asserts the existence or | ||||||
|     non-existence of an LDAP entry, not its attributes. To assert the |     non-existence of an LDAP entry, not its attributes. To assert the | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: ldap_passwd | module: ldap_passwd | ||||||
| short_description: Set passwords in LDAP. | short_description: Set passwords in LDAP | ||||||
| description: | description: | ||||||
|   - Set a password for an LDAP entry.  This module only asserts that |   - Set a password for an LDAP entry.  This module only asserts that | ||||||
|     a given password is valid for a given entry.  To assert the |     a given password is valid for a given entry.  To assert the | ||||||
|  |  | ||||||
|  | @ -12,7 +12,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: librato_annotation | module: librato_annotation | ||||||
| short_description: create an annotation in librato | short_description: Create an annotation in librato | ||||||
| description: | description: | ||||||
|     - Create an annotation event on the given annotation stream :name. If the annotation stream does not exist, it will be created automatically |     - Create an annotation event on the given annotation stream :name. If the annotation stream does not exist, it will be created automatically | ||||||
| author: "Seth Edwards (@Sedward)" | author: "Seth Edwards (@Sedward)" | ||||||
|  |  | ||||||
|  | @ -11,7 +11,7 @@ __metaclass__ = type | ||||||
| DOCUMENTATION = ''' | DOCUMENTATION = ''' | ||||||
| --- | --- | ||||||
| module: linode_v4 | module: linode_v4 | ||||||
| short_description: Manage instances on the Linode cloud. | short_description: Manage instances on the Linode cloud | ||||||
| description: Manage instances on the Linode cloud. | description: Manage instances on the Linode cloud. | ||||||
| requirements: | requirements: | ||||||
|   - python >= 2.7 |   - python >= 2.7 | ||||||
|  |  | ||||||
Some files were not shown because too many files have changed in this diff Show more
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue