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