From 5f7afe9cbfded74f7a67a23469a5c78eecab2968 Mon Sep 17 00:00:00 2001 From: Lenz Weber Date: Sat, 15 Feb 2020 17:25:43 +0100 Subject: [PATCH 1/2] createAsyncThunk return fulfilled/rejected action instead of re-trowing errors --- etc/redux-toolkit.api.md | 9 ++++-- src/createAsyncThunk.ts | 31 +++++++++---------- type-tests/files/createAsyncThunk.typetest.ts | 31 ++++++++++++++----- 3 files changed, 46 insertions(+), 25 deletions(-) diff --git a/etc/redux-toolkit.api.md b/etc/redux-toolkit.api.md index 700363d48d..a7167b7c3f 100644 --- a/etc/redux-toolkit.api.md +++ b/etc/redux-toolkit.api.md @@ -102,8 +102,13 @@ export function createAction

(type: T): Payl export function createAction, T extends string = string>(type: T, prepareAction: PA): PayloadActionCreator['payload'], T, PA>; // @alpha (undocumented) -export function createAsyncThunk = AsyncThunksArgs>(type: ActionType, payloadCreator: (args: ActionParams, thunkArgs: TA) => Promise | Returned): { - (args: ActionParams): (dispatch: TA["dispatch"], getState: TA["getState"], extra: TA["extra"]) => Promise; +export function createAsyncThunk = AsyncThunksArgs>(type: ActionType, payloadCreator: (args: ActionParams, thunkArgs: TA) => Promise | Returned): ((args: ActionParams) => (dispatch: TA["dispatch"], getState: TA["getState"], extra: TA["extra"]) => Promise | import("./createAction").PayloadAction>) & { pending: import("./createAction").ActionCreatorWithPreparedPayload<[string, ActionParams], undefined, string, never, { args: ActionParams; requestId: string; diff --git a/src/createAsyncThunk.ts b/src/createAsyncThunk.ts index 944b894f25..39713b7e97 100644 --- a/src/createAsyncThunk.ts +++ b/src/createAsyncThunk.ts @@ -101,34 +101,33 @@ export function createAsyncThunk< ) => { const requestId = nanoid() - let result: Returned + let finalAction: ReturnType try { dispatch(pending(requestId, args)) - result = (await payloadCreator(args, { - dispatch, - getState, - extra, - requestId - } as TA)) as Returned + finalAction = fulfilled( + await payloadCreator(args, { + dispatch, + getState, + extra, + requestId + } as TA), + requestId, + args + ) } catch (err) { const serializedError = miniSerializeError(err) - dispatch(rejected(serializedError, requestId, args)) - // Rethrow this so the user can handle if desired - throw err + finalAction = rejected(serializedError, requestId, args) } // We dispatch "success" _after_ the catch, to avoid having any errors // here get swallowed by the try/catch block, // per https://twitter.com/dan_abramov/status/770914221638942720 // and https://redux-toolkit.js.org/tutorials/advanced-tutorial#async-error-handling-logic-in-thunks - return dispatch(fulfilled(result!, requestId, args)) + dispatch(finalAction) + return finalAction } } - actionCreator.pending = pending - actionCreator.rejected = rejected - actionCreator.fulfilled = fulfilled - - return actionCreator + return Object.assign(actionCreator, { pending, rejected, fulfilled }) } diff --git a/type-tests/files/createAsyncThunk.typetest.ts b/type-tests/files/createAsyncThunk.typetest.ts index 8badad5d5d..b50b81f472 100644 --- a/type-tests/files/createAsyncThunk.typetest.ts +++ b/type-tests/files/createAsyncThunk.typetest.ts @@ -1,4 +1,4 @@ -import { createAsyncThunk, Dispatch, createReducer } from 'src' +import { createAsyncThunk, Dispatch, createReducer, AnyAction } from 'src' import { ThunkDispatch } from 'redux-thunk' function expectType(t: T) { @@ -7,20 +7,37 @@ function expectType(t: T) { function fn() {} // basic usage -{ - const dispatch = fn as ThunkDispatch +;(async function() { + const dispatch = fn as ThunkDispatch<{}, any, AnyAction> const async = createAsyncThunk('test', (id: number) => Promise.resolve(id * 2) ) - dispatch(async(3)) const reducer = createReducer({}, builder => builder - .addCase(async.pending, (_, action) => {}) + .addCase(async.pending, (_, action) => { + expectType>(action) + }) .addCase(async.fulfilled, (_, action) => { + expectType>(action) expectType(action.payload) }) - .addCase(async.rejected, (_, action) => {}) + .addCase(async.rejected, (_, action) => { + expectType>(action) + expectType(action.error) + }) ) -} + + const result = await dispatch(async(3)) + + if (async.fulfilled.match(result)) { + expectType>(result) + // typings:expect-error + expectType>(result) + } else { + expectType>(result) + // typings:expect-error + expectType>(result) + } +})() From 78acd7d6b6488af927f9f5a3fce1c0ac0a7e8096 Mon Sep 17 00:00:00 2001 From: Lenz Weber Date: Sat, 15 Feb 2020 18:01:23 +0100 Subject: [PATCH 2/2] add unwrapResult helper --- etc/redux-toolkit.api.md | 7 +++++++ src/createAsyncThunk.ts | 18 +++++++++++++++++- type-tests/files/createAsyncThunk.typetest.ts | 15 ++++++++++++++- 3 files changed, 38 insertions(+), 2 deletions(-) diff --git a/etc/redux-toolkit.api.md b/etc/redux-toolkit.api.md index a7167b7c3f..92233aa9e8 100644 --- a/etc/redux-toolkit.api.md +++ b/etc/redux-toolkit.api.md @@ -121,6 +121,13 @@ export function createAsyncThunk; + unwrapResult: (returned: import("./createAction").PayloadAction | import("./createAction").PayloadAction) => Returned; }; // @alpha (undocumented) diff --git a/src/createAsyncThunk.ts b/src/createAsyncThunk.ts index 39713b7e97..1e0c21608e 100644 --- a/src/createAsyncThunk.ts +++ b/src/createAsyncThunk.ts @@ -2,6 +2,8 @@ import { Dispatch } from 'redux' import nanoid from 'nanoid' import { createAction } from './createAction' +type Await

