Merge branch '16592-renaming-fix'
[arvados-workbench2.git] / src / store / collections / collection-partial-copy-actions.ts
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import { Dispatch } from 'redux';
6 import * as _ from "lodash";
7 import { RootState } from '~/store/store';
8 import { FormErrors, initialize, startSubmit, stopSubmit } from 'redux-form';
9 import { resetPickerProjectTree } from '~/store/project-tree-picker/project-tree-picker-actions';
10 import { dialogActions } from '~/store/dialog/dialog-actions';
11 import { ServiceRepository } from '~/services/services';
12 import { filterCollectionFilesBySelection } from '../collection-panel/collection-panel-files/collection-panel-files-state';
13 import { snackbarActions, SnackbarKind } from '~/store/snackbar/snackbar-actions';
14 import { getCommonResourceServiceError, CommonResourceServiceError } from '~/services/common-service/common-resource-service';
15 import { progressIndicatorActions } from "~/store/progress-indicator/progress-indicator-actions";
16 import { initProjectsTreePicker } from '~/store/tree-picker/tree-picker-actions';
17
18 export const COLLECTION_PARTIAL_COPY_FORM_NAME = 'COLLECTION_PARTIAL_COPY_DIALOG';
19 export const COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION = 'COLLECTION_PARTIAL_COPY_TO_SELECTED_DIALOG';
20
21 export interface CollectionPartialCopyFormData {
22     name: string;
23     description: string;
24     projectUuid: string;
25 }
26
27 export interface CollectionPartialCopyToSelectedCollectionFormData {
28     collectionUuid: string;
29 }
30
31 export const openCollectionPartialCopyDialog = () =>
32     (dispatch: Dispatch, getState: () => RootState) => {
33         const currentCollection = getState().collectionPanel.item;
34         if (currentCollection) {
35             const initialData = {
36                 name: currentCollection.name,
37                 description: currentCollection.description,
38                 projectUuid: ''
39             };
40             dispatch(initialize(COLLECTION_PARTIAL_COPY_FORM_NAME, initialData));
41             dispatch<any>(resetPickerProjectTree());
42             dispatch<any>(initProjectsTreePicker(COLLECTION_PARTIAL_COPY_FORM_NAME));
43             dispatch(dialogActions.OPEN_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME, data: {} }));
44         }
45     };
46
47 export const copyCollectionPartial = ({ name, description, projectUuid }: CollectionPartialCopyFormData) =>
48     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
49         dispatch(startSubmit(COLLECTION_PARTIAL_COPY_FORM_NAME));
50         const state = getState();
51         const currentCollection = state.collectionPanel.item;
52         if (currentCollection) {
53             try {
54                 dispatch(progressIndicatorActions.START_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
55                 const collection = await services.collectionService.get(currentCollection.uuid);
56                 const collectionCopy = {
57                     name,
58                     description,
59                     ownerUuid: projectUuid,
60                     uuid: undefined,
61                     manifestText: collection.manifestText,
62                 };
63                 const newCollection = await services.collectionService.create(collectionCopy);
64                 const copiedFiles = await services.collectionService.files(newCollection.uuid);
65                 const paths = filterCollectionFilesBySelection(state.collectionPanelFiles, true).map(file => file.id);
66                 const filesToDelete = copiedFiles.map(({ id }) => id).filter(file => {
67                     return !paths.find(path => path.indexOf(file.replace(newCollection.uuid, '')) > -1);
68                 });
69                 await services.collectionService.deleteFiles(
70                     '',
71                     filesToDelete
72                 );
73                 dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME }));
74                 dispatch(snackbarActions.OPEN_SNACKBAR({
75                     message: 'New collection created.',
76                     hideDuration: 2000,
77                     kind: SnackbarKind.SUCCESS
78                 }));
79                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
80             } catch (e) {
81                 const error = getCommonResourceServiceError(e);
82                 if (error === CommonResourceServiceError.UNIQUE_NAME_VIOLATION) {
83                     dispatch(stopSubmit(COLLECTION_PARTIAL_COPY_FORM_NAME, { name: 'Collection with this name already exists.' } as FormErrors));
84                 } else if (error === CommonResourceServiceError.UNKNOWN) {
85                     dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME }));
86                     dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Could not create a copy of collection', hideDuration: 2000, kind: SnackbarKind.ERROR }));
87                 } else {
88                     dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME }));
89                     dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Collection has been copied but may contain incorrect files.', hideDuration: 2000, kind: SnackbarKind.ERROR }));
90                 }
91                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
92             }
93         }
94     };
95
96 export const openCollectionPartialCopyToSelectedCollectionDialog = () =>
97     (dispatch: Dispatch, getState: () => RootState) => {
98         const currentCollection = getState().collectionPanel.item;
99         if (currentCollection) {
100             const initialData = {
101                 collectionUuid: ''
102             };
103             dispatch(initialize(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION, initialData));
104             dispatch<any>(resetPickerProjectTree());
105             dispatch<any>(initProjectsTreePicker(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
106             dispatch(dialogActions.OPEN_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION, data: {} }));
107         }
108     };
109
110 export const copyCollectionPartialToSelectedCollection = ({ collectionUuid }: CollectionPartialCopyToSelectedCollectionFormData) =>
111     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
112         dispatch(startSubmit(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
113         const state = getState();
114         const currentCollection = state.collectionPanel.item;
115         if (currentCollection) {
116             try {
117                 dispatch(progressIndicatorActions.START_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
118                 const selectedCollection = await services.collectionService.get(collectionUuid);
119                 const paths = filterCollectionFilesBySelection(state.collectionPanelFiles, false).map(file => file.id);
120                 const pathsToRemove = paths.filter(path => {
121                     const a = path.split('/');
122                     const fileExistsInSelectedCollection = selectedCollection.manifestText.includes(a[1]);
123                     if (fileExistsInSelectedCollection) {
124                         return path;
125                     } else {
126                         return;
127                     }
128                 });
129                 const diffPathToRemove = _.difference(paths, pathsToRemove);
130                 await services.collectionService.deleteFiles(selectedCollection.uuid, pathsToRemove);
131                 const collectionWithDeletedFiles = await services.collectionService.get(collectionUuid);
132                 await services.collectionService.update(collectionUuid, { manifestText: `${collectionWithDeletedFiles.manifestText}${currentCollection.manifestText ? currentCollection.manifestText : currentCollection.unsignedManifestText}` });
133                 await services.collectionService.deleteFiles(collectionWithDeletedFiles.uuid, diffPathToRemove);
134                 dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION }));
135                 dispatch(snackbarActions.OPEN_SNACKBAR({
136                     message: 'Files has been copied to selected collection.',
137                     hideDuration: 2000,
138                     kind: SnackbarKind.SUCCESS
139                 }));
140                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
141             } catch (e) {
142                 const error = getCommonResourceServiceError(e);
143                 if (error === CommonResourceServiceError.UNKNOWN) {
144                     dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION }));
145                     dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Could not copy this files to selected collection', hideDuration: 2000, kind: SnackbarKind.ERROR }));
146                 }
147                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
148             }
149         }
150     };