mirror of
				https://github.com/ansible-collections/community.general.git
				synced 2025-10-24 21:14:00 -07:00 
			
		
		
		
	
		
			Some checks are pending
		
		
	
	EOL CI / EOL Sanity (Ⓐ2.17) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.10) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.12) (push) Waiting to run
				
			EOL CI / EOL Units (Ⓐ2.17+py3.7) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+alpine319+py:azp/posix/3/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+fedora39+py:azp/posix/3/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/1/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/2/) (push) Waiting to run
				
			EOL CI / EOL I (Ⓐ2.17+ubuntu2004+py:azp/posix/3/) (push) Waiting to run
				
			nox / Run extra sanity tests (push) Waiting to run
				
			* Adjust all __future__ imports: for i in $(grep -REl "__future__.*absolute_import" plugins/ tests/); do sed -e 's/from __future__ import .*/from __future__ import annotations/g' -i $i; done * Remove all UTF-8 encoding specifications for Python source files: for i in $(grep -REl '[-][*]- coding: utf-8 -[*]-' plugins/ tests/); do sed -e '/^# -\*- coding: utf-8 -\*-/d' -i $i; done * Remove __metaclass__ = type: for i in $(grep -REl '__metaclass__ = type' plugins/ tests/); do sed -e '/^__metaclass__ = type/d' -i $i; done
		
			
				
	
	
		
			276 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			276 lines
		
	
	
	
		
			8.1 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
| #!/usr/bin/python
 | |
| 
 | |
| # Copyright (c) 2022, Alexander Hussey <ahussey@redhat.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
 | |
| """
 | |
| Ansible Module - community.general.keyring
 | |
| """
 | |
| 
 | |
| from __future__ import annotations
 | |
| 
 | |
| 
 | |
| DOCUMENTATION = r"""
 | |
| module: keyring
 | |
| version_added: 5.2.0
 | |
| author:
 | |
|   - Alexander Hussey (@ahussey-redhat)
 | |
| short_description: Set or delete a passphrase using the Operating System's native keyring
 | |
| description: >-
 | |
|   This module uses the L(keyring Python library, https://pypi.org/project/keyring/) to set or delete passphrases for a given
 | |
|   service and username from the OS' native keyring.
 | |
| requirements:
 | |
|   - keyring (Python library)
 | |
|   - gnome-keyring (application - required for headless Gnome keyring access)
 | |
|   - dbus-run-session (application - required for headless Gnome keyring access)
 | |
| extends_documentation_fragment:
 | |
|   - community.general.attributes
 | |
| attributes:
 | |
|   check_mode:
 | |
|     support: full
 | |
|   diff_mode:
 | |
|     support: none
 | |
| options:
 | |
|   service:
 | |
|     description: The name of the service.
 | |
|     required: true
 | |
|     type: str
 | |
|   username:
 | |
|     description: The user belonging to the service.
 | |
|     required: true
 | |
|     type: str
 | |
|   user_password:
 | |
|     description: The password to set.
 | |
|     required: false
 | |
|     type: str
 | |
|     aliases:
 | |
|       - password
 | |
|   keyring_password:
 | |
|     description: Password to unlock keyring.
 | |
|     required: true
 | |
|     type: str
 | |
|   state:
 | |
|     description: Whether the password should exist.
 | |
|     required: false
 | |
|     default: present
 | |
|     type: str
 | |
|     choices:
 | |
|       - present
 | |
|       - absent
 | |
| """
 | |
| 
 | |
| EXAMPLES = r"""
 | |
| - name: Set a password for test/test1
 | |
|   community.general.keyring:
 | |
|     service: test
 | |
|     username: test1
 | |
|     user_password: "{{ user_password }}"
 | |
|     keyring_password: "{{ keyring_password }}"
 | |
| 
 | |
| - name: Delete the password for test/test1
 | |
|   community.general.keyring:
 | |
|     service: test
 | |
|     username: test1
 | |
|     user_password: "{{ user_password }}"
 | |
|     keyring_password: "{{ keyring_password }}"
 | |
|     state: absent
 | |
| """
 | |
