#!/usr/bin/python3
#
# Univention Updater
#  collect statistics
#
# Copyright 2016-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/>.
from typing import Any, Callable, List, NoReturn, Optional, Tuple
from univention.admin.license import _license
from univention.admin.uldap import access, getAdminConnection, position
from univention.config_registry import ConfigRegistry
from univention.config_registry.frontend import ucr_update
[docs]def encode_number(number: int, significant_digits: int = 3) -> str:
	assert 0 <= number <= int('9' * 26)
	assert 1 < significant_digits
	string = str(number)
	return string[:significant_digits] + ' abcdefghijklmnopqrstuvwxyz'[len(string)] 
[docs]def encode_users(users: int) -> str:
	return encode_number(users) 
[docs]def encode_role(role: str) -> str:
	if role == 'domaincontroller_master':
		return 'M'
	if role == 'domaincontroller_backup':
		return 'B'
	if role == 'domaincontroller_slave':
		return 'S'
	if role == 'memberserver':
		return 'm'
	if role == 'basesystem':
		return 'b'
	raise ValueError('Invalid role %r' % (role, )) 
[docs]def encode_additional_info(users: Optional[int] = None, role: Optional[str] = None) -> str:
	data: List[Tuple[str, Callable[[Any], str], Any]] = [
		('U', encode_users, users),
		('R', encode_role, role),
	]
	return ",".join(
		"%s:%s" % (key, encoder(datum))
		for key, encoder, datum in data
		if datum is not None
	) 
[docs]def getReadonlyAdminConnection() -> Tuple[access, position]:
	def do_nothing(*a: Any, **kw: Any) -> NoReturn:
		raise AssertionError('readonly connection')
	lo, position = getAdminConnection()
	lo.add = lo.modify = lo.rename = lo.delete = do_nothing
	return lo, position 
[docs]def main() -> None:
	def get_role() -> Optional[str]:
		return configRegistry.get('server/role', None)
	def get_users() -> Optional[int]:
		if get_role() != 'domaincontroller_master':
			return None
		lo, _ = getReadonlyAdminConnection()
		filter = _license.filters['2'][_license.USERS]
		return len(lo.searchDn(filter=filter))
	configRegistry = ConfigRegistry()
	configRegistry.load()
	ucr_update(
		configRegistry,
		{
			'updater/statistics': encode_additional_info(users=get_users(), role=get_role()),
		}
	) 
if __name__ == "__main__":
	main()