eead4efe6a
* Use user_id: int instead of Optional[CollectionElment] in utils * Rewrote autoupdate system without CollectionElement
90 lines
3.8 KiB
Python
90 lines
3.8 KiB
Python
from typing import Any, Dict, Iterable, List
|
|
|
|
from ..utils.access_permissions import BaseAccessPermissions
|
|
from ..utils.auth import async_has_perm
|
|
|
|
|
|
class ItemAccessPermissions(BaseAccessPermissions):
|
|
"""
|
|
Access permissions container for Item and ItemViewSet.
|
|
"""
|
|
base_permission = 'agenda.can_see'
|
|
|
|
def get_serializer_class(self, user=None):
|
|
"""
|
|
Returns serializer class.
|
|
"""
|
|
from .serializers import ItemSerializer
|
|
|
|
return ItemSerializer
|
|
|
|
# TODO: In the following method we use full_data['is_hidden'] and
|
|
# full_data['is_internal'] but this can be out of date.
|
|
|
|
async def get_restricted_data(
|
|
self,
|
|
full_data: List[Dict[str, Any]],
|
|
user_id: int) -> List[Dict[str, Any]]:
|
|
"""
|
|
Returns the restricted serialized data for the instance prepared
|
|
for the user.
|
|
|
|
Hidden items can only be seen by managers with can_manage permission.
|
|
|
|
We remove comments for non admins/managers and a lot of fields of
|
|
internal items for users without permission to see internal items.
|
|
"""
|
|
def filtered_data(full_data, blocked_keys):
|
|
"""
|
|
Returns a new dict like full_data but with all blocked_keys removed.
|
|
"""
|
|
whitelist = full_data.keys() - blocked_keys
|
|
return {key: full_data[key] for key in whitelist}
|
|
|
|
# Parse data.
|
|
if full_data and await async_has_perm(user_id, 'agenda.can_see'):
|
|
if await async_has_perm(user_id, 'agenda.can_manage') and await async_has_perm(user_id, 'agenda.can_see_internal_items'):
|
|
# Managers with special permission can see everything.
|
|
data = full_data
|
|
elif await async_has_perm(user_id, 'agenda.can_see_internal_items'):
|
|
# Non managers with special permission can see everything but
|
|
# comments and hidden items.
|
|
data = [full for full in full_data if not full['is_hidden']] # filter hidden items
|
|
blocked_keys = ('comment',)
|
|
data = [filtered_data(full, blocked_keys) for full in data] # remove blocked_keys
|
|
else:
|
|
# Users without special permission for internal items.
|
|
|
|
# In internal and hidden case managers and non managers see only some fields
|
|
# so that list of speakers is provided regardless. Hidden items can only be seen by managers.
|
|
# We know that full_data has at least one entry which can be used to parse the keys.
|
|
blocked_keys_internal_hidden_case = set(full_data[0].keys()) - set((
|
|
'id',
|
|
'title',
|
|
'speakers',
|
|
'speaker_list_closed',
|
|
'content_object'))
|
|
|
|
# In non internal case managers see everything and non managers see
|
|
# everything but comments.
|
|
if await async_has_perm(user_id, 'agenda.can_manage'):
|
|
blocked_keys_non_internal_hidden_case: Iterable[str] = []
|
|
can_see_hidden = True
|
|
else:
|
|
blocked_keys_non_internal_hidden_case = ('comment',)
|
|
can_see_hidden = False
|
|
|
|
data = []
|
|
for full in full_data:
|
|
if full['is_hidden'] and can_see_hidden:
|
|
# Same filtering for internal and hidden items
|
|
data.append(filtered_data(full, blocked_keys_internal_hidden_case))
|
|
elif full['is_internal']:
|
|
data.append(filtered_data(full, blocked_keys_internal_hidden_case))
|
|
else: # agenda item
|
|
data.append(filtered_data(full, blocked_keys_non_internal_hidden_case))
|
|
else:
|
|
data = []
|
|
|
|
return data
|