From e5f469edab7e0a9f4c0cd3d7d2e285f6e3be69d8 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?F=C3=A1bio=20Henriques?= Date: Wed, 17 Apr 2024 17:30:22 +0100 Subject: [PATCH 1/7] Change `useOnyx` to return `undefined` instead of `null` --- lib/Onyx.ts | 5 ++-- lib/types.ts | 56 ++++++++++++++++++++++++--------------- lib/useOnyx.ts | 24 ++++++++++++----- tests/unit/useOnyxTest.ts | 10 +++---- 4 files changed, 59 insertions(+), 36 deletions(-) diff --git a/lib/Onyx.ts b/lib/Onyx.ts index ee039a918..77686b385 100644 --- a/lib/Onyx.ts +++ b/lib/Onyx.ts @@ -14,6 +14,7 @@ import type { InitOptions, KeyValueMapping, Mapping, + NonUndefined, NullableKeyValueMapping, NullishDeep, OnyxCollection, @@ -207,7 +208,7 @@ function disconnect(connectionID: number, keyToRemoveFromEvictionBlocklist?: Ony * @param key ONYXKEY to set * @param value value to store */ -function set(key: TKey, value: OnyxEntry): Promise { +function set(key: TKey, value: NonUndefined>): Promise { // If the value is null, we remove the key from storage const {value: valueAfterRemoving, wasRemoved} = OnyxUtils.removeNullValues(key, value); const valueWithoutNullValues = valueAfterRemoving as OnyxValue; @@ -280,7 +281,7 @@ function multiSet(data: Partial): Promise { * Onyx.merge(ONYXKEYS.POLICY, {id: 1}); // -> {id: 1} * Onyx.merge(ONYXKEYS.POLICY, {name: 'My Workspace'}); // -> {id: 1, name: 'My Workspace'} */ -function merge(key: TKey, changes: OnyxEntry>): Promise { +function merge(key: TKey, changes: NonUndefined>>): Promise { const mergeQueue = OnyxUtils.getMergeQueue(); const mergeQueuePromise = OnyxUtils.getMergeQueuePromise(); diff --git a/lib/types.ts b/lib/types.ts index 62a4ec1c5..72c6afbb4 100644 --- a/lib/types.ts +++ b/lib/types.ts @@ -3,6 +3,16 @@ import type {Merge} from 'type-fest'; import type {BuiltIns} from 'type-fest/source/internal'; import type OnyxUtils from './OnyxUtils'; +/** + * Utility type that excludes `null` from the type `TValue`. + */ +type NonNull = TValue extends null ? never : TValue; + +/** + * Utility type that excludes `undefined` from the type `TValue`. + */ +type NonUndefined = TValue extends undefined ? never : TValue; + /** * Represents a deeply nested record. It maps keys to values, * and those values can either be of type `TValue` or further nested `DeepRecord` instances. @@ -141,7 +151,7 @@ type NullableKeyValueMapping = { type Selector = (value: OnyxEntry, state: WithOnyxInstanceState) => TReturnType; /** - * Represents a single Onyx entry, that can be either `TOnyxValue` or `null` if it doesn't exist. + * Represents a single Onyx entry, that can be either `TOnyxValue` or `null` / `undefined` if it doesn't exist. * * It can be used to specify data retrieved from Onyx e.g. `withOnyx` HOC mappings. * @@ -168,10 +178,10 @@ type Selector = (value: OnyxEntry * })(Component); * ``` */ -type OnyxEntry = TOnyxValue | null; +type OnyxEntry = TOnyxValue | null | undefined; /** - * Represents an Onyx collection of entries, that can be either a record of `TOnyxValue`s or `null` if it is empty or doesn't exist. + * Represents an Onyx collection of entries, that can be either a record of `TOnyxValue`s or `null` / `undefined` if it is empty or doesn't exist. * * It can be used to specify collection data retrieved from Onyx e.g. `withOnyx` HOC mappings. * @@ -198,7 +208,7 @@ type OnyxEntry = TOnyxValue | null; * })(Component); * ``` */ -type OnyxCollection = OnyxEntry>; +type OnyxCollection = OnyxEntry>; /** Utility type to extract `TOnyxValue` from `OnyxCollection` */ type ExtractOnyxCollectionValue = TOnyxCollection extends NonNullable> ? U : never; @@ -284,9 +294,9 @@ type WithOnyxConnectOptions = { canEvict?: boolean; }; -type DefaultConnectCallback = (value: OnyxEntry, key: TKey) => void; +type DefaultConnectCallback = (value: NonUndefined>, key: TKey) => void; -type CollectionConnectCallback = (value: OnyxCollection) => void; +type CollectionConnectCallback = (value: NonUndefined>) => void; /** Represents the callback function used in `Onyx.connect()` method with a regular key. */ type DefaultConnectOptions = { @@ -331,12 +341,12 @@ type OnyxUpdate = | { onyxMethod: typeof OnyxUtils.METHOD.SET; key: TKey; - value: OnyxEntry; + value: NonUndefined>; } | { onyxMethod: typeof OnyxUtils.METHOD.MERGE; key: TKey; - value: OnyxEntry>; + value: NonUndefined>>; } | { onyxMethod: typeof OnyxUtils.METHOD.MULTI_SET; @@ -391,31 +401,33 @@ type InitOptions = { }; export type { + BaseConnectOptions, + Collection, + CollectionConnectCallback, + CollectionConnectOptions, CollectionKey, CollectionKeyBase, + ConnectOptions, CustomTypeOptions, DeepRecord, + DefaultConnectCallback, + DefaultConnectOptions, + ExtractOnyxCollectionValue, + InitOptions, Key, KeyValueMapping, + Mapping, + NonNull, + NonUndefined, NullableKeyValueMapping, + NullishDeep, OnyxCollection, OnyxEntry, OnyxKey, + OnyxUpdate, OnyxValue, Selector, - NullishDeep, - WithOnyxInstanceState, - ExtractOnyxCollectionValue, - Collection, - WithOnyxInstance, - BaseConnectOptions, WithOnyxConnectOptions, - DefaultConnectCallback, - CollectionConnectCallback, - DefaultConnectOptions, - CollectionConnectOptions, - ConnectOptions, - Mapping, - OnyxUpdate, - InitOptions, + WithOnyxInstance, + WithOnyxInstanceState, }; diff --git a/lib/useOnyx.ts b/lib/useOnyx.ts index 99de3bfc1..b41e1739a 100644 --- a/lib/useOnyx.ts +++ b/lib/useOnyx.ts @@ -2,11 +2,21 @@ import {deepEqual} from 'fast-equals'; import {useCallback, useEffect, useRef, useSyncExternalStore} from 'react'; import type {IsEqual} from 'type-fest'; import OnyxUtils from './OnyxUtils'; -import type {CollectionKeyBase, OnyxCollection, OnyxKey, OnyxValue, Selector} from './types'; +import type {CollectionKeyBase, KeyValueMapping, NonNull, OnyxCollection, OnyxEntry, OnyxKey, Selector} from './types'; import useLiveRef from './useLiveRef'; import usePrevious from './usePrevious'; import Onyx from './Onyx'; +/** + * Represents a Onyx value that can be either a single entry or a collection of entries, depending on the `TKey` provided. + * It's a variation of `OnyxValue` type that excludes the `null` type. + */ +type UseOnyxValue = string extends TKey + ? unknown + : TKey extends CollectionKeyBase + ? NonNull> + : NonNull>; + type UseOnyxOptions = { /** * Determines if this key in this subscription is safe to be evicted. @@ -39,7 +49,7 @@ type UseOnyxOptions = { type FetchStatus = 'loading' | 'loaded'; -type CachedValue = IsEqual> extends true ? TValue : TKey extends CollectionKeyBase ? NonNullable> : TValue; +type CachedValue = IsEqual> extends true ? TValue : TKey extends CollectionKeyBase ? NonNullable> : TValue; type ResultMetadata = { status: FetchStatus; @@ -51,7 +61,7 @@ function getCachedValue(key: TKey, selector?: Sele return OnyxUtils.tryGetCachedValue(key, {selector}) as CachedValue | undefined; } -function useOnyx>(key: TKey, options?: UseOnyxOptions): UseOnyxResult { +function useOnyx>(key: TKey, options?: UseOnyxOptions): UseOnyxResult { const connectionIDRef = useRef(null); const previousKey = usePrevious(key); @@ -63,10 +73,10 @@ function useOnyx>(key: TKey const cachedValueRef = useRef | undefined>(undefined); // Stores the previously result returned by the hook, containing the data from cache and the fetch status. - // We initialize it to `null` and `loading` fetch status to simulate the initial result when the hook is loading from the cache. + // We initialize it to `undefined` and `loading` fetch status to simulate the initial result when the hook is loading from the cache. // However, if `initWithStoredValues` is `true` we set the fetch status to `loaded` since we want to signal that data is ready. const resultRef = useRef>([ - null as CachedValue, + undefined as CachedValue, { status: options?.initWithStoredValues === false ? 'loaded' : 'loading', }, @@ -131,8 +141,8 @@ function useOnyx>(key: TKey if (!deepEqual(cachedValueRef.current, newValue)) { cachedValueRef.current = newValue; - // If the new value is `undefined` we default it to `null` to ensure the consumer get a consistent result from the hook. - resultRef.current = [(cachedValueRef.current ?? null) as CachedValue, {status: newFetchStatus ?? 'loaded'}]; + // If the new value is `null` we default it to `undefined` to ensure the consumer get a consistent result from the hook. + resultRef.current = [(cachedValueRef.current ?? undefined) as CachedValue, {status: newFetchStatus ?? 'loaded'}]; } return resultRef.current; diff --git a/tests/unit/useOnyxTest.ts b/tests/unit/useOnyxTest.ts index 890a4b7ad..c1a1e2cad 100644 --- a/tests/unit/useOnyxTest.ts +++ b/tests/unit/useOnyxTest.ts @@ -102,7 +102,7 @@ describe('useOnyx', () => { const {result} = renderHook(() => useOnyx(ONYXKEYS.TEST_KEY)); - expect(result.current[0]).toEqual(null); + expect(result.current[0]).toEqual(undefined); expect(result.current[1].status).toEqual('loading'); await act(async () => waitForPromisesToResolve()); @@ -118,7 +118,7 @@ describe('useOnyx', () => { const {result} = renderHook(() => useOnyx(ONYXKEYS.TEST_KEY)); - expect(result.current[0]).toEqual(null); + expect(result.current[0]).toEqual(undefined); expect(result.current[1].status).toEqual('loading'); await act(async () => waitForPromisesToResolve()); @@ -257,7 +257,7 @@ describe('useOnyx', () => { await act(async () => waitForPromisesToResolve()); - expect(result.current[0]).toEqual(null); + expect(result.current[0]).toEqual(undefined); expect(result.current[1].status).toEqual('loaded'); }); @@ -290,7 +290,7 @@ describe('useOnyx', () => { const {result} = renderHook(() => useOnyx(ONYXKEYS.TEST_KEY)); - expect(result.current[0]).toEqual(null); + expect(result.current[0]).toEqual(undefined); expect(result.current[1].status).toEqual('loading'); await act(async () => waitForPromisesToResolve()); @@ -348,7 +348,7 @@ describe('useOnyx', () => { await act(async () => waitForPromisesToResolve()); - expect(result.current[0]).toEqual(null); + expect(result.current[0]).toEqual(undefined); expect(result.current[1].status).toEqual('loaded'); await act(async () => Onyx.merge(ONYXKEYS.TEST_KEY, 'test2')); From b67cb13a52fe56fd45ecc754d580d8d2d9f614a7 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?F=C3=A1bio=20Henriques?= Date: Wed, 17 Apr 2024 17:34:31 +0100 Subject: [PATCH 2/7] Deprecate `withOnyx` --- lib/withOnyx.d.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/lib/withOnyx.d.ts b/lib/withOnyx.d.ts index 8ec7004cf..bc0cd6881 100644 --- a/lib/withOnyx.d.ts +++ b/lib/withOnyx.d.ts @@ -125,6 +125,8 @@ type OnyxPropCollectionMapping Date: Wed, 17 Apr 2024 18:36:09 +0100 Subject: [PATCH 3/7] Make `useOnyx` return read-only value --- lib/useOnyx.ts | 12 ++++++++---- 1 file changed, 8 insertions(+), 4 deletions(-) diff --git a/lib/useOnyx.ts b/lib/useOnyx.ts index b41e1739a..24317a0c8 100644 --- a/lib/useOnyx.ts +++ b/lib/useOnyx.ts @@ -9,13 +9,13 @@ import Onyx from './Onyx'; /** * Represents a Onyx value that can be either a single entry or a collection of entries, depending on the `TKey` provided. - * It's a variation of `OnyxValue` type that excludes the `null` type. + * It's a variation of `OnyxValue` type that is read-only and excludes the `null` type. */ type UseOnyxValue = string extends TKey ? unknown : TKey extends CollectionKeyBase - ? NonNull> - : NonNull>; + ? Readonly>> + : Readonly>>; type UseOnyxOptions = { /** @@ -49,7 +49,11 @@ type UseOnyxOptions = { type FetchStatus = 'loading' | 'loaded'; -type CachedValue = IsEqual> extends true ? TValue : TKey extends CollectionKeyBase ? NonNullable> : TValue; +type CachedValue = IsEqual> extends true + ? TValue + : TKey extends CollectionKeyBase + ? Readonly>> + : Readonly; type ResultMetadata = { status: FetchStatus; From 9e36847fca1678bd79e685b307e99d9433abbf15 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?F=C3=A1bio=20Henriques?= Date: Wed, 17 Apr 2024 19:12:31 +0100 Subject: [PATCH 4/7] Fix prettier --- lib/withOnyx.d.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/withOnyx.d.ts b/lib/withOnyx.d.ts index bc0cd6881..f665cea9f 100644 --- a/lib/withOnyx.d.ts +++ b/lib/withOnyx.d.ts @@ -126,7 +126,7 @@ type OnyxPropCollectionMapping Date: Fri, 19 Apr 2024 08:07:18 +0100 Subject: [PATCH 5/7] Bump TS to 5.4.5 to be able to use NoInfer type --- package-lock.json | 14 +++++++------- package.json | 2 +- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/package-lock.json b/package-lock.json index f1a2b0fa6..9547e0ed8 100644 --- a/package-lock.json +++ b/package-lock.json @@ -52,7 +52,7 @@ "react-test-renderer": "18.1.0", "ts-node": "^10.9.2", "type-fest": "^3.12.0", - "typescript": "^5.3.3" + "typescript": "^5.4.5" }, "engines": { "node": ">=20.10.0", @@ -17132,9 +17132,9 @@ } }, "node_modules/typescript": { - "version": "5.3.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.3.3.tgz", - "integrity": "sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==", + "version": "5.4.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.4.5.tgz", + "integrity": "sha512-vcI4UpRgg81oIRUFwR0WSIHKt11nJ7SAVlYNIu+QpqeyXP+gpQJy/Z4+F0aGxSE4MqwjyXvW/TzgkLAx2AGHwQ==", "dev": true, "bin": { "tsc": "bin/tsc", @@ -31163,9 +31163,9 @@ } }, "typescript": { - "version": "5.3.3", - "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.3.3.tgz", - "integrity": "sha512-pXWcraxM0uxAS+tN0AG/BF2TyqmHO014Z070UsJ+pFvYuRSq8KH8DmWpnbXe0pEPDHXZV3FcAbJkijJ5oNEnWw==", + "version": "5.4.5", + "resolved": "https://registry.npmjs.org/typescript/-/typescript-5.4.5.tgz", + "integrity": "sha512-vcI4UpRgg81oIRUFwR0WSIHKt11nJ7SAVlYNIu+QpqeyXP+gpQJy/Z4+F0aGxSE4MqwjyXvW/TzgkLAx2AGHwQ==", "dev": true }, "typical": { diff --git a/package.json b/package.json index fa2527deb..0c3ec48b1 100644 --- a/package.json +++ b/package.json @@ -82,7 +82,7 @@ "react-test-renderer": "18.1.0", "ts-node": "^10.9.2", "type-fest": "^3.12.0", - "typescript": "^5.3.3" + "typescript": "^5.4.5" }, "peerDependencies": { "idb-keyval": "^6.2.1", From 34c9440e1290264cdc13457d93b0f2d8c2077954 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?F=C3=A1bio=20Henriques?= Date: Fri, 19 Apr 2024 08:07:30 +0100 Subject: [PATCH 6/7] Fix initialValue typings --- lib/useOnyx.ts | 18 ++++++++++++++++-- tests/unit/useOnyxTest.ts | 20 ++++++++++++++++++++ 2 files changed, 36 insertions(+), 2 deletions(-) diff --git a/lib/useOnyx.ts b/lib/useOnyx.ts index 24317a0c8..aa51af64d 100644 --- a/lib/useOnyx.ts +++ b/lib/useOnyx.ts @@ -17,7 +17,7 @@ type UseOnyxValue = string extends TKey ? Readonly>> : Readonly>>; -type UseOnyxOptions = { +type BaseUseOnyxOptions = { /** * Determines if this key in this subscription is safe to be evicted. */ @@ -32,12 +32,16 @@ type UseOnyxOptions = { * If set to true, data will be retrieved from cache during the first render even if there is a pending merge for the key. */ allowStaleData?: boolean; +}; +type UseOnyxInitialValueOption = { /** * This value will be returned by the hook on the first render while the data is being read from Onyx. */ - initialValue?: TReturnValue; + initialValue?: TInitialValue; +}; +type UseOnyxSelectorOption = { /** * This will be used to subscribe to a subset of an Onyx key's data. * Using this setting on `useOnyx` can have very positive performance benefits because the component will only re-render @@ -47,6 +51,8 @@ type UseOnyxOptions = { selector?: Selector; }; +type UseOnyxOptions = BaseUseOnyxOptions & UseOnyxInitialValueOption & UseOnyxSelectorOption; + type FetchStatus = 'loading' | 'loaded'; type CachedValue = IsEqual> extends true @@ -65,6 +71,14 @@ function getCachedValue(key: TKey, selector?: Sele return OnyxUtils.tryGetCachedValue(key, {selector}) as CachedValue | undefined; } +function useOnyx>( + key: TKey, + options?: BaseUseOnyxOptions & UseOnyxInitialValueOption & Required>, +): UseOnyxResult; +function useOnyx>( + key: TKey, + options?: BaseUseOnyxOptions & UseOnyxInitialValueOption>, +): UseOnyxResult; function useOnyx>(key: TKey, options?: UseOnyxOptions): UseOnyxResult { const connectionIDRef = useRef(null); const previousKey = usePrevious(key); diff --git a/tests/unit/useOnyxTest.ts b/tests/unit/useOnyxTest.ts index c1a1e2cad..82f3488ad 100644 --- a/tests/unit/useOnyxTest.ts +++ b/tests/unit/useOnyxTest.ts @@ -242,6 +242,26 @@ describe('useOnyx', () => { expect(result.current[0]).toEqual('id - changed_id, name - changed_name - selector changed'); expect(result.current[1].status).toEqual('loaded'); }); + + it('should return initial value if selected data is undefined', async () => { + Onyx.set(ONYXKEYS.TEST_KEY, 'test_id_1'); + + const {result} = renderHook(() => + useOnyx(ONYXKEYS.TEST_KEY, { + // @ts-expect-error bypass + selector: (_entry: OnyxEntry) => undefined, + initialValue: 'initial value', + }), + ); + + expect(result.current[0]).toEqual('initial value'); + expect(result.current[1].status).toEqual('loaded'); + + await act(async () => waitForPromisesToResolve()); + + expect(result.current[0]).toEqual(undefined); + expect(result.current[1].status).toEqual('loaded'); + }); }); describe('initialValue', () => { From 96f8483029eeea94b6e8fec12beff0e489b9561c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?F=C3=A1bio=20Henriques?= Date: Fri, 19 Apr 2024 09:23:54 +0100 Subject: [PATCH 7/7] Use toBeUndefined() on tests --- tests/unit/useOnyxTest.ts | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tests/unit/useOnyxTest.ts b/tests/unit/useOnyxTest.ts index 82f3488ad..04160c514 100644 --- a/tests/unit/useOnyxTest.ts +++ b/tests/unit/useOnyxTest.ts @@ -102,7 +102,7 @@ describe('useOnyx', () => { const {result} = renderHook(() => useOnyx(ONYXKEYS.TEST_KEY)); - expect(result.current[0]).toEqual(undefined); + expect(result.current[0]).toBeUndefined(); expect(result.current[1].status).toEqual('loading'); await act(async () => waitForPromisesToResolve()); @@ -118,7 +118,7 @@ describe('useOnyx', () => { const {result} = renderHook(() => useOnyx(ONYXKEYS.TEST_KEY)); - expect(result.current[0]).toEqual(undefined); + expect(result.current[0]).toBeUndefined(); expect(result.current[1].status).toEqual('loading'); await act(async () => waitForPromisesToResolve()); @@ -259,7 +259,7 @@ describe('useOnyx', () => { await act(async () => waitForPromisesToResolve()); - expect(result.current[0]).toEqual(undefined); + expect(result.current[0]).toBeUndefined(); expect(result.current[1].status).toEqual('loaded'); }); }); @@ -277,7 +277,7 @@ describe('useOnyx', () => { await act(async () => waitForPromisesToResolve()); - expect(result.current[0]).toEqual(undefined); + expect(result.current[0]).toBeUndefined(); expect(result.current[1].status).toEqual('loaded'); }); @@ -310,7 +310,7 @@ describe('useOnyx', () => { const {result} = renderHook(() => useOnyx(ONYXKEYS.TEST_KEY)); - expect(result.current[0]).toEqual(undefined); + expect(result.current[0]).toBeUndefined(); expect(result.current[1].status).toEqual('loading'); await act(async () => waitForPromisesToResolve()); @@ -368,7 +368,7 @@ describe('useOnyx', () => { await act(async () => waitForPromisesToResolve()); - expect(result.current[0]).toEqual(undefined); + expect(result.current[0]).toBeUndefined(); expect(result.current[1].status).toEqual('loaded'); await act(async () => Onyx.merge(ONYXKEYS.TEST_KEY, 'test2'));