1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import { Dispatch } from 'redux';
6 import { difference } 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";
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';
21 export interface CollectionPartialCopyFormData {
27 export interface CollectionPartialCopyToSelectedCollectionFormData {
28 collectionUuid: string;
31 export const openCollectionPartialCopyDialog = () =>
32 (dispatch: Dispatch, getState: () => RootState) => {
33 const currentCollection = getState().collectionPanel.item;
34 if (currentCollection) {
36 name: `Files extracted from: ${currentCollection.name}`,
37 description: currentCollection.description,
38 projectUuid: undefined
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: {} }));
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) {
54 dispatch(progressIndicatorActions.START_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
55 const collectionManifestText = await services.collectionService.get(currentCollection.uuid, undefined, ['manifestText']);
56 const collectionCopy = {
59 ownerUuid: projectUuid,
61 manifestText: collectionManifestText.manifestText,
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);
69 await services.collectionService.deleteFiles(
73 dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_FORM_NAME }));
74 dispatch(snackbarActions.OPEN_SNACKBAR({
75 message: 'New collection created.',
77 kind: SnackbarKind.SUCCESS
79 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
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 }));
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 }));
91 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_FORM_NAME));
96 export const openCollectionPartialCopyToSelectedCollectionDialog = () =>
97 (dispatch: Dispatch, getState: () => RootState) => {
98 const currentCollection = getState().collectionPanel.item;
99 if (currentCollection) {
100 const initialData = {
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: {} }));
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;
116 if (currentCollection && !currentCollection.manifestText) {
117 const fetchedCurrentCollection = await services.collectionService.get(currentCollection.uuid, undefined, ['manifestText']);
118 currentCollection.manifestText = fetchedCurrentCollection.manifestText;
119 currentCollection.unsignedManifestText = fetchedCurrentCollection.unsignedManifestText;
122 if (currentCollection) {
124 dispatch(progressIndicatorActions.START_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
125 const selectedCollection = await services.collectionService.get(collectionUuid);
126 const paths = filterCollectionFilesBySelection(state.collectionPanelFiles, false).map(file => file.id);
127 const pathsToRemove = paths.filter(path => {
128 const a = path.split('/');
129 const fileExistsInSelectedCollection = selectedCollection.manifestText.includes(a[1]);
130 if (fileExistsInSelectedCollection) {
136 const diffPathToRemove = difference(paths, pathsToRemove);
137 await services.collectionService.deleteFiles(selectedCollection.uuid, pathsToRemove.map(path => path.replace(currentCollection.uuid, collectionUuid)));
138 const collectionWithDeletedFiles = await services.collectionService.get(collectionUuid, undefined, ['uuid', 'manifestText']);
139 await services.collectionService.update(collectionUuid, { manifestText: `${collectionWithDeletedFiles.manifestText}${(currentCollection.manifestText ? currentCollection.manifestText : currentCollection.unsignedManifestText) || ''}` });
140 await services.collectionService.deleteFiles(collectionWithDeletedFiles.uuid, diffPathToRemove.map(path => path.replace(currentCollection.uuid, collectionUuid)));
141 dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION }));
142 dispatch(snackbarActions.OPEN_SNACKBAR({
143 message: 'Files has been copied to selected collection.',
145 kind: SnackbarKind.SUCCESS
147 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));
149 const error = getCommonResourceServiceError(e);
150 if (error === CommonResourceServiceError.UNKNOWN) {
151 dispatch(dialogActions.CLOSE_DIALOG({ id: COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION }));
152 dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Could not copy this files to selected collection', hideDuration: 2000, kind: SnackbarKind.ERROR }));
154 dispatch(progressIndicatorActions.STOP_WORKING(COLLECTION_PARTIAL_COPY_TO_SELECTED_COLLECTION));