18123: Add group details permissions columns and tweaked renderers
[arvados-workbench2.git] / src / store / group-details-panel / group-details-panel-actions.ts
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import { bindDataExplorerActions } from 'store/data-explorer/data-explorer-action';
6 import { Dispatch } from 'redux';
7 import { propertiesActions } from 'store/properties/properties-actions';
8 import { getProperty } from 'store/properties/properties';
9 import { Participant } from 'views-components/sharing-dialog/participant-select';
10 import { dialogActions } from 'store/dialog/dialog-actions';
11 import { reset, startSubmit } from 'redux-form';
12 import { addGroupMember, deleteGroupMember } from 'store/groups-panel/groups-panel-actions';
13 import { getResource } from 'store/resources/resources';
14 import { GroupResource } from 'models/group';
15 import { RootState } from 'store/store';
16 import { ServiceRepository } from 'services/services';
17 import { PermissionResource } from 'models/permission';
18 import { snackbarActions, SnackbarKind } from 'store/snackbar/snackbar-actions';
19 import { UserResource, getUserDisplayName } from 'models/user';
20
21 export const GROUP_DETAILS_MEMBERS_PANEL_ID = 'groupDetailsMembersPanel';
22 export const GROUP_DETAILS_PERMISSIONS_PANEL_ID = 'groupDetailsPermissionsPanel';
23 export const ADD_GROUP_MEMBERS_DIALOG = 'addGrupMembers';
24 export const ADD_GROUP_MEMBERS_FORM = 'addGrupMembers';
25 export const ADD_GROUP_MEMBERS_USERS_FIELD_NAME = 'users';
26 export const MEMBER_ATTRIBUTES_DIALOG = 'memberAttributesDialog';
27 export const MEMBER_REMOVE_DIALOG = 'memberRemoveDialog';
28
29 export const GroupMembersPanelActions = bindDataExplorerActions(GROUP_DETAILS_MEMBERS_PANEL_ID);
30 export const GroupPermissionsPanelActions = bindDataExplorerActions(GROUP_DETAILS_PERMISSIONS_PANEL_ID);
31
32 export const loadGroupDetailsPanel = (groupUuid: string) =>
33     (dispatch: Dispatch) => {
34         dispatch(propertiesActions.SET_PROPERTY({ key: GROUP_DETAILS_MEMBERS_PANEL_ID, value: groupUuid }));
35         dispatch(GroupMembersPanelActions.REQUEST_ITEMS());
36         dispatch(propertiesActions.SET_PROPERTY({ key: GROUP_DETAILS_PERMISSIONS_PANEL_ID, value: groupUuid }));
37         dispatch(GroupPermissionsPanelActions.REQUEST_ITEMS());
38     };
39
40 export const getCurrentGroupDetailsPanelUuid = getProperty<string>(GROUP_DETAILS_MEMBERS_PANEL_ID);
41
42 export interface AddGroupMembersFormData {
43     [ADD_GROUP_MEMBERS_USERS_FIELD_NAME]: Participant[];
44 }
45
46 export const openAddGroupMembersDialog = () =>
47     (dispatch: Dispatch) => {
48         dispatch(dialogActions.OPEN_DIALOG({ id: ADD_GROUP_MEMBERS_DIALOG, data: {} }));
49         dispatch(reset(ADD_GROUP_MEMBERS_FORM));
50     };
51
52 export const addGroupMembers = ({ users }: AddGroupMembersFormData) =>
53
54     async (dispatch: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
55
56         const groupUuid = getCurrentGroupDetailsPanelUuid(getState().properties);
57
58         if (groupUuid) {
59
60             dispatch(startSubmit(ADD_GROUP_MEMBERS_FORM));
61
62             const group = getResource<GroupResource>(groupUuid)(getState().resources);
63
64             for (const user of users) {
65
66                 await addGroupMember({
67                     user,
68                     group: {
69                         uuid: groupUuid,
70                         name: group ? group.name : groupUuid,
71                     },
72                     dispatch,
73                     permissionService,
74                 });
75
76             }
77
78             dispatch(dialogActions.CLOSE_DIALOG({ id: ADD_GROUP_MEMBERS_FORM }));
79             dispatch(GroupMembersPanelActions.REQUEST_ITEMS());
80
81         }
82     };
83
84 export const openGroupMemberAttributes = (uuid: string) =>
85     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
86         const { resources } = getState();
87         const data = getResource<PermissionResource>(uuid)(resources);
88         dispatch(dialogActions.OPEN_DIALOG({ id: MEMBER_ATTRIBUTES_DIALOG, data }));
89     };
90
91 export const openRemoveGroupMemberDialog = (uuid: string) =>
92     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
93         dispatch(dialogActions.OPEN_DIALOG({
94             id: MEMBER_REMOVE_DIALOG,
95             data: {
96                 title: 'Remove member',
97                 text: 'Are you sure you want to remove this member from this group?',
98                 confirmButtonLabel: 'Remove',
99                 uuid
100             }
101         }));
102     };
103
104 export const removeGroupMember = (uuid: string) =>
105
106     async (dispatch: Dispatch, getState: () => RootState, { permissionService }: ServiceRepository) => {
107
108         const groupUuid = getCurrentGroupDetailsPanelUuid(getState().properties);
109
110         if (groupUuid) {
111
112             const group = getResource<GroupResource>(groupUuid)(getState().resources);
113             const user = getResource<UserResource>(groupUuid)(getState().resources);
114
115             dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removing ...', kind: SnackbarKind.INFO }));
116
117             await deleteGroupMember({
118                 user: {
119                     uuid,
120                     name: user ? getUserDisplayName(user) : uuid,
121                 },
122                 group: {
123                     uuid: groupUuid,
124                     name: group ? group.name : groupUuid,
125                 },
126                 permissionService,
127                 dispatch,
128             });
129
130             dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removed.', hideDuration: 2000, kind: SnackbarKind.SUCCESS }));
131             dispatch(GroupMembersPanelActions.REQUEST_ITEMS());
132
133         }
134
135     };