Merge branch '19142-avoid-loading-unneeded-mounts' into main. Closes #19142
[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 { getWorkflow, getWorkflowInputs } from "models/workflow";
21 import { ProjectResource } from "models/project";
22 import { UserResource } from "models/user";
23
24 export const loadProcess = (containerRequestUuid: string) =>
25     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository): Promise<Process> => {
26         const containerRequest = await services.containerRequestService.get(containerRequestUuid);
27         dispatch<any>(updateResources([containerRequest]));
28
29         if (containerRequest.containerUuid) {
30             const container = await services.containerService.get(containerRequest.containerUuid);
31             dispatch<any>(updateResources([container]));
32             return { containerRequest, container };
33         }
34         return { containerRequest };
35     };
36
37 export const loadContainers = (filters: string, loadMounts: boolean = true) =>
38     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
39         let args: any = { filters };
40         if (!loadMounts) {
41             args.select = containerFieldsNoMounts;
42         }
43         const { items } = await services.containerService.list(args);
44         dispatch<any>(updateResources(items));
45         return items;
46     };
47
48 // Until the api supports unselecting fields, we need a list of all other fields to omit mounts
49 const containerFieldsNoMounts = [
50     "auth_uuid",
51     "command",
52     "container_image",
53     "created_at",
54     "cwd",
55     "environment",
56     "etag",
57     "exit_code",
58     "finished_at",
59     "gateway_address",
60     "href",
61     "interactive_session_started",
62     "kind",
63     "lock_count",
64     "locked_by_uuid",
65     "log",
66     "modified_at",
67     "modified_by_client_uuid",
68     "modified_by_user_uuid",
69     "output_path",
70     "output_properties",
71     "output_storage_classes",
72     "output",
73     "owner_uuid",
74     "priority",
75     "progress",
76     "runtime_auth_scopes",
77     "runtime_constraints",
78     "runtime_status",
79     "runtime_user_uuid",
80     "scheduling_parameters",
81     "started_at",
82     "state",
83     "uuid",
84 ]
85
86 export const cancelRunningWorkflow = (uuid: string) =>
87     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
88         try {
89             const process = await services.containerRequestService.update(uuid, { priority: 0 });
90             return process;
91         } catch (e) {
92             throw new Error('Could not cancel the process.');
93         }
94     };
95
96 export const reRunProcess = (processUuid: string, workflowUuid: string) =>
97     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
98         const process = getResource<any>(processUuid)(getState().resources);
99         const workflows = getState().runProcessPanel.searchWorkflows;
100         const workflow = workflows.find(workflow => workflow.uuid === workflowUuid);
101         if (workflow && process) {
102             const mainWf = getWorkflow(process.mounts[MOUNT_PATH_CWL_WORKFLOW]);
103             if (mainWf) { mainWf.inputs = getInputs(process); }
104             const stringifiedDefinition = JSON.stringify(process.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
105             const newWorkflow = { ...workflow, definition: stringifiedDefinition };
106
107             const owner = getResource<ProjectResource | UserResource>(workflow.ownerUuid)(getState().resources);
108             const basicInitialData: RunProcessBasicFormData = { name: `Copy of: ${process.name}`, description: process.description, owner };
109             dispatch<any>(initialize(RUN_PROCESS_BASIC_FORM, basicInitialData));
110
111             const advancedInitialData: RunProcessAdvancedFormData = {
112                 output: process.outputName,
113                 runtime: process.schedulingParameters.max_run_time,
114                 ram: process.runtimeConstraints.ram,
115                 vcpus: process.runtimeConstraints.vcpus,
116                 keep_cache_ram: process.runtimeConstraints.keep_cache_ram,
117                 acr_container_image: process.containerImage
118             };
119             dispatch<any>(initialize(RUN_PROCESS_ADVANCED_FORM, advancedInitialData));
120
121             dispatch<any>(navigateToRunProcess);
122             dispatch<any>(goToStep(1));
123             dispatch(runProcessPanelActions.SET_STEP_CHANGED(true));
124             dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(newWorkflow));
125         } else {
126             dispatch<any>(snackbarActions.OPEN_SNACKBAR({ message: `You can't re-run this process`, kind: SnackbarKind.ERROR }));
127         }
128     };
129
130 const getInputs = (data: any) => {
131     if (!data || !data.mounts || !data.mounts[MOUNT_PATH_CWL_WORKFLOW]) { return []; }
132     const inputs = getWorkflowInputs(data.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
133     return inputs ? inputs.map(
134         (it: any) => (
135             {
136                 type: it.type,
137                 id: it.id,
138                 label: it.label,
139                 default: data.mounts[MOUNT_PATH_CWL_INPUT].content[it.id],
140                 doc: it.doc
141             }
142         )
143     ) : [];
144 };
145
146 export const openRemoveProcessDialog = (uuid: string) =>
147     (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
148         dispatch(dialogActions.OPEN_DIALOG({
149             id: REMOVE_PROCESS_DIALOG,
150             data: {
151                 title: 'Remove process permanently',
152                 text: 'Are you sure you want to remove this process?',
153                 confirmButtonLabel: 'Remove',
154                 uuid
155             }
156         }));
157     };
158
159 export const REMOVE_PROCESS_DIALOG = 'removeProcessDialog';
160
161 export const removeProcessPermanently = (uuid: string) =>
162     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
163         dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removing ...', kind: SnackbarKind.INFO }));
164         await services.containerRequestService.delete(uuid);
165         dispatch(projectPanelActions.REQUEST_ITEMS());
166         dispatch(snackbarActions.OPEN_SNACKBAR({ message: 'Removed.', hideDuration: 2000, kind: SnackbarKind.SUCCESS }));
167     };