From d2ee084bf50f8326155270a287dcb2ededd10257 Mon Sep 17 00:00:00 2001 From: Michael Lin Date: Mon, 27 Mar 2023 00:10:00 +0800 Subject: [PATCH] refactor(returning): refactor handling return value via rawReturn() and remove safeReturn() (#11) --- README.md | 51 +- package.json | 2 +- src/constant.ts | 1 + src/create.ts | 20 +- src/current.ts | 40 +- src/index.ts | 2 +- src/rawReturn.ts | 40 ++ src/safeReturn.ts | 24 - test/immer/base.test.ts | 63 ++- test/immer/patch.test.ts | 16 +- test/immer/produce.test.ts | 13 +- test/index.test.ts | 2 +- test/jsdoc.test.ts | 13 +- test/performance/benchmark.ts | 17 +- .../{justReturn.ts => rawReturn.ts} | 18 +- .../performance/benchmarks/returnWithDraft.ts | 47 +- test/rawReturn.0.snap.ts | 10 + test/rawReturn.test.ts | 508 ++++++++++++++++++ test/safeReturn.test.ts | 375 ------------- yarn.lock | 8 +- 20 files changed, 744 insertions(+), 526 deletions(-) create mode 100644 src/rawReturn.ts delete mode 100644 src/safeReturn.ts rename test/performance/benchmarks/{justReturn.ts => rawReturn.ts} (95%) create mode 100644 test/rawReturn.0.snap.ts create mode 100644 test/rawReturn.test.ts delete mode 100644 test/safeReturn.test.ts diff --git a/README.md b/README.md index ec63d768..996bf415 100644 --- a/README.md +++ b/README.md @@ -146,7 +146,7 @@ In this basic example, the changes to the draft are 'mutative' within the draft > Forbid accessing non-draftable values in strict mode(unless using [unsafe()](#unsafe)). - > It is recommended to enable `strict` in development mode and disable `strict` in production mode. This will ensure safe returns and also keep good performance in the production build. If the return value is mixed drafts or `undefined`, then use [safeReturn()](#safereturn). + > It is recommended to enable `strict` in development mode and disable `strict` in production mode. This will ensure safe explicit returns and also keep good performance in the production build. If the value that does not mix any current draft or is `undefined` is returned, then use [rawReturn()](#rawreturn). - enablePatches - `boolean | { pathAsArray?: boolean; arrayLengthAssignment?: boolean; }`, the default is false. @@ -302,19 +302,56 @@ expect(isDraftable(baseState.list)).toBeTruthy(); > You can set a mark to determine if the value is draftable, and the mark function should be the same as passing in `create()` mark option. -### `safeReturn()` +### `rawReturn()` -It is used as a safe return value to ensure that this value replaces the finalized draft value or use it to return `undefined` explicitly. +For return values that do not contain any drafts, you can use `rawReturn()` to wrap this return value to improve performance. It ensure that the return value is only returned explicitly. ```ts const baseState = { id: 'test' }; const state = create(baseState as { id: string } | undefined, (draft) => { - return safeReturn(undefined); + return rawReturn(undefined); }); expect(state).toBe(undefined); ``` -> You don't need to use `safeReturn()` when the return value doesn't have any draft. +> You don't need to use `rawReturn()` when the return value have any draft. + +```ts +const baseState = { a: 1, b: { c: 1 } }; +const state = create(baseState, (draft) => { + if (draft.b.c === 1) { + return { + ...draft, + a: 2, + }; + } +}); +expect(state).toEqual({ a: 2, b: { c: 1 } }); +expect(isDraft(state.b)).toBeFalsy(); +``` + + If you use `rawReturn()`, we recommend that you enable `strict` mode in development. + +```ts +const baseState = { a: 1, b: { c: 1 } }; +const state = create( + baseState, + (draft) => { + if (draft.b.c === 1) { + return rawReturn({ + ...draft, + a: 2, + }); + } + }, + { + strict: true, + } +); +// it will warn `The return value contains drafts, please don't use 'rawReturn()' to wrap the return value.` in strict mode. +expect(state).toEqual({ a: 2, b: { c: 1 } }); +expect(isDraft(state.b)).toBeFalsy(); +``` [View more API docs](./docs/README.md). @@ -440,10 +477,10 @@ const nextState = produce(baseState, (draft) => { Use Mutative ```ts -import { create, safeReturn } from 'mutative'; +import { create, rawReturn } from 'mutative'; const nextState = create(baseState, (draft) => { - return safeReturn(undefined); + return rawReturn(undefined); }); ``` diff --git a/package.json b/package.json index e1644799..344d33fd 100644 --- a/package.json +++ b/package.json @@ -95,7 +95,7 @@ "immer": "^9.0.19", "immutable": "^4.0.0", "jest": "^29.4.0", - "jsdoc-tests": "^0.1.1", + "jsdoc-tests": "^1.1.0", "json2csv": "^5.0.7", "lodash": "^4.17.21", "prettier": "^2.5.1", diff --git a/src/constant.ts b/src/constant.ts index 4d37dc85..abc1b23d 100644 --- a/src/constant.ts +++ b/src/constant.ts @@ -1,5 +1,6 @@ // Don't use `Symbol()` just for 3rd party access the draft export const PROXY_DRAFT = Symbol.for('__MUTATIVE_PROXY_DRAFT__'); +export const RAW_RETURN_SYMBOL = Symbol('__MUTATIVE_RAW_RETURN_SYMBOL__'); export const iteratorSymbol: typeof Symbol.iterator = Symbol.iterator; diff --git a/src/create.ts b/src/create.ts index 6bf75cfa..43f0122c 100644 --- a/src/create.ts +++ b/src/create.ts @@ -14,7 +14,7 @@ import { revokeProxy, } from './utils'; import { current, handleReturnValue } from './current'; -import { safeReturnValue } from './safeReturn'; +import { RAW_RETURN_SYMBOL } from './constant'; /** * `create(baseState, callback, options)` to create the next state @@ -108,7 +108,6 @@ function create(arg0: any, arg1: any, arg2?: any): any { strict, enablePatches, }; - safeReturnValue.length = 0; if ( !isDraftable(state, _options) && typeof state === 'object' && @@ -146,16 +145,21 @@ function create(arg0: any, arg1: any, arg2?: any): any { `Either the value is returned as a new non-draft value, or only the draft is modified without returning any value.` ); } - if (safeReturnValue.length) { - const _value = safeReturnValue.pop(); - if (typeof value === 'object' && value !== null) { - handleReturnValue(value); + const rawReturnValue = value?.[RAW_RETURN_SYMBOL] as [any] | undefined; + if (rawReturnValue) { + const _value = rawReturnValue[0]; + if (_options.strict && typeof value === 'object' && value !== null) { + handleReturnValue({ + rootDraft: proxyDraft, + value, + useRawReturn: true, + }); } return finalize([_value]); } if (value !== undefined) { - if (_options.strict && typeof value === 'object' && value !== null) { - handleReturnValue(value, true); + if (typeof value === 'object' && value !== null) { + handleReturnValue({ rootDraft: proxyDraft, value }); } return finalize([value]); } diff --git a/src/current.ts b/src/current.ts index acc1bec1..5c14b818 100644 --- a/src/current.ts +++ b/src/current.ts @@ -1,4 +1,4 @@ -import { DraftType } from './interface'; +import { DraftType, ProxyDraft } from './interface'; import { forEach, get, @@ -11,15 +11,23 @@ import { shallowCopy, } from './utils'; -export function handleReturnValue(value: T, warning = false) { +export function handleReturnValue(options: { + rootDraft: ProxyDraft | undefined; + value: T; + useRawReturn?: boolean; + isContainDraft?: boolean; + isRoot?: boolean; +}) { + const { rootDraft, value, useRawReturn = false, isRoot = true } = options; forEach(value, (key, item, source) => { const proxyDraft = getProxyDraft(item); - if (proxyDraft) { - if (__DEV__ && warning) { - console.warn( - `The return value contains drafts, please use safeReturn() to wrap the return value.` - ); - } + // just handle the draft which is created by the same rootDraft + if ( + proxyDraft && + rootDraft && + proxyDraft.finalities === rootDraft.finalities + ) { + options.isContainDraft = true; const currentValue = proxyDraft.original; if (source instanceof Set) { const arr = Array.from(source); @@ -31,9 +39,23 @@ export function handleReturnValue(value: T, warning = false) { set(source, key, currentValue); } } else if (typeof item === 'object' && item !== null) { - handleReturnValue(item, warning); + options.value = item; + options.isRoot = false; + handleReturnValue(options); } }); + if (__DEV__ && isRoot) { + if (!options.isContainDraft) + console.warn( + `The return value does not contain any draft, please use 'rawReturn()' to wrap the return value to improve performance.` + ); + + if (useRawReturn) { + console.warn( + `The return value contains drafts, please don't use 'rawReturn()' to wrap the return value.` + ); + } + } } function getCurrent(target: any) { diff --git a/src/index.ts b/src/index.ts index 64f9e86a..620f9bee 100644 --- a/src/index.ts +++ b/src/index.ts @@ -3,7 +3,7 @@ export { apply } from './apply'; export { original } from './original'; export { current } from './current'; export { unsafe } from './unsafe'; -export { safeReturn } from './safeReturn'; +export { rawReturn } from './rawReturn'; export { isDraft } from './utils/draft'; export { isDraftable } from './utils/draft'; diff --git a/src/rawReturn.ts b/src/rawReturn.ts new file mode 100644 index 00000000..b9c9da0f --- /dev/null +++ b/src/rawReturn.ts @@ -0,0 +1,40 @@ +import { RAW_RETURN_SYMBOL } from './constant'; + +/** + * Use rawReturn() to wrap the return value to skip the draft check and thus improve performance. + * + * ## Example + * + * ```ts + * import { create, rawReturn } from '../index'; + * + * const baseState = { foo: { bar: 'str' }, arr: [] }; + * const state = create( + * baseState, + * (draft) => { + * return rawReturn(baseState); + * }, + * ); + * expect(state).toBe(baseState); + * ``` + */ +export function rawReturn(value: T): T { + if (arguments.length === 0) { + throw new Error('rawReturn() must be called with a value.'); + } + if (arguments.length > 1) { + throw new Error('rawReturn() must be called with one argument.'); + } + if ( + __DEV__ && + value !== undefined && + (typeof value !== 'object' || value === null) + ) { + console.warn( + 'rawReturn() must be called with an object(including plain object, arrays, Set, Map, etc.) or `undefined`, other types do not need to be returned via rawReturn().' + ); + } + return { + [RAW_RETURN_SYMBOL]: [value], + } as never; +} diff --git a/src/safeReturn.ts b/src/safeReturn.ts deleted file mode 100644 index c66774ba..00000000 --- a/src/safeReturn.ts +++ /dev/null @@ -1,24 +0,0 @@ -export const safeReturnValue: unknown[] = []; - -/** - * It is used as a safe return value to ensure that this value replaces the finalized value. - */ -export function safeReturn(value: T): T { - if (arguments.length === 0) { - throw new Error('safeReturn() must be called with a value.'); - } - if (arguments.length > 1) { - throw new Error('safeReturn() must be called with one argument.'); - } - if ( - __DEV__ && - value !== undefined && - (typeof value !== 'object' || value === null) - ) { - console.warn( - 'safeReturn() must be called with an object or undefined, other types do not need to be returned via safeReturn().' - ); - } - safeReturnValue[0] = value; - return value; -} diff --git a/test/immer/base.test.ts b/test/immer/base.test.ts index 3e252018..be7419d8 100644 --- a/test/immer/base.test.ts +++ b/test/immer/base.test.ts @@ -1,6 +1,6 @@ // @ts-nocheck 'use strict'; -import { create, original, isDraft, safeReturn } from '../../src'; +import { create, original, isDraft, rawReturn } from '../../src'; import deepFreeze from 'deep-freeze'; import * as lodash from 'lodash'; import { getType } from '../../src/utils'; @@ -1522,7 +1522,7 @@ function runBaseTest( // Modified parent test parent.c = 1; - expect(produce({}, () => [parent.b])[0]).toBe(parent.b); + expect(produce(null, () => [parent.b])[0]).toBe(parent.b); parent.b.x; // Ensure proxy not revoked. }); }); @@ -1531,28 +1531,26 @@ function runBaseTest( const base = {}; const result = create(base, (s1) => // ! it's different from mutative - safeReturn( - create( - { s1 }, - (s2) => { - expect(original(s2.s1)).toBe(s1); - s2.n = 1; - s2.s1 = create( - { s2 }, - (s3) => { - expect(original(s3.s2)).toBe(s2); - expect(original(s3.s2.s1)).toBe(s2.s1); - return s3.s2.s1; - }, - // ! it's different from mutative - { enableAutoFreeze: false } - ); - }, - { + create( + { s1 }, + (s2) => { + expect(original(s2.s1)).toBe(s1); + s2.n = 1; + s2.s1 = create( + { s2 }, + (s3) => { + expect(original(s3.s2)).toBe(s2); + expect(original(s3.s2.s1)).toBe(s2.s1); + return s3.s2.s1; + }, // ! it's different from mutative - enableAutoFreeze: false, - } - ) + { enableAutoFreeze: false } + ); + }, + { + // ! it's different from mutative + enableAutoFreeze: false, + } ) ); expect(result.n).toBe(1); @@ -1805,8 +1803,7 @@ function runBaseTest( it('can return an object with two references to an unmodified draft', () => { const base = { a: {} }; const next = produce(base, (d) => { - // ! it's different from mutative - return safeReturn([d.a, d.a]); + return [d.a, d.a]; }); expect(next[0]).toBe(base.a); expect(next[0]).toBe(next[1]); @@ -1928,7 +1925,7 @@ function runBaseTest( produce(state, (draft) => { switch (action.type) { case 'SET_STARTING_DOTS': - return safeReturn(draft.availableStartingDots.map((a) => a)); + return draft.availableStartingDots.map((a) => a); default: break; } @@ -1951,9 +1948,9 @@ function runBaseTest( produce(state, (draft) => { switch (action.type) { case 'SET_STARTING_DOTS': - return safeReturn({ + return { dots: draft.availableStartingDots.map((a) => a), - }); + }; default: break; } @@ -2053,15 +2050,15 @@ function runBaseTest( expect(create(base, () => null)).toBe(null); expect(create(base, () => undefined)).toBe(3); expect(create(base, () => {})).toBe(3); - expect(create(base, () => safeReturn(undefined))).toBe(undefined); + expect(create(base, () => rawReturn(undefined))).toBe(undefined); expect(create({}, () => undefined)).toEqual({}); - expect(create({}, () => safeReturn(undefined))).toBe(undefined); - expect(create(3, () => safeReturn(undefined))).toBe(undefined); + expect(create({}, () => rawReturn(undefined))).toBe(undefined); + expect(create(3, () => rawReturn(undefined))).toBe(undefined); expect(create(() => undefined)({})).toEqual({}); - expect(create(() => safeReturn(undefined))({})).toBe(undefined); - expect(create(() => safeReturn(undefined))(3)).toBe(undefined); + expect(create(() => rawReturn(undefined))({})).toBe(undefined); + expect(create(() => rawReturn(undefined))(3)).toBe(undefined); }); describe('base state type', () => { diff --git a/test/immer/patch.test.ts b/test/immer/patch.test.ts index ee562619..6a4ea4ab 100644 --- a/test/immer/patch.test.ts +++ b/test/immer/patch.test.ts @@ -1,5 +1,5 @@ 'use strict'; -import { apply, create, isDraft, safeReturn } from '../../src'; +import { apply, create, isDraft, rawReturn } from '../../src'; jest.setTimeout(1000); @@ -1403,7 +1403,7 @@ test('#648 assigning object to itself should not change patches', () => { test('#791 patch for returning `undefined` is stored as undefined', () => { const [newState, patches] = create( { abc: 123 }, - (draft) => safeReturn(undefined), + (draft) => rawReturn(undefined), { enablePatches: true, } @@ -1447,7 +1447,7 @@ test('#888 patch to a primitive produces the primitive', () => { { const [res, patches] = create( { abc: 123 }, - (draft) => safeReturn(undefined), + (draft) => rawReturn(undefined), { enablePatches: true, } @@ -1456,35 +1456,35 @@ test('#888 patch to a primitive produces the primitive', () => { expect(patches).toEqual([{ op: 'replace', path: [], value: undefined }]); } { - const [res, patches] = create(null, (draft) => safeReturn(undefined), { + const [res, patches] = create(null, (draft) => rawReturn(undefined), { enablePatches: true, }); expect(res).toEqual(undefined); expect(patches).toEqual([{ op: 'replace', path: [], value: undefined }]); } { - const [res, patches] = create(0, (draft) => safeReturn(undefined), { + const [res, patches] = create(0, (draft) => rawReturn(undefined), { enablePatches: true, }); expect(res).toEqual(undefined); expect(patches).toEqual([{ op: 'replace', path: [], value: undefined }]); } { - const [res, patches] = create('foobar', (draft) => safeReturn(undefined), { + const [res, patches] = create('foobar', (draft) => rawReturn(undefined), { enablePatches: true, }); expect(res).toEqual(undefined); expect(patches).toEqual([{ op: 'replace', path: [], value: undefined }]); } { - const [res, patches] = create([], (draft) => safeReturn(undefined), { + const [res, patches] = create([], (draft) => rawReturn(undefined), { enablePatches: true, }); expect(res).toEqual(undefined); expect(patches).toEqual([{ op: 'replace', path: [], value: undefined }]); } { - const [res, patches] = create(false, (draft) => safeReturn(undefined), { + const [res, patches] = create(false, (draft) => rawReturn(undefined), { enablePatches: true, }); expect(res).toEqual(undefined); diff --git a/test/immer/produce.test.ts b/test/immer/produce.test.ts index eb84e7b4..4ddd81a8 100644 --- a/test/immer/produce.test.ts +++ b/test/immer/produce.test.ts @@ -5,8 +5,7 @@ import { Immutable, apply, castDraft, - castImmutable, - safeReturn, + rawReturn, } from '../../src'; interface State { @@ -269,12 +268,12 @@ it('can produce an undefined value', () => { const base = { a: 0 } as State; // Return only nothing. - let result = create(base, (_) => safeReturn(undefined)); + let result = create(base, (_) => rawReturn(undefined)); assert(result, _ as State); // Return maybe nothing. let result2 = create(base, (draft) => { - if (draft?.a ?? 0 > 0) return safeReturn(undefined); + if (draft?.a ?? 0 > 0) return rawReturn(undefined); }); assert(result2, _ as State); }); @@ -742,21 +741,21 @@ it('infers async curried', async () => { { // nothing allowed const res = create(base as State | undefined, (draft) => { - return safeReturn(undefined); + return rawReturn(undefined); }); assert(res, _ as State | undefined); } { // as any const res = create(base as State, (draft) => { - return safeReturn(undefined); + return rawReturn(undefined); }); assert(res, _ as State); } { // nothing not allowed create(base as State, (draft) => { - return safeReturn(undefined); + return rawReturn(undefined); }); } { diff --git a/test/index.test.ts b/test/index.test.ts index d78d88ae..43c2c839 100644 --- a/test/index.test.ts +++ b/test/index.test.ts @@ -2697,5 +2697,5 @@ test('can return an object that references itself', () => { expect(() => { // @ts-expect-error create(res, (draft) => res.self, { enableAutoFreeze: true }); - }).toThrowErrorMatchingInlineSnapshot(`"Forbids circular reference"`); + }).toThrowErrorMatchingInlineSnapshot(`"Maximum call stack size exceeded"`); }); diff --git a/test/jsdoc.test.ts b/test/jsdoc.test.ts index dbcc5f72..56a87392 100644 --- a/test/jsdoc.test.ts +++ b/test/jsdoc.test.ts @@ -2,18 +2,21 @@ import { jsdocTests } from 'jsdoc-tests'; describe('jsdoc', () => { test('create()', () => { - jsdocTests('../src/create.ts', __dirname, require); + jsdocTests('../src/create.ts', __dirname); }); test('apply()', () => { - jsdocTests('../src/apply.ts', __dirname, require); + jsdocTests('../src/apply.ts', __dirname); }); test('current()', () => { - jsdocTests('../src/current.ts', __dirname, require); + jsdocTests('../src/current.ts', __dirname); }); test('original()', () => { - jsdocTests('../src/original.ts', __dirname, require); + jsdocTests('../src/original.ts', __dirname); }); test('unsafe()', () => { - jsdocTests('../src/unsafe.ts', __dirname, require); + jsdocTests('../src/unsafe.ts', __dirname); + }); + test('rawReturn()', () => { + jsdocTests('../src/rawReturn.ts', __dirname); }); }); diff --git a/test/performance/benchmark.ts b/test/performance/benchmark.ts index 933e4ba9..efd7cd3a 100644 --- a/test/performance/benchmark.ts +++ b/test/performance/benchmark.ts @@ -1,3 +1,4 @@ +/* eslint-disable import/no-relative-packages */ /* eslint-disable prefer-template */ // @ts-nocheck import fs from 'fs'; @@ -5,13 +6,13 @@ import https from 'https'; import { Suite } from 'benchmark'; import { parse } from 'json2csv'; import deepFreeze from 'deep-freeze'; -import produce, { +import QuickChart from 'quickchart-js'; +import { + produce, enablePatches, produceWithPatches, setAutoFreeze, - setUseProxies, -} from 'immer'; -import QuickChart from 'quickchart-js'; +} from '../../../temp/immer/dist'; import { create } from '../..'; const labels = []; @@ -105,7 +106,7 @@ suite { onStart: () => { setAutoFreeze(false); - setUseProxies(true); + // setUseProxies(true); i = Math.random(); baseState = getData(); }, @@ -144,7 +145,7 @@ suite { onStart: () => { setAutoFreeze(true); - setUseProxies(true); + // setUseProxies(true); i = Math.random(); baseState = getData(); }, @@ -183,7 +184,7 @@ suite { onStart: () => { setAutoFreeze(false); - setUseProxies(true); + // setUseProxies(true); enablePatches(); i = Math.random(); baseState = getData(); @@ -223,7 +224,7 @@ suite { onStart: () => { setAutoFreeze(true); - setUseProxies(true); + // setUseProxies(true); enablePatches(); i = Math.random(); baseState = getData(); diff --git a/test/performance/benchmarks/justReturn.ts b/test/performance/benchmarks/rawReturn.ts similarity index 95% rename from test/performance/benchmarks/justReturn.ts rename to test/performance/benchmarks/rawReturn.ts index 18e3cc05..fc730a76 100644 --- a/test/performance/benchmarks/justReturn.ts +++ b/test/performance/benchmarks/rawReturn.ts @@ -12,7 +12,7 @@ import produce, { setAutoFreeze, setUseProxies, } from 'immer'; -import { create, safeReturn } from '../../..'; +import { create, rawReturn } from '../../..'; const getData = () => { const baseState: { arr: any[]; map: Record } = { @@ -49,9 +49,9 @@ suite () => { const state = create(baseState, (draft) => { draft.arr; - return { + return rawReturn({ ...baseState, - }; + }); }); }, { @@ -87,9 +87,9 @@ suite baseState, (draft) => { draft.arr; - return { + return rawReturn({ ...baseState, - }; + }); }, { enableAutoFreeze: true, @@ -130,9 +130,9 @@ suite baseState, (draft) => { draft.arr; - return { + return rawReturn({ ...baseState, - }; + }); }, { enableAutoFreeze: false, @@ -174,9 +174,9 @@ suite baseState, (draft) => { draft.arr; - return { + return rawReturn({ ...baseState, - }; + }); }, { enableAutoFreeze: true, diff --git a/test/performance/benchmarks/returnWithDraft.ts b/test/performance/benchmarks/returnWithDraft.ts index c465151d..0453ef08 100644 --- a/test/performance/benchmarks/returnWithDraft.ts +++ b/test/performance/benchmarks/returnWithDraft.ts @@ -11,7 +11,7 @@ import produce, { setAutoFreeze, setUseProxies, } from 'immer'; -import { create, safeReturn } from '../../..'; +import { create } from '../../..'; const getData = () => { const baseState: { arr: any[]; map: Record } = { @@ -46,13 +46,11 @@ suite .add( 'Mutative - No Freeze(by default)', () => { - const state = create(baseState, (draft) => - safeReturn({ - ...baseState, - arr: [...draft.arr, i], - map: { ...draft.map, [i]: { i } }, - }) - ); + const state = create(baseState, (draft) => ({ + ...baseState, + arr: [...draft.arr, i], + map: { ...draft.map, [i]: { i } }, + })); }, { onStart: () => { @@ -84,12 +82,11 @@ suite () => { const state = create( baseState, - (draft) => - safeReturn({ - ...baseState, - arr: [...draft.arr, i], - map: { ...draft.map, [i]: { i } }, - }), + (draft) => ({ + ...baseState, + arr: [...draft.arr, i], + map: { ...draft.map, [i]: { i } }, + }), { enableAutoFreeze: true, enablePatches: false, @@ -126,12 +123,11 @@ suite () => { const state = create( baseState, - (draft) => - safeReturn({ - ...baseState, - arr: [...draft.arr, i], - map: { ...draft.map, [i]: { i } }, - }), + (draft) => ({ + ...baseState, + arr: [...draft.arr, i], + map: { ...draft.map, [i]: { i } }, + }), { enableAutoFreeze: false, enablePatches: true, @@ -169,12 +165,11 @@ suite () => { const state = create( baseState, - (draft) => - safeReturn({ - ...baseState, - arr: [...draft.arr, i], - map: { ...draft.map, [i]: { i } }, - }), + (draft) => ({ + ...baseState, + arr: [...draft.arr, i], + map: { ...draft.map, [i]: { i } }, + }), { enableAutoFreeze: true, enablePatches: true, diff --git a/test/rawReturn.0.snap.ts b/test/rawReturn.0.snap.ts new file mode 100644 index 00000000..c6e89f30 --- /dev/null +++ b/test/rawReturn.0.snap.ts @@ -0,0 +1,10 @@ +import { create, rawReturn } from '../index'; + +const baseState = { foo: { bar: 'str' }, arr: [] }; +const state = create( + baseState, + (draft) => { + return rawReturn(baseState); + }, +); +expect(state).toBe(baseState); diff --git a/test/rawReturn.test.ts b/test/rawReturn.test.ts new file mode 100644 index 00000000..fbf2d06d --- /dev/null +++ b/test/rawReturn.test.ts @@ -0,0 +1,508 @@ +/* eslint-disable symbol-description */ +/* eslint-disable object-shorthand */ +/* eslint-disable @typescript-eslint/no-empty-function */ +/* eslint-disable no-unused-expressions */ +/* eslint-disable arrow-body-style */ +/* eslint-disable no-self-assign */ +/* eslint-disable no-param-reassign */ +/* eslint-disable @typescript-eslint/ban-ts-comment */ +import { create, isDraft, rawReturn } from '../src'; + +afterEach(() => { + jest.clearAllMocks(); +}); + +test('base', () => { + const base = 3; + expect(create(base, () => 4)).toBe(4); + // @ts-expect-error + expect(create(base, () => null)).toBe(null); + expect(create(base, () => undefined)).toBe(3); + expect(create(base, () => {})).toBe(3); + expect(create(base, () => rawReturn(undefined))).toBe(undefined); + + expect(create({}, () => undefined)).toEqual({}); + expect(create({}, () => rawReturn(undefined))).toBe(undefined); + expect(create(3, () => rawReturn(undefined))).toBe(undefined); + + expect(create(() => undefined)({})).toEqual({}); + expect(create(() => rawReturn(undefined))({})).toBe(undefined); + expect(create(() => rawReturn(undefined))(3)).toBe(undefined); +}); + +test('base enableAutoFreeze: true', () => { + create( + { a: { b: 1 } }, + (draft) => { + return { + a: draft.a, + }; + }, + { enableAutoFreeze: true } + ); +}); + +test('base enableAutoFreeze: true - with rawReturn()', () => { + expect(() => { + create( + { a: { b: 1 } }, + (draft) => { + return rawReturn({ + a: draft.a, + }); + }, + { enableAutoFreeze: true } + ); + }).toThrowError(); +}); + +describe.each([{ useRawReturn: true }, { useRawReturn: false }])( + 'check Primitive type $useRawReturn', + ({ useRawReturn }) => { + test(`useRawReturn ${useRawReturn}: check Primitive type with returning`, () => { + [ + -1, + 1, + 0, + NaN, + BigInt(1), + Infinity, + '', + 'test', + null, + true, + false, + Symbol('foo'), + ].forEach((value: any) => { + expect( + create(value, (draft) => { + return useRawReturn ? rawReturn(undefined) : ''; + }) + ).toBe(useRawReturn ? undefined : ''); + }); + }); + + test(`useRawReturn ${useRawReturn}: check Primitive type with returning and patches`, () => { + [ + -1, + 1, + 0, + NaN, + BigInt(1), + Infinity, + '', + 'test', + null, + true, + false, + undefined, + Symbol('foo'), + ].forEach((value: any) => { + expect( + create( + value, + (draft) => { + return useRawReturn ? rawReturn(undefined) : ''; + }, + { + enablePatches: true, + } + ) + ).toEqual([ + useRawReturn ? undefined : '', + [{ op: 'replace', path: [], value: useRawReturn ? undefined : '' }], + [{ op: 'replace', path: [], value: value }], + ]); + }); + }); + + test(`useRawReturn ${useRawReturn}: check Primitive type with returning, patches and freeze`, () => { + [ + -1, + 1, + 0, + NaN, + BigInt(1), + Infinity, + '', + 'test', + null, + true, + false, + undefined, + Symbol('foo'), + ].forEach((value: any) => { + expect( + create( + value, + (draft) => { + return useRawReturn ? rawReturn(undefined) : ''; + }, + { + enableAutoFreeze: true, + enablePatches: true, + } + ) + ).toEqual([ + useRawReturn ? undefined : '', + [{ op: 'replace', path: [], value: useRawReturn ? undefined : '' }], + [{ op: 'replace', path: [], value: value }], + ]); + }); + }); + + test(`useRawReturn ${useRawReturn}: check Primitive type with returning, patches, freeze and async`, async () => { + for (const value of [ + -1, + 1, + 0, + NaN, + BigInt(1), + Infinity, + '', + 'test', + null, + true, + false, + undefined, + Symbol('foo'), + ]) { + await expect( + await create( + value, + async (draft) => { + return useRawReturn ? rawReturn(undefined) : ''; + }, + { + enableAutoFreeze: true, + enablePatches: true, + } + ) + ).toEqual([ + useRawReturn ? undefined : '', + [{ op: 'replace', path: [], value: useRawReturn ? undefined : '' }], + [{ op: 'replace', path: [], value: value }], + ]); + } + }); + } +); + +test('error args', () => { + expect(() => + // @ts-expect-error + create(3, () => rawReturn(undefined, undefined)) + ).toThrowErrorMatchingInlineSnapshot( + `"rawReturn() must be called with one argument."` + ); + + expect(() => + // @ts-expect-error + create({}, () => rawReturn()) + ).toThrowErrorMatchingInlineSnapshot( + `"rawReturn() must be called with a value."` + ); + + const logSpy = jest.spyOn(console, 'warn').mockImplementation(() => {}); + + [ + -1, + 1, + 0, + NaN, + BigInt(1), + Infinity, + '', + 'test', + null, + false, + true, + Symbol('foo'), + ].forEach((value: any) => { + create({}, () => rawReturn(value)); + expect(logSpy).toHaveBeenCalledWith( + 'rawReturn() must be called with an object(including plain object, arrays, Set, Map, etc.) or `undefined`, other types do not need to be returned via rawReturn().' + ); + logSpy.mockClear(); + }); + + logSpy.mockReset(); +}); + +test('check warning rawReturn() in strict mode', () => { + class Foo { + a?: any; + } + const warnSpy = jest.spyOn(console, 'warn').mockImplementation(() => {}); + [ + (draft: any) => { + return rawReturn({ + a: draft.a, + }); + }, + (draft: any) => { + return rawReturn([draft.a]); + }, + (draft: any) => { + return rawReturn( + new Map([ + [ + 1, + { + a: draft.a, + }, + ], + ]) + ); + }, + (draft: any) => { + return rawReturn( + new Set([ + 1, + { + a: draft.a, + }, + ]) + ); + }, + (draft: any) => { + const foo = new Foo(); + foo.a = draft.a; + return rawReturn(foo); + }, + ].forEach((callback: any) => { + create({ a: { b: 1 } }, callback, { + strict: true, + }); + expect(warnSpy).toHaveBeenCalledWith( + `The return value contains drafts, please don't use 'rawReturn()' to wrap the return value.` + ); + warnSpy.mockClear(); + }); + warnSpy.mockReset(); +}); + +test('return parent draft', () => { + expect( + create({ a: 1 }, (draft) => { + const _draft = create({}, () => draft) as any; + _draft.a = 2; + return _draft; + }) + ).toEqual({ a: 2 }); +}); +test('mix more type draft without rawReturn()', () => { + [ + (draft: any) => ({ + a: { + b: draft.a, + }, + }), + (draft: any) => [{ c: draft.a }], + (draft: any) => new Map([[1, draft.a]]), + (draft: any) => new Set([1, draft.a]), + ].forEach((callback: any) => { + expect(() => create({ a: { b: 1 } }, callback)).not.toThrowError(); + }); +}); + +test('mix more type draft with rawReturn()', () => { + [ + (draft: any) => + rawReturn({ + a: { + b: draft.a, + }, + }), + (draft: any) => rawReturn([{ c: draft.a }]), + (draft: any) => rawReturn(new Map([[1, draft.a]])), + (draft: any) => rawReturn(new Set([1, draft.a])), + ].forEach((callback: any) => { + expect(() => + expect(create({ a: { b: 1 } }, callback)).toEqual({}) + ).toThrowError(); + }); +}); + +test(`safe returning with non-enumerable or symbolic properties`, () => { + const component = {}; + Object.defineProperty(component, 'state', { + value: { x: 1 }, + enumerable: false, + writable: true, + configurable: true, + }); + + const state = { + x: 2, + }; + + const key = Symbol(); + const state2: any = create( + state, + (draft) => { + return Object.assign(component, { + [key]: draft, + }) as any; + }, + { + enableAutoFreeze: true, + } + ); + + expect(Object.isFrozen(state2)).toBeTruthy(); + // Do not auto freeze non-enumerable or symbolic properties + expect(Object.isFrozen(state2.state)).toBeFalsy(); + expect(Object.isFrozen(state2[key])).toBeFalsy(); + + // @ts-expect-error + expect(state2.state).toBe(component.state); + expect(state2[key]).toBe(state); +}); + +test('works with interweaved Immer instances with disable Freeze', () => { + const base = {}; + const result = create(base, (s1) => { + const f = create( + { s1 }, + (s2) => { + s2.s1 = s2.s1; + }, + { + enableAutoFreeze: false, + } + ); + return f; + }); + // @ts-expect-error + expect(result.s1).toBe(base); +}); + +test('deep draft', () => { + const state = create({ a: { b: { c: 1 } } }, (draft) => { + draft.a.b.c; + return { + a: draft.a, + }; + }); + expect(state).toEqual({ a: { b: { c: 1 } } }); +}); + +test('case', () => { + const baseState = { foo: 'bar' }; + const state = create(baseState as { foo: string } | undefined, (draft) => { + return rawReturn(undefined); + }); + expect(state).toBe(undefined); +}); + +test('does not finalize upvalue drafts', () => { + create({ a: {}, b: {} }, (parent) => { + expect(create({}, () => parent)).toBe(parent); + // @ts-ignore + parent.x; // Ensure proxy not revoked. + + // @ts-ignore + expect(create({}, () => [parent])[0]).toBe(parent); + // @ts-ignore + parent.x; // Ensure proxy not revoked. + + expect(create({}, () => parent.a)).toBe(parent.a); + // @ts-ignore + + parent.a.x; // Ensure proxy not revoked. + // @ts-ignore + // Modified parent test + parent.c = 1; + // @ts-ignore + expect(create({}, () => [parent.b])[0]).toBe(parent.b); + // @ts-ignore + parent.b.x; // Ensure proxy not revoked. + }); +}); + +test('mixed draft', () => { + jest.spyOn(console, 'warn').mockImplementation(() => {}); + const baseState = { a: 1, b: { c: 1 } }; + const state = create(baseState, (draft) => { + if (draft.b.c === 1) { + return { + ...draft, + a: 2, + }; + } + }); + expect(state).toEqual({ a: 2, b: { c: 1 } }); + expect(isDraft(state.b)).toBeFalsy(); + expect(console.warn).toBeCalledTimes(0); +}); + +test('mixed draft with rawReturn()', () => { + jest.spyOn(console, 'warn').mockImplementation(() => {}); + const baseState = { a: 1, b: { c: 1 } }; + const state = create(baseState, (draft) => { + if (draft.b.c === 1) { + return rawReturn({ + ...draft, + a: 2, + }); + } + }); + expect(() => { + JSON.stringify(state); + }).toThrowError(); + expect(() => { + isDraft(state.b); + }).toThrowError(); + expect(console.warn).toBeCalledTimes(0); +}); + +test('mixed draft with rawReturn() and strict mode', () => { + jest.spyOn(console, 'warn').mockImplementation(() => {}); + const baseState = { a: 1, b: { c: 1 } }; + const state = create( + baseState, + (draft) => { + if (draft.b.c === 1) { + return rawReturn({ + ...draft, + a: 2, + }); + } + }, + { + strict: true, + } + ); + expect(state).toEqual({ a: 2, b: { c: 1 } }); + expect(isDraft(state.b)).toBeFalsy(); + expect(console.warn).toBeCalledTimes(1); + expect(console.warn).toBeCalledWith( + `The return value contains drafts, please don't use 'rawReturn()' to wrap the return value.` + ); +}); + +test('no mixed draft with strict mode', () => { + jest.spyOn(console, 'warn').mockImplementation(() => {}); + const baseState = { a: 1, b: { c: 1 } }; + const state = create( + baseState, + (draft) => { + if (draft.b.c === 1) { + return { + ...baseState, + a: 2, + }; + } + }, + { + strict: true, + } + ); + expect(state).toEqual({ a: 2, b: { c: 1 } }); + expect(isDraft(state.b)).toBeFalsy(); + expect(console.warn).toBeCalledTimes(1); + expect(console.warn).toBeCalledWith( + `The return value does not contain any draft, please use 'rawReturn()' to wrap the return value to improve performance.` + ); +}); diff --git a/test/safeReturn.test.ts b/test/safeReturn.test.ts deleted file mode 100644 index e6ef878e..00000000 --- a/test/safeReturn.test.ts +++ /dev/null @@ -1,375 +0,0 @@ -/* eslint-disable symbol-description */ -/* eslint-disable object-shorthand */ -/* eslint-disable @typescript-eslint/no-empty-function */ -/* eslint-disable no-unused-expressions */ -/* eslint-disable arrow-body-style */ -/* eslint-disable no-self-assign */ -/* eslint-disable no-param-reassign */ -/* eslint-disable @typescript-eslint/ban-ts-comment */ -import { create, safeReturn } from '../src'; - -test('base', () => { - const base = 3; - expect(create(base, () => 4)).toBe(4); - // @ts-expect-error - expect(create(base, () => null)).toBe(null); - expect(create(base, () => undefined)).toBe(3); - expect(create(base, () => {})).toBe(3); - expect(create(base, () => safeReturn(undefined))).toBe(undefined); - - expect(create({}, () => undefined)).toEqual({}); - expect(create({}, () => safeReturn(undefined))).toBe(undefined); - expect(create(3, () => safeReturn(undefined))).toBe(undefined); - - expect(create(() => undefined)({})).toEqual({}); - expect(create(() => safeReturn(undefined))({})).toBe(undefined); - expect(create(() => safeReturn(undefined))(3)).toBe(undefined); -}); - -test('base enableAutoFreeze: true', () => { - create( - { a: { b: 1 } }, - (draft) => { - return safeReturn({ - a: draft.a, - }); - }, - { enableAutoFreeze: true } - ); -}); - -test('base enableAutoFreeze: true - without safeReturn()', () => { - expect(() => { - create( - { a: { b: 1 } }, - (draft) => { - return { - a: draft.a, - }; - }, - { enableAutoFreeze: true } - ); - }).toThrowError(); -}); - -describe.each([{ useSafeReturn: true }, { useSafeReturn: false }])( - 'check Primitive type $useSafeReturn', - ({ useSafeReturn }) => { - test(`useSafeReturn ${useSafeReturn}: check Primitive type with returning`, () => { - [ - -1, - 1, - 0, - NaN, - BigInt(1), - Infinity, - '', - 'test', - null, - true, - false, - Symbol('foo'), - ].forEach((value: any) => { - expect( - create(value, (draft) => { - return useSafeReturn ? safeReturn(undefined) : ''; - }) - ).toBe(useSafeReturn ? undefined : ''); - }); - }); - - test(`useSafeReturn ${useSafeReturn}: check Primitive type with returning and patches`, () => { - [ - -1, - 1, - 0, - NaN, - BigInt(1), - Infinity, - '', - 'test', - null, - true, - false, - undefined, - Symbol('foo'), - ].forEach((value: any) => { - expect( - create( - value, - (draft) => { - return useSafeReturn ? safeReturn(undefined) : ''; - }, - { - enablePatches: true, - } - ) - ).toEqual([ - useSafeReturn ? undefined : '', - [{ op: 'replace', path: [], value: useSafeReturn ? undefined : '' }], - [{ op: 'replace', path: [], value: value }], - ]); - }); - }); - - test(`useSafeReturn ${useSafeReturn}: check Primitive type with returning, patches and freeze`, () => { - [ - -1, - 1, - 0, - NaN, - BigInt(1), - Infinity, - '', - 'test', - null, - true, - false, - undefined, - Symbol('foo'), - ].forEach((value: any) => { - expect( - create( - value, - (draft) => { - return useSafeReturn ? safeReturn(undefined) : ''; - }, - { - enableAutoFreeze: true, - enablePatches: true, - } - ) - ).toEqual([ - useSafeReturn ? undefined : '', - [{ op: 'replace', path: [], value: useSafeReturn ? undefined : '' }], - [{ op: 'replace', path: [], value: value }], - ]); - }); - }); - - test(`useSafeReturn ${useSafeReturn}: check Primitive type with returning, patches, freeze and async`, async () => { - for (const value of [ - -1, - 1, - 0, - NaN, - BigInt(1), - Infinity, - '', - 'test', - null, - true, - false, - undefined, - Symbol('foo'), - ]) { - await expect( - await create( - value, - async (draft) => { - return useSafeReturn ? safeReturn(undefined) : ''; - }, - { - enableAutoFreeze: true, - enablePatches: true, - } - ) - ).toEqual([ - useSafeReturn ? undefined : '', - [{ op: 'replace', path: [], value: useSafeReturn ? undefined : '' }], - [{ op: 'replace', path: [], value: value }], - ]); - } - }); - } -); - -test('error args', () => { - expect(() => - // @ts-expect-error - create(3, () => safeReturn(undefined, undefined)) - ).toThrowErrorMatchingInlineSnapshot( - `"safeReturn() must be called with one argument."` - ); - - expect(() => - // @ts-expect-error - create({}, () => safeReturn()) - ).toThrowErrorMatchingInlineSnapshot( - `"safeReturn() must be called with a value."` - ); - - const logSpy = jest.spyOn(console, 'warn').mockImplementation(() => {}); - - [ - -1, - 1, - 0, - NaN, - BigInt(1), - Infinity, - '', - 'test', - null, - false, - true, - Symbol('foo'), - ].forEach((value: any) => { - create({}, () => safeReturn(value)); - expect(logSpy).toHaveBeenCalledWith( - 'safeReturn() must be called with an object or undefined, other types do not need to be returned via safeReturn().' - ); - logSpy.mockClear(); - }); - - logSpy.mockReset(); -}); - -test('check warning in strict mode', () => { - class Foo { - a?: any; - } - const warnSpy = jest.spyOn(console, 'warn').mockImplementation(() => {}); - [ - (draft: any) => { - return { - a: draft.a, - }; - }, - (draft: any) => { - return [draft.a]; - }, - (draft: any) => { - return new Map([ - [ - 1, - { - a: draft.a, - }, - ], - ]); - }, - (draft: any) => { - return new Set([ - 1, - { - a: draft.a, - }, - ]); - }, - (draft: any) => { - const foo = new Foo(); - foo.a = draft.a; - return foo; - }, - ].forEach((callback: any) => { - create({ a: { b: 1 } }, callback, { - strict: true, - }); - expect(warnSpy).toHaveBeenCalledWith( - `The return value contains drafts, please use safeReturn() to wrap the return value.` - ); - warnSpy.mockClear(); - }); - warnSpy.mockReset(); -}); - -test('return parent draft', () => { - expect( - create({ a: 1 }, (draft) => { - const _draft = create({}, () => draft) as any; - _draft.a = 2; - return _draft; - }) - ).toEqual({ a: 2 }); -}); - -test('mix more type draft', () => { - [ - (draft: any) => - safeReturn({ - a: { - b: draft.a, - }, - }), - (draft: any) => safeReturn([{ c: draft.a }]), - (draft: any) => safeReturn(new Map([[1, draft.a]])), - (draft: any) => safeReturn(new Set([1, draft.a])), - ].forEach((callback: any) => { - expect(() => create({ a: { b: 1 } }, callback)).not.toThrowError(); - }); -}); - -test(`safe returning with non-enumerable or symbolic properties`, () => { - const component = {}; - Object.defineProperty(component, 'state', { - value: { x: 1 }, - enumerable: false, - writable: true, - configurable: true, - }); - - const state = { - x: 2, - }; - - const key = Symbol(); - const state2: any = create( - state, - (draft) => { - return safeReturn( - Object.assign(component, { - [key]: draft, - }) - ) as any; - }, - { - enableAutoFreeze: true, - } - ); - - expect(Object.isFrozen(state2)).toBeTruthy(); - // Do not auto freeze non-enumerable or symbolic properties - expect(Object.isFrozen(state2.state)).toBeFalsy(); - expect(Object.isFrozen(state2[key])).toBeFalsy(); - - // @ts-expect-error - expect(state2.state).toBe(component.state); - expect(state2[key]).toBe(state); -}); - -test('works with interweaved Immer instances with disable Freeze', () => { - const base = {}; - const result = create(base, (s1) => { - const f = create( - { s1 }, - (s2) => { - s2.s1 = s2.s1; - }, - { - enableAutoFreeze: false, - } - ); - return safeReturn(f); - }); - // @ts-expect-error - expect(result.s1).toBe(base); -}); - -test('deep draft', () => { - const state = create({ a: { b: { c: 1 } } }, (draft) => { - draft.a.b.c; - return safeReturn({ - a: draft.a, - }); - }); - expect(state).toEqual({ a: { b: { c: 1 } } }); -}); - -test('case', () => { - const baseState = { foo: 'bar' }; - const state = create(baseState as { foo: string } | undefined, (draft) => { - return safeReturn(undefined); - }); - expect(state).toBe(undefined); -}); diff --git a/yarn.lock b/yarn.lock index d97f99c2..b41b5672 100644 --- a/yarn.lock +++ b/yarn.lock @@ -6347,10 +6347,10 @@ jsbn@~0.1.0: resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-0.1.1.tgz#a5e654c2e5a2deb5f201d96cefbca80c0ef2f513" integrity sha512-UVU9dibq2JcFWxQPA6KCqj5O42VOmAY3zQUfEKxU0KpTGXwNoCjkX1e13eHNvw/xPynt6pU0rZ1htjWTNTSXsg== -jsdoc-tests@^0.1.1: - version "0.1.1" - resolved "https://registry.yarnpkg.com/jsdoc-tests/-/jsdoc-tests-0.1.1.tgz#c5f9c25b365a599c3830f5b21fb2b07146485612" - integrity sha512-l6IflN82QMMlLX9/RBcdYFWnmZl3FnXe2kPl8S4DldHrt6AlsEkD7iQnPEerEpbx0IQhjAKM52kiD6YUeOp5JA== +jsdoc-tests@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/jsdoc-tests/-/jsdoc-tests-1.1.0.tgz#e353b65f284df07d1ff15ca1a43be951bce1c5fd" + integrity sha512-kXWJm517vOlAHResJ813eyDpC32pGsfOfzlx+2ftJAAsiAp6eSBdiOBEU6Oa5PDALR+FmOc8WJ2kYRYgT35isw== dependencies: "@babel/parser" "^7.12.7" markdown-it "^12.0.2"