Skip to content

Commit 3f51efd

Browse files
committed
Merge branch 'master' into next
2 parents 1223710 + 8aa1417 commit 3f51efd

File tree

8 files changed

+73
-70
lines changed

8 files changed

+73
-70
lines changed

README.md

+2-3
Original file line numberDiff line numberDiff line change
@@ -88,11 +88,10 @@ Use it with `fetch`, Axios or other data fetching libraries, even GraphQL.
8888

8989
[abortable fetch]: https://developers.google.com/web/updates/2017/09/abortable-fetch
9090

91-
> ## Upgrading to v8
91+
> ## Upgrading to v9
9292
>
93-
> Version 8 comes with breaking changes.
93+
> Version 9 comes with a minor breaking change.
9494
> See [Upgrading](https://docs.react-async.com/installation#upgrading) for details.
95-
> A [codemod](https://github.com/async-library/react-async/tree/master/codemods) is available.
9695
9796
# Documentation
9897

docs/state.md

+12-6
Original file line numberDiff line numberDiff line change
@@ -108,9 +108,12 @@ The number of times a promise was started.
108108

109109
> `Promise`
110110
111-
A reference to the internal wrapper promise created when starting a new promise \(either automatically or by invoking `run` / `reload`\). It fulfills or rejects along with the provided `promise` / `promiseFn` / `deferFn`. Useful as a chainable alternative to the `onResolve` / `onReject` callbacks.
111+
A reference to the internal wrapper promise created when starting a new promise \(either automatically or by invoking
112+
`run` / `reload`\). It fulfills or rejects along with the provided `promise` / `promiseFn` / `deferFn`. Useful as a
113+
chainable alternative to the `onResolve` / `onReject` callbacks.
112114

113-
Warning! If you chain on `promise`, you MUST provide a rejection handler \(e.g. `.catch(...)`\). Otherwise React will throw an exception and crash if the promise rejects.
115+
Warning! If you chain on `promise`, you MUST provide a rejection handler \(e.g. `.catch(...)`\). Otherwise React will
116+
throw an exception and crash if the promise rejects.
114117

115118
## `run`
116119

@@ -120,15 +123,18 @@ Runs the `deferFn`, passing any arguments provided as an array.
120123

121124
When used with `useFetch`, `run` has several overloaded signatures:
122125

123-
> `function(resource: String | Resource, init: Object | (init: Object) => Object): void`
124-
>
125-
> `function(init: Object | (init: Object) => Object): void`
126+
> `function(override: OverrideParams | (params: OverrideParams) => OverrideParams): void`
126127
>
127128
> `function(event: SyntheticEvent | Event): void`
128129
>
129130
> `function(): void`
130131
131-
This way you can run the `fetch` request using the provided `resource` and `init`. `resource` can be omitted. If `init` is an object it will be spread over the default `init` \(`useFetch`'s 2nd argument\). If it's a function it will be invoked with the default `init` and should return a new `init` object. This way you can either extend or override the value of `init`, for example to set request headers.
132+
Where `type OverrideParams = { resource?: RequestInfo } & Partial<RequestInit>`.
133+
134+
This way you can run the `fetch` request with custom `resource` and `init`. If `override` is an object it will be spread
135+
over the default `resource` and `init` for `fetch`. If it's a function it will be invoked with the params defined with
136+
`useFetch`, and should return an `override` object. This way you can either extend or override the value of `resource`
137+
and `init`, for example to change the URL or set custom request headers.
132138

133139
## `reload`
134140

docs/upgrading.md

+8-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,11 @@
11
# Upgrading
22

3+
## Upgrade to v9
4+
5+
The rejection value for failed requests with `useFetch` was changed. Previously it was the Response object. Now it's an
6+
Error object with `response` property. If you are using `useFetch` and are using the `error` value, expecting it to be
7+
of type Response, you must now use `error.response` instead.
8+
39
## Upgrade to v8
410

511
All standalone helper components were renamed to avoid import naming collision.
@@ -12,7 +18,8 @@ All standalone helper components were renamed to avoid import naming collision.
1218

1319
> A [codemod](https://github.com/async-library/react-async/tree/master/codemods) is available to automate the upgrade.
1420
15-
The return type for `run` was changed from `Promise` to `undefined`. You should now use the `promise` prop instead. This is a manual upgrade. See [`promise`](installation.md#promise-1) for details.
21+
The return type for `run` was changed from `Promise` to `undefined`. You should now use the `promise` prop instead. This
22+
is a manual upgrade. See [`promise`](state.md#promise) for details.
1623

1724
## Upgrade to v6
1825

@@ -25,11 +32,8 @@ The return type for `run` was changed from `Promise` to `undefined`. You should
2532
## Upgrade to v4
2633

2734
- `deferFn` now receives an `args` array as the first argument, instead of arguments to `run` being spread at the front
28-
2935
of the arguments list. This enables better interop with TypeScript. You can use destructuring to keep using your
30-
3136
existing variables.
3237

3338
- The shorthand version of `useAsync` now takes the `options` object as optional second argument. This used to be
34-
3539
`initialValue`, but was undocumented and inflexible.

packages/react-async/src/index.d.ts

+4-4
Original file line numberDiff line numberDiff line change
@@ -227,11 +227,11 @@ type AsyncInitialWithout<K extends keyof AsyncInitial<T>, T> =
227227
| Omit<AsyncFulfilled<T>, K>
228228
| Omit<AsyncRejected<T>, K>
229229

230+
type OverrideParams = { resource?: RequestInfo } & Partial<RequestInit>
231+
230232
type FetchRun<T> = {
231-
run(overrideResource: RequestInfo, overrideInit: (init: RequestInit) => RequestInit): void
232-
run(overrideResource: RequestInfo, overrideInit: Partial<RequestInit>): void
233-
run(overrideInit: (init: RequestInit) => RequestInit): void
234-
run(overrideInit: Partial<RequestInit>): void
233+
run(overrideParams: (params?: OverrideParams) => OverrideParams): void
234+
run(overrideParams: OverrideParams): void
235235
run(ignoredEvent: React.SyntheticEvent): void
236236
run(ignoredEvent: Event): void
237237
run(): void

packages/react-async/src/reducer.js

+22-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,27 @@
11
import { getInitialStatus, getIdleStatus, getStatusProps, statusTypes } from "./status"
22

3-
export const neverSettle = new Promise(() => {})
3+
// This exists to make sure we don't hold any references to user-provided functions
4+
class NeverSettle extends Promise {
5+
constructor() {
6+
super(() => {}, () => {})
7+
/* istanbul ignore next */
8+
if (Object.setPrototypeOf) {
9+
// Not available in IE 10, but can be polyfilled
10+
Object.setPrototypeOf(this, NeverSettle.prototype)
11+
}
12+
}
13+
finally() {
14+
return this
15+
}
16+
catch() {
17+
return this
18+
}
19+
then() {
20+
return this
21+
}
22+
}
23+
24+
export const neverSettle = new NeverSettle()
425

526
export const actionTypes = {
627
start: "start",

packages/react-async/src/useAsync.js

+16-10
Original file line numberDiff line numberDiff line change
@@ -168,6 +168,11 @@ const useAsync = (arg1, arg2) => {
168168
export class FetchError extends Error {
169169
constructor(response) {
170170
super(`${response.status} ${response.statusText}`)
171+
/* istanbul ignore next */
172+
if (Object.setPrototypeOf) {
173+
// Not available in IE 10, but can be polyfilled
174+
Object.setPrototypeOf(this, FetchError.prototype)
175+
}
171176
this.response = response
172177
}
173178
}
@@ -178,8 +183,6 @@ const parseResponse = (accept, json) => res => {
178183
return accept === "application/json" ? res.json() : res
179184
}
180185

181-
const isResource = value => typeof value === "string" || (typeof value === "object" && value.url)
182-
183186
const useAsyncFetch = (resource, init, { defer, json, ...options } = {}) => {
184187
const method = resource.method || (init && init.method)
185188
const headers = resource.headers || (init && init.headers) || {}
@@ -194,15 +197,17 @@ const useAsyncFetch = (resource, init, { defer, json, ...options } = {}) => {
194197
...options,
195198
[fn]: useCallback(
196199
(arg1, arg2, arg3) => {
197-
const [runArgs, signal] = isDefer ? [arg1, arg3.signal] : [[], arg2.signal]
198-
const [runResource, runInit] = isResource(runArgs[0]) ? runArgs : [, runArgs[0]]
199-
if (typeof runInit === "object" && "preventDefault" in runInit) {
200-
// Don't spread Events or SyntheticEvents
201-
return doFetch(runResource || resource, { signal, ...init })
200+
const [override, signal] = isDefer ? [arg1[0], arg3.signal] : [undefined, arg2.signal]
201+
const isEvent = typeof override === "object" && "preventDefault" in override
202+
if (!override || isEvent) {
203+
return doFetch(resource, { signal, ...init })
204+
}
205+
if (typeof override === "function") {
206+
const { resource: runResource, ...runInit } = override({ resource, signal, ...init })
207+
return doFetch(runResource || resource, { signal, ...runInit })
202208
}
203-
return typeof runInit === "function"
204-
? doFetch(runResource || resource, { signal, ...runInit(init) })
205-
: doFetch(runResource || resource, { signal, ...init, ...runInit })
209+
const { resource: runResource, ...runInit } = override
210+
return doFetch(runResource || resource, { signal, ...init, ...runInit })
206211
},
207212
[identity] // eslint-disable-line react-hooks/exhaustive-deps
208213
),
@@ -211,6 +216,7 @@ const useAsyncFetch = (resource, init, { defer, json, ...options } = {}) => {
211216
return state
212217
}
213218

219+
/* istanbul ignore next */
214220
const unsupported = () => {
215221
throw new Error(
216222
"useAsync requires React v16.8 or up. Upgrade your React version or use the <Async> component instead."

packages/react-async/src/useAsync.spec.js

+8-42
Original file line numberDiff line numberDiff line change
@@ -203,61 +203,27 @@ describe("useFetch", () => {
203203
expect(json).toHaveBeenCalled()
204204
})
205205

206-
test("calling `run` with a callback as argument allows to override `init` parameters", () => {
206+
test("calling `run` with a callback as argument allows to override fetch parameters", () => {
207+
const override = params => ({ ...params, resource: "/bar", body: '{"name":"bar"}' })
207208
const component = (
208-
<Fetch input="/test" init={{ method: "POST", body: '{"name":"foo"}' }}>
209-
{({ run }) => (
210-
<button onClick={() => run(init => ({ ...init, body: '{"name":"bar"}' }))}>run</button>
211-
)}
212-
</Fetch>
213-
)
214-
const { getByText } = render(component)
215-
expect(globalScope.fetch).not.toHaveBeenCalled()
216-
fireEvent.click(getByText("run"))
217-
expect(globalScope.fetch).toHaveBeenCalledWith(
218-
"/test",
219-
expect.objectContaining({ method: "POST", signal: abortCtrl.signal, body: '{"name":"bar"}' })
220-
)
221-
})
222-
223-
test("calling `run` with an object as argument allows to override `init` parameters", () => {
224-
const component = (
225-
<Fetch input="/test" init={{ method: "POST", body: '{"name":"foo"}' }}>
226-
{({ run }) => <button onClick={() => run({ body: '{"name":"bar"}' })}>run</button>}
227-
</Fetch>
228-
)
229-
const { getByText } = render(component)
230-
expect(globalScope.fetch).not.toHaveBeenCalled()
231-
fireEvent.click(getByText("run"))
232-
expect(globalScope.fetch).toHaveBeenCalledWith(
233-
"/test",
234-
expect.objectContaining({ method: "POST", signal: abortCtrl.signal, body: '{"name":"bar"}' })
235-
)
236-
})
237-
238-
test("calling `run` with a url allows to override fetch's `resource` parameter", () => {
239-
const component = (
240-
<Fetch input="/foo" options={{ defer: true }}>
241-
{({ run }) => <button onClick={() => run("/bar")}>run</button>}
209+
<Fetch input="/foo" init={{ method: "POST", body: '{"name":"foo"}' }}>
210+
{({ run }) => <button onClick={() => run(override)}>run</button>}
242211
</Fetch>
243212
)
244213
const { getByText } = render(component)
245214
expect(globalScope.fetch).not.toHaveBeenCalled()
246215
fireEvent.click(getByText("run"))
247216
expect(globalScope.fetch).toHaveBeenCalledWith(
248217
"/bar",
249-
expect.objectContaining({ signal: abortCtrl.signal })
218+
expect.objectContaining({ method: "POST", signal: abortCtrl.signal, body: '{"name":"bar"}' })
250219
)
251220
})
252221

253-
test("overriding the `resource` can be combined with overriding `init`", () => {
222+
test("calling `run` with an object as argument allows to override fetch parameters", () => {
223+
const override = { resource: "/bar", body: '{"name":"bar"}' }
254224
const component = (
255225
<Fetch input="/foo" init={{ method: "POST", body: '{"name":"foo"}' }}>
256-
{({ run }) => (
257-
<button onClick={() => run("/bar", init => ({ ...init, body: '{"name":"bar"}' }))}>
258-
run
259-
</button>
260-
)}
226+
{({ run }) => <button onClick={() => run(override)}>run</button>}
261227
</Fetch>
262228
)
263229
const { getByText } = render(component)

stories/index.stories.js

+1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
1+
/* eslint-disable react/prop-types */
12
import React, { Suspense } from "react"
23
import { storiesOf } from "@storybook/react"
34

0 commit comments

Comments
 (0)