From 31de2ce7f6e5967b17b6269f7f07b225b737c3b3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Adam=20Zieli=C5=84ski?= Date: Wed, 18 May 2022 14:41:34 +0200 Subject: [PATCH] Squashed commit of the following: MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit commit c86f5616a89cb1998ead4ea1f34694795e019267 Author: Adam Zieliński Date: Wed Feb 16 12:17:33 2022 +0100 Move the return type to TS commit eefec13ba88c7aa63e950b71dc242de4b9a46b16 Author: Adam Zieliński Date: Wed Feb 16 12:16:36 2022 +0100 Adjust TS types commit 12c74338a1b63d8b796b50e11160bf89fc2f5c49 Author: Adam Zieliński Date: Wed Feb 16 12:13:22 2022 +0100 Make useResourcePermissions return a tuple and make hasResolved the first item of the tuple forces the users to think about that variable and it's not easy to accidentally miss it. commit caa48a72800e6b47934a8cef5356b32bb8361788 Author: Adam Zieliński Date: Tue Feb 15 17:03:31 2022 +0100 Fix typo in the tests commit 33cdb4e59fd3a81c743430fa6016903373aff666 Author: Adam Zieliński Date: Mon Feb 14 15:05:09 2022 +0100 Expose __experimentalUseResourcePermissions as a public API commit cf30c5aa85dae7d7a488573b5e6459abc05086d5 Author: Adam Zieliński Date: Mon Feb 14 14:58:42 2022 +0100 Distinguish between global and local resoluion commit 0db7bd25a7106400188addc23892d77bd33b63ed Author: Adam Zieliński Date: Mon Feb 14 14:42:36 2022 +0100 Propose useResourcePermissions hook commit 2d5e270d67781783c02d1d8ff1fa0d3a671e3aff Author: Adam Zieliński Date: Mon Feb 14 14:48:09 2022 +0100 Move the status computation inside the enriched selectors commit e7ac34e5e9fdb3051d9d093fad5d4ab73833ba0a Author: Adam Zieliński Date: Mon Feb 14 14:01:11 2022 +0100 Propose useEntityRecords --- .../hooks/test/use-resource-permissions.js | 117 +++++++++++++++++ .../src/hooks/use-resource-permissions.ts | 120 ++++++++++++++++++ packages/core-data/src/index.js | 3 + 3 files changed, 240 insertions(+) create mode 100644 packages/core-data/src/hooks/test/use-resource-permissions.js create mode 100644 packages/core-data/src/hooks/use-resource-permissions.ts diff --git a/packages/core-data/src/hooks/test/use-resource-permissions.js b/packages/core-data/src/hooks/test/use-resource-permissions.js new file mode 100644 index 00000000000000..d1d41db3ddf360 --- /dev/null +++ b/packages/core-data/src/hooks/test/use-resource-permissions.js @@ -0,0 +1,117 @@ +/** + * WordPress dependencies + */ +import triggerFetch from '@wordpress/api-fetch'; +import { createRegistry, RegistryProvider } from '@wordpress/data'; + +jest.mock( '@wordpress/api-fetch' ); + +/** + * External dependencies + */ +import { act, render } from '@testing-library/react'; + +/** + * Internal dependencies + */ +import { store as coreDataStore } from '../../index'; +import useResourcePermissions from '../use-resource-permissions'; + +describe( 'useResourcePermissions', () => { + let registry; + beforeEach( () => { + jest.useFakeTimers(); + + registry = createRegistry(); + registry.register( coreDataStore ); + } ); + + afterEach( () => { + jest.runOnlyPendingTimers(); + jest.useRealTimers(); + } ); + + it( 'retrieves the relevant permissions for a key-less resource', async () => { + triggerFetch.mockImplementation( () => ( { + headers: { + Allow: 'POST', + }, + } ) ); + let data; + const TestComponent = () => { + data = useResourcePermissions( 'widgets' ); + return
; + }; + render( + + + + ); + expect( data ).toEqual( [ + false, + { + status: 'IDLE', + isResolving: false, + canCreate: false, + }, + ] ); + + // Required to make sure no updates happen outside of act() + await act( async () => { + jest.advanceTimersByTime( 1 ); + } ); + + expect( data ).toEqual( [ + true, + { + status: 'SUCCESS', + isResolving: false, + canCreate: true, + }, + ] ); + } ); + + it( 'retrieves the relevant permissions for a resource with a key', async () => { + triggerFetch.mockImplementation( () => ( { + headers: { + Allow: 'POST', + }, + } ) ); + let data; + const TestComponent = () => { + data = useResourcePermissions( 'widgets', 1 ); + return
; + }; + render( + + + + ); + expect( data ).toEqual( [ + false, + { + status: 'IDLE', + isResolving: false, + canCreate: false, + canUpdate: false, + canDelete: false, + }, + ] ); + + // Required to make sure no updates happen outside of act() + await act( async () => { + jest.advanceTimersByTime( 1 ); + } ); + + expect( data ).toEqual( [ + true, + { + status: 'SUCCESS', + isResolving: false, + canCreate: true, + canUpdate: false, + canDelete: false, + }, + ] ); + } ); +} ); diff --git a/packages/core-data/src/hooks/use-resource-permissions.ts b/packages/core-data/src/hooks/use-resource-permissions.ts new file mode 100644 index 00000000000000..122eadc5759805 --- /dev/null +++ b/packages/core-data/src/hooks/use-resource-permissions.ts @@ -0,0 +1,120 @@ +/** + * Internal dependencies + */ +import { store as coreStore } from '../'; +import { Status } from './constants'; +import useQuerySelect from './use-query-select'; + +interface GlobalResourcePermissionsResolution { + /** Can the current user create new resources of this type? */ + canCreate: boolean; +} +interface SpecificResourcePermissionsResolution { + /** Can the current user update resources of this type? */ + canUpdate: boolean; + /** Can the current user delete resources of this type? */ + canDelete: boolean; +} +interface ResolutionDetails { + /** Resolution status */ + status: Status; + /** + * Is the data still being resolved? + */ + isResolving: boolean; +} + +/** + * Is the data resolved by now? + */ +type HasResolved = boolean; + +type ResourcePermissionsResolution< IdType > = [ + HasResolved, + ResolutionDetails & + GlobalResourcePermissionsResolution & + ( IdType extends void ? SpecificResourcePermissionsResolution : {} ) +]; + +/** + * Resolves resource permissions. + * + * @param resource The resource in question, e.g. media. + * @param id ID of a specific resource entry, if needed, e.g. 10. + * + * @example + * ```js + * import { useResourcePermissions } from '@wordpress/core-data'; + * + * function PagesList() { + * const { canCreate, isResolving } = useResourcePermissions( 'pages' ); + * + * if ( isResolving ) { + * return 'Loading ...'; + * } + * + * return ( + *
+ * {canCreate ? () : false} + * // ... + *
+ * ); + * } + * + * // Rendered in the application: + * // + * ``` + * + * In the above example, when `PagesList` is rendered into an + * application, the appropriate permissions and the resolution details will be retrieved from + * the store state using `canUser()`, or resolved if missing. + * + * @return Entity records data. + * @template IdType + */ +export default function __experimentalUseResourcePermissions< IdType = void >( + resource: string, + id: IdType +): ResourcePermissionsResolution< IdType > { + return useQuerySelect( + ( resolve ) => { + const { canUser } = resolve( coreStore ); + const create = canUser( 'create', resource ); + if ( ! id ) { + return [ + create.hasResolved, + { + status: create.status, + isResolving: create.isResolving, + canCreate: create.hasResolved && create.data, + }, + ]; + } + + const update = canUser( 'update', resource, id ); + const _delete = canUser( 'delete', resource, id ); + const isResolving = + create.isResolving || update.isResolving || _delete.isResolving; + const hasResolved = + create.hasResolved && update.hasResolved && _delete.hasResolved; + + let status = Status.Idle; + if ( isResolving ) { + status = Status.Resolving; + } else if ( hasResolved ) { + status = Status.Success; + } + return [ + hasResolved, + { + status, + isResolving, + canCreate: hasResolved && create.data, + canUpdate: hasResolved && update.data, + canDelete: hasResolved && _delete.data, + }, + ]; + }, + [ resource, id ] + ); +} diff --git a/packages/core-data/src/index.js b/packages/core-data/src/index.js index 05cace07c992b4..6914f376d72e43 100644 --- a/packages/core-data/src/index.js +++ b/packages/core-data/src/index.js @@ -68,6 +68,9 @@ export const store = createReduxStore( STORE_NAME, storeConfig() ); register( store ); export { default as EntityProvider } from './entity-provider'; +export { default as useEntityRecord } from './hooks/use-entity-record'; +export { default as useEntityRecords } from './hooks/use-entity-records'; +export { default as useResourcePermissions } from './hooks/use-resource-permissions'; export * from './entity-provider'; export * from './entity-types'; export * from './fetch';