cf0034726b98f33805d5f6ae1c9db75b02334b48
[arvados-workbench2.git] / src / store / subprocess-panel / subprocess-panel-middleware-service.ts
1 // Copyright (C) The Arvados Authors. All rights reserved.
2 //
3 // SPDX-License-Identifier: AGPL-3.0
4
5 import { ServiceRepository } from '~/services/services';
6 import { MiddlewareAPI, Dispatch } from 'redux';
7 import {
8     DataExplorerMiddlewareService, dataExplorerToListParams, listResultsToDataExplorerItemsMeta, getDataExplorerColumnFilters
9 } from '~/store/data-explorer/data-explorer-middleware-service';
10 import { RootState } from '~/store/store';
11 import { snackbarActions, SnackbarKind } from '~/store/snackbar/snackbar-actions';
12 import { DataExplorer, getDataExplorer } from '~/store/data-explorer/data-explorer-reducer';
13 import { updateResources } from '~/store/resources/resources-actions';
14 import { SortDirection } from '~/components/data-table/data-column';
15 import { OrderDirection, OrderBuilder } from '~/services/api/order-builder';
16 import { ListResults } from '~/services/common-service/common-service';
17 import { getSortColumn } from "~/store/data-explorer/data-explorer-reducer";
18 import { ProcessResource } from '~/models/process';
19 import { SubprocessPanelColumnNames } from '~/views/subprocess-panel/subprocess-panel-root';
20 import { FilterBuilder, joinFilters } from '~/services/api/filter-builder';
21 import { subprocessPanelActions } from './subprocess-panel-actions';
22 import { DataColumns } from '~/components/data-table/data-table';
23 import { ProcessStatusFilter } from '../resource-type-filters/resource-type-filters';
24 import { ContainerRequestResource } from '~/models/container-request';
25
26 export class SubprocessMiddlewareService extends DataExplorerMiddlewareService {
27     constructor(private services: ServiceRepository, id: string) {
28         super(id);
29     }
30
31     async requestItems(api: MiddlewareAPI<Dispatch, RootState>) {
32         const state = api.getState();
33         const dataExplorer = getDataExplorer(state.dataExplorer, this.getId());
34
35         try {
36             const parentContainerRequestUuid = state.processPanel.containerRequestUuid;
37             if (parentContainerRequestUuid === "") {
38                 api.dispatch(subprocessPanelActions.CLEAR());
39                 return;
40             }
41             const parentContainerRequest = await this.services.containerRequestService.get(parentContainerRequestUuid);
42             if (!parentContainerRequest.containerUuid) {
43                 api.dispatch(subprocessPanelActions.CLEAR());
44                 return;
45             }
46             const containerRequests = await this.services.containerRequestService.list(
47                 { ...getParams(dataExplorer, parentContainerRequest) });
48             if (containerRequests.items.length === 0) {
49                 api.dispatch(subprocessPanelActions.CLEAR());
50                 return;
51             }
52             const containerUuids: string[] = containerRequests.items.reduce(
53                 (uuids, { containerUuid }) =>
54                     containerUuid
55                         ? [...uuids, containerUuid]
56                         : uuids, []);
57             const containers = await this.services.containerService.list({
58                 filters: new FilterBuilder().addIn('uuid', containerUuids).getFilters()
59             });
60
61             // Populate the actual user view
62             api.dispatch(updateResources(containerRequests.items));
63             api.dispatch(updateResources(containers.items));
64             api.dispatch(setItems(containerRequests));
65         } catch {
66             api.dispatch(couldNotFetchSubprocesses());
67         }
68     }
69 }
70
71 export const getParams = (
72     dataExplorer: DataExplorer,
73     parentContainerRequest: ContainerRequestResource) => ({
74         ...dataExplorerToListParams(dataExplorer),
75         order: getOrder(dataExplorer),
76         filters: getFilters(dataExplorer, parentContainerRequest)
77     });
78
79 const getOrder = (dataExplorer: DataExplorer) => {
80     const sortColumn = getSortColumn(dataExplorer);
81     const order = new OrderBuilder<ProcessResource>();
82     if (sortColumn) {
83         const sortDirection = sortColumn && sortColumn.sortDirection === SortDirection.ASC
84             ? OrderDirection.ASC
85             : OrderDirection.DESC;
86
87         const columnName = sortColumn && sortColumn.name === SubprocessPanelColumnNames.NAME ? "name" : "modifiedAt";
88         return order
89             .addOrder(sortDirection, columnName)
90             .getOrder();
91     } else {
92         return order.getOrder();
93     }
94 };
95
96 export const getFilters = (
97     dataExplorer: DataExplorer,
98     parentContainerRequest: ContainerRequestResource) => {
99         const columns = dataExplorer.columns as DataColumns<string>;
100         const statusColumnFilters = getDataExplorerColumnFilters(columns, 'Status');
101         const activeStatusFilter = Object.keys(statusColumnFilters).find(
102             filterName => statusColumnFilters[filterName].selected
103         );
104
105         // Get all the subprocess' container requests and containers.
106         const fb = new FilterBuilder().addEqual('requesting_container_uuid', parentContainerRequest.containerUuid);
107         switch (activeStatusFilter) {
108             case ProcessStatusFilter.COMPLETED: {
109                 fb.addEqual('container.state', 'Complete');
110                 fb.addEqual('container.exit_code', '0');
111                 break;
112             }
113             case ProcessStatusFilter.FAILED: {
114                 fb.addEqual('container.state', 'Complete');
115                 fb.addDistinct('container.exit_code', '0');
116                 break;
117             }
118             case ProcessStatusFilter.CANCELLED:
119             case ProcessStatusFilter.FAILED:
120             case ProcessStatusFilter.LOCKED:
121             case ProcessStatusFilter.QUEUED:
122             case ProcessStatusFilter.RUNNING: {
123                 fb.addEqual('container.state', activeStatusFilter);
124                 break;
125             }
126         }
127         const statusFilters = fb.getFilters();
128
129         const nameFilters = dataExplorer.searchValue
130             ? new FilterBuilder()
131                 .addILike("name", dataExplorer.searchValue)
132                 .getFilters()
133             : '';
134
135         return joinFilters(
136             nameFilters,
137             statusFilters
138         );
139     };
140
141 export const setItems = (listResults: ListResults<ProcessResource>) =>
142     subprocessPanelActions.SET_ITEMS({
143         ...listResultsToDataExplorerItemsMeta(listResults),
144         items: listResults.items.map(resource => resource.uuid),
145     });
146
147 const couldNotFetchSubprocesses = () =>
148     snackbarActions.OPEN_SNACKBAR({
149         message: 'Could not fetch subprocesses.',
150         kind: SnackbarKind.ERROR
151     });