#!/usr/bin/python3
# -*- coding: utf-8 -*-
#
# Univention Common Python Library
#
# Copyright 2010-2022 Univention GmbH
#
# https://www.univention.de/
#
# All rights reserved.
#
# The source code of this program is made available
# under the terms of the GNU Affero General Public License version 3
# (GNU AGPL V3) as published by the Free Software Foundation.
#
# Binary versions of this program provided by Univention to you as
# well as other copyrighted, protected or trademarked materials like
# Logos, graphics, fonts, specific documentations and configurations,
# cryptographic keys etc. are subject to a license agreement between
# you and Univention and not subject to the GNU AGPL V3.
#
# In the case you use this program under the terms of the GNU AGPL V3,
# the program is provided 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 Affero General Public License for more details.
#
# You should have received a copy of the GNU Affero General Public
# License with the Debian GNU/Linux or Univention distribution in file
# /usr/share/common-licenses/AGPL-3; if not, see
# <https://www.gnu.org/licenses/>.
"""
python -m univention.lib.account lock --dn "$user_dn" --lock-time "$(date --utc '+%Y%m%d%H%M%SZ')"
"""
import argparse
import univention.admin.uldap
import univention.admin.objects
import univention.admin.modules
import univention.admin.handlers.users.user
import univention.debug as ud1
univention.admin.modules.update()
# Ensure univention debug is initialized
[docs]def initialize_debug():
# Use a little hack to determine if univention.debug has been initialized
# get_level(..) returns always ud.ERROR if univention.debug is not initialized
oldLevel = ud1.get_level(ud1.ADMIN)
if oldLevel == ud1.PROCESS:
ud1.set_level(ud1.ADMIN, ud1.DEBUG)
is_ready = (ud1.get_level(ud1.ADMIN) == ud1.DEBUG)
else:
ud1.set_level(ud1.ADMIN, ud1.PROCESS)
is_ready = (ud1.get_level(ud1.ADMIN) == ud1.PROCESS)
if not is_ready:
ud1.init('/var/log/univention/directory-manager-cmd.log', ud1.FLUSH, 0)
ud1.set_level(ud1.LDAP, ud1.PROCESS)
ud1.set_level(ud1.ADMIN, ud1.PROCESS)
else:
ud1.set_level(ud1.ADMIN, oldLevel)
[docs]def lock(userdn, lock_timestamp):
"""
Lock a user account
* used by ppolicy OpenLDAP overlay
* used by PAM tally
>>> from univention.lib.account import lock # doctest: +SKIP
>>> lock('uid=user1,dc=example,dc=com', '20141006192950Z') # doctest: +SKIP
"""
if not lock_timestamp: # timed unlocking via ppolicy not implemented yet, so block it.
return
try:
lo, pos = univention.admin.uldap.getAdminConnection()
except Exception:
lo, pos = univention.admin.uldap.getMachineConnection()
module = univention.admin.modules.get('users/user')
univention.admin.modules.init(lo, pos, module)
object = module.object(None, lo, pos, userdn)
object.open()
states = (object.descriptions['locked'].editable, object.descriptions['locked'].may_change, object.descriptions['lockedTime'].editable, object.descriptions['lockedTime'].may_change)
object.descriptions['locked'].editable, object.descriptions['locked'].may_change, object.descriptions['lockedTime'].editable, object.descriptions['lockedTime'].may_change = (True, True, True, True)
object['locked'] = "1"
try:
if lock_timestamp:
object['lockedTime'] = lock_timestamp
object.modify()
finally:
object.descriptions['locked'].editable, object.descriptions['locked'].may_change, object.descriptions['lockedTime'].editable, object.descriptions['lockedTime'].may_change = states
if __name__ == '__main__':
parser = argparse.ArgumentParser(description=__doc__)
subparsers = parser.add_subparsers()
subparser = subparsers.add_parser('lock', help='Locks a user account')
subparser.add_argument('--dn', required=True, help='The DN of the user account to be locked.')
subparser.add_argument('--lock-time', required=True, help='The time when the user account was locked.')
args = parser.parse_args()
initialize_debug()
ud1.debug(ud1.ADMIN, ud1.PROCESS, "univention.lib.account.lock was called for %s (%s)" % (args.dn, args.lock_time))
lock(args.dn, args.lock_time)