| 
 | |
| try:
 | |
|     from shlex import quote
 | |
| except ImportError:
 | |
|     from pipes import quote
 | |
| import traceback
 | |
| 
 | |
| from ansible.module_utils.basic import AnsibleModule, missing_required_lib
 | |
| 
 | |
| try:
 | |
|     import keyring
 | |
| 
 | |
|     HAS_KEYRING = True
 | |
|     KEYRING_IMP_ERR = None
 | |
| except ImportError:
 | |
|     HAS_KEYRING = False
 | |
|     KEYRING_IMP_ERR = traceback.format_exc()
 | |
| 
 | |
| 
 | |
| def del_passphrase(module):
 | |
|     """
 | |
|     Attempt to delete a passphrase in the keyring using the Python API and fallback to using a shell.
 | |
|     """
 | |
|     if module.check_mode:
 | |
|         return None
 | |
|     try:
 | |
|         keyring.delete_password(module.params["service"], module.params["username"])
 | |
|         return None
 | |
|     except keyring.errors.KeyringLocked:
 | |
|         delete_argument = (
 | |
|             'echo "%s" | gnome-keyring-daemon --unlock\nkeyring del %s %s\n'
 | |
|             % (
 | |
|                 quote(module.params["keyring_password"]),
 | |
|                 quote(module.params["service"]),
 | |
|                 quote(module.params["username"]),
 | |
|             )
 | |
|         )
 | |
|         dummy, dummy, stderr = module.run_command(
 | |
|             "dbus-run-session -- /bin/bash",
 | |
|             use_unsafe_shell=True,
 | |
|             data=delete_argument,
 | |
|             encoding=None,
 | |
|         )
 | |
| 
 | |
|         if not stderr.decode("UTF-8"):
 | |
|             return None
 | |
|         return stderr.decode("UTF-8")
 | |
| 
 | |
| 
 | |
| def set_passphrase(module):
 | |
|     """
 | |
|     Attempt to set passphrase in the keyring using the Python API and fallback to using a shell.
 | |
|     """
 | |
|     if module.check_mode:
 | |
|         return None
 | |
|     try:
 | |
|         keyring.set_password(
 | |
|             module.params["service"],
 | |
|             module.params["username"],
 | |
|             module.params["user_password"],
 | |
|         )
 | |
|         return None
 | |
|     except keyring.errors.KeyringLocked:
 | |
|         set_argument = (
 | |
|             'echo "%s" | gnome-keyring-daemon --unlock\nkeyring set %s %s\n%s\n'
 | |
|             % (
 | |
|                 quote(module.params["keyring_password"]),
 | |
|                 quote(module.params["service"]),
 | |
|                 quote(module.params["username"]),
 | |
|                 quote(module.params["user_password"]),
 | |
|             )
 | |
|         )
 | |
|         dummy, dummy, stderr = module.run_command(
 | |
|             "dbus-run-session -- /bin/bash",
 | |
|             use_unsafe_shell=True,
 | |
|             data=set_argument,
 | |
|             encoding=None,
 | |
|         )
 | |
|         if not stderr.decode("UTF-8"):
 | |
|             return None
 | |
|         return stderr.decode("UTF-8")
 | |
| 
 | |
| 
 | |
| def get_passphrase(module):
 | |
|     """
 | |
|     Attempt to retrieve passphrase from keyring using the Python API and fallback to using a shell.
 | |
|     """
 | |
|     try:
 | |
|         passphrase = keyring.get_password(
 | |
|             module.params["service"], module.params["username"]
 | |
|         )
 | |
|         return passphrase
 | |
|     except keyring.errors.KeyringLocked:
 | |
|         pass
 | |
|     except keyring.errors.InitError:
 | |
|         pass
 | |
|     except AttributeError:
 | |
|         pass
 | |
