From b7aa426a9dcbda36416afbd5ecab60d1be295577 Mon Sep 17 00:00:00 2001 From: Tejas Kumar Date: Mon, 9 Jul 2018 17:32:56 +0200 Subject: [PATCH] Throw error responses and leave error handling to the user --- src/Context.tsx | 2 +- src/Poll.tsx | 5 +- src/index.tsx | 130 +++++++++++++++++++++++++----------------------- 3 files changed, 72 insertions(+), 65 deletions(-) diff --git a/src/Context.tsx b/src/Context.tsx index 6af37011..7acc2b2a 100644 --- a/src/Context.tsx +++ b/src/Context.tsx @@ -13,7 +13,7 @@ export interface RestfulReactProviderProps { /** * Options passed to the fetch request. */ - requestOptions?: Partial; + requestOptions?: (() => Partial) | Partial; } const { Provider, Consumer: RestfulReactConsumer } = React.createContext({ diff --git a/src/Poll.tsx b/src/Poll.tsx index 58e4c59b..b7cf93b7 100644 --- a/src/Poll.tsx +++ b/src/Poll.tsx @@ -158,7 +158,10 @@ class ContextlessPoll extends React.Component, Readonly = (data: any) => T; /** * HTTP Verbs: POST/GET/PUT/PATCH/DELETE. */ -export type RequestMethod = "POST" | "GET" | "PUT" | "DELETE" | "PATCH"; +export type RequestMethod = "GET" | "POST" | "PUT" | "PATCH" | "DELETE"; /** * A collection of actions that map to @@ -18,15 +18,15 @@ export type RequestMethod = "POST" | "GET" | "PUT" | "DELETE" | "PATCH"; */ export interface Verbs { /** GET a resource */ - get: (path?: string, requestOptions?: Partial) => Promise; + get: (path?: string, extraRequestOptions?: RequestInit) => Promise; /** DELETE a resource */ - destroy: (id?: string, requestOptions?: Partial) => Promise; + destroy: (id?: string, extraRequestOptions?: RequestInit) => Promise; /** POST a resource */ - post: (data?: string, requestOptions?: Partial) => Promise; + post: (data?: string, extraRequestOptions?: RequestInit) => Promise; /** PUT a resource */ - put: (data?: string, requestOptions?: Partial) => Promise; + put: (data?: string, extraRequestOptions?: RequestInit) => Promise; /** PATCH a resource */ - patch: (data?: string, requestOptions?: Partial) => Promise; + patch: (data?: string, extraRequestOptions?: RequestInit) => Promise; } /** @@ -69,7 +69,7 @@ export interface GetComponentProps { */ children: (data: T | null, states: States, actions: Verbs, meta: Meta) => React.ReactNode; /** Options passed into the fetch call. */ - requestOptions?: Partial; + requestOptions?: RestfulReactProviderProps["requestOptions"]; /** * A function to resolve data return from the backend, most typically * used when the backend response needs to be adapted in some way. @@ -120,91 +120,99 @@ class ContextlessGet extends React.Component, Readonly) { // If the path or base prop changes, refetch! const { path, base } = this.props; if (prevProps.path !== path || prevProps.base !== base) { - this.shouldFetchImmediately() && this.fetch("GET")(); + this.shouldFetchImmediately() && this.fetch()(); } } - setError = (erroredResponse: Response) => { - this.setState(() => ({ - response: erroredResponse, - error: erroredResponse.statusText, - loading: false, - })); - return null; + getRequestOptions = (extraOptions?: Partial, extraHeaders?: boolean | { [key: string]: string }) => { + const { requestOptions } = this.props; + + if (typeof requestOptions === "function") { + return { + ...extraOptions, + ...requestOptions(), + headers: new Headers({ + ...(typeof extraHeaders !== "boolean" ? extraHeaders : {}), + ...(extraOptions || {}).headers, + ...(requestOptions() || {}).headers, + }), + }; + } + + return { + ...extraOptions, + ...requestOptions, + headers: new Headers({ + ...(typeof extraHeaders !== "boolean" ? extraHeaders : {}), + ...(extraOptions || {}).headers, + ...(requestOptions || {}).headers, + }), + }; }; - fetch = (method?: RequestMethod) => { - const { base, path, requestOptions: options } = this.props; - this.setState(() => ({ error: "", loading: true })); + fetch = (method: RequestMethod = "GET") => { + const { base, path } = this.props; switch (method) { case "POST": case "PUT": case "PATCH": case "DELETE": - return async (data?: string, requestOptions?: Partial) => { + return async (body?: string, thisRequestOptions?: RequestInit) => { + this.setState(() => ({ error: "", loading: true })); const isJSON = () => { try { - return data ? Boolean(JSON.parse(data)) : false; + return body ? Boolean(JSON.parse(body)) : false; } catch (e) { return false; } }; - try { - const response = await fetch(`${base}${path}`, { - ...options, - ...requestOptions, - headers: new Headers({ - ...(isJSON() && { "content-type": "application/json" }), - ...(options || {}).headers, - ...(requestOptions || {}).headers, - }), - method, - body: data, - }); + const response = await fetch(`${base}${path}`, { + ...this.getRequestOptions(thisRequestOptions, isJSON() && { "content-type": "application/json" }), + method, + body, + }); - if (!response.ok) { - throw response; - } + this.setState({ loading: false }); - this.setState(() => ({ loading: false })); - const responseData: Promise = - response.headers.get("content-type") === "application/json" ? response.json() : response.text(); - return responseData; - } catch (erroredResponse) { - return this.setError(erroredResponse); + if (!response.ok) { + throw response; } + + const responseData: Promise = + response.headers.get("content-type") === "application/json" ? response.json() : response.text(); + return responseData; }; default: - return async (requestPath?: string, requestOptions?: Partial) => { + return async (requestPath?: string, thisRequestOptions?: RequestInit) => { + this.setState(() => ({ error: "", loading: true })); + const { resolve } = this.props; const foolProofResolve = resolve || (data => data); - try { - const response = await fetch(`${base}${requestPath || path || ""}`, { ...options, ...requestOptions }); + const response = await fetch( + `${base}${requestPath || path || ""}`, + this.getRequestOptions(thisRequestOptions), + ); - if (!response.ok) { - throw `Failed to fetch: ${response.status}${response.statusText}`; - } + if (!response.ok) { + this.setState({ loading: false, error: `Failed to fetch: ${response.status} ${response.statusText}` }); + throw response; + } - const data: T = - response.headers.get("content-type") === "application/json" - ? await response.json() - : await response.text(); + const data: T = + response.headers.get("content-type") === "application/json" ? await response.json() : await response.text(); - this.setState({ data: foolProofResolve(data), loading: false }); - return data; - } catch (erroredResponse) { - return this.setError(erroredResponse); - } + this.setState({ data: foolProofResolve(data) }); + return data; }; } }; @@ -244,11 +252,7 @@ function Get(props: GetComponentProps) { {contextProps => ( - + )}