| 
									
										
										
										
											2020-10-30 03:05:59 +00:00
										 |  |  | """LDAP Authentication Provider Plugin""" | 
					
						
							| 
									
										
										
										
											2020-11-16 12:35:23 +00:00
										 |  |  | import io | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2020-10-15 19:58:56 +00:00
										 |  |  | import ssl | 
					
						
							| 
									
										
										
										
											2020-11-15 00:21:32 +00:00
										 |  |  | from typing import Optional | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  | from flask_ldapconn import LDAPConn | 
					
						
							|  |  |  | from flask import current_app as app | 
					
						
							| 
									
										
										
										
											2020-10-15 19:58:56 +00:00
										 |  |  | from ldap3.core.exceptions import LDAPPasswordIsMandatoryError, LDAPBindError | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  | from ldap3 import SUBTREE, MODIFY_REPLACE, MODIFY_ADD, MODIFY_DELETE | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  | from werkzeug.exceptions import BadRequest, InternalServerError, NotFound | 
					
						
							| 
									
										
										
										
											2020-08-25 02:36:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-13 00:20:25 +00:00
										 |  |  | from flaschengeist import logger | 
					
						
							| 
									
										
										
										
											2020-11-17 23:39:25 +00:00
										 |  |  | from flaschengeist.plugins import AuthPlugin, after_role_updated | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  | from flaschengeist.models.user import User, Role, _Avatar | 
					
						
							| 
									
										
										
										
											2020-10-30 02:30:46 +00:00
										 |  |  | import flaschengeist.controller.userController as userController | 
					
						
							| 
									
										
										
										
											2020-09-03 22:55:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-01 23:09:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-15 19:58:56 +00:00
										 |  |  | class AuthLDAP(AuthPlugin): | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |     def __init__(self, config): | 
					
						
							| 
									
										
										
										
											2020-10-15 19:58:56 +00:00
										 |  |  |         super().__init__() | 
					
						
							| 
									
										
										
										
											2020-08-23 21:58:26 +00:00
										 |  |  |         app.config.update( | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             LDAP_SERVER=config.get("host", "localhost"), | 
					
						
							|  |  |  |             LDAP_PORT=config.get("port", 389), | 
					
						
							|  |  |  |             LDAP_BINDDN=config.get("bind_dn", None), | 
					
						
							|  |  |  |             LDAP_SECRET=config.get("secret", None), | 
					
						
							|  |  |  |             LDAP_USE_SSL=config.get("use_ssl", False), | 
					
						
							|  |  |  |             # That's not TLS, its dirty StartTLS on unencrypted LDAP | 
					
						
							| 
									
										
										
										
											2020-09-01 23:09:24 +00:00
										 |  |  |             LDAP_USE_TLS=False, | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             LDAP_TLS_VERSION=ssl.PROTOCOL_TLS, | 
					
						
							| 
									
										
										
										
											2020-09-01 23:09:24 +00:00
										 |  |  |             FORCE_ATTRIBUTE_VALUE_AS_LIST=True, | 
					
						
							| 
									
										
										
										
											2020-08-23 21:58:26 +00:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         logger.warning(app.config.get("LDAP_USE_SSL")) | 
					
						
							|  |  |  |         if "ca_cert" in config: | 
					
						
							|  |  |  |             app.config["LDAP_CA_CERTS_FILE"] = config["ca_cert"] | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # Default is CERT_REQUIRED | 
					
						
							|  |  |  |             app.config["LDAP_REQUIRE_CERT"] = ssl.CERT_OPTIONAL | 
					
						
							| 
									
										
										
										
											2020-08-25 02:36:05 +00:00
										 |  |  |         self.ldap = LDAPConn(app) | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         self.base_dn = config["base_dn"] | 
					
						
							|  |  |  |         self.search_dn = config.get("search_dn", "ou=people,{base_dn}").format(base_dn=self.base_dn) | 
					
						
							|  |  |  |         self.group_dn = config.get("group_dn", "ou=group,{base_dn}").format(base_dn=self.base_dn) | 
					
						
							|  |  |  |         self.password_hash = config.get("password_hash", "SSHA").upper() | 
					
						
							|  |  |  |         self.object_classes = config.get("object_classes", ["inetOrgPerson"]) | 
					
						
							|  |  |  |         self.user_attributes: dict = config.get("user_attributes", {}) | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-29 01:07:40 +00:00
										 |  |  |         # TODO: might not be set if modify is called | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         self.root_dn = config.get("root_dn", None) | 
					
						
							|  |  |  |         self.root_secret = self.root_dn = config.get("root_secret", None) | 
					
						
							| 
									
										
										
										
											2020-08-23 21:58:26 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 23:39:25 +00:00
										 |  |  |         @after_role_updated | 
					
						
							|  |  |  |         def _role_updated(role, new_name): | 
					
						
							|  |  |  |             self.__modify_role(role, new_name) | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-23 21:58:26 +00:00
										 |  |  |     def login(self, user, password): | 
					
						
							|  |  |  |         if not user: | 
					
						
							|  |  |  |             return False | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         return self.ldap.authenticate(user.userid, password, "uid", self.base_dn) | 
					
						
							| 
									
										
										
										
											2020-08-25 02:36:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-19 02:29:26 +00:00
										 |  |  |     def find_user(self, userid, mail=None): | 
					
						
							|  |  |  |         attr = self.__find(userid, mail) | 
					
						
							| 
									
										
										
										
											2021-03-14 14:53:14 +00:00
										 |  |  |         if attr is not None: | 
					
						
							| 
									
										
										
										
											2021-01-19 02:29:26 +00:00
										 |  |  |             user = User(userid=attr["uid"][0]) | 
					
						
							|  |  |  |             self.__update(user, attr) | 
					
						
							|  |  |  |             return user | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-03 15:56:12 +00:00
										 |  |  |     def update_user(self, user): | 
					
						
							| 
									
										
										
										
											2021-01-19 02:29:26 +00:00
										 |  |  |         attr = self.__find(user.userid) | 
					
						
							|  |  |  |         self.__update(user, attr) | 
					
						
							| 
									
										
										
										
											2020-08-25 02:36:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-12 18:29:24 +00:00
										 |  |  |     def create_user(self, user, password): | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         if self.root_dn is None: | 
					
						
							|  |  |  |             logger.error("root_dn missing in ldap config!") | 
					
						
							| 
									
										
										
										
											2020-11-13 00:20:25 +00:00
										 |  |  |             raise InternalServerError | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-12 18:29:24 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             ldap_conn = self.ldap.connect(self.root_dn, self.root_secret) | 
					
						
							|  |  |  |             attributes = self.user_attributes.copy() | 
					
						
							|  |  |  |             if "uidNumber" in attributes: | 
					
						
							|  |  |  |                 self.ldap.connection.search( | 
					
						
							|  |  |  |                     self.search_dn, | 
					
						
							|  |  |  |                     "(uidNumber=*)", | 
					
						
							|  |  |  |                     SUBTREE, | 
					
						
							|  |  |  |                     attributes=["uidNumber"], | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 resp = sorted( | 
					
						
							|  |  |  |                     self.ldap.response(), | 
					
						
							|  |  |  |                     key=lambda i: i["attributes"]["uidNumber"], | 
					
						
							|  |  |  |                     reverse=True, | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |                 attributes = resp[0]["attributes"]["uidNumber"] + 1 if resp else attributes["uidNumber"] | 
					
						
							|  |  |  |             dn = self.dn_template.format( | 
					
						
							|  |  |  |                 firstname=user.firstname, | 
					
						
							|  |  |  |                 lastname=user.lastname, | 
					
						
							|  |  |  |                 userid=user.userid, | 
					
						
							|  |  |  |                 mail=user.mail, | 
					
						
							|  |  |  |                 display_name=user.display_name, | 
					
						
							|  |  |  |                 base_dn=self.base_dn, | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |             ) | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             attributes.update({ | 
					
						
							|  |  |  |                 "sn": user.lastname, | 
					
						
							|  |  |  |                 "givenName": user.firstname, | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |                 "uid": user.userid, | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |                 "userPassword": self.__hash(password), | 
					
						
							|  |  |  |             }) | 
					
						
							|  |  |  |             ldap_conn.add(dn, self.object_classes, attributes) | 
					
						
							| 
									
										
										
										
											2020-11-13 00:20:25 +00:00
										 |  |  |             self._set_roles(user) | 
					
						
							| 
									
										
										
										
											2020-11-12 18:29:24 +00:00
										 |  |  |         except (LDAPPasswordIsMandatoryError, LDAPBindError): | 
					
						
							|  |  |  |             raise BadRequest | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |     def modify_user(self, user: User, password=None, new_password=None): | 
					
						
							|  |  |  |         try: | 
					
						
							|  |  |  |             dn = user.get_attribute("DN") | 
					
						
							|  |  |  |             if password: | 
					
						
							|  |  |  |                 ldap_conn = self.ldap.connect(dn, password) | 
					
						
							|  |  |  |             else: | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |                 if self.root_dn is None: | 
					
						
							|  |  |  |                     logger.error("root_dn missing in ldap config!") | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |                     raise InternalServerError | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |                 ldap_conn = self.ldap.connect(self.root_dn, self.root_secret) | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |             modifier = {} | 
					
						
							|  |  |  |             for name, ldap_name in [ | 
					
						
							|  |  |  |                 ("firstname", "givenName"), | 
					
						
							|  |  |  |                 ("lastname", "sn"), | 
					
						
							|  |  |  |                 ("mail", "mail"), | 
					
						
							|  |  |  |                 ("display_name", "displayName"), | 
					
						
							|  |  |  |             ]: | 
					
						
							|  |  |  |                 if hasattr(user, name): | 
					
						
							|  |  |  |                     modifier[ldap_name] = [(MODIFY_REPLACE, [getattr(user, name)])] | 
					
						
							|  |  |  |             if new_password: | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |                 modifier["userPassword"] = [(MODIFY_REPLACE, [self.__hash(new_password)])] | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |             ldap_conn.modify(dn, modifier) | 
					
						
							|  |  |  |             self._set_roles(user) | 
					
						
							|  |  |  |         except (LDAPPasswordIsMandatoryError, LDAPBindError): | 
					
						
							|  |  |  |             raise BadRequest | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |     def get_avatar(self, user): | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |         self.ldap.connection.search( | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             self.search_dn, | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |             "(uid={})".format(user.userid), | 
					
						
							|  |  |  |             SUBTREE, | 
					
						
							| 
									
										
										
										
											2021-01-27 01:34:04 +00:00
										 |  |  |             attributes=["jpegPhoto"], | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |         ) | 
					
						
							|  |  |  |         r = self.ldap.connection.response[0]["attributes"] | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-27 01:34:04 +00:00
										 |  |  |         if "jpegPhoto" in r and len(r["jpegPhoto"]) > 0: | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |             avatar = _Avatar() | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |             avatar.mimetype = "image/jpeg" | 
					
						
							|  |  |  |             avatar.binary.extend(r["jpegPhoto"][0]) | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |             return avatar | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             raise NotFound | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |     def set_avatar(self, user, avatar: _Avatar): | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         if self.root_dn is None: | 
					
						
							|  |  |  |             logger.error("root_dn missing in ldap config!") | 
					
						
							| 
									
										
										
										
											2021-01-18 17:31:13 +00:00
										 |  |  |             raise InternalServerError | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |         if avatar.mimetype != "image/jpeg": | 
					
						
							| 
									
										
										
										
											2020-11-16 12:35:23 +00:00
										 |  |  |             # Try converting using Pillow (if installed) | 
					
						
							|  |  |  |             try: | 
					
						
							|  |  |  |                 from PIL import Image | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-16 12:35:23 +00:00
										 |  |  |                 image = Image.open(io.BytesIO(avatar.binary)) | 
					
						
							|  |  |  |                 image_bytes = io.BytesIO() | 
					
						
							|  |  |  |                 image.save(image_bytes, format="JPEG") | 
					
						
							|  |  |  |                 avatar.binary = image_bytes.getvalue() | 
					
						
							|  |  |  |                 avatar.mimetype = "image/jpeg" | 
					
						
							|  |  |  |             except ImportError: | 
					
						
							|  |  |  |                 logger.debug("Pillow not installed for image conversion") | 
					
						
							|  |  |  |                 raise BadRequest("Unsupported image format") | 
					
						
							|  |  |  |             except IOError: | 
					
						
							|  |  |  |                 logger.debug(f"Could not convert avatar from '{avatar.mimetype}' to JPEG") | 
					
						
							|  |  |  |                 raise BadRequest("Unsupported image format") | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |         dn = user.get_attribute("DN") | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         ldap_conn = self.ldap.connect(self.root_dn, self.root_secret) | 
					
						
							| 
									
										
										
										
											2020-11-16 01:30:24 +00:00
										 |  |  |         ldap_conn.modify(dn, {"jpegPhoto": [(MODIFY_REPLACE, [avatar.binary])]}) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-01-19 02:29:26 +00:00
										 |  |  |     def __find(self, userid, mail=None): | 
					
						
							|  |  |  |         """Find attributes of an user by uid or mail in LDAP""" | 
					
						
							|  |  |  |         con = self.ldap.connection | 
					
						
							|  |  |  |         if not con: | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             con = self.ldap.connect(self.root_dn, self.root_secret) | 
					
						
							| 
									
										
										
										
											2021-01-19 02:29:26 +00:00
										 |  |  |         con.search( | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             self.search_dn, | 
					
						
							| 
									
										
										
										
											2021-01-19 02:29:26 +00:00
										 |  |  |             f"(| (uid={userid})(mail={mail}))" if mail else f"(uid={userid})", | 
					
						
							|  |  |  |             SUBTREE, | 
					
						
							|  |  |  |             attributes=["uid", "givenName", "sn", "mail"], | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-03-14 14:53:14 +00:00
										 |  |  |         return con.response[0]["attributes"] if len(con.response) > 0 else None | 
					
						
							| 
									
										
										
										
											2021-01-19 02:29:26 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __update(self, user, attr): | 
					
						
							|  |  |  |         """Update an User object with LDAP attributes""" | 
					
						
							|  |  |  |         if attr["uid"][0] == user.userid: | 
					
						
							|  |  |  |             user.set_attribute("DN", self.ldap.connection.response[0]["dn"]) | 
					
						
							|  |  |  |             user.firstname = attr["givenName"][0] | 
					
						
							|  |  |  |             user.lastname = attr["sn"][0] | 
					
						
							|  |  |  |             if attr["mail"]: | 
					
						
							|  |  |  |                 user.mail = attr["mail"][0] | 
					
						
							|  |  |  |             if "displayName" in attr: | 
					
						
							|  |  |  |                 user.display_name = attr["displayName"][0] | 
					
						
							|  |  |  |             userController.set_roles(user, self._get_groups(user.userid), create=True) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |     def __modify_role( | 
					
						
							|  |  |  |         self, | 
					
						
							| 
									
										
										
										
											2020-11-17 23:39:25 +00:00
										 |  |  |         role: Role, | 
					
						
							|  |  |  |         new_name: Optional[str], | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |     ): | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |         if self.root_dn is None: | 
					
						
							|  |  |  |             logger.error("root_dn missing in ldap config!") | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |             raise InternalServerError | 
					
						
							|  |  |  |         try: | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             ldap_conn = self.ldap.connect(self.root_dn, self.root_secret) | 
					
						
							|  |  |  |             ldap_conn.search(self.group_dn, f"(cn={role.name})", SUBTREE, attributes=["cn"]) | 
					
						
							| 
									
										
										
										
											2020-11-17 23:39:25 +00:00
										 |  |  |             if len(ldap_conn.response) > 0: | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |                 dn = ldap_conn.response[0]["dn"] | 
					
						
							| 
									
										
										
										
											2020-11-17 23:39:25 +00:00
										 |  |  |                 if new_name: | 
					
						
							|  |  |  |                     ldap_conn.modify_dn(dn, f"cn={new_name}") | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |                 else: | 
					
						
							|  |  |  |                     ldap_conn.delete(dn) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         except (LDAPPasswordIsMandatoryError, LDAPBindError): | 
					
						
							|  |  |  |             raise BadRequest | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |     def __hash(self, password): | 
					
						
							|  |  |  |         if self.password_hash == "ARGON2": | 
					
						
							|  |  |  |             from argon2 import PasswordHasher | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return f"{{ARGON2}}{PasswordHasher().hash(password)}" | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             from hashlib import pbkdf2_hmac, sha1 | 
					
						
							|  |  |  |             import base64 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             salt = os.urandom(16) | 
					
						
							|  |  |  |             if self.password_hash == "PBKDF2": | 
					
						
							|  |  |  |                 rounds = 200000 | 
					
						
							|  |  |  |                 password_hash = base64.b64encode(pbkdf2_hmac("sha512", password.encode("utf-8"), salt, rounds)).decode() | 
					
						
							|  |  |  |                 return f"{{PBKDF2-SHA512}}{rounds}${base64.b64encode(salt).decode()}${password_hash}" | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return f"{{SSHA}}{base64.b64encode(sha1(password + salt) + salt)}" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-09-02 11:07:21 +00:00
										 |  |  |     def _get_groups(self, uid): | 
					
						
							| 
									
										
										
										
											2020-09-03 22:55:23 +00:00
										 |  |  |         groups = [] | 
					
						
							|  |  |  |         self.ldap.connection.search( | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             self.group_dn, | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |             "(memberUID={})".format(uid), | 
					
						
							|  |  |  |             SUBTREE, | 
					
						
							|  |  |  |             attributes=["cn"], | 
					
						
							| 
									
										
										
										
											2020-09-03 22:55:23 +00:00
										 |  |  |         ) | 
					
						
							|  |  |  |         groups_data = self.ldap.connection.response | 
					
						
							|  |  |  |         for data in groups_data: | 
					
						
							|  |  |  |             groups.append(data["attributes"]["cn"][0]) | 
					
						
							|  |  |  |         return groups | 
					
						
							| 
									
										
										
										
											2020-08-25 02:36:05 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |     def _get_all_roles(self): | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |         self.ldap.connection.search( | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             self.group_dn, | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |             "(cn=*)", | 
					
						
							|  |  |  |             SUBTREE, | 
					
						
							|  |  |  |             attributes=["cn", "gidNumber", "memberUid"], | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         return self.ldap.response() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-13 00:20:25 +00:00
										 |  |  |     def _set_roles(self, user: User): | 
					
						
							| 
									
										
										
										
											2020-11-12 21:47:10 +00:00
										 |  |  |         try: | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |             ldap_conn = self.ldap.connect(self.root_dn, self.root_secret) | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |             ldap_roles = self._get_all_roles() | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-15 18:44:49 +00:00
										 |  |  |             gid_numbers = sorted(ldap_roles, key=lambda i: i["attributes"]["gidNumber"], reverse=True) | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |             gid_number = gid_numbers[0]["attributes"]["gidNumber"] + 1 | 
					
						
							| 
									
										
										
										
											2020-11-12 22:42:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             for user_role in user.roles: | 
					
						
							| 
									
										
										
										
											2020-11-15 18:44:49 +00:00
										 |  |  |                 if user_role not in [role["attributes"]["cn"][0] for role in ldap_roles]: | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |                     ldap_conn.add( | 
					
						
							| 
									
										
										
										
											2021-07-29 13:50:03 +00:00
										 |  |  |                         f"cn={user_role},{self.group_dn}", | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |                         ["posixGroup"], | 
					
						
							|  |  |  |                         attributes={"gidNumber": gid_number}, | 
					
						
							|  |  |  |                     ) | 
					
						
							| 
									
										
										
										
											2020-11-12 22:42:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-17 02:28:04 +00:00
										 |  |  |             ldap_roles = self._get_all_roles() | 
					
						
							| 
									
										
										
										
											2020-11-12 22:42:03 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-12 21:47:10 +00:00
										 |  |  |             for ldap_role in ldap_roles: | 
					
						
							|  |  |  |                 if ldap_role["attributes"]["cn"][0] in user.roles: | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |                     modify = {"memberUid": [(MODIFY_ADD, [user.userid])]} | 
					
						
							| 
									
										
										
										
											2020-11-12 21:47:10 +00:00
										 |  |  |                 else: | 
					
						
							| 
									
										
										
										
											2020-11-14 12:16:30 +00:00
										 |  |  |                     modify = {"memberUid": [(MODIFY_DELETE, [user.userid])]} | 
					
						
							| 
									
										
										
										
											2020-11-12 22:42:03 +00:00
										 |  |  |                 ldap_conn.modify(ldap_role["dn"], modify) | 
					
						
							| 
									
										
										
										
											2020-11-12 21:47:10 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-15 00:21:32 +00:00
										 |  |  |         except (LDAPPasswordIsMandatoryError, LDAPBindError): | 
					
						
							|  |  |  |             raise BadRequest |