Merge branch 'main' into 21842-improve-sharing
[arvados.git] / services / workbench2 / cypress / e2e / group-manage.cy.js
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 describe('Group manage tests', function() {
6     let activeUser;
7     let adminUser;
8     let otherUser;
9     let userThree;
10     const groupName = `Test group (${Math.floor(999999 * Math.random())})`;
11
12     before(function() {
13         // Only set up common users once. These aren't set up as aliases because
14         // aliases are cleaned up after every test. Also it doesn't make sense
15         // to set the same users on beforeEach() over and over again, so we
16         // separate a little from Cypress' 'Best Practices' here.
17         cy.getUser('admin', 'Admin', 'User', true, true)
18             .as('adminUser').then(function() {
19                 adminUser = this.adminUser;
20             }
21         );
22         cy.getUser('user', 'Active', 'User', false, true)
23             .as('activeUser').then(function() {
24                 activeUser = this.activeUser;
25             }
26         );
27         cy.getUser('otheruser', 'Other', 'User', false, true)
28             .as('otherUser').then(function() {
29                 otherUser = this.otherUser;
30             }
31         );
32         cy.getUser('userThree', 'User', 'Three', false, true)
33             .as('userThree').then(function() {
34                 userThree = this.userThree;
35             }
36         );
37     });
38
39     it('creates a new group, add users to it and changes permission level', function() {
40         cy.loginAs(activeUser);
41
42         // Navigate to Groups
43         cy.get('[data-cy=side-panel-tree]').contains('Groups').click();
44
45         // Create new group
46         cy.get('[data-cy=groups-panel-new-group]').click();
47         cy.get('[data-cy=form-dialog]')
48             .should('contain', 'New Group')
49             .within(() => {
50                 cy.get('input[name=name]').type(groupName);
51                 cy.get('[data-cy=users-field] input').type("three");
52             });
53         cy.get('[role=tooltip]').click();
54         cy.get('[data-cy=form-dialog]').within(() => {
55             cy.get('[data-cy=form-submit-btn]').click();
56         })
57
58         // Check that the group was created
59         cy.get('[data-cy=groups-panel-data-explorer]').contains(groupName).click();
60         cy.get('[data-cy=group-members-data-explorer]').contains(activeUser.user.full_name);
61         cy.get('[data-cy=group-members-data-explorer]').contains(userThree.user.full_name);
62
63         // Add other user to the group
64         cy.get('[data-cy=group-member-add]').click();
65         cy.get('.sharing-dialog')
66             .should('contain', 'Sharing settings')
67             .within(() => {
68                 cy.get('[data-cy=invite-people-field] input').type("other");
69             });
70         cy.get('[data-cy="users-tab-label"]').click();
71         cy.waitForDom();
72         cy.get('[data-cy=sharing-suggestion]').click();
73         // Add admin to the group
74         cy.get('.sharing-dialog')
75             .should('contain', 'Sharing settings')
76             .within(() => {
77                 cy.get('[data-cy=invite-people-field] input').type("admin");
78             });
79         cy.get('[data-cy="users-tab-label"]').click();
80         cy.waitForDom();
81         cy.get('[data-cy=sharing-suggestion]').click();
82         cy.get('.sharing-dialog').get('[data-cy=add-invited-people]').click();
83         cy.get('.sharing-dialog').contains('Close').click();
84
85         // Check that both users are present with appropriate permissions
86         cy.get('[data-cy=group-members-data-explorer]')
87             .contains(otherUser.user.full_name)
88             .parents('tr')
89             .within(() => {
90                 cy.contains('Read');
91             });
92         cy.get('[data-cy=group-members-data-explorer] tr')
93             .contains(activeUser.user.full_name)
94             .parents('tr')
95             .within(() => {
96                 cy.contains('Manage');
97             });
98
99         // Test change permission level
100         cy.get('[data-cy=group-members-data-explorer]')
101             .contains(otherUser.user.full_name)
102             .parents('tr')
103             .within(() => {
104                 cy.contains('Read')
105                     .parents('td')
106                     .within(() => {
107                         cy.get('button').click();
108                     });
109             });
110         cy.get('[data-cy=context-menu]')
111             .contains('Write')
112             .click();
113         cy.get('[data-cy=group-members-data-explorer]')
114             .contains(otherUser.user.full_name)
115             .parents('tr')
116             .within(() => {
117                 cy.contains('Write');
118             });
119
120         // Change admin to manage
121         cy.get('[data-cy=group-members-data-explorer]')
122             .contains(adminUser.user.full_name)
123             .parents('tr')
124             .within(() => {
125                 cy.contains('Read')
126                     .parents('td')
127                     .within(() => {
128                         cy.get('button').click();
129                     });
130             });
131         cy.get('[data-cy=context-menu]')
132             .contains('Manage')
133             .click();
134         cy.get('[data-cy=group-members-data-explorer]')
135             .contains(adminUser.user.full_name)
136             .parents('tr')
137             .within(() => {
138                 cy.contains('Manage');
139             });
140     });
141
142     it('can unhide and re-hide users', function() {
143         // Must use admin user to have manage permission on user
144         cy.loginAs(adminUser);
145         cy.get('[data-cy=side-panel-tree]').contains('Groups').click();
146         cy.get('[data-cy=groups-panel-data-explorer]').contains(groupName).click();
147
148         // Check that other user is hidden
149         cy.get('[data-cy=group-details-permissions-tab]').click();
150         cy.get('[data-cy=group-permissions-data-explorer]')
151             .should('not.contain', otherUser.user.full_name)
152         cy.get('[data-cy=group-details-members-tab]').click();
153
154         // Test unhide
155         cy.get('[data-cy=group-members-data-explorer]')
156             .contains(otherUser.user.full_name)
157             .parents('tr')
158             .within(() => {
159                 cy.get('[data-cy=user-visible-checkbox]').click();
160             });
161         // Check that other user is visible
162         cy.get('[data-cy=group-details-permissions-tab]').click();
163         cy.get('[data-cy=group-permissions-data-explorer]')
164             .contains(otherUser.user.full_name)
165             .parents('tr')
166             .within(() => {
167                 cy.contains('Read');
168             });
169         // Test re-hide
170         cy.get('[data-cy=group-details-members-tab]').click();
171         cy.get('[data-cy=group-members-data-explorer]')
172             .contains(otherUser.user.full_name)
173             .parents('tr')
174             .within(() => {
175                 cy.get('[data-cy=user-visible-checkbox]').click();
176             });
177         // Check that other user is hidden
178         cy.get('[data-cy=group-details-permissions-tab]').click();
179         cy.get('[data-cy=group-permissions-data-explorer]')
180             .should('not.contain', otherUser.user.full_name)
181     });
182
183     it('displays resources shared with the group', function() {
184         // Switch to activeUser
185         cy.loginAs(activeUser);
186         cy.get('[data-cy=side-panel-tree]').contains('Groups').click();
187
188         // Get groupUuid and create shared project
189         cy.get('[data-cy=groups-panel-data-explorer]')
190             .contains(groupName)
191             .parents('tr')
192             .find('[data-cy=uuid]')
193             .invoke('text')
194             .as('groupUuid')
195             .then((groupUuid) => {
196                 cy.createProject({
197                     owningUser: activeUser,
198                     projectName: 'test-project',
199                 }).as('testProject').then((testProject) => {
200                     cy.shareWith(activeUser.token, groupUuid, testProject.uuid, 'can_read');
201                 });
202             });
203
204         // Check that the project is listed in permissions
205         cy.get('[data-cy=groups-panel-data-explorer]').contains(groupName).click();
206         cy.get('[data-cy=group-details-permissions-tab]').click();
207         cy.get('[data-cy=group-permissions-data-explorer]')
208             .contains('test-project')
209             .parents('tr')
210             .within(() => {
211                 cy.contains('Read');
212             });
213     });
214
215     it('removes users from the group', function() {
216         cy.loginAs(activeUser);
217
218         cy.get('[data-cy=side-panel-tree]').contains('Groups').click();
219         cy.get('[data-cy=groups-panel-data-explorer]').contains(groupName).click();
220
221         // Remove other user
222         cy.get('[data-cy=group-members-data-explorer]')
223             .contains(otherUser.user.full_name)
224             .parents('tr')
225             .within(() => {
226                 cy.get('[data-cy=resource-delete-button]').click();
227             });
228         cy.get('[data-cy=confirmation-dialog-ok-btn]').click();
229         cy.get('[data-cy=group-members-data-explorer]')
230             .should('not.contain', otherUser.user.full_name);
231
232         // Remove user three
233         cy.get('[data-cy=group-members-data-explorer]')
234             .contains(userThree.user.full_name)
235             .parents('tr')
236             .within(() => {
237                 cy.get('[data-cy=resource-delete-button]').click();
238             });
239         cy.get('[data-cy=confirmation-dialog-ok-btn]').click();
240         cy.get('[data-cy=group-members-data-explorer]')
241             .should('not.contain', userThree.user.full_name);
242     });
243
244     it('renames the group', function() {
245         cy.loginAs(adminUser);
246         // Navigate to Groups
247         cy.get('[data-cy=side-panel-tree]').contains('Groups').click();
248
249         // Open rename dialog
250         cy.get('[data-cy=groups-panel-data-explorer]')
251             .contains(groupName)
252             .rightclick();
253         cy.get('[data-cy=context-menu]')
254             .contains('Rename')
255             .click();
256
257         // Rename the group
258         cy.get('[data-cy=form-dialog]')
259             .should('contain', 'Edit Group')
260             .within(() => {
261                 cy.get('input[name=name]').clear().type(groupName + ' (renamed)');
262                 cy.get('button').contains('Save').click();
263             });
264
265         // Check that the group was renamed
266         cy.get('[data-cy=groups-panel-data-explorer]')
267             .contains(groupName + ' (renamed)');
268     });
269
270     it('deletes the group', function() {
271         cy.loginAs(adminUser);
272
273         // Navigate to Groups
274         cy.get('[data-cy=side-panel-tree]').contains('Groups').click();
275
276         // Delete the group
277         cy.get('[data-cy=groups-panel-data-explorer]')
278             .contains(groupName + ' (renamed)')
279             .rightclick();
280         cy.get('[data-cy=context-menu]')
281             .contains('Remove')
282             .click();
283         cy.get('[data-cy=confirmation-dialog-ok-btn]').click();
284
285         // Check that the group was deleted
286         cy.get('[data-cy=groups-panel-data-explorer]')
287             .should('not.contain', groupName + ' (renamed)');
288     });
289
290     it('disables group-related controls for built-in groups', function() {
291         cy.loginAs(adminUser);
292
293         ['All users', 'Anonymous users', 'System group'].forEach((builtInGroup) => {
294             cy.get('[data-cy=side-panel-tree]').contains('Groups').click();
295             cy.get('[data-cy=groups-panel-data-explorer]').contains(builtInGroup).click();
296
297             // Check group member actions
298             cy.get('[data-cy=group-members-data-explorer]')
299                 .within(() => {
300                     cy.get('[data-cy=group-member-add]').should('not.exist');
301                     cy.get('[data-cy=user-visible-checkbox] input').should('be.disabled');
302                     cy.get('[data-cy=resource-delete-button]').should('be.disabled');
303                     cy.get('[data-cy=edit-permission-button]').should('not.exist');
304                 });
305
306             // Check permissions actions
307             cy.get('[data-cy=group-details-permissions-tab]').click();
308             cy.get('[data-cy=group-permissions-data-explorer]').within(() => {
309                 cy.get('[data-cy=resource-delete-button]').should('be.disabled');
310                 cy.get('[data-cy=edit-permission-button]').should('not.exist');
311             });
312         });
313     });
314
315 });