Implement sharing save operation
[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 } 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, isDirty, 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
20 export const openSharingDialog = (resourceUuid: string) =>
21     (dispatch: Dispatch) => {
22         dispatch(dialogActions.OPEN_DIALOG({ id: SHARING_DIALOG_NAME, data: resourceUuid }));
23         dispatch<any>(loadSharingDialog);
24     };
25
26 export const closeSharingDialog = () =>
27     dialogActions.CLOSE_DIALOG({ id: SHARING_DIALOG_NAME });
28
29 export const connectSharingDialog = withDialog(SHARING_DIALOG_NAME);
30
31 export const saveSharingDialogChanges = async (dispatch: Dispatch) => {
32     await Promise.all([
33         dispatch<any>(savePublicPermissionChanges),
34         dispatch<any>(saveManagementChanges),
35         dispatch<any>(sendInvitations),
36     ]);
37     dispatch(reset(SHARING_INVITATION_FORM_NAME));
38     await dispatch<any>(loadSharingDialog);
39 };
40
41 export const hasChanges = (state: RootState) =>
42     isDirty(SHARING_PUBLIC_ACCESS_FORM_NAME)(state) ||
43     isDirty(SHARING_MANAGEMENT_FORM_NAME)(state) ||
44     isDirty(SHARING_INVITATION_FORM_NAME)(state);
45
46 const loadSharingDialog = async (dispatch: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
47
48     const dialog = getDialog<string>(getState().dialog, SHARING_DIALOG_NAME);
49
50     if (dialog) {
51         const { items } = await permissionService.listResourcePermissions(dialog.data);
52         dispatch<any>(initializePublicAccessForm(items));
53         await dispatch<any>(initializeManagementForm(items));
54     }
55 };
56
57 const initializeManagementForm = (permissionLinks: PermissionResource[]) =>
58     async (dispatch: Dispatch, getState: () => RootState, { userService }: ServiceRepository) => {
59
60         const filters = new FilterBuilder()
61             .addIn('uuid', permissionLinks.map(({ tailUuid }) => tailUuid))
62             .getFilters();
63
64         const { items: users } = await userService.list({ filters });
65
66         const getEmail = (tailUuid: string) => {
67             const user = users.find(({ uuid }) => uuid === tailUuid);
68             return user
69                 ? user.email
70                 : tailUuid;
71         };
72
73         const managementPermissions = permissionLinks
74             .filter(item =>
75                 item.tailUuid !== getPublicGroupUuid(getState()))
76             .map(({ tailUuid, name, uuid }) => ({
77                 email: getEmail(tailUuid),
78                 permissions: name as PermissionLevel,
79                 permissionUuid: uuid,
80             }));
81
82         const managementFormData: SharingManagementFormData = {
83             permissions: managementPermissions,
84             initialPermissions: managementPermissions,
85         };
86
87         dispatch(initialize(SHARING_MANAGEMENT_FORM_NAME, managementFormData));
88     };
89
90 const initializePublicAccessForm = (permissionLinks: PermissionResource[]) =>
91     (dispatch: Dispatch, getState: () => RootState, ) => {
92
93         const [publicPermission] = permissionLinks
94             .filter(item => item.tailUuid === getPublicGroupUuid(getState()));
95
96         const publicAccessFormData: SharingPublicAccessFormData = publicPermission
97             ? {
98                 enabled: publicPermission.name !== PermissionLevel.NONE,
99                 permissions: publicPermission.name as PermissionLevel,
100                 permissionUuid: publicPermission.uuid,
101             }
102             : {
103                 enabled: false,
104                 permissions: PermissionLevel.CAN_READ,
105                 permissionUuid: '',
106             };
107
108         dispatch(initialize(SHARING_PUBLIC_ACCESS_FORM_NAME, publicAccessFormData));
109     };
110
111 const savePublicPermissionChanges = async (_: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
112     const state = getState();
113     const { user } = state.auth;
114     const dialog = getDialog<string>(state.dialog, SHARING_DIALOG_NAME);
115     if (dialog && user) {
116         const { permissionUuid, enabled, permissions } = getFormValues(SHARING_PUBLIC_ACCESS_FORM_NAME)(state) as SharingPublicAccessFormData;
117
118         if (permissionUuid) {
119             if (enabled) {
120                 await permissionService.update(permissionUuid, {
121                     name: enabled ? permissions : PermissionLevel.NONE
122                 });
123             } else {
124                 await permissionService.delete(permissionUuid);
125             }
126
127         } else if (enabled) {
128
129             await permissionService.create({
130                 ownerUuid: user.uuid,
131                 headUuid: dialog.data,
132                 tailUuid: getPublicGroupUuid(state),
133                 name: permissions,
134             });
135         }
136     }
137 };
138
139 const saveManagementChanges = async (_: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
140     const state = getState();
141     const { user } = state.auth;
142     const dialog = getDialog<string>(state.dialog, SHARING_DIALOG_NAME);
143     if (dialog && user) {
144
145         const { initialPermissions, permissions } = getFormValues(SHARING_MANAGEMENT_FORM_NAME)(state) as SharingManagementFormData;
146
147         const cancelledPermissions = differenceWith(
148             initialPermissions,
149             permissions,
150             (a, b) => a.permissionUuid === b.permissionUuid
151         );
152
153         await Promise.all(cancelledPermissions.map(({ permissionUuid }) =>
154             permissionService.delete(permissionUuid)
155         ));
156
157         await Promise.all(permissions.map(({ permissionUuid, permissions }) =>
158             permissionService.update(permissionUuid, { name: permissions })
159         ));
160
161     }
162 };
163
164 const sendInvitations = async (_: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
165     const state = getState();
166     const { user } = state.auth;
167     const dialog = getDialog<string>(state.dialog, SHARING_DIALOG_NAME);
168     if (dialog && user) {
169
170         const invitations = getFormValues(SHARING_INVITATION_FORM_NAME)(state) as SharingInvitationFormData;
171
172         const promises = invitations.invitedPeople
173             .map(person => ({
174                 ownerUuid: user.uuid,
175                 headUuid: dialog.data,
176                 tailUuid: person.uuid,
177                 name: invitations.permissions
178             }))
179             .map(data => permissionService.create(data));
180
181         await Promise.all(promises);
182     }
183 };