eb04ed676f7644026cb8462760b47c064336dc87
[arvados-workbench2.git] / src / store / processes / processes-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 { RootState } from 'store/store';
7 import { ServiceRepository } from 'services/services';
8 import { updateResources } from 'store/resources/resources-actions';
9 import { Process } from './process';
10 import { dialogActions } from 'store/dialog/dialog-actions';
11 import { snackbarActions, SnackbarKind } from 'store/snackbar/snackbar-actions';
12 import { projectPanelActions } from 'store/project-panel/project-panel-action';
13 import { navigateToRunProcess } from 'store/navigation/navigation-action';
14 import { goToStep, runProcessPanelActions } from 'store/run-process-panel/run-process-panel-actions';
15 import { getResource } from 'store/resources/resources';
16 import { initialize } from "redux-form";
17 import { RUN_PROCESS_BASIC_FORM, RunProcessBasicFormData } from "views/run-process-panel/run-process-basic-form";
18 import { RunProcessAdvancedFormData, RUN_PROCESS_ADVANCED_FORM } from "views/run-process-panel/run-process-advanced-form";
19 import { MOUNT_PATH_CWL_WORKFLOW, MOUNT_PATH_CWL_INPUT } from 'models/process';
20 import { CommandInputParameter, getWorkflow, getWorkflowInputs, getWorkflowOutputs } from "models/workflow";
21 import { ProjectResource } from "models/project";
22 import { UserResource } from "models/user";
23 import { CommandOutputParameter } from "cwlts/mappings/v1.0/CommandOutputParameter";
24
25 export const loadProcess = (containerRequestUuid: string) =>
26     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository): Promise<Process> => {
27         const containerRequest = await services.containerRequestService.get(containerRequestUuid);
28         dispatch<any>(updateResources([containerRequest]));
29
30         if (containerRequest.outputUuid) {
31             const collection = await services.collectionService.get(containerRequest.outputUuid);
32             dispatch<any>(updateResources([collection]));
33         }
34
35         if (containerRequest.containerUuid) {
36             const container = await services.containerService.get(containerRequest.containerUuid);
37             dispatch<any>(updateResources([container]));
38             return { containerRequest, container };
39         }
40         return { containerRequest };
41     };
42
43 export const loadContainers = (filters: string, loadMounts: boolean = true) =>
44     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
45         let args: any = { filters };
46         if (!loadMounts) {
47             args.select = containerFieldsNoMounts;
48         }
49         const { items } = await services.containerService.list(args);
50         dispatch<any>(updateResources(items));
51         return items;
52     };
53
54 // Until the api supports unselecting fields, we need a list of all other fields to omit mounts
55 const containerFieldsNoMounts = [
56     "auth_uuid",
57     "command",
58     "container_image",
59     "created_at",
60     "cwd",
61     "environment",
62     "etag",
63     "exit_code",
64     "finished_at",
65     "gateway_address",
66     "href",
67     "interactive_session_started",
68     "kind",
69     "lock_count",
70     "locked_by_uuid",
71     "log",
72     "modified_at",
73     "modified_by_client_uuid",
74     "modified_by_user_uuid",
75     "output_path",
76     "output_properties",
77     "output_storage_classes",
78     "output",
79     "owner_uuid",
80     "priority",
81     "progress",
82     "runtime_auth_scopes",
83     "runtime_constraints",
84     "runtime_status",
85     "runtime_user_uuid",
86     "scheduling_parameters",
87     "started_at",
88     "state",
89     "uuid",
90 ]
91
92 export const cancelRunningWorkflow = (uuid: string) =>
93     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
94         try {
95             const process = await services.containerRequestService.update(uuid, { priority: 0 });
96             return process;
97         } catch (e) {
98             throw new Error('Could not cancel the process.');
99         }
100     };
101
102 export const reRunProcess = (processUuid: string, workflowUuid: string) =>
103     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
104         const process = getResource<any>(processUuid)(getState().resources);
105         const workflows = getState().runProcessPanel.searchWorkflows;
106         const workflow = workflows.find(workflow => workflow.uuid === workflowUuid);
107         if (workflow && process) {
108             const mainWf = getWorkflow(process.mounts[MOUNT_PATH_CWL_WORKFLOW]);
109             if (mainWf) { mainWf.inputs = getInputs(process); }
110             const stringifiedDefinition = JSON.stringify(process.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
111             const newWorkflow = { ...workflow, definition: stringifiedDefinition };
112
113             const owner = getResource<ProjectResource | UserResource>(workflow.ownerUuid)(getState().resources);
114             const basicInitialData: RunProcessBasicFormData = { name: `Copy of: ${process.name}`, description: process.description, owner };
115             dispatch<any>(initialize(RUN_PROCESS_BASIC_FORM, basicInitialData));
116
117             const advancedInitialData: RunProcessAdvancedFormData = {
118                 output: process.outputName,
119                 runtime: process.schedulingParameters.max_run_time,
120                 ram: process.runtimeConstraints.ram,
121                 vcpus: process.runtimeConstraints.vcpus,
122                 keep_cache_ram: process.runtimeConstraints.keep_cache_ram,
123                 acr_container_image: process.containerImage
124             };
125             dispatch<any>(initialize(RUN_PROCESS_ADVANCED_FORM, advancedInitialData));
126
127             dispatch<any>(navigateToRunProcess);
128             dispatch<any>(goToStep(1));
129             dispatch(runProcessPanelActions.SET_STEP_CHANGED(true));
130             dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(newWorkflow));
131         } else {
132             dispatch<any>(snackbarActions.OPEN_SNACKBAR({ message: `You can't re-run this process`, kind: SnackbarKind.ERROR }));
133         }
134     };
135
136 export const getRawInputs = (data: any): CommandInputParameter[] | undefined => {
137     if (!data || !data.mounts || !data.mounts[MOUNT_PATH_CWL_INPUT]) { return undefined; }
138     return (data.mounts[MOUNT_PATH_CWL_INPUT].content);
139 }
140
141 export const getInputs = (data: any): CommandInputParameter[] => {
142     if (!data || !data.mounts || !data.mounts[MOUNT_PATH_CWL_WORKFLOW]) { return []; }
143     const inputs = getWorkflowInputs(data.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
144     return inputs ? inputs.map(
145         (it: any) => (
146             {
147                 type: it.type,
148                 id: it.id,
149                 label: it.label,
150                 default: data.mounts[MOUNT_PATH_CWL_INPUT].content[it.id],
151                 value: data.mounts[MOUNT_PATH_CWL_INPUT].content[it.id.split('/').pop()] || [],
152                 doc: it.doc
153             }
154         )
155     ) : [];
156 };
157
158 export type InputCollectionMount = {
159     path: string;
160     pdh: string;
161 }
162
163 export const getInputCollectionMounts = (data: any): InputCollectionMount[] => {
164     if (!data || !data.mounts) { return []; }
165     return Object.keys(data.mounts)
166         .map(key => ({
167             ...data.mounts[key],
168             path: key,
169         }))
170         .filter(mount => mount.kind === 'collection' &&
171                 mount.portable_data_hash &&
172                 mount.path)
173         .map(mount => ({
174             path: mount.path,
175             pdh: mount.portable_data_hash,
176         }));
177 };
178
179 export const getOutputParameters = (data: any): CommandOutputParameter[] => {
180     if (!data || !data.mounts || !data.mounts[MOUNT_PATH_CWL_WORKFLOW]) { return []; }
181     const outputs = getWorkflowOutputs(data.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
182     return outputs ? outputs.map(
183         (it: any) => (
184             {
185                 type: it.type,
186                 id: it.id,
187                 label: it.label,
188                 doc: it.doc
189             }
190         )
191     ) : [];
192 };
193
194 export const openRemoveProcessDialog = (uuid: string) =>
195     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
196         dispatch(dialogActions.OPEN_DIALOG({
197             id: REMOVE_PROCESS_DIALOG,
198             data: {
199                 title: 'Remove process permanently',
200                 text: 'Are you sure you want to remove this process?',
201                 confirmButtonLabel: 'Remove',
202                 uuid
203             }
204         }));
205     };
206
207 export const REMOVE_PROCESS_DIALOG = 'removeProcessDialog';
208
209 export const removeProcessPermanently = (uuid: string) =>
210     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
211         dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removing ...', kind: SnackbarKind.INFO }));
212         await services.containerRequestService.delete(uuid);
213         dispatch(projectPanelActions.REQUEST_ITEMS());
214         dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removed.', hideDuration: 2000, kind: SnackbarKind.SUCCESS }));
215     };