UNPKG

@wordpress/core-data

Version:
1,108 lines (998 loc) 25.7 kB
/** * WordPress dependencies */ import apiFetch from '@wordpress/api-fetch'; jest.mock( '@wordpress/api-fetch' ); /** * Internal dependencies */ import { editEntityRecord, clearEntityRecordEdits, saveEntityRecord, saveEditedEntityRecord, deleteEntityRecord, receiveUserPermission, receiveAutosaves, receiveCurrentUser, __experimentalBatch, } from '../actions'; import { getSyncManager } from '../sync'; jest.mock( '../batch', () => { const { createBatch } = jest.requireActual( '../batch' ); return { createBatch() { return createBatch( ( inputs ) => Promise.resolve( inputs ) ); }, }; } ); jest.mock( '../sync', () => ( { getSyncManager: jest.fn(), LOCAL_EDITOR_ORIGIN: 'local-editor', } ) ); describe( 'editEntityRecord', () => { it( 'throws when the edited entity does not have a loaded config.', async () => { const entityConfig = { kind: 'someKind', name: 'someName', id: 'someId', }; const select = { getEntityConfig: jest.fn(), }; const fulfillment = async () => editEntityRecord( entityConfig.kind, entityConfig.name, entityConfig.id, {} )( { select } ); await expect( fulfillment ).rejects.toThrow( `The entity being edited (${ entityConfig.kind }, ${ entityConfig.name }) does not have a loaded config.` ); expect( select.getEntityConfig ).toHaveBeenCalledTimes( 1 ); } ); it( 'dispatches the correct action for non-merged edits', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: {}, } ), getRawEntityRecord: () => ( { id: 1, title: 'Original Title', content: 'Original Content', } ), getEditedEntityRecord: () => ( { id: 1, title: 'Original Title', content: 'Original Content', } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; editEntityRecord( 'postType', 'post', 1, { title: 'New Title' } )( { select, dispatch, } ); expect( dispatch ).toHaveBeenCalledWith( { type: 'EDIT_ENTITY_RECORD', kind: 'postType', name: 'post', recordId: 1, edits: { title: 'New Title' }, } ); } ); it( 'merges edits for fields defined in mergedEdits config', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: { meta: true }, } ), getRawEntityRecord: () => ( { id: 1, meta: { existingKey: 'existingValue' }, } ), getEditedEntityRecord: () => ( { id: 1, meta: { existingKey: 'existingValue', editedKey: 'editedValue', }, } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; editEntityRecord( 'postType', 'post', 1, { meta: { newKey: 'newValue' }, } )( { select, dispatch, } ); expect( dispatch ).toHaveBeenCalledWith( { type: 'EDIT_ENTITY_RECORD', kind: 'postType', name: 'post', recordId: 1, edits: { meta: { existingKey: 'existingValue', editedKey: 'editedValue', newKey: 'newValue', }, }, } ); } ); it( 'handles both merged and non-merged edits together', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: { meta: true }, } ), getRawEntityRecord: () => ( { id: 1, title: 'Original Title', meta: { existingKey: 'existingValue' }, } ), getEditedEntityRecord: () => ( { id: 1, title: 'Original Title', meta: { existingKey: 'existingValue' }, } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; editEntityRecord( 'postType', 'post', 1, { title: 'New Title', meta: { newKey: 'newValue' }, } )( { select, dispatch } ); expect( dispatch ).toHaveBeenCalledWith( { type: 'EDIT_ENTITY_RECORD', kind: 'postType', name: 'post', recordId: 1, edits: { title: 'New Title', meta: { existingKey: 'existingValue', newKey: 'newValue', }, }, } ); } ); it( 'clears edit when merged value equals persisted record', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: { meta: true }, } ), getRawEntityRecord: () => ( { id: 1, meta: { key1: 'value1', key2: 'value2' }, } ), getEditedEntityRecord: () => ( { id: 1, meta: { key1: 'value1' }, } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; // Editing meta to add key2 back should result in a value equal to the persisted record editEntityRecord( 'postType', 'post', 1, { meta: { key2: 'value2' }, } )( { select, dispatch } ); expect( dispatch ).toHaveBeenCalledWith( { type: 'EDIT_ENTITY_RECORD', kind: 'postType', name: 'post', recordId: 1, edits: { // meta should be undefined because merged value equals persisted record meta: undefined, }, } ); } ); it( 'clears non-merged edit when value equals persisted record', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: {}, } ), getRawEntityRecord: () => ( { id: 1, title: 'Original Title', } ), getEditedEntityRecord: () => ( { id: 1, title: 'Edited Title', } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; // Editing title back to original should clear the edit editEntityRecord( 'postType', 'post', 1, { title: 'Original Title', } )( { select, dispatch } ); expect( dispatch ).toHaveBeenCalledWith( { type: 'EDIT_ENTITY_RECORD', kind: 'postType', name: 'post', recordId: 1, edits: { title: undefined, }, } ); } ); describe( 'with SyncManager', () => { let syncManager; beforeEach( () => { // Create a mock sync manager syncManager = { update: jest.fn(), }; getSyncManager.mockReturnValue( syncManager ); } ); afterEach( () => { getSyncManager.mockReset(); } ); it( 'passes merged edits to SyncManager#update for merged fields', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: { meta: true }, syncConfig: {}, } ), getRawEntityRecord: () => ( { id: 1, meta: { existingKey: 'existingValue' }, } ), getEditedEntityRecord: () => ( { id: 1, meta: { existingKey: 'existingValue', editedKey: 'editedValue', }, } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; editEntityRecord( 'postType', 'post', 1, { meta: { newKey: 'newValue' }, } )( { select, dispatch, } ); // Verify SyncManager#update was called with merged edits expect( syncManager.update ).toHaveBeenCalledWith( 'postType/post', 1, { meta: { existingKey: 'existingValue', editedKey: 'editedValue', newKey: 'newValue', }, }, 'local-editor', { isNewUndoLevel: true } ); } ); it( 'passes merged edits to SyncManager#update even when value equals persisted record', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: { meta: true }, syncConfig: {}, } ), getRawEntityRecord: () => ( { id: 1, meta: { key1: 'value1', key2: 'value2' }, } ), getEditedEntityRecord: () => ( { id: 1, meta: { key1: 'value1' }, } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; // Editing meta to add key2 back results in a value equal to the persisted record editEntityRecord( 'postType', 'post', 1, { meta: { key2: 'value2' }, } )( { select, dispatch } ); // Verify SyncManager#update was called with merged edits (not cleaned/undefined) expect( syncManager.update ).toHaveBeenCalledWith( 'postType/post', 1, { meta: { key1: 'value1', key2: 'value2', }, }, 'local-editor', { isNewUndoLevel: true } ); // But the local store dispatch should still receive undefined for the cleaned edit expect( dispatch ).toHaveBeenCalledWith( { type: 'EDIT_ENTITY_RECORD', kind: 'postType', name: 'post', recordId: 1, edits: { meta: undefined, }, } ); } ); it( 'passes merged and non-merged edits correctly to SyncManager#update', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: { meta: true }, syncConfig: {}, } ), getRawEntityRecord: () => ( { id: 1, title: 'Original Title', meta: { existingKey: 'existingValue' }, } ), getEditedEntityRecord: () => ( { id: 1, title: 'Original Title', meta: { existingKey: 'existingValue' }, } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; editEntityRecord( 'postType', 'post', 1, { title: 'New Title', meta: { newKey: 'newValue' }, } )( { select, dispatch } ); // Verify SyncManager#update was called with merged meta but non-merged title expect( syncManager.update ).toHaveBeenCalledWith( 'postType/post', 1, { title: 'New Title', meta: { existingKey: 'existingValue', newKey: 'newValue', }, }, 'local-editor', { isNewUndoLevel: true } ); } ); it( 'does not call SyncManager#update when syncConfig is not defined', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', mergedEdits: { meta: true }, // No syncConfig } ), getRawEntityRecord: () => ( { id: 1, meta: { existingKey: 'existingValue' }, } ), getEditedEntityRecord: () => ( { id: 1, meta: { existingKey: 'existingValue' }, } ), getUndoManager: () => ( { addRecord: jest.fn(), } ), }; editEntityRecord( 'postType', 'post', 1, { meta: { newKey: 'newValue' }, } )( { select, dispatch, } ); // Verify SyncManager#update was NOT called expect( syncManager.update ).not.toHaveBeenCalled(); } ); } ); } ); describe( 'clearEntityRecordEdits', () => { it( 'throws when the entity does not have a loaded config.', async () => { const select = { getEntityConfig: jest.fn(), }; const fulfillment = async () => clearEntityRecordEdits( 'someKind', 'someName', 'someId' )( { select } ); await expect( fulfillment ).rejects.toThrow( `The entity being edited (someKind, someName) does not have a loaded config.` ); } ); it( 'does nothing when there are no edits', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', } ), getEntityRecordEdits: () => undefined, }; clearEntityRecordEdits( 'postType', 'post', 1 )( { select, dispatch, } ); expect( dispatch ).not.toHaveBeenCalled(); } ); it( 'clears all edits for an entity record', () => { const dispatch = jest.fn(); const select = { getEntityConfig: () => ( { kind: 'postType', name: 'post', } ), getEntityRecordEdits: () => ( { title: 'New Title', content: 'New Content', } ), getEditedEntityRecord: () => ( { id: 1, title: 'New Title', content: 'New Content', } ), }; clearEntityRecordEdits( 'postType', 'post', 1 )( { select, dispatch, } ); expect( dispatch ).toHaveBeenCalledWith( { type: 'EDIT_ENTITY_RECORD', kind: 'postType', name: 'post', recordId: 1, edits: { title: undefined, content: undefined, }, } ); } ); } ); describe( 'deleteEntityRecord', () => { beforeEach( async () => { apiFetch.mockReset(); } ); it( 'triggers a DELETE request for an existing record', async () => { const deletedRecord = { title: 'new post', id: 10 }; const configs = [ { name: 'post', kind: 'postType', baseURL: '/wp/v2/posts' }, ]; const dispatch = Object.assign( jest.fn(), { receiveEntityRecords: jest.fn(), __unstableAcquireStoreLock: jest.fn(), __unstableReleaseStoreLock: jest.fn(), } ); const resolveSelect = { getEntitiesConfig: jest.fn( () => configs ) }; // Provide response apiFetch.mockImplementation( () => deletedRecord ); const result = await deleteEntityRecord( 'postType', 'post', deletedRecord.id )( { dispatch, resolveSelect } ); expect( apiFetch ).toHaveBeenCalledTimes( 1 ); expect( apiFetch ).toHaveBeenCalledWith( { path: '/wp/v2/posts/10', method: 'DELETE', } ); expect( dispatch ).toHaveBeenCalledTimes( 3 ); expect( dispatch ).toHaveBeenCalledWith( { type: 'DELETE_ENTITY_RECORD_START', kind: 'postType', name: 'post', recordId: 10, } ); expect( dispatch ).toHaveBeenCalledWith( { type: 'DELETE_ENTITY_RECORD_FINISH', kind: 'postType', name: 'post', recordId: 10, error: undefined, } ); expect( dispatch.__unstableAcquireStoreLock ).toHaveBeenCalledTimes( 1 ); expect( dispatch.__unstableReleaseStoreLock ).toHaveBeenCalledTimes( 1 ); expect( result ).toBe( deletedRecord ); } ); it( 'throws on error when throwOnError is true', async () => { const entities = [ { name: 'post', kind: 'postType', baseURL: '/wp/v2/posts' }, ]; const dispatch = Object.assign( jest.fn(), { receiveEntityRecords: jest.fn(), __unstableAcquireStoreLock: jest.fn(), __unstableReleaseStoreLock: jest.fn(), } ); const resolveSelect = { getEntitiesConfig: jest.fn( () => entities ) }; // Provide response apiFetch.mockImplementation( () => { throw new Error( 'API error' ); } ); await expect( deleteEntityRecord( 'postType', 'post', 10, {}, { throwOnError: true, } )( { dispatch, resolveSelect } ) ).rejects.toEqual( new Error( 'API error' ) ); } ); it( 'resolves on error when throwOnError is false', async () => { const entities = [ { name: 'post', kind: 'postType', baseURL: '/wp/v2/posts' }, ]; const dispatch = Object.assign( jest.fn(), { receiveEntityRecords: jest.fn(), __unstableAcquireStoreLock: jest.fn(), __unstableReleaseStoreLock: jest.fn(), } ); const resolveSelect = { getEntitiesConfig: jest.fn( () => entities ) }; // Provide response apiFetch.mockImplementation( () => { throw new Error( 'API error' ); } ); await expect( deleteEntityRecord( 'postType', 'post', 10, {}, { throwOnError: false, } )( { dispatch, resolveSelect } ) ).resolves.toBe( false ); } ); } ); describe( 'saveEditedEntityRecord', () => { beforeEach( async () => { apiFetch.mockReset(); } ); it( 'Uses "id" as a key when no entity key is provided', async () => { const item = { id: 1, menu: 0 }; const configs = [ { kind: 'root', name: 'menuItem', baseURL: '/wp/v2/menu-items', }, ]; const select = { getEntityRecordNonTransientEdits: () => [], hasEditsForEntityRecord: () => true, }; const dispatch = Object.assign( jest.fn(), { saveEntityRecord: jest.fn(), } ); const resolveSelect = { getEntitiesConfig: jest.fn( () => configs ) }; // Provide response const updatedRecord = { ...item, menu: 10 }; apiFetch.mockImplementation( () => { return updatedRecord; } ); await saveEditedEntityRecord( 'root', 'menuItem', 1 )( { dispatch, select, resolveSelect } ); expect( dispatch.saveEntityRecord ).toHaveBeenCalledWith( 'root', 'menuItem', { id: 1 }, undefined ); } ); it( 'Uses the entity key when provided', async () => { const item = { name: 'primary', menu: 0 }; const configs = [ { kind: 'root', name: 'menuLocation', baseURL: '/wp/v2/menu-items', key: 'name', }, ]; const select = { getEntityRecordNonTransientEdits: () => [], hasEditsForEntityRecord: () => true, }; const dispatch = Object.assign( jest.fn(), { saveEntityRecord: jest.fn(), } ); const resolveSelect = { getEntitiesConfig: jest.fn( () => configs ) }; // Provide response const updatedRecord = { ...item, menu: 10 }; apiFetch.mockImplementation( () => { return updatedRecord; } ); await saveEditedEntityRecord( 'root', 'menuLocation', 'primary' )( { dispatch, select, resolveSelect } ); expect( dispatch.saveEntityRecord ).toHaveBeenCalledWith( 'root', 'menuLocation', { name: 'primary' }, undefined ); } ); } ); describe( 'saveEntityRecord', () => { let dispatch; beforeEach( async () => { apiFetch.mockReset(); dispatch = Object.assign( jest.fn(), { receiveEntityRecords: jest.fn(), __unstableAcquireStoreLock: jest.fn(), __unstableReleaseStoreLock: jest.fn(), } ); } ); it( 'triggers a POST request for a new record', async () => { const post = { title: 'new post' }; const configs = [ { name: 'post', kind: 'postType', baseURL: '/wp/v2/posts' }, ]; const select = { getRawEntityRecord: () => post, }; const resolveSelect = { getEntitiesConfig: jest.fn( () => configs ) }; // Provide response const updatedRecord = { ...post, id: 10 }; apiFetch.mockImplementation( () => { return updatedRecord; } ); const result = await saveEntityRecord( 'postType', 'post', post )( { select, dispatch, resolveSelect } ); expect( apiFetch ).toHaveBeenCalledTimes( 1 ); expect( apiFetch ).toHaveBeenCalledWith( { path: '/wp/v2/posts', method: 'POST', data: post, } ); expect( dispatch ).toHaveBeenCalledTimes( 2 ); expect( dispatch ).toHaveBeenCalledWith( { type: 'SAVE_ENTITY_RECORD_START', kind: 'postType', name: 'post', recordId: undefined, isAutosave: false, } ); expect( dispatch.__unstableAcquireStoreLock ).toHaveBeenCalledTimes( 1 ); expect( dispatch ).toHaveBeenCalledWith( { type: 'SAVE_ENTITY_RECORD_FINISH', kind: 'postType', name: 'post', recordId: undefined, error: undefined, isAutosave: false, } ); expect( dispatch.__unstableReleaseStoreLock ).toHaveBeenCalledTimes( 1 ); expect( dispatch.receiveEntityRecords ).toHaveBeenCalledTimes( 1 ); expect( dispatch.receiveEntityRecords ).toHaveBeenCalledWith( 'postType', 'post', updatedRecord, undefined, true, post ); expect( result ).toBe( updatedRecord ); } ); it( 'throws on error when throwOnError is true', async () => { const post = { title: 'new post' }; const entities = [ { name: 'post', kind: 'postType', baseURL: '/wp/v2/posts' }, ]; const select = { getRawEntityRecord: () => post, }; const resolveSelect = { getEntitiesConfig: jest.fn( () => entities ) }; // Provide response apiFetch.mockImplementation( () => { throw new Error( 'API error' ); } ); await expect( saveEntityRecord( 'postType', 'post', post, { throwOnError: true, } )( { select, dispatch, resolveSelect } ) ).rejects.toEqual( new Error( 'API error' ) ); } ); it( 'resolves on error when throwOnError is false', async () => { const post = { title: 'new post' }; const entities = [ { name: 'post', kind: 'postType', baseURL: '/wp/v2/posts' }, ]; const select = { getRawEntityRecord: () => post, }; const resolveSelect = { getEntitiesConfig: jest.fn( () => entities ) }; // Provide response apiFetch.mockImplementation( () => { throw new Error( 'API error' ); } ); await expect( saveEntityRecord( 'postType', 'post', post, { throwOnError: false, } )( { select, dispatch, resolveSelect } ) ).resolves.toEqual( undefined ); } ); it( 'triggers a PUT request for an existing record', async () => { const post = { id: 10, title: 'new post' }; const configs = [ { name: 'post', kind: 'postType', baseURL: '/wp/v2/posts' }, ]; const select = { getRawEntityRecord: () => post, }; const resolveSelect = { getEntitiesConfig: jest.fn( () => configs ) }; // Provide response const updatedRecord = { ...post, id: 10 }; apiFetch.mockImplementation( () => { return updatedRecord; } ); const result = await saveEntityRecord( 'postType', 'post', post )( { select, dispatch, resolveSelect } ); expect( apiFetch ).toHaveBeenCalledTimes( 1 ); expect( apiFetch ).toHaveBeenCalledWith( { path: '/wp/v2/posts/10', method: 'PUT', data: post, } ); expect( dispatch ).toHaveBeenCalledTimes( 2 ); expect( dispatch ).toHaveBeenCalledWith( { type: 'SAVE_ENTITY_RECORD_START', kind: 'postType', name: 'post', recordId: 10, isAutosave: false, } ); expect( dispatch.__unstableAcquireStoreLock ).toHaveBeenCalledTimes( 1 ); expect( dispatch ).toHaveBeenCalledWith( { type: 'SAVE_ENTITY_RECORD_FINISH', kind: 'postType', name: 'post', recordId: 10, error: undefined, isAutosave: false, } ); expect( dispatch.__unstableReleaseStoreLock ).toHaveBeenCalledTimes( 1 ); expect( dispatch.receiveEntityRecords ).toHaveBeenCalledTimes( 1 ); expect( dispatch.receiveEntityRecords ).toHaveBeenCalledWith( 'postType', 'post', updatedRecord, undefined, true, post ); expect( result ).toBe( updatedRecord ); } ); it( 'triggers a PUT request for an existing record with a custom key', async () => { const postType = { slug: 'page', title: 'Pages' }; const configs = [ { name: 'postType', kind: 'root', baseURL: '/wp/v2/types', key: 'slug', }, ]; const select = { getRawEntityRecord: () => ( {} ), }; const resolveSelect = { getEntitiesConfig: jest.fn( () => configs ) }; // Provide response apiFetch.mockImplementation( () => postType ); const result = await saveEntityRecord( 'root', 'postType', postType )( { select, dispatch, resolveSelect } ); expect( apiFetch ).toHaveBeenCalledTimes( 1 ); expect( apiFetch ).toHaveBeenCalledWith( { path: '/wp/v2/types/page', method: 'PUT', data: postType, } ); expect( dispatch ).toHaveBeenCalledTimes( 2 ); expect( dispatch ).toHaveBeenCalledWith( { type: 'SAVE_ENTITY_RECORD_START', kind: 'root', name: 'postType', recordId: 'page', isAutosave: false, } ); expect( dispatch.__unstableAcquireStoreLock ).toHaveBeenCalledTimes( 1 ); expect( dispatch ).toHaveBeenCalledWith( { type: 'SAVE_ENTITY_RECORD_FINISH', kind: 'root', name: 'postType', recordId: 'page', error: undefined, isAutosave: false, } ); expect( dispatch.__unstableReleaseStoreLock ).toHaveBeenCalledTimes( 1 ); expect( dispatch.receiveEntityRecords ).toHaveBeenCalledTimes( 1 ); expect( dispatch.receiveEntityRecords ).toHaveBeenCalledWith( 'root', 'postType', postType, undefined, true, { slug: 'page', title: 'Pages' } ); expect( result ).toBe( postType ); } ); } ); describe( 'receiveUserPermission', () => { it( 'builds an action object', () => { expect( receiveUserPermission( 'create/media', true ) ).toEqual( { type: 'RECEIVE_USER_PERMISSION', key: 'create/media', isAllowed: true, } ); } ); } ); describe( 'receiveAutosaves', () => { it( 'builds an action object', () => { const postId = 1; const autosaves = [ { content: 'test 1', }, { content: 'test 2', }, ]; expect( receiveAutosaves( postId, autosaves ) ).toEqual( { type: 'RECEIVE_AUTOSAVES', postId, autosaves, } ); } ); it( 'converts singular autosaves into an array', () => { const postId = 1; const autosave = { content: 'test 1', }; expect( receiveAutosaves( postId, autosave ) ).toEqual( { type: 'RECEIVE_AUTOSAVES', postId, autosaves: [ autosave ], } ); } ); } ); describe( 'receiveCurrentUser', () => { it( 'builds an action object', () => { const currentUser = { id: 1 }; expect( receiveCurrentUser( currentUser ) ).toEqual( { type: 'RECEIVE_CURRENT_USER', currentUser, } ); } ); } ); describe( '__experimentalBatch', () => { it( 'batches multiple actions together', async () => { const dispatch = { saveEntityRecord: jest.fn( ( kind, name, record, { __unstableFetch } ) => { __unstableFetch( {} ); return { id: 123, created: true }; } ), saveEditedEntityRecord: jest.fn( ( kind, name, recordId, { __unstableFetch } ) => { __unstableFetch( {} ); return { id: 123, updated: true }; } ), deleteEntityRecord: jest.fn( ( kind, name, recordId, query, { __unstableFetch } ) => { __unstableFetch( {} ); return { id: 123, deleted: true }; } ), }; const results = await __experimentalBatch( [ ( { saveEntityRecord: _saveEntityRecord } ) => _saveEntityRecord( 'root', 'widget', {} ), ( { saveEditedEntityRecord: _saveEditedEntityRecord } ) => _saveEditedEntityRecord( 'root', 'widget', 123 ), ( { deleteEntityRecord: _deleteEntityRecord } ) => _deleteEntityRecord( 'root', 'widget', 123, {} ), ], { __unstableProcessor: ( inputs ) => Promise.resolve( inputs ) } )( { dispatch } ); expect( dispatch.saveEntityRecord ).toHaveBeenCalledWith( 'root', 'widget', {}, { __unstableFetch: expect.any( Function ) } ); expect( dispatch.saveEditedEntityRecord ).toHaveBeenCalledWith( 'root', 'widget', 123, { __unstableFetch: expect.any( Function ) } ); expect( dispatch.deleteEntityRecord ).toHaveBeenCalledWith( 'root', 'widget', 123, {}, { __unstableFetch: expect.any( Function ) } ); expect( results ).toEqual( [ { id: 123, created: true }, { id: 123, updated: true }, { id: 123, deleted: true }, ] ); } ); } );