from typing import Optional, Union from django.contrib.auth import get_user_model from django.contrib.auth.models import AnonymousUser from django.db.models import Model from .collection import CollectionElement def has_perm(user: Optional[CollectionElement], perm: str) -> bool: """ Checks that user has a specific permission. User can be a CollectionElement of a user or None. """ group_collection_string = 'users/group' # This is the hard coded collection string for openslides.users.models.Group # Convert user to right type user = user_to_collection_user(user) if user is None and not anonymous_is_enabled(): has_perm = False elif user is None: # Use the permissions from the default group with id 1. default_group = CollectionElement.from_values(group_collection_string, 1) has_perm = perm in default_group.get_full_data()['permissions'] else: # Get all groups of the user and then see, if one group has the required # permission. If the user has no groups, then use group 1. group_ids = user.get_full_data()['groups_id'] or [1] for group_id in group_ids: group = CollectionElement.from_values(group_collection_string, group_id) if perm in group.get_full_data()['permissions']: has_perm = True break else: has_perm = False return has_perm def anonymous_is_enabled() -> bool: """ Returns True if the anonymous user is enabled in the settings. """ from ..core.config import config return config['general_system_enable_anonymous'] AnyUser = Union[Model, CollectionElement, int, AnonymousUser, None] def user_to_collection_user(user: AnyUser) -> Optional[CollectionElement]: """ Takes an object, that represents a user and converts it to a CollectionElement or to None, if it is an anonymous user. User can be * an user object, * a CollectionElement of an user, * an user id or * an anonymous user. Raises an TypeError, if the given user object can not be converted. """ User = get_user_model() if user is None: # Nothing to do pass elif isinstance(user, CollectionElement) and user.collection_string == User.get_collection_string(): # Nothing to do pass elif isinstance(user, CollectionElement): raise TypeError( "Unsupported type for user. Only CollectionElements for users can be" "used. Not {}".format(user.collection_string)) elif isinstance(user, int): user = CollectionElement.from_values(User.get_collection_string(), user) elif isinstance(user, AnonymousUser): user = None elif isinstance(user, User): # Converts a user object to a collection element. # from_instance can not be used because the user serializer loads # the group from the db. So each call to from_instance(user) costs # one db query. user = CollectionElement.from_values(User.get_collection_string(), user.id) else: raise TypeError( "Unsupported type for user. User {} has type {}.".format(user, type(user))) return user