Add presets actions to run-process-panel
[arvados-workbench2.git] / src / store / run-process-panel / run-process-panel-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 { unionize, ofType, UnionOf } from "~/common/unionize";
7 import { ServiceRepository } from "~/services/services";
8 import { RootState } from '~/store/store';
9 import { WorkflowResource } from '~/models/workflow';
10 import { getFormValues } from 'redux-form';
11 import { RUN_PROCESS_BASIC_FORM, RunProcessBasicFormData } from '~/views/run-process-panel/run-process-basic-form';
12 import { RUN_PROCESS_INPUTS_FORM } from '~/views/run-process-panel/run-process-inputs-form';
13 import { WorkflowInputsData } from '~/models/workflow';
14 import { createWorkflowMounts } from '~/models/process';
15 import { ContainerRequestState } from '~/models/container-request';
16 import { navigateToProcess } from '../navigation/navigation-action';
17 import { RunProcessAdvancedFormData, RUN_PROCESS_ADVANCED_FORM } from '~/views/run-process-panel/run-process-advanced-form';
18 import { isItemNotInProject, isProjectOrRunProcessRoute } from '~/store/projects/project-create-actions';
19 import { dialogActions } from '~/store/dialog/dialog-actions';
20 import { setBreadcrumbs } from '~/store/breadcrumbs/breadcrumbs-actions';
21
22 export const runProcessPanelActions = unionize({
23     SET_PROCESS_OWNER_UUID: ofType<string>(),
24     SET_CURRENT_STEP: ofType<number>(),
25     SET_STEP_CHANGED: ofType<boolean>(),
26     SET_WORKFLOWS: ofType<WorkflowResource[]>(),
27     SET_SELECTED_WORKFLOW: ofType<WorkflowResource>(),
28     SET_WORKFLOW_PRESETS: ofType<WorkflowResource[]>(),
29     SEARCH_WORKFLOWS: ofType<string>(),
30     RESET_RUN_PROCESS_PANEL: ofType<{}>(),
31 });
32
33 export interface RunProcessSecondStepDataFormProps {
34     name: string;
35     description: string;
36 }
37
38 export const SET_WORKFLOW_DIALOG = 'setWorkflowDialog';
39 export const RUN_PROCESS_SECOND_STEP_FORM_NAME = 'runProcessSecondStepFormName';
40
41 export type RunProcessPanelAction = UnionOf<typeof runProcessPanelActions>;
42
43 export const loadRunProcessPanel = () =>
44     async (dispatch: Dispatch<any>, getState: () => RootState, services: ServiceRepository) => {
45         try {
46             dispatch(setBreadcrumbs([{ label: 'Run Process' }]));
47             dispatch(runProcessPanelActions.RESET_RUN_PROCESS_PANEL());
48             const response = await services.workflowService.list();
49             dispatch(runProcessPanelActions.SET_WORKFLOWS(response.items));
50         } catch (e) {
51             return;
52         }
53     };
54
55 export const openSetWorkflowDialog = (workflow: WorkflowResource) =>
56     (dispatch: Dispatch, getState: () => RootState) => {
57         const selectedWorkflow = getState().runProcessPanel.selectedWorkflow;
58         const isStepChanged = getState().runProcessPanel.isStepChanged;
59         if (isStepChanged && selectedWorkflow && selectedWorkflow.uuid !== workflow.uuid) {
60             dispatch(dialogActions.OPEN_DIALOG({
61                 id: SET_WORKFLOW_DIALOG,
62                 data: {
63                     title: 'Form will be cleared',
64                     text: 'Changing a workflow will clean all input fields in next step.',
65                     confirmButtonLabel: 'Change Workflow',
66                     workflow
67                 }
68             }));
69         } else {
70             dispatch<any>(setWorkflow(workflow, false));
71         }
72     };
73
74 export const setWorkflow = (workflow: WorkflowResource, isWorkflowChanged = true) =>
75     (dispatch: Dispatch<any>, getState: () => RootState) => {
76         const isStepChanged = getState().runProcessPanel.isStepChanged;
77         if (isStepChanged && isWorkflowChanged) {
78             dispatch(runProcessPanelActions.SET_STEP_CHANGED(false));
79             dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(workflow));
80             dispatch<any>(loadPresets(workflow.uuid));
81         }
82         if (!isWorkflowChanged) {
83             dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(workflow));
84         }
85     };
86
87 const loadPresets = (workflowUuid: string) =>
88     async (dispatch: Dispatch<any>, _: () => RootState, { workflowService }: ServiceRepository) => {
89         const { items } = await workflowService.presets(workflowUuid);
90         dispatch(runProcessPanelActions.SET_WORKFLOW_PRESETS(items));
91     };
92
93
94 export const goToStep = (step: number) =>
95     (dispatch: Dispatch, getState: () => RootState) => {
96         if (step === 1) {
97             dispatch(runProcessPanelActions.SET_STEP_CHANGED(true));
98         }
99         dispatch(runProcessPanelActions.SET_CURRENT_STEP(step));
100     };
101
102 export const runProcess = async (dispatch: Dispatch<any>, getState: () => RootState, services: ServiceRepository) => {
103     const state = getState();
104     const basicForm = getFormValues(RUN_PROCESS_BASIC_FORM)(state) as RunProcessBasicFormData;
105     const inputsForm = getFormValues(RUN_PROCESS_INPUTS_FORM)(state) as WorkflowInputsData;
106     const advancedForm = getFormValues(RUN_PROCESS_ADVANCED_FORM)(state) as RunProcessAdvancedFormData;
107     const userUuid = getState().auth.user!.uuid;
108     const router = getState();
109     const properties = getState().properties;
110     const { processOwnerUuid, selectedWorkflow } = state.runProcessPanel;
111     if (selectedWorkflow) {
112         const newProcessData = {
113             ownerUuid: isItemNotInProject(properties) || !isProjectOrRunProcessRoute(router) ? userUuid : processOwnerUuid,
114             name: basicForm.name,
115             description: basicForm.description,
116             state: ContainerRequestState.COMMITTED,
117             mounts: createWorkflowMounts(selectedWorkflow, normalizeInputKeys(inputsForm)),
118             runtimeConstraints: {
119                 API: true,
120                 vcpus: 1,
121                 ram: 1073741824,
122             },
123             containerImage: 'arvados/jobs',
124             cwd: '/var/spool/cwl',
125             command: [
126                 'arvados-cwl-runner',
127                 '--local',
128                 '--api=containers',
129                 `--project-uuid=${processOwnerUuid}`,
130                 '/var/lib/cwl/workflow.json#main',
131                 '/var/lib/cwl/cwl.input.json'
132             ],
133             outputPath: '/var/spool/cwl',
134             priority: 1,
135             outputName: advancedForm && advancedForm.output ? advancedForm.output : undefined,
136         };
137         const newProcess = await services.containerRequestService.create(newProcessData);
138         dispatch(navigateToProcess(newProcess.uuid));
139     }
140 };
141
142 const normalizeInputKeys = (inputs: WorkflowInputsData): WorkflowInputsData =>
143     Object.keys(inputs).reduce((normalizedInputs, key) => ({
144         ...normalizedInputs,
145         [key.split('/').slice(1).join('/')]: inputs[key],
146     }), {});
147 export const searchWorkflows = (term: string) => runProcessPanelActions.SEARCH_WORKFLOWS(term);