19900: Don't dispay errors when fetching process output collection, container, or...
[arvados.git] / src / store / processes / process.ts
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import { ContainerRequestResource, ContainerRequestState } from '../../models/container-request';
6 import { ContainerResource, ContainerState } from '../../models/container';
7 import { ResourcesState, getResource } from 'store/resources/resources';
8 import { filterResources } from '../resources/resources';
9 import { ResourceKind, Resource, extractUuidKind } from 'models/resource';
10 import { getTimeDiff } from 'common/formatters';
11 import { ArvadosTheme } from 'common/custom-theme';
12
13 export interface Process {
14     containerRequest: ContainerRequestResource;
15     container?: ContainerResource;
16 }
17
18 export enum ProcessStatus {
19     CANCELLED = 'Cancelled',
20     COMPLETED = 'Completed',
21     DRAFT = 'Draft',
22     FAILING = 'Failing',
23     FAILED = 'Failed',
24     ONHOLD = 'On hold',
25     QUEUED = 'Queued',
26     RUNNING = 'Running',
27     WARNING = 'Warning',
28     UNKNOWN = 'Unknown',
29 }
30
31 export const getProcess = (uuid: string) => (resources: ResourcesState): Process | undefined => {
32     if (extractUuidKind(uuid) === ResourceKind.CONTAINER_REQUEST) {
33         const containerRequest = getResource<ContainerRequestResource>(uuid)(resources);
34         if (containerRequest) {
35             if (containerRequest.containerUuid) {
36                 const container = getResource<ContainerResource>(containerRequest.containerUuid)(resources);
37                 if (container) {
38                     return { containerRequest, container };
39                 }
40             }
41             return { containerRequest };
42         }
43     }
44     return;
45 };
46
47 export const getSubprocesses = (uuid: string) => (resources: ResourcesState) => {
48     const process = getProcess(uuid)(resources);
49     if (process && process.container) {
50         const containerRequests = filterResources(isSubprocess(process.container.uuid))(resources) as ContainerRequestResource[];
51         return containerRequests.reduce((subprocesses, { uuid }) => {
52             const process = getProcess(uuid)(resources);
53             return process
54                 ? [...subprocesses, process]
55                 : subprocesses;
56         }, []);
57     }
58     return [];
59 };
60
61 export const getProcessRuntime = ({ container }: Process) => {
62     if (container) {
63         if (container.startedAt === null) {
64             return 0;
65         }
66         if (container.finishedAt === null) {
67             // Count it from now
68             return new Date().getTime() - new Date(container.startedAt).getTime();
69         }
70         return getTimeDiff(container.finishedAt, container.startedAt);
71     } else {
72         return 0;
73     }
74 };
75
76
77 export const getProcessStatusStyles = (status: string, theme: ArvadosTheme): React.CSSProperties => {
78     let color = theme.customs.colors.grey500;
79     let running = false;
80     switch (status) {
81         case ProcessStatus.RUNNING:
82             color = theme.customs.colors.green800;
83             running = true;
84             break;
85         case ProcessStatus.COMPLETED:
86             color = theme.customs.colors.green800;
87             break;
88         case ProcessStatus.WARNING:
89             color = theme.customs.colors.green800;
90             running = true;
91             break;
92         case ProcessStatus.FAILING:
93             color = theme.customs.colors.red900;
94             running = true;
95             break;
96         case ProcessStatus.CANCELLED:
97         case ProcessStatus.FAILED:
98             color = theme.customs.colors.red900;
99             break;
100         case ProcessStatus.QUEUED:
101             color = theme.customs.colors.grey600;
102             running = true;
103             break;
104         default:
105             color = theme.customs.colors.grey600;
106             break;
107     }
108
109     // Using color and running we build the text, border, and background style properties
110     return {
111         // Set background color when not running, otherwise use white
112         backgroundColor: running ? theme.palette.common.white : color,
113         // Set text color to status color when running, else use white text for solid button
114         color: running ? color : theme.palette.common.white,
115         // Set border color when running, else omit the style entirely
116         ...(running ? {border: `2px solid ${color}`} : {}),
117     };
118 };
119
120 export const getProcessStatus = ({ containerRequest, container }: Process): ProcessStatus => {
121     switch (true) {
122         case containerRequest.state === ContainerRequestState.FINAL &&
123             container?.state !== ContainerState.COMPLETE:
124             // Request was finalized before its container started (or the
125             // container was cancelled)
126             return ProcessStatus.CANCELLED;
127
128         case containerRequest.state === ContainerRequestState.UNCOMMITTED:
129             return ProcessStatus.DRAFT;
130
131         case container?.state === ContainerState.COMPLETE:
132             if (container?.exitCode === 0) {
133                 return ProcessStatus.COMPLETED;
134             }
135             return ProcessStatus.FAILED;
136
137         case container?.state === ContainerState.CANCELLED:
138             return ProcessStatus.CANCELLED;
139
140         case container?.state === ContainerState.QUEUED ||
141             container?.state === ContainerState.LOCKED:
142             if (containerRequest.priority === 0) {
143                 return ProcessStatus.ONHOLD;
144             }
145             return ProcessStatus.QUEUED;
146
147         case container?.state === ContainerState.RUNNING:
148             if (!!container?.runtimeStatus.error) {
149                 return ProcessStatus.FAILING;
150             }
151             if (!!container?.runtimeStatus.warning) {
152                 return ProcessStatus.WARNING;
153             }
154             return ProcessStatus.RUNNING;
155
156         default:
157             return ProcessStatus.UNKNOWN;
158     }
159 };
160
161 const isSubprocess = (containerUuid: string) => (resource: Resource) =>
162     resource.kind === ResourceKind.CONTAINER_REQUEST
163     && (resource as ContainerRequestResource).requestingContainerUuid === containerUuid;