= P extends PromiseLike ? T : P + type AsyncThunksArgs = { dispatch: D getState: S @@ -129,5 +131,19 @@ export function createAsyncThunk< } } - return Object.assign(actionCreator, { pending, rejected, fulfilled }) + function unwrapResult( + returned: Await>> + ) { + if (rejected.match(returned)) { + throw returned.error + } + return returned.payload + } + + return Object.assign(actionCreator, { + pending, + rejected, + fulfilled, + unwrapResult + }) } diff --git a/type-tests/files/createAsyncThunk.typetest.ts b/type-tests/files/createAsyncThunk.typetest.ts index b50b81f472..04a0bf113d 100644 --- a/type-tests/files/createAsyncThunk.typetest.ts +++ b/type-tests/files/createAsyncThunk.typetest.ts @@ -1,5 +1,6 @@ import { createAsyncThunk, Dispatch, createReducer, AnyAction } from 'src' import { ThunkDispatch } from 'redux-thunk' +import { promises } from 'fs' function expectType(t: T) { return t @@ -29,7 +30,8 @@ function fn() {} }) ) - const result = await dispatch(async(3)) + const promise = dispatch(async(3)) + const result = await promise if (async.fulfilled.match(result)) { expectType>(result) @@ -40,4 +42,15 @@ function fn() {} // typings:expect-error expectType>(result) } + + promise + .then(async.unwrapResult) + .then(result => { + expectType(result) + // typings:expect-error + expectType(result) + }) + .catch(error => { + // catch is always any-typed, nothing we can do here + }) })()