1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
5 import { camelCase, isPlainObject, isArray, snakeCase } from "lodash";
6 import { AxiosInstance, AxiosPromise } from "axios";
7 import uuid from "uuid/v4";
8 import { ApiActions } from "services/api/api-actions";
9 import QueryString from "query-string";
17 export interface ListArguments {
25 includeOldVersions?: boolean;
28 export interface ListResults<T> {
34 itemsAvailable: number;
37 export class CommonService<T> {
38 protected serverApi: AxiosInstance;
39 protected resourceType: string;
40 protected actions: ApiActions;
41 protected readOnlyFields: string[];
43 constructor(serverApi: AxiosInstance, resourceType: string, actions: ApiActions, readOnlyFields: string[] = []) {
44 this.serverApi = serverApi;
45 this.resourceType = resourceType;
46 this.actions = actions;
47 this.readOnlyFields = readOnlyFields;
50 static mapResponseKeys = (response: { data: any }) =>
51 CommonService.mapKeys(camelCase)(response.data)
53 static mapKeys = (mapFn: (key: string) => string) =>
54 (value: any): any => {
56 case isPlainObject(value):
59 .map(key => [key, mapFn(key)])
60 .reduce((newValue, [key, newKey]) => ({
62 [newKey]: (key === 'items') ? CommonService.mapKeys(mapFn)(value[key]) : value[key]
65 return value.map(CommonService.mapKeys(mapFn));
71 protected validateUuid(uuid: string) {
73 throw new Error('UUID cannot be empty string');
77 static defaultResponse<R>(promise: AxiosPromise<R>, actions: ApiActions, mapKeys = true, showErrors = true): Promise<R> {
79 actions.progressFn(reqId, true);
82 actions.progressFn(reqId, false);
85 .then((response: { data: any }) => {
86 return mapKeys ? CommonService.mapResponseKeys(response) : response.data;
88 .catch(({ response }) => {
89 actions.progressFn(reqId, false);
90 const errors = CommonService.mapResponseKeys(response) as Errors;
91 errors.status = response.status;
92 actions.errorFn(reqId, errors, showErrors);
97 create(data?: Partial<T>, showErrors?: boolean) {
98 return CommonService.defaultResponse(
100 .post<T>(`/${this.resourceType}`, data && CommonService.mapKeys(snakeCase)(data)),
107 delete(uuid: string): Promise<T> {
108 this.validateUuid(uuid);
109 return CommonService.defaultResponse(
111 .delete(`/${this.resourceType}/${uuid}`),
116 get(uuid: string, showErrors?: boolean) {
117 this.validateUuid(uuid);
118 return CommonService.defaultResponse(
120 .get<T>(`/${this.resourceType}/${uuid}`),
127 list(args: ListArguments = {}, showErrors?: boolean): Promise<ListResults<T>> {
128 const { filters, select, ...other } = args;
130 ...CommonService.mapKeys(snakeCase)(other),
131 filters: filters ? `[${filters}]` : undefined,
133 ? `[${select.map(snakeCase).map(s => `"${s}"`).join(', ')}]`
137 if (QueryString.stringify(params).length <= 1500) {
138 return CommonService.defaultResponse(
139 this.serverApi.get(`/${this.resourceType}`, { params }),
144 // Using the POST special case to avoid URI length 414 errors.
145 const formData = new FormData();
146 formData.append("_method", "GET");
147 Object.keys(params).forEach(key => {
148 if (params[key] !== undefined) {
149 formData.append(key, params[key]);
152 return CommonService.defaultResponse(
153 this.serverApi.post(`/${this.resourceType}`, formData, {
164 update(uuid: string, data: Partial<T>) {
165 this.validateUuid(uuid);
166 return CommonService.defaultResponse(
168 .put<T>(`/${this.resourceType}/${uuid}`, data && CommonService.mapKeys(snakeCase)(data)),