|     get_argument = 'echo "%s" | gnome-keyring-daemon --unlock\nkeyring get %s %s\n' % (
 | |
|         quote(module.params["keyring_password"]),
 | |
|         quote(module.params["service"]),
 | |
|         quote(module.params["username"]),
 | |
|     )
 | |
|     dummy, stdout, dummy = module.run_command(
 | |
|         "dbus-run-session -- /bin/bash",
 | |
|         use_unsafe_shell=True,
 | |
|         data=get_argument,
 | |
|         encoding=None,
 | |
|     )
 | |
|     try:
 | |
|         return stdout.decode("UTF-8").splitlines()[1]  # Only return the line containing the password
 | |
|     except IndexError:
 | |
|         return None
 | |
| 
 | |
| 
 | |
| def run_module():
 | |
|     """
 | |
|     Attempts to retrieve a passphrase from a keyring.
 | |
|     """
 | |
|     result = dict(
 | |
|         changed=False,
 | |
|         msg="",
 | |
|     )
 | |
| 
 | |
|     module_args = dict(
 | |
|         service=dict(type="str", required=True),
 | |
|         username=dict(type="str", required=True),
 | |
|         keyring_password=dict(type="str", required=True, no_log=True),
 | |
|         user_password=dict(
 | |
|             type="str", no_log=True, aliases=["password"]
 | |
|         ),
 | |
|         state=dict(
 | |
|             type="str", default="present", choices=["absent", "present"]
 | |
|         ),
 | |
|     )
 | |
| 
 | |
|     module = AnsibleModule(argument_spec=module_args, supports_check_mode=True)
 | |
| 
 | |
|     if not HAS_KEYRING:
 | |
|         module.fail_json(msg=missing_required_lib("keyring"), exception=KEYRING_IMP_ERR)
 | |
| 
 | |
|     passphrase = get_passphrase(module)
 | |
|     if module.params["state"] == "present":
 | |
|         if passphrase is not None:
 | |
|             if passphrase == module.params["user_password"]:
 | |
|                 result["msg"] = "Passphrase already set for %s@%s" % (
 | |
|                     module.params["service"],
 | |
|                     module.params["username"],
 | |
|                 )
 | |
|             if passphrase != module.params["user_password"]:
 | |
|                 set_result = set_passphrase(module)
 | |
|                 if set_result is None:
 | |
|                     result["changed"] = True
 | |
|                     result["msg"] = "Passphrase has been updated for %s@%s" % (
 | |
|                         module.params["service"],
 | |
|                         module.params["username"],
 | |
|                     )
 | |
|                 if set_result is not None:
 | |
|                     module.fail_json(msg=set_result)
 | |
|         if passphrase is None:
 | |
|             set_result = set_passphrase(module)
 | |
|             if set_result is None:
 | |
|                 result["changed"] = True
 | |
|                 result["msg"] = "Passphrase has been updated for %s@%s" % (
 | |
|                     module.params["service"],
 | |
|                     module.params["username"],
 | |
|                 )
 | |
|             if set_result is not None:
 | |
|                 module.fail_json(msg=set_result)
 | |
| 
 | |
|     if module.params["state"] == "absent":
 | |
|         if not passphrase:
 | |
|             result["result"] = "Passphrase already absent for %s@%s" % (
 | |
|                 module.params["service"],
 | |
|                 module.params["username"],
 | |
|             )
 | |
|         if passphrase:
 | |
|             del_result = del_passphrase(module)
 | |
|             if del_result is None:
 | |
|                 result["changed"] = True
 | |
|                 result["msg"] = "Passphrase has been removed for %s@%s" % (
 | |
|                     module.params["service"],
 | |
|                     module.params["username"],
 | |
|                 )
 | |
|             if del_result is not None:
 | |
|                 module.fail_json(msg=del_result)
 | |
| 
 | |
|     module.exit_json(**result)
 | |
| 
 | |
| 
 | |
| def main():
 | |
|     """
 | |
|     main module loop
 | |
|     """
 | |
|     run_module()
 | |
| 
 | |
| 
 | |
| if __name__ == "__main__":
 | |
|     main()
 |