1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
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";
25 export const openSharingDialog = (resourceUuid: string) =>
26 (dispatch: Dispatch) => {
27 dispatch(dialogActions.OPEN_DIALOG({ id: SHARING_DIALOG_NAME, data: resourceUuid }));
28 dispatch<any>(loadSharingDialog);
31 export const closeSharingDialog = () =>
32 dialogActions.CLOSE_DIALOG({ id: SHARING_DIALOG_NAME });
34 export const connectSharingDialog = withDialog(SHARING_DIALOG_NAME);
35 export const connectSharingDialogProgress = withProgress(SHARING_DIALOG_NAME);
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);
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,
55 dispatch(progressIndicatorActions.STOP_WORKING(SHARING_DIALOG_NAME));
58 const loadSharingDialog = async (dispatch: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
60 const dialog = getDialog<string>(getState().dialog, SHARING_DIALOG_NAME);
62 dispatch(progressIndicatorActions.START_WORKING(SHARING_DIALOG_NAME));
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));
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));
76 const initializeManagementForm = (permissionLinks: PermissionResource[]) =>
77 async (dispatch: Dispatch, getState: () => RootState, { userService, groupsService }: ServiceRepository) => {
79 const filters = new FilterBuilder()
80 .addIn('uuid', permissionLinks.map(({ tailUuid }) => tailUuid))
83 const { items: users } = await userService.list({ filters });
84 const { items: groups} = await groupsService.list({ filters });
86 const getEmail = (tailUuid: string) => {
87 const user = users.find(({ uuid }) => uuid === tailUuid);
88 const group = groups.find(({ uuid }) => uuid === tailUuid);
96 const managementPermissions = permissionLinks
98 item.tailUuid !== getPublicGroupUuid(getState()))
99 .map(({ tailUuid, name, uuid }) => ({
100 email: getEmail(tailUuid),
101 permissions: name as PermissionLevel,
102 permissionUuid: uuid,
105 const managementFormData: SharingManagementFormData = {
106 permissions: managementPermissions,
107 initialPermissions: managementPermissions,
110 dispatch(initialize(SHARING_MANAGEMENT_FORM_NAME, managementFormData));
113 const initializePublicAccessForm = (permissionLinks: PermissionResource[]) =>
114 (dispatch: Dispatch, getState: () => RootState, ) => {
116 const [publicPermission] = permissionLinks
117 .filter(item => item.tailUuid === getPublicGroupUuid(getState()));
119 const publicAccessFormData: SharingPublicAccessFormData = publicPermission
121 visibility: VisibilityLevel.PUBLIC,
122 permissionUuid: publicPermission.uuid,
125 visibility: permissionLinks.length > 0
126 ? VisibilityLevel.SHARED
127 : VisibilityLevel.PRIVATE,
131 dispatch(initialize(SHARING_PUBLIC_ACCESS_FORM_NAME, publicAccessFormData));
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);
141 if (permissionUuid) {
142 if (visibility === VisibilityLevel.PUBLIC) {
143 await permissionService.update(permissionUuid, {
144 name: PermissionLevel.CAN_READ
147 await permissionService.delete(permissionUuid);
150 } else if (visibility === VisibilityLevel.PUBLIC) {
152 await permissionService.create({
153 ownerUuid: user.uuid,
154 headUuid: dialog.data,
155 tailUuid: getPublicGroupUuid(state),
156 name: PermissionLevel.CAN_READ,
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) {
168 const { initialPermissions, permissions } = getSharingMangementFormData(state);
169 const { visibility } = getSharingPublicAccessFormData(state);
172 if (visibility === VisibilityLevel.PRIVATE) {
174 for (const permission of initialPermissions) {
175 await permissionService.delete(permission.permissionUuid);
180 const cancelledPermissions = differenceWith(
183 (a, b) => a.permissionUuid === b.permissionUuid
186 for (const { permissionUuid } of cancelledPermissions) {
187 await permissionService.delete(permissionUuid);
190 for (const permission of permissions) {
191 await permissionService.update(permission.permissionUuid, { name: permission.permissions });
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;
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);
209 const permissions = await permissionService.list({
210 filters: new FilterBuilder()
211 .addIn('tailUuid', uuids)
212 .addEqual('linkClass', LinkClass.PERMISSION)
216 const usersFromGroups = await userService.list({
217 filters: new FilterBuilder()
218 .addIn('uuid', permissions.items.map(item => item.headUuid))
223 const invitationDataUsers = getUsersFromForm
225 ownerUuid: user.uuid,
226 headUuid: dialog.data,
227 tailUuid: person.uuid,
228 name: invitations.permissions
231 const invitationsDataGroups = usersFromGroups.items.map(
233 ownerUuid: user.uuid,
234 headUuid: dialog.data,
235 tailUuid: person.uuid,
236 name: invitations.permissions
240 const data = invitationDataUsers.concat(invitationsDataGroups);
242 for (const invitation of data) {
243 await permissionService.create(invitation);