mirror of
https://github.com/hay-kot/homebox.git
synced 2024-11-17 06:08:42 +00:00
102 lines
2.5 KiB
TypeScript
102 lines
2.5 KiB
TypeScript
export enum Method {
|
|
GET = 'GET',
|
|
POST = 'POST',
|
|
PUT = 'PUT',
|
|
DELETE = 'DELETE',
|
|
}
|
|
|
|
export type RequestInterceptor = (r: Response) => void;
|
|
export type ResponseInterceptor = (r: Response) => void;
|
|
|
|
export interface TResponse<T> {
|
|
status: number;
|
|
error: boolean;
|
|
data: T;
|
|
response: Response;
|
|
}
|
|
|
|
export class Requests {
|
|
private baseUrl: string;
|
|
private token: () => string;
|
|
private headers: Record<string, string> = {};
|
|
private responseInterceptors: ResponseInterceptor[] = [];
|
|
|
|
addResponseInterceptor(interceptor: ResponseInterceptor) {
|
|
this.responseInterceptors.push(interceptor);
|
|
}
|
|
|
|
private callResponseInterceptors(response: Response) {
|
|
this.responseInterceptors.forEach(i => i(response));
|
|
}
|
|
|
|
private url(rest: string): string {
|
|
return this.baseUrl + rest;
|
|
}
|
|
|
|
constructor(baseUrl: string, token: string | (() => string) = '', headers: Record<string, string> = {}) {
|
|
this.baseUrl = baseUrl;
|
|
this.token = typeof token === 'string' ? () => token : token;
|
|
this.headers = headers;
|
|
}
|
|
|
|
public get<T>(url: string): Promise<TResponse<T>> {
|
|
return this.do<T>(Method.GET, url);
|
|
}
|
|
|
|
public post<T, U>(url: string, payload: T): Promise<TResponse<U>> {
|
|
return this.do<U>(Method.POST, url, payload);
|
|
}
|
|
|
|
public put<T, U>(url: string, payload: T): Promise<TResponse<U>> {
|
|
return this.do<U>(Method.PUT, url, payload);
|
|
}
|
|
|
|
public delete<T>(url: string): Promise<TResponse<T>> {
|
|
return this.do<T>(Method.DELETE, url);
|
|
}
|
|
|
|
private methodSupportsBody(method: Method): boolean {
|
|
return method === Method.POST || method === Method.PUT;
|
|
}
|
|
|
|
private async do<T>(method: Method, url: string, payload: Object = {}): Promise<TResponse<T>> {
|
|
const args: RequestInit = {
|
|
method,
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
...this.headers,
|
|
},
|
|
};
|
|
|
|
const token = this.token();
|
|
if (token !== '' && args.headers !== undefined) {
|
|
args.headers['Authorization'] = token;
|
|
}
|
|
|
|
if (this.methodSupportsBody(method)) {
|
|
args.body = JSON.stringify(payload);
|
|
}
|
|
|
|
const response = await fetch(this.url(url), args);
|
|
this.callResponseInterceptors(response);
|
|
|
|
const data: T = await (async () => {
|
|
if (response.status === 204) {
|
|
return {} as T;
|
|
}
|
|
|
|
try {
|
|
return await response.json();
|
|
} catch (e) {
|
|
return {} as T;
|
|
}
|
|
})();
|
|
|
|
return {
|
|
status: response.status,
|
|
error: !response.ok,
|
|
data,
|
|
response,
|
|
};
|
|
}
|
|
}
|