22141: Refactoring to reduce circular import dependencies
[arvados.git] / services / workbench2 / src / store / workflow-panel / workflow-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 { RootState } from 'store/store';
7 import { ServiceRepository } from 'services/services';
8 import { bindDataExplorerActions } from 'store/data-explorer/data-explorer-action';
9 import { propertiesActions } from 'store/properties/properties-actions';
10 import { getProperty } from 'store/properties/properties';
11 import { navigateToRunProcess, navigateTo } from 'store/navigation/navigation-action';
12 import {
13     goToStep,
14     runProcessPanelActions,
15     loadPresets,
16     getWorkflowRunnerSettings
17 } from 'store/run-process-panel/run-process-panel-actions';
18 import { snackbarActions, SnackbarKind } from 'store/snackbar/snackbar-actions';
19 import { initialize } from 'redux-form';
20 import { RUN_PROCESS_BASIC_FORM, RUN_PROCESS_INPUTS_FORM, RUN_PROCESS_ADVANCED_FORM } from 'store/run-process-panel/run-process-panel-actions';
21 import { getResource } from 'store/resources/resources';
22 import { ProjectResource } from 'models/project';
23 import { UserResource } from 'models/user';
24 import { getWorkflowInputs, parseWorkflowDefinition } from 'models/workflow';
25 import { ContextMenuResource } from 'store/context-menu/context-menu-actions';
26 import { dialogActions } from 'store/dialog/dialog-actions';
27 import { ResourceKind, Resource } from 'models/resource';
28 import { selectedToArray } from "components/multiselect-toolbar/MultiselectToolbar";
29 import { CommonResourceServiceError, getCommonResourceServiceError } from "services/common-service/common-resource-service";
30 import { projectPanelDataActions } from "store/project-panel/project-panel-action-bind";
31
32 export const WORKFLOW_PANEL_ID = "workflowPanel";
33 const UUID_PREFIX_PROPERTY_NAME = 'uuidPrefix';
34 const WORKFLOW_PANEL_DETAILS_UUID = 'workflowPanelDetailsUuid';
35 export const workflowPanelActions = bindDataExplorerActions(WORKFLOW_PANEL_ID);
36
37 export const WORKFLOW_PROCESSES_PANEL_ID = "workflowProcessesPanel";
38 export const workflowProcessesPanelActions = bindDataExplorerActions(WORKFLOW_PROCESSES_PANEL_ID);
39
40 export const loadWorkflowPanel = () =>
41     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
42         dispatch(workflowPanelActions.REQUEST_ITEMS());
43         const response = await services.workflowService.list();
44         dispatch(runProcessPanelActions.SET_WORKFLOWS(response.items));
45     };
46
47 export const setUuidPrefix = (uuidPrefix: string) =>
48     propertiesActions.SET_PROPERTY({ key: UUID_PREFIX_PROPERTY_NAME, value: uuidPrefix });
49
50 export const getUuidPrefix = (state: RootState) => {
51     return state.properties.uuidPrefix;
52 };
53
54 export const openRunProcess = (workflowUuid: string, ownerUuid?: string, name?: string, inputObj?: { [key: string]: any }) =>
55     async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
56         const response = await services.workflowService.list();
57         dispatch(runProcessPanelActions.SET_WORKFLOWS(response.items));
58
59         const workflows = getState().runProcessPanel.searchWorkflows;
60         const listedWorkflow = workflows.find(workflow => workflow.uuid === workflowUuid);
61         const workflow = listedWorkflow || (await services.workflowService.get(workflowUuid));
62         if (workflow) {
63             dispatch<any>(navigateToRunProcess);
64             dispatch<any>(goToStep(1));
65             dispatch(runProcessPanelActions.SET_STEP_CHANGED(true));
66             dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(workflow));
67             dispatch<any>(loadPresets(workflow.uuid));
68
69             dispatch(initialize(RUN_PROCESS_ADVANCED_FORM, getWorkflowRunnerSettings(workflow)));
70             let owner;
71             if (ownerUuid) {
72                 // Must be writable.
73                 owner = getResource<ProjectResource | UserResource>(ownerUuid)(getState().resources);
74                 if (!owner || !owner.canWrite) {
75                     owner = undefined;
76                 }
77             }
78             if (owner) {
79                 dispatch(runProcessPanelActions.SET_PROCESS_OWNER_UUID(owner.uuid));
80             }
81
82             dispatch(initialize(RUN_PROCESS_BASIC_FORM, { name, owner }));
83
84             const definition = parseWorkflowDefinition(workflow);
85             if (definition) {
86                 const inputs = getWorkflowInputs(definition);
87                 if (inputs) {
88                     const values = inputs.reduce((values, input) => ({
89                         ...values,
90                         [input.id]: input.default,
91                     }), {});
92                     dispatch(initialize(RUN_PROCESS_INPUTS_FORM, values));
93                 }
94             }
95
96             if (inputObj) {
97                 dispatch(initialize(RUN_PROCESS_INPUTS_FORM, inputObj));
98             }
99         } else {
100             dispatch<any>(snackbarActions.OPEN_SNACKBAR({ message: `You can't run this process` }));
101         }
102     };
103
104 export const getPublicUserUuid = (state: RootState) => {
105     const prefix = state.auth.localCluster;
106     return `${prefix}-tpzed-anonymouspublic`;
107 };
108 export const getPublicGroupUuid = (state: RootState) => {
109     const prefix = state.auth.localCluster;
110     return `${prefix}-j7d0g-anonymouspublic`;
111 };
112 export const getAllUsersGroupUuid = (state: RootState) => {
113     const prefix = state.auth.localCluster;
114     return `${prefix}-j7d0g-fffffffffffffff`;
115 };
116
117 export const showWorkflowDetails = (uuid: string) =>
118     propertiesActions.SET_PROPERTY({ key: WORKFLOW_PANEL_DETAILS_UUID, value: uuid });
119
120 export const getWorkflowDetails = (state: RootState) => {
121     const uuid = getProperty<string>(WORKFLOW_PANEL_DETAILS_UUID)(state.properties);
122     const workflows = state.runProcessPanel.workflows;
123     const workflow = workflows.find(workflow => workflow.uuid === uuid);
124     return workflow || undefined;
125 };
126
127 export const openRemoveWorkflowDialog =
128 (resource: ContextMenuResource, numOfWorkflows: Number) => (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
129     const confirmationText =
130         numOfWorkflows === 1
131             ? "Are you sure you want to remove this workflow?"
132             : `Are you sure you want to remove these ${numOfWorkflows} workflows?`;
133     const titleText = numOfWorkflows === 1 ? "Remove workflow permanently" : "Remove workflows permanently";
134
135     dispatch(
136         dialogActions.OPEN_DIALOG({
137             id: REMOVE_WORKFLOW_DIALOG,
138             data: {
139                 title: titleText,
140                 text: confirmationText,
141                 confirmButtonLabel: "Remove",
142                 uuid: resource.uuid,
143                 resource,
144             },
145         })
146     );
147 };
148
149 export const REMOVE_WORKFLOW_DIALOG = "removeWorkflowDialog";
150
151 export const removeWorkflowPermanently = (uuid: string, ownerUuid?: string) => async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
152     const resource = getState().dialog.removeWorkflowDialog.data.resource;
153     const checkedList = getState().multiselect.checkedList;
154
155     const uuidsToRemove: string[] = resource.fromContextMenu ? [resource.uuid] : selectedToArray(checkedList);
156
157     //if no items in checkedlist, default to normal context menu behavior
158     if (!uuidsToRemove.length) uuidsToRemove.push(uuid);
159     if(ownerUuid) dispatch<any>(navigateTo(ownerUuid));
160
161     const workflowsToRemove = uuidsToRemove
162         .map(uuid => getResource(uuid)(getState().resources) as Resource)
163         .filter(resource => resource.kind === ResourceKind.WORKFLOW);
164
165     for (const workflow of workflowsToRemove) {
166         try {
167             dispatch(snackbarActions.OPEN_SNACKBAR({ message: "Removing ...", kind: SnackbarKind.INFO }));
168             await services.workflowService.delete(workflow.uuid);
169             dispatch(projectPanelDataActions.REQUEST_ITEMS());
170             dispatch(snackbarActions.OPEN_SNACKBAR({ message: "Removed.", hideDuration: 2000, kind: SnackbarKind.SUCCESS }));
171         } catch (e) {
172             const error = getCommonResourceServiceError(e);
173             if (error === CommonResourceServiceError.PERMISSION_ERROR_FORBIDDEN) {
174                 dispatch(snackbarActions.OPEN_SNACKBAR({ message: `Access denied`, hideDuration: 2000, kind: SnackbarKind.ERROR }));
175             } else {
176                 dispatch(snackbarActions.OPEN_SNACKBAR({ message: `Deletion failed`, hideDuration: 2000, kind: SnackbarKind.ERROR }));
177             }
178         }
179     }
180 };