2013-07-17 16:37:04 -04:00
|
|
|
#!/usr/bin/python
|
2013-06-24 21:18:05 -04:00
|
|
|
# -*- coding: utf-8 -*-
|
|
|
|
|
|
|
|
# (c) 2013, Nimbis Services, Inc.
|
|
|
|
#
|
|
|
|
# This file is part of Ansible
|
|
|
|
#
|
|
|
|
# Ansible is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# Ansible is distributed in the hope that it will be useful,
|
|
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
# GNU General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with Ansible. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#
|
2013-06-24 16:24:55 -04:00
|
|
|
DOCUMENTATION = """
|
|
|
|
module: htpasswd
|
2013-11-18 18:55:49 -05:00
|
|
|
version_added: "1.3"
|
2013-06-24 16:24:55 -04:00
|
|
|
short_description: manage user files for basic authentication
|
|
|
|
description:
|
|
|
|
- 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.
|
|
|
|
options:
|
|
|
|
path:
|
|
|
|
required: true
|
|
|
|
aliases: [ dest, destfile ]
|
|
|
|
description:
|
|
|
|
- Path to the file that contains the usernames and passwords
|
|
|
|
name:
|
|
|
|
required: true
|
|
|
|
aliases: [ username ]
|
|
|
|
description:
|
|
|
|
- User name to add or remove
|
|
|
|
password:
|
|
|
|
required: false
|
|
|
|
description:
|
2014-02-03 11:52:37 -06:00
|
|
|
- Password associated with user.
|
2014-01-30 01:05:05 -05:00
|
|
|
- Must be specified if user does not exist yet.
|
2013-09-05 12:05:01 +02:00
|
|
|
crypt_scheme:
|
|
|
|
required: false
|
2014-01-30 01:05:05 -05:00
|
|
|
choices: ["apr_md5_crypt", "des_crypt", "ldap_sha1", "plaintext"]
|
2013-09-05 12:05:01 +02:00
|
|
|
default: "apr_md5_crypt"
|
|
|
|
description:
|
2014-01-30 01:05:05 -05:00
|
|
|
- Encryption scheme to be used.
|
2013-06-24 16:24:55 -04:00
|
|
|
state:
|
|
|
|
required: false
|
|
|
|
choices: [ present, absent ]
|
|
|
|
default: "present"
|
|
|
|
description:
|
|
|
|
- Whether the user entry should be present or not
|
|
|
|
create:
|
|
|
|
required: false
|
|
|
|
choices: [ "yes", "no" ]
|
|
|
|
default: "yes"
|
|
|
|
description:
|
|
|
|
- Used with C(state=present). If specified, the file will be created
|
|
|
|
if it does not already exist. If set to "no", will fail if the
|
|
|
|
file does not exist
|
2013-09-18 21:43:13 -04:00
|
|
|
notes:
|
2013-10-02 08:24:21 -04:00
|
|
|
- "This module depends on the I(passlib) Python library, which needs to be installed on all target systems."
|
|
|
|
- "On Debian, Ubuntu, or Fedora: install I(python-passlib)."
|
|
|
|
- "On RHEL or CentOS: Enable EPEL, then install I(python-passlib)."
|
2013-06-24 16:24:55 -04:00
|
|
|
requires: [ passlib>=1.6 ]
|
2015-06-15 15:53:30 -04:00
|
|
|
author: "Lorin Hochstein (@lorin)"
|
2013-06-24 16:24:55 -04:00
|
|
|
"""
|
|
|
|
|
|
|
|
EXAMPLES = """
|
|
|
|
# Add a user to a password file and ensure permissions are set
|
|
|
|
- htpasswd: path=/etc/nginx/passwdfile name=janedoe password=9s36?;fyNp owner=root group=www-data mode=0640
|
|
|
|
# Remove a user from a password file
|
|
|
|
- htpasswd: path=/etc/apache2/passwdfile name=foobar state=absent
|
|
|
|
"""
|
|
|
|
|
|
|
|
|
|
|
|
import os
|
2015-06-24 11:23:34 -07:00
|
|
|
import tempfile
|
2013-10-04 03:18:17 +02:00
|
|
|
from distutils.version import StrictVersion
|
2013-06-24 16:24:55 -04:00
|
|
|
|
|
|
|
try:
|
|
|
|
from passlib.apache import HtpasswdFile
|
2013-10-04 03:18:17 +02:00
|
|
|
import passlib
|
2013-06-24 16:24:55 -04:00
|
|
|
except ImportError:
|
|
|
|
passlib_installed = False
|
|
|
|
else:
|
|
|
|
passlib_installed = True
|
|
|
|
|
|
|
|
|
|
|
|
def create_missing_directories(dest):
|
|
|
|
destpath = os.path.dirname(dest)
|
|
|
|
if not os.path.exists(destpath):
|
|
|
|
os.makedirs(destpath)
|
|
|
|
|
|
|
|
|
2013-09-05 12:05:01 +02:00
|
|
|
def present(dest, username, password, crypt_scheme, create, check_mode):
|
2013-06-24 16:24:55 -04:00
|
|
|
""" Ensures user is present
|
|
|
|
|
|
|
|
Returns (msg, changed) """
|
|
|
|
if not os.path.exists(dest):
|
|
|
|
if not create:
|
|
|
|
raise ValueError('Destination %s does not exist' % dest)
|
|
|
|
if check_mode:
|
|
|
|
return ("Create %s" % dest, True)
|
|
|
|
create_missing_directories(dest)
|
2013-10-04 03:18:17 +02:00
|
|
|
if StrictVersion(passlib.__version__) >= StrictVersion('1.6'):
|
2013-09-05 12:05:01 +02:00
|
|
|
ht = HtpasswdFile(dest, new=True, default_scheme=crypt_scheme)
|
2013-10-04 03:18:17 +02:00
|
|
|
else:
|
|
|
|
ht = HtpasswdFile(dest, autoload=False, default=crypt_scheme)
|
2013-07-20 18:22:58 -04:00
|
|
|
if getattr(ht, 'set_password', None):
|
|
|
|
ht.set_password(username, password)
|
|
|
|
else:
|
|
|
|
ht.update(username, password)
|
2013-06-24 16:24:55 -04:00
|
|
|
ht.save()
|
|
|
|
return ("Created %s and added %s" % (dest, username), True)
|
|
|
|
else:
|
2013-10-04 03:18:17 +02:00
|
|
|
if StrictVersion(passlib.__version__) >= StrictVersion('1.6'):
|
2013-09-05 12:05:01 +02:00
|
|
|
ht = HtpasswdFile(dest, new=False, default_scheme=crypt_scheme)
|
2013-10-04 03:18:17 +02:00
|
|
|
else:
|
|
|
|
ht = HtpasswdFile(dest, default=crypt_scheme)
|
2013-09-05 12:05:01 +02:00
|
|
|
|
2013-07-20 18:22:58 -04:00
|
|
|
found = None
|
|
|
|
if getattr(ht, 'check_password', None):
|
|
|
|
found = ht.check_password(username, password)
|
|
|
|
else:
|
|
|
|
found = ht.verify(username, password)
|
|
|
|
|
|
|
|
if found:
|
2013-06-24 16:24:55 -04:00
|
|
|
return ("%s already present" % username, False)
|
|
|
|
else:
|
|
|
|
if not check_mode:
|
2013-07-20 18:22:58 -04:00
|
|
|
if getattr(ht, 'set_password', None):
|
|
|
|
ht.set_password(username, password)
|
|
|
|
else:
|
|
|
|
ht.update(username, password)
|
2013-06-24 16:24:55 -04:00
|
|
|
ht.save()
|
|
|
|
return ("Add/update %s" % username, True)
|
|
|
|
|
|
|
|
|
|
|
|
def absent(dest, username, check_mode):
|
|
|
|
""" Ensures user is absent
|
|
|
|
|
|
|
|
Returns (msg, changed) """
|
|
|
|
if not os.path.exists(dest):
|
|
|
|
raise ValueError("%s does not exists" % dest)
|
|
|
|
|
2013-10-04 03:18:17 +02:00
|
|
|
if StrictVersion(passlib.__version__) >= StrictVersion('1.6'):
|
2013-07-20 18:22:58 -04:00
|
|
|
ht = HtpasswdFile(dest, new=False)
|
2013-10-04 03:18:17 +02:00
|
|
|
else:
|
2013-07-20 18:22:58 -04:00
|
|
|
ht = HtpasswdFile(dest)
|
|
|
|
|
2013-06-24 16:24:55 -04:00
|
|
|
if username not in ht.users():
|
|
|
|
return ("%s not present" % username, False)
|
|
|
|
else:
|
|
|
|
if not check_mode:
|
|
|
|
ht.delete(username)
|
|
|
|
ht.save()
|
|
|
|
return ("Remove %s" % username, True)
|
|
|
|
|
|
|
|
|
|
|
|
def check_file_attrs(module, changed, message):
|
|
|
|
|
|
|
|
file_args = module.load_file_common_arguments(module.params)
|
2014-03-18 22:39:45 -04:00
|
|
|
if module.set_fs_attributes_if_different(file_args, False):
|
2013-06-24 16:24:55 -04:00
|
|
|
|
|
|
|
if changed:
|
|
|
|
message += " and "
|
|
|
|
changed = True
|
|
|
|
message += "ownership, perms or SE linux context changed"
|
|
|
|
|
|
|
|
return message, changed
|
|
|
|
|
|
|
|
|
|
|
|
def main():
|
|
|
|
arg_spec = dict(
|
|
|
|
path=dict(required=True, aliases=["dest", "destfile"]),
|
|
|
|
name=dict(required=True, aliases=["username"]),
|
|
|
|
password=dict(required=False, default=None),
|
2013-09-05 12:05:01 +02:00
|
|
|
crypt_scheme=dict(required=False, default=None),
|
2013-06-24 16:24:55 -04:00
|
|
|
state=dict(required=False, default="present"),
|
2013-10-11 08:45:13 -04:00
|
|
|
create=dict(type='bool', default='yes'),
|
2013-06-24 16:24:55 -04:00
|
|
|
|
|
|
|
)
|
|
|
|
module = AnsibleModule(argument_spec=arg_spec,
|
|
|
|
add_file_common_args=True,
|
|
|
|
supports_check_mode=True)
|
|
|
|
|
|
|
|
path = module.params['path']
|
|
|
|
username = module.params['name']
|
|
|
|
password = module.params['password']
|
2013-09-05 12:05:01 +02:00
|
|
|
crypt_scheme = module.params['crypt_scheme']
|
2013-06-24 16:24:55 -04:00
|
|
|
state = module.params['state']
|
|
|
|
create = module.params['create']
|
|
|
|
check_mode = module.check_mode
|
|
|
|
|
|
|
|
if not passlib_installed:
|
|
|
|
module.fail_json(msg="This module requires the passlib Python library")
|
|
|
|
|
2015-05-22 18:57:06 -04:00
|
|
|
# Check file for blank lines in effort to avoid "need more than 1 value to unpack" error.
|
|
|
|
try:
|
2015-06-24 11:23:34 -07:00
|
|
|
f = open(path, "r")
|
|
|
|
except IOError:
|
|
|
|
# No preexisting file to remove blank lines from
|
|
|
|
f = None
|
|
|
|
else:
|
2015-05-22 18:57:06 -04:00
|
|
|
try:
|
2015-06-24 11:23:34 -07:00
|
|
|
lines = f.readlines()
|
2015-05-22 18:57:06 -04:00
|
|
|
finally:
|
2015-06-24 11:23:34 -07:00
|
|
|
f.close()
|
|
|
|
|
|
|
|
# If the file gets edited, it returns true, so only edit the file if it has blank lines
|
|
|
|
strip = False
|
|
|
|
for line in lines:
|
|
|
|
if not line.strip():
|
|
|
|
strip = True
|
|
|
|
break
|
|
|
|
|
|
|
|
if strip:
|
|
|
|
# If check mode, create a temporary file
|
|
|
|
if check_mode:
|
|
|
|
temp = tempfile.NamedTemporaryFile()
|
|
|
|
path = temp.name
|
|
|
|
f = open(path, "w")
|
|
|
|
try:
|
|
|
|
[ f.write(line) for line in lines if line.strip() ]
|
|
|
|
finally:
|
|
|
|
f.close()
|
2015-05-22 18:57:06 -04:00
|
|
|
|
2013-06-24 16:24:55 -04:00
|
|
|
try:
|
|
|
|
if state == 'present':
|
2013-09-05 12:05:01 +02:00
|
|
|
(msg, changed) = present(path, username, password, crypt_scheme, create, check_mode)
|
2013-06-24 16:24:55 -04:00
|
|
|
elif state == 'absent':
|
|
|
|
(msg, changed) = absent(path, username, check_mode)
|
|
|
|
else:
|
|
|
|
module.fail_json(msg="Invalid state: %s" % state)
|
|
|
|
|
|
|
|
check_file_attrs(module, changed, msg)
|
|
|
|
module.exit_json(msg=msg, changed=changed)
|
|
|
|
except Exception, e:
|
|
|
|
module.fail_json(msg=str(e))
|
|
|
|
|
|
|
|
|
2013-12-02 15:13:49 -05:00
|
|
|
# import module snippets
|
2013-12-02 15:11:23 -05:00
|
|
|
from ansible.module_utils.basic import *
|
2013-06-24 16:24:55 -04:00
|
|
|
|
|
|
|
if __name__ == '__main__':
|
|
|
|
main()
|