X-Git-Url: https://git.arvados.org/arvados-workbench2.git/blobdiff_plain/a92ac33f876a928d9542fe08f2a567ce86f3e890..f0a64666816383d2641d5fa7ea22019441ac4464:/src/store/run-process-panel/run-process-panel-actions.ts diff --git a/src/store/run-process-panel/run-process-panel-actions.ts b/src/store/run-process-panel/run-process-panel-actions.ts index de18e70f..66c784f9 100644 --- a/src/store/run-process-panel/run-process-panel-actions.ts +++ b/src/store/run-process-panel/run-process-panel-actions.ts @@ -6,12 +6,31 @@ import { Dispatch } from 'redux'; import { unionize, ofType, UnionOf } from "~/common/unionize"; import { ServiceRepository } from "~/services/services"; import { RootState } from '~/store/store'; -import { WorkflowResource } from '~/models/workflow'; +import { getUserUuid } from "~/common/getuser"; +import { WorkflowResource, getWorkflowInputs, parseWorkflowDefinition } from '~/models/workflow'; +import { getFormValues, initialize } from 'redux-form'; +import { RUN_PROCESS_BASIC_FORM, RunProcessBasicFormData } from '~/views/run-process-panel/run-process-basic-form'; +import { RUN_PROCESS_INPUTS_FORM } from '~/views/run-process-panel/run-process-inputs-form'; +import { WorkflowInputsData } from '~/models/workflow'; +import { createWorkflowMounts } from '~/models/process'; +import { ContainerRequestState } from '~/models/container-request'; +import { navigateTo } from '../navigation/navigation-action'; +import { RunProcessAdvancedFormData, RUN_PROCESS_ADVANCED_FORM, VCPUS_FIELD, RAM_FIELD, RUNTIME_FIELD, OUTPUT_FIELD, API_FIELD } from '~/views/run-process-panel/run-process-advanced-form'; +import { dialogActions } from '~/store/dialog/dialog-actions'; +import { setBreadcrumbs } from '~/store/breadcrumbs/breadcrumbs-actions'; +import { matchProjectRoute } from '~/routes/routes'; export const runProcessPanelActions = unionize({ + SET_PROCESS_PATHNAME: ofType(), + SET_PROCESS_OWNER_UUID: ofType(), SET_CURRENT_STEP: ofType(), + SET_STEP_CHANGED: ofType(), SET_WORKFLOWS: ofType(), SET_SELECTED_WORKFLOW: ofType(), + SET_WORKFLOW_PRESETS: ofType(), + SELECT_WORKFLOW_PRESET: ofType(), + SEARCH_WORKFLOWS: ofType(), + RESET_RUN_PROCESS_PANEL: ofType<{}>(), }); export interface RunProcessSecondStepDataFormProps { @@ -19,6 +38,7 @@ export interface RunProcessSecondStepDataFormProps { description: string; } +export const SET_WORKFLOW_DIALOG = 'setWorkflowDialog'; export const RUN_PROCESS_SECOND_STEP_FORM_NAME = 'runProcessSecondStepFormName'; export type RunProcessPanelAction = UnionOf; @@ -26,6 +46,7 @@ export type RunProcessPanelAction = UnionOf; export const loadRunProcessPanel = () => async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => { try { + dispatch(setBreadcrumbs([{ label: 'Run Process' }])); const response = await services.workflowService.list(); dispatch(runProcessPanelActions.SET_WORKFLOWS(response.items)); } catch (e) { @@ -33,9 +54,122 @@ export const loadRunProcessPanel = () => } }; -export const setWorkflow = (workflow: WorkflowResource) => - async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => { - dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(workflow)); +export const openSetWorkflowDialog = (workflow: WorkflowResource) => + (dispatch: Dispatch, getState: () => RootState) => { + const selectedWorkflow = getState().runProcessPanel.selectedWorkflow; + const isStepChanged = getState().runProcessPanel.isStepChanged; + if (isStepChanged && selectedWorkflow && selectedWorkflow.uuid !== workflow.uuid) { + dispatch(dialogActions.OPEN_DIALOG({ + id: SET_WORKFLOW_DIALOG, + data: { + title: 'Form will be cleared', + text: 'Changing a workflow will clean all input fields in next step.', + confirmButtonLabel: 'Change Workflow', + workflow + } + })); + } else { + dispatch(setWorkflow(workflow, false)); + } + }; + +export const setWorkflow = (workflow: WorkflowResource, isWorkflowChanged = true) => + (dispatch: Dispatch, getState: () => RootState) => { + const isStepChanged = getState().runProcessPanel.isStepChanged; + if (isStepChanged && isWorkflowChanged) { + dispatch(runProcessPanelActions.SET_STEP_CHANGED(false)); + dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(workflow)); + dispatch(loadPresets(workflow.uuid)); + dispatch(initialize(RUN_PROCESS_ADVANCED_FORM, DEFAULT_ADVANCED_FORM_VALUES)); + } + if (!isWorkflowChanged) { + dispatch(runProcessPanelActions.SET_SELECTED_WORKFLOW(workflow)); + dispatch(loadPresets(workflow.uuid)); + dispatch(initialize(RUN_PROCESS_ADVANCED_FORM, DEFAULT_ADVANCED_FORM_VALUES)); + } + }; + +export const loadPresets = (workflowUuid: string) => + async (dispatch: Dispatch, _: () => RootState, { workflowService }: ServiceRepository) => { + const { items } = await workflowService.presets(workflowUuid); + dispatch(runProcessPanelActions.SET_WORKFLOW_PRESETS(items)); }; -export const goToStep = (step: number) => runProcessPanelActions.SET_CURRENT_STEP(step); \ No newline at end of file +export const selectPreset = (preset: WorkflowResource) => + (dispatch: Dispatch) => { + dispatch(runProcessPanelActions.SELECT_WORKFLOW_PRESET(preset)); + const inputs = getWorkflowInputs(parseWorkflowDefinition(preset)) || []; + const values = inputs.reduce((values, input) => ({ + ...values, + [input.id]: input.default, + }), {}); + dispatch(initialize(RUN_PROCESS_INPUTS_FORM, values)); + }; + +export const goToStep = (step: number) => + (dispatch: Dispatch) => { + if (step === 1) { + dispatch(runProcessPanelActions.SET_STEP_CHANGED(true)); + } + dispatch(runProcessPanelActions.SET_CURRENT_STEP(step)); + }; + +export const runProcess = async (dispatch: Dispatch, getState: () => RootState, services: ServiceRepository) => { + const state = getState(); + const basicForm = getFormValues(RUN_PROCESS_BASIC_FORM)(state) as RunProcessBasicFormData; + const inputsForm = getFormValues(RUN_PROCESS_INPUTS_FORM)(state) as WorkflowInputsData; + const advancedForm = getFormValues(RUN_PROCESS_ADVANCED_FORM)(state) as RunProcessAdvancedFormData || DEFAULT_ADVANCED_FORM_VALUES; + const userUuid = getUserUuid(getState()); + if (!userUuid) { return; } + const pathname = getState().runProcessPanel.processPathname; + const { processOwnerUuid, selectedWorkflow } = state.runProcessPanel; + if (selectedWorkflow) { + const newProcessData = { + ownerUuid: !matchProjectRoute(pathname) ? userUuid : processOwnerUuid, + name: basicForm.name, + description: basicForm.description, + state: ContainerRequestState.COMMITTED, + mounts: createWorkflowMounts(selectedWorkflow, normalizeInputKeys(inputsForm)), + runtimeConstraints: { + API: true, + vcpus: advancedForm[VCPUS_FIELD], + ram: advancedForm[RAM_FIELD], + api: advancedForm[API_FIELD], + }, + schedulingParameters: { + max_run_time: advancedForm[RUNTIME_FIELD] + }, + containerImage: 'arvados/jobs', + cwd: '/var/spool/cwl', + command: [ + 'arvados-cwl-runner', + '--api=containers', + '--local', + '/var/lib/cwl/workflow.json#main', + '/var/lib/cwl/cwl.input.json' + ], + outputPath: '/var/spool/cwl', + priority: 1, + outputName: advancedForm[OUTPUT_FIELD] ? advancedForm[OUTPUT_FIELD] : undefined, + properties: { + workflowUuid: selectedWorkflow.uuid, + workflowName: selectedWorkflow.name + } + }; + const newProcess = await services.containerRequestService.create(newProcessData); + dispatch(navigateTo(newProcess.uuid)); + } +}; + +export const DEFAULT_ADVANCED_FORM_VALUES: Partial = { + [VCPUS_FIELD]: 1, + [RAM_FIELD]: 1073741824, + [API_FIELD]: true, +}; + +const normalizeInputKeys = (inputs: WorkflowInputsData): WorkflowInputsData => + Object.keys(inputs).reduce((normalizedInputs, key) => ({ + ...normalizedInputs, + [key.split('/').slice(1).join('/')]: inputs[key], + }), {}); +export const searchWorkflows = (term: string) => runProcessPanelActions.SEARCH_WORKFLOWS(term);