16243: Fixed copy of selected items to new collection
[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                 // console.log(paths.length, filesToDelete.length, copiedFiles.length);
70                 await services.collectionService.deleteFiles(
71                     '',
72                     filesToDelete
73                 );
74                 dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME }));
75                 dispatch(snackbarActions.OPEN_SNACKBAR({
76                     message: 'New collection created.',
77                     hideDuration: 2000,
78                     kind: SnackbarKind.SUCCESS
79                 }));
80                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
81             } catch (e) {
82                 const error = getCommonResourceServiceError(e);
83                 if (error === CommonResourceServiceError.UNIQUE_NAME_VIOLATION) {
84                     dispatch(stopSubmit(COLLECTION_PARTIAL_COPY_FORM_NAME, { name: 'Collection with this name already exists.' } as FormErrors));
85                 } else if (error === CommonResourceServiceError.UNKNOWN) {
86                     dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME }));
87                     dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Could not create a copy of collection', hideDuration: 2000, kind: SnackbarKind.ERROR }));
88                 } else {
89                     dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME }));
90                     dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Collection has been copied but may contain incorrect files.', hideDuration: 2000, kind: SnackbarKind.ERROR }));
91                 }
92                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
93             }
94         }
95     };
96
97 export const openCollectionPartialCopyToSelectedCollectionDialog = () =>
98     (dispatch: Dispatch, getState: () => RootState) => {
99         const currentCollection = getState().collectionPanel.item;
100         if (currentCollection) {
101             const initialData = {
102                 collectionUuid: ''
103             };
104             dispatch(initialize(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION, initialData));
105             dispatch<any>(resetPickerProjectTree());
106             dispatch<any>(initProjectsTreePicker(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
107             dispatch(dialogActions.OPEN_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION, data: {} }));
108         }
109     };
110
111 export const copyCollectionPartialToSelectedCollection = ({ collectionUuid }: CollectionPartialCopyToSelectedCollectionFormData) =>
112     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
113         dispatch(startSubmit(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
114         const state = getState();
115         const currentCollection = state.collectionPanel.item;
116         if (currentCollection) {
117             try {
118                 dispatch(progressIndicatorActions.START_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
119                 const selectedCollection = await services.collectionService.get(collectionUuid);
120                 const paths = filterCollectionFilesBySelection(state.collectionPanelFiles, false).map(file => file.id);
121                 const pathsToRemove = paths.filter(path => {
122                     const a = path.split('/');
123                     const fileExistsInSelectedCollection = selectedCollection.manifestText.includes(a[1]);
124                     if (fileExistsInSelectedCollection) {
125                         return path;
126                     } else {
127                         return;
128                     }
129                 });
130                 const diffPathToRemove = _.difference(paths, pathsToRemove);
131                 await services.collectionService.deleteFiles(selectedCollection.uuid, pathsToRemove);
132                 const collectionWithDeletedFiles = await services.collectionService.get(collectionUuid);
133                 await services.collectionService.update(collectionUuid, { manifestText: `${collectionWithDeletedFiles.manifestText}${currentCollection.manifestText ? currentCollection.manifestText : currentCollection.unsignedManifestText}` });
134                 await services.collectionService.deleteFiles(collectionWithDeletedFiles.uuid, diffPathToRemove);
135                 dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION }));
136                 dispatch(snackbarActions.OPEN_SNACKBAR({
137                     message: 'Files has been copied to selected collection.',
138                     hideDuration: 2000,
139                     kind: SnackbarKind.SUCCESS
140                 }));
141                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
142             } catch (e) {
143                 const error = getCommonResourceServiceError(e);
144                 if (error === CommonResourceServiceError.UNKNOWN) {
145                     dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION }));
146                     dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Could not copy this files to selected collection', hideDuration: 2000, kind: SnackbarKind.ERROR }));
147                 }
148                 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
149             }
150         }
151     };