2022-04-03 11:04:27 +02:00
|
|
|
#!/usr/bin/python
|
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
2023-10-19 11:10:04 +02:00
|
|
|
# Copyright (c) 2017-2018, Keller Fuchs <kellerfuchs@hashbang.sh>
|
|
|
|
# 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
|
2022-04-03 11:04:27 +02:00
|
|
|
|
|
|
|
from __future__ import absolute_import, division, print_function
|
|
|
|
__metaclass__ = type
|
|
|
|
|
|
|
|
|
|
|
|
DOCUMENTATION = '''
|
|
|
|
---
|
|
|
|
module: ldap_passwd
|
2023-10-19 11:10:04 +02:00
|
|
|
short_description: Set passwords in LDAP
|
2022-04-03 11:04:27 +02:00
|
|
|
description:
|
|
|
|
- Set a password for an LDAP entry. This module only asserts that
|
|
|
|
a given password is valid for a given entry. To assert the
|
|
|
|
existence of an entry, see M(community.general.ldap_entry).
|
|
|
|
notes:
|
|
|
|
- The default authentication settings will attempt to use a SASL EXTERNAL
|
|
|
|
bind over a UNIX domain socket. This works well with the default Ubuntu
|
2023-10-19 11:10:04 +02:00
|
|
|
install for example, which includes a C(cn=peercred,cn=external,cn=auth) ACL
|
2022-04-03 11:04:27 +02:00
|
|
|
rule allowing root to modify the server configuration. If you need to use
|
2023-10-19 11:10:04 +02:00
|
|
|
a simple bind to access your server, pass the credentials in O(bind_dn)
|
|
|
|
and O(bind_pw).
|
2022-04-03 11:04:27 +02:00
|
|
|
author:
|
|
|
|
- Keller Fuchs (@KellerFuchs)
|
|
|
|
requirements:
|
|
|
|
- python-ldap
|
2023-10-19 11:10:04 +02:00
|
|
|
attributes:
|
|
|
|
check_mode:
|
|
|
|
support: full
|
|
|
|
diff_mode:
|
|
|
|
support: none
|
2022-04-03 11:04:27 +02:00
|
|
|
options:
|
|
|
|
passwd:
|
|
|
|
description:
|
2023-10-19 11:10:04 +02:00
|
|
|
- The (plaintext) password to be set for O(dn).
|
2022-04-03 11:04:27 +02:00
|
|
|
type: str
|
|
|
|
extends_documentation_fragment:
|
2023-10-19 11:10:04 +02:00
|
|
|
- community.general.ldap.documentation
|
|
|
|
- community.general.attributes
|
2022-04-03 11:04:27 +02:00
|
|
|
|
|
|
|
'''
|
|
|
|
|
|
|
|
EXAMPLES = """
|
|
|
|
- name: Set a password for the admin user
|
|
|
|
community.general.ldap_passwd:
|
|
|
|
dn: cn=admin,dc=example,dc=com
|
|
|
|
passwd: "{{ vault_secret }}"
|
|
|
|
|
|
|
|
- name: Setting passwords in bulk
|
|
|
|
community.general.ldap_passwd:
|
|
|
|
dn: "{{ item.key }}"
|
|
|
|
passwd: "{{ item.value }}"
|
|
|
|
with_dict:
|
|
|
|
alice: alice123123
|
|
|
|
bob: "|30b!"
|
|
|
|
admin: "{{ vault_secret }}"
|
|
|
|
"""
|
|
|
|
|
|
|
|
RETURN = """
|
|
|
|
modlist:
|
|
|
|
description: list of modified parameters
|
|
|
|
returned: success
|
|
|
|
type: list
|
2023-10-19 11:10:04 +02:00
|
|
|
sample:
|
|
|
|
- [2, "olcRootDN", ["cn=root,dc=example,dc=com"]]
|
2022-04-03 11:04:27 +02:00
|
|
|
"""
|
|
|
|
|
|
|
|
import traceback
|
|
|
|
|
|
|
|
from ansible.module_utils.basic import AnsibleModule, missing_required_lib
|
2023-10-19 11:10:04 +02:00
|
|
|
from ansible_collections.community.general.plugins.module_utils.ldap import LdapGeneric, gen_specs, ldap_required_together
|
2022-04-03 11:04:27 +02:00
|
|
|
|
|
|
|
LDAP_IMP_ERR = None
|
|
|
|
try:
|
|
|
|
import ldap
|
|
|
|
|
|
|
|
HAS_LDAP = True
|
|
|
|
except ImportError:
|
|
|
|
LDAP_IMP_ERR = traceback.format_exc()
|
|
|
|
HAS_LDAP = False
|
|
|
|
|
|
|
|
|
|
|
|
class LdapPasswd(LdapGeneric):
|
|
|
|
def __init__(self, module):
|
|
|
|
LdapGeneric.__init__(self, module)
|
|
|
|
|
|
|
|
# Shortcuts
|
|
|
|
self.passwd = self.module.params['passwd']
|
|
|
|
|
|
|
|
def passwd_check(self):
|
|
|
|
try:
|
|
|
|
tmp_con = ldap.initialize(self.server_uri)
|
|
|
|
except ldap.LDAPError as e:
|
|
|
|
self.fail("Cannot initialize LDAP connection", e)
|
|
|
|
|
|
|
|
if self.start_tls:
|
|
|
|
try:
|
|
|
|
tmp_con.start_tls_s()
|
|
|
|
except ldap.LDAPError as e:
|
|
|
|
self.fail("Cannot start TLS.", e)
|
|
|
|
|
|
|
|
try:
|
|
|
|
tmp_con.simple_bind_s(self.dn, self.passwd)
|
|
|
|
except ldap.INVALID_CREDENTIALS:
|
|
|
|
return True
|
|
|
|
except ldap.LDAPError as e:
|
|
|
|
self.fail("Cannot bind to the server.", e)
|
|
|
|
else:
|
|
|
|
return False
|
|
|
|
finally:
|
|
|
|
tmp_con.unbind()
|
|
|
|
|
|
|
|
def passwd_set(self):
|
|
|
|
# Exit early if the password is already valid
|
|
|
|
if not self.passwd_check():
|
|
|
|
return False
|
|
|
|
|
|
|
|
# Change the password (or throw an exception)
|
|
|
|
try:
|
|
|
|
self.connection.passwd_s(self.dn, None, self.passwd)
|
|
|
|
except ldap.LDAPError as e:
|
|
|
|
self.fail("Unable to set password", e)
|
|
|
|
|
|
|
|
# Password successfully changed
|
|
|
|
return True
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
module = AnsibleModule(
|
|
|
|
argument_spec=gen_specs(passwd=dict(no_log=True)),
|
|
|
|
supports_check_mode=True,
|
2023-10-19 11:10:04 +02:00
|
|
|
required_together=ldap_required_together(),
|
2022-04-03 11:04:27 +02:00
|
|
|
)
|
|
|
|
|
|
|
|
if not HAS_LDAP:
|
|
|
|
module.fail_json(msg=missing_required_lib('python-ldap'),
|
|
|
|
exception=LDAP_IMP_ERR)
|
|
|
|
|
|
|
|
ldap = LdapPasswd(module)
|
|
|
|
|
|
|
|
if module.check_mode:
|
|
|
|
module.exit_json(changed=ldap.passwd_check())
|
|
|
|
|
|
|
|
module.exit_json(changed=ldap.passwd_set())
|
|
|
|
|
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|