2017-09-25 09:47:15 +02:00
|
|
|
# -*- coding: utf-8 -*-
|
2018-05-11 01:39:38 +02:00
|
|
|
from typing import Any, List, Optional
|
2017-09-25 09:47:15 +02:00
|
|
|
|
2017-11-01 10:04:16 +01:00
|
|
|
import ujson
|
2017-09-25 09:47:15 +02:00
|
|
|
import django
|
|
|
|
import mock
|
|
|
|
|
|
|
|
from zerver.lib.test_classes import ZulipTestCase
|
|
|
|
from zerver.lib.user_groups import (
|
|
|
|
check_add_user_to_user_group,
|
|
|
|
check_remove_user_from_user_group,
|
|
|
|
create_user_group,
|
|
|
|
get_user_groups,
|
|
|
|
user_groups_in_realm,
|
2017-11-02 08:53:30 +01:00
|
|
|
get_memberships_of_users,
|
2017-11-07 07:56:26 +01:00
|
|
|
user_groups_in_realm_serialized,
|
2017-09-25 09:47:15 +02:00
|
|
|
)
|
2017-11-02 08:15:14 +01:00
|
|
|
from zerver.models import UserProfile, UserGroup, get_realm, Realm, \
|
|
|
|
UserGroupMembership
|
2017-09-25 09:47:15 +02:00
|
|
|
|
|
|
|
class UserGroupTestCase(ZulipTestCase):
|
2018-05-11 01:39:38 +02:00
|
|
|
def create_user_group_for_test(self, group_name: str,
|
2017-11-20 03:22:57 +01:00
|
|
|
realm: Realm=get_realm('zulip')) -> UserGroup:
|
2017-09-25 09:47:15 +02:00
|
|
|
members = [self.example_user('othello')]
|
|
|
|
return create_user_group(group_name, members, realm)
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_user_groups_in_realm(self) -> None:
|
2017-09-25 09:47:15 +02:00
|
|
|
realm = get_realm('zulip')
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(len(user_groups_in_realm(realm)), 1)
|
2017-09-25 09:47:15 +02:00
|
|
|
self.create_user_group_for_test('support')
|
|
|
|
user_groups = user_groups_in_realm(realm)
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(len(user_groups), 2)
|
|
|
|
names = set([ug.name for ug in user_groups])
|
|
|
|
self.assertEqual(names, set(['hamletcharacters', 'support']))
|
2017-09-25 09:47:15 +02:00
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_user_groups_in_realm_serialized(self) -> None:
|
2017-11-07 07:56:26 +01:00
|
|
|
realm = get_realm('zulip')
|
2017-11-13 07:49:01 +01:00
|
|
|
user_group = UserGroup.objects.first()
|
|
|
|
membership = UserGroupMembership.objects.filter(user_group=user_group)
|
|
|
|
membership = membership.values_list('user_profile_id', flat=True)
|
2017-11-30 01:09:23 +01:00
|
|
|
empty_user_group = create_user_group('newgroup', [], realm)
|
|
|
|
|
2017-11-13 07:49:01 +01:00
|
|
|
user_groups = user_groups_in_realm_serialized(realm)
|
2017-11-30 01:09:23 +01:00
|
|
|
self.assertEqual(len(user_groups), 2)
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(user_groups[0]['id'], user_group.id)
|
|
|
|
self.assertEqual(user_groups[0]['name'], 'hamletcharacters')
|
|
|
|
self.assertEqual(user_groups[0]['description'], 'Characters of Hamlet')
|
|
|
|
self.assertEqual(set(user_groups[0]['members']), set(membership))
|
2017-11-07 07:56:26 +01:00
|
|
|
|
2017-11-30 01:09:23 +01:00
|
|
|
self.assertEqual(user_groups[1]['id'], empty_user_group.id)
|
|
|
|
self.assertEqual(user_groups[1]['name'], 'newgroup')
|
|
|
|
self.assertEqual(user_groups[1]['description'], '')
|
|
|
|
self.assertEqual(user_groups[1]['members'], [])
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_get_user_groups(self) -> None:
|
2017-09-25 09:47:15 +02:00
|
|
|
othello = self.example_user('othello')
|
|
|
|
self.create_user_group_for_test('support')
|
|
|
|
user_groups = get_user_groups(othello)
|
|
|
|
self.assertEqual(len(user_groups), 1)
|
|
|
|
self.assertEqual(user_groups[0].name, 'support')
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_check_add_user_to_user_group(self) -> None:
|
2017-09-25 09:47:15 +02:00
|
|
|
user_group = self.create_user_group_for_test('support')
|
|
|
|
hamlet = self.example_user('hamlet')
|
|
|
|
self.assertTrue(check_add_user_to_user_group(hamlet, user_group))
|
|
|
|
self.assertFalse(check_add_user_to_user_group(hamlet, user_group))
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_check_remove_user_from_user_group(self) -> None:
|
2017-09-25 09:47:15 +02:00
|
|
|
user_group = self.create_user_group_for_test('support')
|
|
|
|
othello = self.example_user('othello')
|
|
|
|
self.assertTrue(check_remove_user_from_user_group(othello, user_group))
|
|
|
|
self.assertFalse(check_remove_user_from_user_group(othello, user_group))
|
|
|
|
|
|
|
|
with mock.patch('zerver.lib.user_groups.remove_user_from_user_group',
|
|
|
|
side_effect=Exception):
|
|
|
|
self.assertFalse(check_remove_user_from_user_group(othello, user_group))
|
2017-11-01 10:04:16 +01:00
|
|
|
|
|
|
|
class UserGroupAPITestCase(ZulipTestCase):
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_user_group_create(self) -> None:
|
2017-11-01 10:04:16 +01:00
|
|
|
hamlet = self.example_user('hamlet')
|
|
|
|
|
|
|
|
# Test success
|
|
|
|
self.login(self.example_email("hamlet"))
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([hamlet.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
result = self.client_post('/json/user_groups/create', info=params)
|
|
|
|
self.assert_json_success(result)
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assert_length(UserGroup.objects.all(), 2)
|
2017-11-01 10:04:16 +01:00
|
|
|
|
|
|
|
# Test invalid member error
|
|
|
|
params = {
|
|
|
|
'name': 'backend',
|
|
|
|
'members': ujson.dumps([1111]),
|
|
|
|
'description': 'Backend team',
|
|
|
|
}
|
|
|
|
result = self.client_post('/json/user_groups/create', info=params)
|
|
|
|
self.assert_json_error(result, "Invalid user ID: 1111")
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assert_length(UserGroup.objects.all(), 2)
|
2017-11-01 10:04:16 +01:00
|
|
|
|
|
|
|
# Test we cannot add hamlet again
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([hamlet.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
result = self.client_post('/json/user_groups/create', info=params)
|
|
|
|
self.assert_json_error(result, "User group 'support' already exists.")
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assert_length(UserGroup.objects.all(), 2)
|
2017-11-02 07:53:08 +01:00
|
|
|
|
2018-06-14 08:35:05 +02:00
|
|
|
def test_user_group_create_by_guest_user(self) -> None:
|
|
|
|
guest_user = self.example_user('polonius')
|
|
|
|
|
|
|
|
# Guest users can't create user group
|
|
|
|
self.login(guest_user.email)
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([guest_user.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
result = self.client_post('/json/user_groups/create', info=params)
|
|
|
|
self.assert_json_error(result, "Not allowed for guest users")
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_user_group_update(self) -> None:
|
2017-11-02 07:53:08 +01:00
|
|
|
hamlet = self.example_user('hamlet')
|
|
|
|
self.login(self.example_email("hamlet"))
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([hamlet.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
self.client_post('/json/user_groups/create', info=params)
|
2018-02-19 13:38:18 +01:00
|
|
|
user_group = UserGroup.objects.get(name='support')
|
2017-11-02 07:53:08 +01:00
|
|
|
# Test success
|
|
|
|
params = {
|
|
|
|
'name': 'help',
|
|
|
|
'description': 'Troubleshooting team',
|
|
|
|
}
|
|
|
|
result = self.client_patch('/json/user_groups/{}'.format(user_group.id), info=params)
|
|
|
|
self.assert_json_success(result)
|
|
|
|
self.assertEqual(result.json()['name'], 'Name successfully updated.')
|
|
|
|
self.assertEqual(result.json()['description'], 'Description successfully updated.')
|
|
|
|
|
|
|
|
# Test when new data is not supplied.
|
|
|
|
result = self.client_patch('/json/user_groups/{}'.format(user_group.id), info={})
|
|
|
|
self.assert_json_error(result, "No new data supplied")
|
|
|
|
|
|
|
|
# Test when invalid user group is supplied
|
|
|
|
params = {'name': 'help'}
|
|
|
|
result = self.client_patch('/json/user_groups/1111', info=params)
|
|
|
|
self.assert_json_error(result, "Invalid user group")
|
2017-11-02 08:15:14 +01:00
|
|
|
|
2018-02-19 13:38:18 +01:00
|
|
|
self.logout()
|
|
|
|
# Test when user not a member of user group tries to modify it
|
|
|
|
cordelia = self.example_user('cordelia')
|
|
|
|
self.login(cordelia.email)
|
|
|
|
params = {
|
|
|
|
'name': 'help',
|
|
|
|
'description': 'Troubleshooting',
|
|
|
|
}
|
|
|
|
result = self.client_patch('/json/user_groups/{}'.format(user_group.id), info=params)
|
|
|
|
self.assert_json_error(result, "Only group members and organization administrators can administer this group.")
|
|
|
|
|
|
|
|
self.logout()
|
|
|
|
# Test when organization admin tries to modify group
|
|
|
|
iago = self.example_user('iago')
|
|
|
|
self.login(iago.email)
|
|
|
|
params = {
|
|
|
|
'name': 'help',
|
|
|
|
'description': 'Troubleshooting',
|
|
|
|
}
|
|
|
|
result = self.client_patch('/json/user_groups/{}'.format(user_group.id), info=params)
|
|
|
|
self.assert_json_success(result)
|
|
|
|
self.assertEqual(result.json()['description'], 'Description successfully updated.')
|
|
|
|
|
2018-06-14 08:35:05 +02:00
|
|
|
def test_user_group_update_by_guest_user(self) -> None:
|
|
|
|
hamlet = self.example_user('hamlet')
|
|
|
|
guest_user = self.example_user('polonius')
|
|
|
|
self.login(hamlet.email)
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([hamlet.id, guest_user.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
result = self.client_post('/json/user_groups/create', info=params)
|
|
|
|
self.assert_json_success(result)
|
|
|
|
user_group = UserGroup.objects.get(name='support')
|
|
|
|
|
|
|
|
# Guest user can't edit any detail of an user group
|
|
|
|
self.login(guest_user.email)
|
|
|
|
params = {
|
|
|
|
'name': 'help',
|
|
|
|
'description': 'Troubleshooting team',
|
|
|
|
}
|
|
|
|
result = self.client_patch('/json/user_groups/{}'.format(user_group.id), info=params)
|
|
|
|
self.assert_json_error(result, "Not allowed for guest users")
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_user_group_delete(self) -> None:
|
2017-11-02 08:15:14 +01:00
|
|
|
hamlet = self.example_user('hamlet')
|
|
|
|
self.login(self.example_email("hamlet"))
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([hamlet.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
self.client_post('/json/user_groups/create', info=params)
|
2017-11-13 07:49:01 +01:00
|
|
|
user_group = UserGroup.objects.get(name='support')
|
2017-11-02 08:15:14 +01:00
|
|
|
# Test success
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(UserGroup.objects.count(), 2)
|
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 3)
|
2017-11-02 08:15:14 +01:00
|
|
|
result = self.client_delete('/json/user_groups/{}'.format(user_group.id))
|
|
|
|
self.assert_json_success(result)
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(UserGroup.objects.count(), 1)
|
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 2)
|
2017-11-02 08:15:14 +01:00
|
|
|
|
|
|
|
# Test when invalid user group is supplied
|
|
|
|
result = self.client_delete('/json/user_groups/1111')
|
|
|
|
self.assert_json_error(result, "Invalid user group")
|
2017-11-02 08:53:30 +01:00
|
|
|
|
2018-02-19 13:38:18 +01:00
|
|
|
# Test when user not a member of user group tries to delete it
|
|
|
|
params = {
|
|
|
|
'name': 'Development',
|
|
|
|
'members': ujson.dumps([hamlet.id]),
|
|
|
|
'description': 'Development team',
|
|
|
|
}
|
|
|
|
self.client_post('/json/user_groups/create', info=params)
|
|
|
|
user_group = UserGroup.objects.get(name='Development')
|
|
|
|
self.assertEqual(UserGroup.objects.count(), 2)
|
|
|
|
self.logout()
|
|
|
|
cordelia = self.example_user('cordelia')
|
|
|
|
self.login(cordelia.email)
|
|
|
|
|
|
|
|
result = self.client_delete('/json/user_groups/{}'.format(user_group.id))
|
|
|
|
self.assert_json_error(result, "Only group members and organization administrators can administer this group.")
|
|
|
|
self.assertEqual(UserGroup.objects.count(), 2)
|
|
|
|
|
|
|
|
self.logout()
|
|
|
|
# Test when organization admin tries to delete group
|
|
|
|
iago = self.example_user('iago')
|
|
|
|
self.login(iago.email)
|
|
|
|
|
|
|
|
result = self.client_delete('/json/user_groups/{}'.format(user_group.id))
|
|
|
|
self.assert_json_success(result)
|
|
|
|
self.assertEqual(UserGroup.objects.count(), 1)
|
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 2)
|
|
|
|
|
2018-06-14 08:35:05 +02:00
|
|
|
def test_user_group_delete_by_guest_user(self) -> None:
|
|
|
|
hamlet = self.example_user('hamlet')
|
|
|
|
guest_user = self.example_user('polonius')
|
|
|
|
self.login(hamlet.email)
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([hamlet.id, guest_user.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
result = self.client_post('/json/user_groups/create', info=params)
|
|
|
|
self.assert_json_success(result)
|
|
|
|
user_group = UserGroup.objects.get(name='support')
|
|
|
|
|
|
|
|
# Guest users can't delete any user group(not even those of which they are a member)
|
|
|
|
self.login(guest_user.email)
|
|
|
|
result = self.client_delete('/json/user_groups/{}'.format(user_group.id))
|
|
|
|
self.assert_json_error(result, "Not allowed for guest users")
|
|
|
|
|
2017-11-05 10:51:25 +01:00
|
|
|
def test_update_members_of_user_group(self) -> None:
|
2017-11-02 08:53:30 +01:00
|
|
|
hamlet = self.example_user('hamlet')
|
|
|
|
self.login(self.example_email("hamlet"))
|
|
|
|
params = {
|
|
|
|
'name': 'support',
|
|
|
|
'members': ujson.dumps([hamlet.id]),
|
|
|
|
'description': 'Support team',
|
|
|
|
}
|
|
|
|
self.client_post('/json/user_groups/create', info=params)
|
2018-02-19 13:38:18 +01:00
|
|
|
user_group = UserGroup.objects.get(name='support')
|
2017-11-02 08:53:30 +01:00
|
|
|
# Test add members
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 3)
|
2018-02-19 13:38:18 +01:00
|
|
|
|
2017-11-02 08:53:30 +01:00
|
|
|
othello = self.example_user('othello')
|
|
|
|
add = [othello.id]
|
|
|
|
params = {'add': ujson.dumps(add)}
|
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_success(result)
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 4)
|
2017-11-02 08:53:30 +01:00
|
|
|
members = get_memberships_of_users(user_group, [hamlet, othello])
|
|
|
|
self.assertEqual(len(members), 2)
|
|
|
|
|
|
|
|
# Test adding a member already there.
|
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_error(result, "User 6 is already a member of this group")
|
2017-11-13 07:49:01 +01:00
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 4)
|
2017-11-02 08:53:30 +01:00
|
|
|
members = get_memberships_of_users(user_group, [hamlet, othello])
|
|
|
|
self.assertEqual(len(members), 2)
|
|
|
|
|
2018-02-19 13:38:18 +01:00
|
|
|
self.logout()
|
|
|
|
# Test when user not a member of user group tries to add members to it
|
|
|
|
cordelia = self.example_user('cordelia')
|
|
|
|
self.login(cordelia.email)
|
|
|
|
add = [cordelia.id]
|
|
|
|
params = {'add': ujson.dumps(add)}
|
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_error(result, "Only group members and organization administrators can administer this group.")
|
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 4)
|
|
|
|
|
|
|
|
self.logout()
|
|
|
|
# Test when organization admin tries to add members to group
|
|
|
|
iago = self.example_user('iago')
|
|
|
|
self.login(iago.email)
|
|
|
|
aaron = self.example_user('aaron')
|
|
|
|
add = [aaron.id]
|
|
|
|
params = {'add': ujson.dumps(add)}
|
2017-11-02 08:53:30 +01:00
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_success(result)
|
2018-02-19 13:38:18 +01:00
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 5)
|
|
|
|
members = get_memberships_of_users(user_group, [hamlet, othello, aaron])
|
|
|
|
self.assertEqual(len(members), 3)
|
2017-11-02 08:53:30 +01:00
|
|
|
|
2018-02-19 13:38:18 +01:00
|
|
|
# For normal testing we again login with hamlet
|
|
|
|
self.logout()
|
|
|
|
self.login(hamlet.email)
|
|
|
|
# Test remove members
|
|
|
|
params = {'delete': ujson.dumps([othello.id])}
|
2017-11-02 08:53:30 +01:00
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_success(result)
|
2018-02-19 13:38:18 +01:00
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 4)
|
|
|
|
members = get_memberships_of_users(user_group, [hamlet, othello, aaron])
|
|
|
|
self.assertEqual(len(members), 2)
|
|
|
|
|
|
|
|
# Test remove a member that's already removed
|
|
|
|
params = {'delete': ujson.dumps([othello.id])}
|
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_error(result, "There is no member '6' in this user group")
|
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 4)
|
|
|
|
members = get_memberships_of_users(user_group, [hamlet, othello, aaron])
|
|
|
|
self.assertEqual(len(members), 2)
|
2017-11-02 08:53:30 +01:00
|
|
|
|
|
|
|
# Test when nothing is provided
|
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info={})
|
|
|
|
msg = 'Nothing to do. Specify at least one of "add" or "delete".'
|
|
|
|
self.assert_json_error(result, msg)
|
2018-02-19 13:38:18 +01:00
|
|
|
|
|
|
|
# Test when user not a member of user group tries to remove members
|
|
|
|
self.logout()
|
|
|
|
self.login(cordelia.email)
|
|
|
|
params = {'delete': ujson.dumps([hamlet.id])}
|
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_error(result, "Only group members and organization administrators can administer this group.")
|
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 4)
|
|
|
|
|
|
|
|
self.logout()
|
|
|
|
# Test when organization admin tries to remove members from group
|
|
|
|
iago = self.example_user('iago')
|
|
|
|
self.login(iago.email)
|
|
|
|
result = self.client_post('/json/user_groups/{}/members'.format(user_group.id),
|
|
|
|
info=params)
|
|
|
|
self.assert_json_success(result)
|
|
|
|
self.assertEqual(UserGroupMembership.objects.count(), 3)
|
|
|
|
members = get_memberships_of_users(user_group, [hamlet, othello, aaron])
|
|
|
|
self.assertEqual(len(members), 1)
|