15524: Shows groups names on sharing dialog's permission rows where applicable.
[arvados-workbench2.git] / src / store / sharing-dialog / sharing-dialog-actions.ts
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import { dialogActions } from "~/store/dialog/dialog-actions";
6 import { withDialog } from "~/store/dialog/with-dialog";
7 import { SHARING_DIALOG_NAME, SharingPublicAccessFormData, SHARING_PUBLIC_ACCESS_FORM_NAME, SHARING_INVITATION_FORM_NAME, SharingManagementFormData, SharingInvitationFormData, VisibilityLevel, getSharingMangementFormData, getSharingPublicAccessFormData } from './sharing-dialog-types';
8 import { Dispatch } from 'redux';
9 import { ServiceRepository } from "~/services/services";
10 import { FilterBuilder } from '~/services/api/filter-builder';
11 import { initialize, getFormValues, reset } from 'redux-form';
12 import { SHARING_MANAGEMENT_FORM_NAME } from '~/store/sharing-dialog/sharing-dialog-types';
13 import { RootState } from '~/store/store';
14 import { getDialog } from '~/store/dialog/dialog-reducer';
15 import { PermissionLevel } from '~/models/permission';
16 import { getPublicGroupUuid } from "~/store/workflow-panel/workflow-panel-actions";
17 import { PermissionResource } from '~/models/permission';
18 import { differenceWith } from "lodash";
19 import { withProgress } from "~/store/progress-indicator/with-progress";
20 import { progressIndicatorActions } from '~/store/progress-indicator/progress-indicator-actions.ts';
21 import { snackbarActions, SnackbarKind } from "../snackbar/snackbar-actions";
22 import { extractUuidKind, ResourceKind } from "~/models/resource";
23 import { LinkClass } from "~/models/link";
24
25 export const openSharingDialog = (resourceUuid: string) =>
26     (dispatch: Dispatch) => {
27         dispatch(dialogActions.OPEN_DIALOG({ id: SHARING_DIALOG_NAME, data: resourceUuid }));
28         dispatch<any>(loadSharingDialog);
29     };
30
31 export const closeSharingDialog = () =>
32     dialogActions.CLOSE_DIALOG({ id: SHARING_DIALOG_NAME });
33
34 export const connectSharingDialog = withDialog(SHARING_DIALOG_NAME);
35 export const connectSharingDialogProgress = withProgress(SHARING_DIALOG_NAME);
36
37
38 export const saveSharingDialogChanges = async (dispatch: Dispatch) => {
39     dispatch(progressIndicatorActions.START_WORKING(SHARING_DIALOG_NAME));
40     await dispatch<any>(savePublicPermissionChanges);
41     await dispatch<any>(saveManagementChanges);
42     await dispatch<any>(sendInvitations);
43     dispatch(reset(SHARING_INVITATION_FORM_NAME));
44     await dispatch<any>(loadSharingDialog);
45 };
46
47 export const sendSharingInvitations = async (dispatch: Dispatch) => {
48     dispatch(progressIndicatorActions.START_WORKING(SHARING_DIALOG_NAME));
49     await dispatch<any>(sendInvitations);
50     dispatch(closeSharingDialog());
51     dispatch(snackbarActions.OPEN_SNACKBAR({
52         message: 'Resource has been shared',
53         kind: SnackbarKind.SUCCESS,
54     }));
55     dispatch(progressIndicatorActions.STOP_WORKING(SHARING_DIALOG_NAME));
56 };
57
58 const loadSharingDialog = async (dispatch: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
59
60     const dialog = getDialog<string>(getState().dialog, SHARING_DIALOG_NAME);
61     if (dialog) {
62         dispatch(progressIndicatorActions.START_WORKING(SHARING_DIALOG_NAME));
63         try {
64             const { items } = await permissionService.listResourcePermissions(dialog.data);
65             dispatch<any>(initializePublicAccessForm(items));
66             await dispatch<any>(initializeManagementForm(items));
67             dispatch(progressIndicatorActions.STOP_WORKING(SHARING_DIALOG_NAME));
68         } catch (e) {
69             dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'You do not have access to share this item', hideDuration: 2000, kind: SnackbarKind.ERROR }));
70             dispatch(dialogActions.CLOSE_DIALOG({ id: SHARING_DIALOG_NAME }));
71             dispatch(progressIndicatorActions.STOP_WORKING(SHARING_DIALOG_NAME));
72         }
73     }
74 };
75
76 const initializeManagementForm = (permissionLinks: PermissionResource[]) =>
77     async (dispatch: Dispatch, getState: () => RootState, { userService, groupsService }: ServiceRepository) => {
78
79         const filters = new FilterBuilder()
80             .addIn('uuid', permissionLinks.map(({ tailUuid }) => tailUuid))
81             .getFilters();
82
83         const { items: users } = await userService.list({ filters });
84         const { items: groups} = await groupsService.list({ filters });
85
86         const getEmail = (tailUuid: string) => {
87             const user = users.find(({ uuid }) => uuid === tailUuid);
88             const group = groups.find(({ uuid }) => uuid === tailUuid);
89             return user
90                 ? user.email
91                 : group
92                     ? group.name
93                     : tailUuid;
94         };
95
96         const managementPermissions = permissionLinks
97             .filter(item =>
98                 item.tailUuid !== getPublicGroupUuid(getState()))
99             .map(({ tailUuid, name, uuid }) => ({
100                 email: getEmail(tailUuid),
101                 permissions: name as PermissionLevel,
102                 permissionUuid: uuid,
103             }));
104
105         const managementFormData: SharingManagementFormData = {
106             permissions: managementPermissions,
107             initialPermissions: managementPermissions,
108         };
109
110         dispatch(initialize(SHARING_MANAGEMENT_FORM_NAME, managementFormData));
111     };
112
113 const initializePublicAccessForm = (permissionLinks: PermissionResource[]) =>
114     (dispatch: Dispatch, getState: () => RootState, ) => {
115
116         const [publicPermission] = permissionLinks
117             .filter(item => item.tailUuid === getPublicGroupUuid(getState()));
118
119         const publicAccessFormData: SharingPublicAccessFormData = publicPermission
120             ? {
121                 visibility: VisibilityLevel.PUBLIC,
122                 permissionUuid: publicPermission.uuid,
123             }
124             : {
125                 visibility: permissionLinks.length > 0
126                     ? VisibilityLevel.SHARED
127                     : VisibilityLevel.PRIVATE,
128                 permissionUuid: '',
129             };
130
131         dispatch(initialize(SHARING_PUBLIC_ACCESS_FORM_NAME, publicAccessFormData));
132     };
133
134 const savePublicPermissionChanges = async (_: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
135     const state = getState();
136     const { user } = state.auth;
137     const dialog = getDialog<string>(state.dialog, SHARING_DIALOG_NAME);
138     if (dialog && user) {
139         const { permissionUuid, visibility } = getSharingPublicAccessFormData(state);
140
141         if (permissionUuid) {
142             if (visibility === VisibilityLevel.PUBLIC) {
143                 await permissionService.update(permissionUuid, {
144                     name: PermissionLevel.CAN_READ
145                 });
146             } else {
147                 await permissionService.delete(permissionUuid);
148             }
149
150         } else if (visibility === VisibilityLevel.PUBLIC) {
151
152             await permissionService.create({
153                 ownerUuid: user.uuid,
154                 headUuid: dialog.data,
155                 tailUuid: getPublicGroupUuid(state),
156                 name: PermissionLevel.CAN_READ,
157             });
158         }
159     }
160 };
161
162 const saveManagementChanges = async (_: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
163     const state = getState();
164     const { user } = state.auth;
165     const dialog = getDialog<string>(state.dialog, SHARING_DIALOG_NAME);
166     if (dialog && user) {
167
168         const { initialPermissions, permissions } = getSharingMangementFormData(state);
169         const { visibility } = getSharingPublicAccessFormData(state);
170
171
172         if (visibility === VisibilityLevel.PRIVATE) {
173
174             for (const permission of initialPermissions) {
175                 await permissionService.delete(permission.permissionUuid);
176             }
177
178         } else {
179
180             const cancelledPermissions = differenceWith(
181                 initialPermissions,
182                 permissions,
183                 (a, b) => a.permissionUuid === b.permissionUuid
184             );
185
186             for (const { permissionUuid } of cancelledPermissions) {
187                 await permissionService.delete(permissionUuid);
188             }
189
190             for (const permission of permissions) {
191                 await permissionService.update(permission.permissionUuid, { name: permission.permissions });
192             }
193
194         }
195     }
196 };
197
198 const sendInvitations = async (_: Dispatch, getState: () => RootState, { permissionService, userService }: ServiceRepository) => {
199     const state = getState();
200     const { user } = state.auth;
201     const dialog = getDialog<string>(state.dialog, SHARING_DIALOG_NAME);
202     if (dialog && user) {
203         const invitations = getFormValues(SHARING_INVITATION_FORM_NAME)(state) as SharingInvitationFormData;
204
205         const getGroupsFromForm = invitations.invitedPeople.filter((invitation) => extractUuidKind(invitation.uuid) === ResourceKind.GROUP);
206         const getUsersFromForm = invitations.invitedPeople.filter((invitation) => extractUuidKind(invitation.uuid) === ResourceKind.USER);
207         const uuids = getGroupsFromForm.map(group => group.uuid);
208
209         const permissions = await permissionService.list({
210             filters: new FilterBuilder()
211                 .addIn('tailUuid', uuids)
212                 .addEqual('linkClass', LinkClass.PERMISSION)
213                 .getFilters()
214         });
215
216         const usersFromGroups = await userService.list({
217             filters: new FilterBuilder()
218                 .addIn('uuid', permissions.items.map(item => item.headUuid))
219                 .getFilters()
220
221         });
222
223         const invitationDataUsers = getUsersFromForm
224             .map(person => ({
225                 ownerUuid: user.uuid,
226                 headUuid: dialog.data,
227                 tailUuid: person.uuid,
228                 name: invitations.permissions
229             }));
230
231         const invitationsDataGroups = usersFromGroups.items.map(
232             person => ({
233                 ownerUuid: user.uuid,
234                 headUuid: dialog.data,
235                 tailUuid: person.uuid,
236                 name: invitations.permissions
237             })
238         );
239
240         const data = invitationDataUsers.concat(invitationsDataGroups);
241
242         for (const invitation of data) {
243             await permissionService.create(invitation);
244         }
245     }
246 };