1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
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, WorkflowInputsData } 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 import { ContainerResource } from "models/container";
25 import { ContainerRequestResource, ContainerRequestState } from "models/container-request";
26 import { FilterBuilder } from "services/api/filter-builder";
27 import { selectedToArray } from "components/multiselect-toolbar/MultiselectToolbar";
28 import { Resource, ResourceKind } from "models/resource";
29 import { ContextMenuResource } from "store/context-menu/context-menu-actions";
31 export const loadProcess =
32 (containerRequestUuid: string) =>
33 async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository): Promise<Process | undefined> => {
34 let containerRequest: ContainerRequestResource | undefined = undefined;
36 containerRequest = await services.containerRequestService.get(containerRequestUuid);
37 dispatch<any>(updateResources([containerRequest]));
42 if (containerRequest.outputUuid) {
44 const collection = await services.collectionService.get(containerRequest.outputUuid, false);
45 dispatch<any>(updateResources([collection]));
49 if (containerRequest.containerUuid) {
50 let container: ContainerResource | undefined = undefined;
52 container = await services.containerService.get(containerRequest.containerUuid, false);
53 dispatch<any>(updateResources([container]));
57 if (container && container.runtimeUserUuid) {
58 const runtimeUser = await services.userService.get(container.runtimeUserUuid, false);
59 dispatch<any>(updateResources([runtimeUser]));
63 return { containerRequest, container };
65 return { containerRequest };
68 export const loadContainers =
69 (containerUuids: string[], loadMounts: boolean = true) =>
70 async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
72 filters: new FilterBuilder().addIn("uuid", containerUuids).getFilters(),
73 limit: containerUuids.length,
76 args.select = containerFieldsNoMounts;
78 const { items } = await services.containerService.list(args);
79 dispatch<any>(updateResources(items));
83 // Until the api supports unselecting fields, we need a list of all other fields to omit mounts
84 const containerFieldsNoMounts = [
97 "interactive_session_started",
103 "modified_by_client_uuid",
104 "modified_by_user_uuid",
107 "output_storage_classes",
112 "runtime_auth_scopes",
113 "runtime_constraints",
116 "scheduling_parameters",
122 export const cancelRunningWorkflow = (uuid: string) => async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
124 const process = await services.containerRequestService.update(uuid, { priority: 0 });
125 dispatch<any>(updateResources([process]));
126 if (process.containerUuid) {
127 const container = await services.containerService.get(process.containerUuid, false);
128 dispatch<any>(updateResources([container]));
132 throw new Error("Could not cancel the process.");
136 export const resumeOnHoldWorkflow = (uuid: string) => async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
138 const process = await services.containerRequestService.update(uuid, { priority: 500 });
139 dispatch<any>(updateResources([process]));
140 if (process.containerUuid) {
141 const container = await services.containerService.get(process.containerUuid, false);
142 dispatch<any>(updateResources([container]));
146 throw new Error("Could not resume the process.");
150 export const startWorkflow = (uuid: string) => async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
152 const process = await services.containerRequestService.update(uuid, { state: ContainerRequestState.COMMITTED });
154 dispatch<any>(updateResources([process]));
155 dispatch(snackbarActions.OPEN_SNACKBAR({ message: "Process started", hideDuration: 2000, kind: SnackbarKind.SUCCESS }));
157 dispatch<any>(snackbarActions.OPEN_SNACKBAR({ message: `Failed to start process`, kind: SnackbarKind.ERROR }));
160 dispatch<any>(snackbarActions.OPEN_SNACKBAR({ message: `Failed to start process`, kind: SnackbarKind.ERROR }));
164 export const reRunProcess =
165 (processUuid: string, workflowUuid: string) => (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
166 const process = getResource<any>(processUuid)(getState().resources);
167 const workflows = getState().runProcessPanel.searchWorkflows;
168 const workflow = workflows.find(workflow => workflow.uuid === workflowUuid);
169 if (workflow && process) {
170 const mainWf = getWorkflow(process.mounts[MOUNT_PATH_CWL_WORKFLOW]);
172 mainWf.inputs = getInputs(process);
174 const stringifiedDefinition = JSON.stringify(process.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
175 const newWorkflow = { ...workflow, definition: stringifiedDefinition };
177 const owner = getResource<ProjectResource | UserResource>(workflow.ownerUuid)(getState().resources);
178 const basicInitialData: RunProcessBasicFormData = { name: `Copy of: ${process.name}`, description: process.description, owner };
179 dispatch<any>(initialize(RUN_PROCESS_BASIC_FORM, basicInitialData));
181 const advancedInitialData: RunProcessAdvancedFormData = {
182 output: process.outputName,
183 runtime: process.schedulingParameters.max_run_time,
184 ram: process.runtimeConstraints.ram,
185 vcpus: process.runtimeConstraints.vcpus,
186 keep_cache_ram: process.runtimeConstraints.keep_cache_ram,
187 acr_container_image: process.containerImage,
189 dispatch<any>(initialize(RUN_PROCESS_ADVANCED_FORM, advancedInitialData));
191 dispatch<any>(navigateToRunProcess);
192 dispatch<any>(goToStep(1));
193 dispatch(runProcessPanelActions.SET_STEP_CHANGED(true));
194 dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(newWorkflow));
196 dispatch<any>(snackbarActions.OPEN_SNACKBAR({ message: `You can't re-run this process`, kind: SnackbarKind.ERROR }));
201 * Fetches raw inputs from containerRequest mounts with fallback to properties
202 * Returns undefined if containerRequest not loaded
203 * Returns {} if inputs not found in mounts or props
205 export const getRawInputs = (data: any): WorkflowInputsData | undefined => {
209 const mountInput = data.mounts?.[MOUNT_PATH_CWL_INPUT]?.content;
210 const propsInput = data.properties?.cwl_input;
211 if (!mountInput && !propsInput) {
214 return mountInput || propsInput;
217 export const getInputs = (data: any): CommandInputParameter[] => {
218 // Definitions from mounts are needed so we return early if missing
219 if (!data || !data.mounts || !data.mounts[MOUNT_PATH_CWL_WORKFLOW]) {
222 const content = getRawInputs(data) as any;
223 // Only escape if content is falsy to allow displaying definitions if no inputs are present
224 // (Don't check raw content length)
229 const inputs = getWorkflowInputs(data.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
231 ? inputs.map((it: any) => ({
235 default: content[it.id],
236 value: content[it.id.split("/").pop()] || [],
243 * Fetches raw outputs from containerRequest properties
244 * Assumes containerRequest is loaded
246 export const getRawOutputs = (data: any): CommandInputParameter[] | undefined => {
247 if (!data || !data.properties || !data.properties.cwl_output) {
250 return data.properties.cwl_output;
253 export type InputCollectionMount = {
258 export const getInputCollectionMounts = (data: any): InputCollectionMount[] => {
259 if (!data || !data.mounts) {
262 return Object.keys(data.mounts)
267 .filter(mount => mount.kind === "collection" && mount.portable_data_hash && mount.path)
270 pdh: mount.portable_data_hash,
274 export const getOutputParameters = (data: any): CommandOutputParameter[] => {
275 if (!data || !data.mounts || !data.mounts[MOUNT_PATH_CWL_WORKFLOW]) {
278 const outputs = getWorkflowOutputs(data.mounts[MOUNT_PATH_CWL_WORKFLOW].content);
280 ? outputs.map((it: any) => ({
289 export const openRemoveProcessDialog =
290 (resource: ContextMenuResource, numOfProcesses: Number) => (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
291 const confirmationText =
293 ? "Are you sure you want to remove this process?"
294 : `Are you sure you want to remove these ${numOfProcesses} processes?`;
295 const titleText = numOfProcesses === 1 ? "Remove process permanently" : "Remove processes permanently";
298 dialogActions.OPEN_DIALOG({
299 id: REMOVE_PROCESS_DIALOG,
302 text: confirmationText,
303 confirmButtonLabel: "Remove",
311 export const REMOVE_PROCESS_DIALOG = "removeProcessDialog";
313 export const removeProcessPermanently = (uuid: string) => async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => {
314 const resource = getState().dialog.removeProcessDialog.data.resource;
315 const checkedList = getState().multiselect.checkedList;
317 const uuidsToRemove: string[] = resource.isSingle ? [resource.uuid] : selectedToArray(checkedList);
319 //if no items in checkedlist, default to normal context menu behavior
320 if (!uuidsToRemove.length) uuidsToRemove.push(uuid);
322 const processesToRemove = uuidsToRemove
323 .map(uuid => getResource(uuid)(getState().resources) as Resource)
324 .filter(resource => resource.kind === ResourceKind.PROCESS);
326 for (const process of processesToRemove) {
327 dispatch(snackbarActions.OPEN_SNACKBAR({ message: "Removing ...", kind: SnackbarKind.INFO }));
328 await services.containerRequestService.delete(process.uuid);
329 dispatch(projectPanelActions.REQUEST_ITEMS());
330 dispatch(snackbarActions.OPEN_SNACKBAR({ message: "Removed.", hideDuration: 2000, kind: SnackbarKind.SUCCESS }));