From 50d216cb4b7a98995b327cb997bcf6bbd873ea18 Mon Sep 17 00:00:00 2001 From: Kai Vandivier <49666798+KaiVandivier@users.noreply.github.com> Date: Wed, 27 Nov 2024 14:13:52 +0100 Subject: [PATCH] fix(cacheable-section): synchronously flush recording state for UI consistency (#1394) --- services/data/src/react/hooks/useDataQuery.ts | 11 ++-- .../src/__tests__/integration.test.tsx | 20 ++++---- .../cacheable-section-state.test.tsx | 4 +- .../__tests__/use-cacheable-section.test.tsx | 50 +++++++++++-------- .../offline/src/lib/cacheable-section.tsx | 15 ++++-- 5 files changed, 58 insertions(+), 42 deletions(-) diff --git a/services/data/src/react/hooks/useDataQuery.ts b/services/data/src/react/hooks/useDataQuery.ts index 6e4d3862..a7d16fb2 100644 --- a/services/data/src/react/hooks/useDataQuery.ts +++ b/services/data/src/react/hooks/useDataQuery.ts @@ -1,11 +1,10 @@ import { useState, useRef, useCallback, useDebugValue } from 'react' import { useQuery, setLogger } from 'react-query' -import { - JsonMap, - type Query, - type QueryOptions, - type QueryResult, - type QueryVariables, +import type { + Query, + QueryOptions, + QueryResult, + QueryVariables, } from '../../engine' import type { FetchError } from '../../engine/types/FetchError' import type { QueryRenderInput, QueryRefetchFunction } from '../../types' diff --git a/services/offline/src/__tests__/integration.test.tsx b/services/offline/src/__tests__/integration.test.tsx index db760d77..0d92427c 100644 --- a/services/offline/src/__tests__/integration.test.tsx +++ b/services/offline/src/__tests__/integration.test.tsx @@ -43,7 +43,7 @@ const TestControls = ({ />
{isCached ? 'yes' : 'no'}
- {lastUpdated || 'never'} + {lastUpdated?.toISOString() ?? 'never'}
{recordingState}
@@ -112,7 +112,7 @@ describe('Coordination between useCacheableSection and CacheableSection', () => expect(getByTestId(/controls-rc/)).toBeInTheDocument() }) - it.skip('handles a successful recording', async (done) => { + it('handles a successful recording', async (done) => { const { getByTestId, queryByTestId } = screen const onStarted = () => { @@ -147,7 +147,7 @@ describe('Coordination between useCacheableSection and CacheableSection', () => expect.assertions(7) }) - it.skip('handles a recording that encounters an error', async (done) => { + it('handles a recording that encounters an error', async (done) => { // Suppress the expected error from console (in addition to 'act' warning) jest.spyOn(console, 'error').mockImplementation((...args) => { const actPattern = @@ -255,7 +255,7 @@ describe('Performant state management', () => { expect(getByTestId('section-rc-2')).toHaveTextContent('1') }) - it.skip('isolates rerenders from other consumers', async (done) => { + it('isolates rerenders from other consumers', async (done) => { const { getByTestId } = screen // Make assertions const onCompleted = () => { @@ -300,12 +300,12 @@ describe('useCacheableSection can be used inside a child of CacheableSection', ( ) } - it.skip('handles a successful recording', async (done) => { - const { getByTestId, queryByTestId, findByTestId } = screen + it('handles a successful recording', async (done) => { + const { getByTestId, queryByTestId } = screen const onStarted = async () => { await waitFor(() => { - expect(getByTestId(/recording-state/)).toBeInTheDocument( + expect(getByTestId(/recording-state/)).toHaveTextContent( 'recording' ) expect(getByTestId(/loading-mask/)).toBeInTheDocument() @@ -326,9 +326,9 @@ describe('useCacheableSection can be used inside a child of CacheableSection', ( render() - // await act(async () => { - await fireEvent.click(getByTestId(/start-recording/)) - // }) + await act(async () => { + await fireEvent.click(getByTestId(/start-recording/)) + }) await waitFor(() => { // At this stage, should be pending diff --git a/services/offline/src/lib/__tests__/cacheable-section-state.test.tsx b/services/offline/src/lib/__tests__/cacheable-section-state.test.tsx index 5bd9079c..ff897ca2 100644 --- a/services/offline/src/lib/__tests__/cacheable-section-state.test.tsx +++ b/services/offline/src/lib/__tests__/cacheable-section-state.test.tsx @@ -1,10 +1,10 @@ import { renderHook } from '@testing-library/react' -import React, { FC } from 'react' +import React, { FC, PropsWithChildren } from 'react' import { mockOfflineInterface } from '../../utils/test-mocks' import { useCachedSection, useRecordingState } from '../cacheable-section-state' import { OfflineProvider } from '../offline-provider' -const wrapper: FC = ({ children }) => ( +const wrapper: FC = ({ children }) => ( {children} diff --git a/services/offline/src/lib/__tests__/use-cacheable-section.test.tsx b/services/offline/src/lib/__tests__/use-cacheable-section.test.tsx index cfc2cc1b..8e8743a6 100644 --- a/services/offline/src/lib/__tests__/use-cacheable-section.test.tsx +++ b/services/offline/src/lib/__tests__/use-cacheable-section.test.tsx @@ -27,7 +27,7 @@ afterEach(() => { ;(console.error as jest.Mock).mockRestore() }) -it.skip('renders in the default state initially', () => { +it('renders in the default state initially', () => { const wrapper: FC = ({ children }) => ( {children} @@ -42,7 +42,7 @@ it.skip('renders in the default state initially', () => { }) it('has stable references', () => { - const wrapper: FC = ({ children }) => ( + const wrapper: FC = ({ children }) => ( {children} @@ -66,9 +66,9 @@ it('has stable references', () => { expect(result.current.remove).toBe(origRemove) }) -it.skip('handles a successful recording', async (done) => { +it('handles a successful recording', async (done) => { const [sectionId, timeoutDelay] = ['one', 1234] - const testOfflineInterface = { + const recordingSuccessOfflineInterface = { ...mockOfflineInterface, getCachedSections: jest .fn() @@ -78,7 +78,7 @@ it.skip('handles a successful recording', async (done) => { ]), } const wrapper: FC = ({ children }) => ( - + {children} ) @@ -93,8 +93,16 @@ it.skip('handles a successful recording', async (done) => { expect(result.current.recordingState).toBe('default') // Test that 'isCached' gets updated - expect(testOfflineInterface.getCachedSections).toBeCalledTimes(2) - await waitFor(() => expect(result.current.isCached).toBe(true)) + expect( + recordingSuccessOfflineInterface.getCachedSections + ).toBeCalledTimes(2) + // Recording states are updated synchronously, but getting isCached + // state is asynchronous -- need to wait for that here. + // An assertion is not used as the waitFor condition because it may skew + // the total number assertions in this test if it needs to retry. Number + // of assertions is checked at the bottom of this test to make sure both + // of these callbacks are called. + await waitFor(() => result.current.isCached === true) expect(result.current.isCached).toBe(true) expect(result.current.lastUpdated).toBeInstanceOf(Date) @@ -125,7 +133,7 @@ it.skip('handles a successful recording', async (done) => { expect.assertions(11) }) -it.skip('handles a recording that encounters an error', async (done) => { +it('handles a recording that encounters an error', async (done) => { // Suppress the expected error from console (in addition to 'act' warning) jest.spyOn(console, 'error').mockImplementation((...args) => { const actPattern = @@ -138,12 +146,12 @@ it.skip('handles a recording that encounters an error', async (done) => { } return originalError.call(console, ...args) }) - const testOfflineInterface = { + const recordingErrorOfflineInterface = { ...mockOfflineInterface, startRecording: errorRecordingMock, } const wrapper: FC = ({ children }) => ( - + {children} ) @@ -181,13 +189,13 @@ it.skip('handles a recording that encounters an error', async (done) => { expect.assertions(6) }) -it.skip('handles an error starting the recording', async () => { - const testOfflineInterface = { +it('handles an error starting the recording', async () => { + const messageErrorOfflineInterface = { ...mockOfflineInterface, startRecording: failedMessageRecordingMock, } const wrapper: FC = ({ children }) => ( - + {children} ) @@ -198,9 +206,9 @@ it.skip('handles an error starting the recording', async () => { ) }) -it.skip('handles remove and updates sections', async () => { +it('handles remove and updates sections', async () => { const sectionId = 'one' - const testOfflineInterface = { + const sectionOpsOfflineInterface = { ...mockOfflineInterface, getCachedSections: jest .fn() @@ -210,7 +218,7 @@ it.skip('handles remove and updates sections', async () => { .mockResolvedValueOnce([]), } const wrapper: FC = ({ children }) => ( - + {children} ) @@ -228,14 +236,14 @@ it.skip('handles remove and updates sections', async () => { expect(success).toBe(true) // Test that 'isCached' gets updated - expect(testOfflineInterface.getCachedSections).toBeCalledTimes(2) + expect(sectionOpsOfflineInterface.getCachedSections).toBeCalledTimes(2) await waitFor(() => expect(result.current.isCached).toBe(false)) expect(result.current.isCached).toBe(false) expect(result.current.lastUpdated).toBeUndefined() }) -it.skip('handles a change in ID', async () => { - const testOfflineInterface = { +it('handles a change in ID', async () => { + const idChangeOfflineInterface = { ...mockOfflineInterface, getCachedSections: jest .fn() @@ -244,7 +252,7 @@ it.skip('handles a change in ID', async () => { ]), } const wrapper: FC = ({ children }) => ( - + {children} ) @@ -259,7 +267,7 @@ it.skip('handles a change in ID', async () => { rerender('id-two') // Test that 'isCached' gets updated - // expect(testOfflineInterface.getCachedSections).toBeCalledTimes(2) + // expect(idChangeOfflineInterface.getCachedSections).toBeCalledTimes(2) await waitFor(() => expect(result.current.isCached).toBe(false)) expect(result.current.isCached).toBe(false) expect(result.current.lastUpdated).toBeUndefined() diff --git a/services/offline/src/lib/cacheable-section.tsx b/services/offline/src/lib/cacheable-section.tsx index 98b1fcdd..57484d52 100644 --- a/services/offline/src/lib/cacheable-section.tsx +++ b/services/offline/src/lib/cacheable-section.tsx @@ -1,5 +1,6 @@ import PropTypes from 'prop-types' import React, { useCallback, useEffect, useMemo } from 'react' +import { flushSync } from 'react-dom' import { RecordingState } from '../types' import { useRecordingState, useCachedSection } from './cacheable-section-state' import { useOfflineInterface } from './offline-interface' @@ -95,15 +96,23 @@ export function useCacheableSection(id: string): CacheableSectionControls { sectionId: id, recordingTimeoutDelay, onStarted: () => { - onRecordingStarted() + // Flush this state update synchronously so that the + // right recordingState is set before any other callbacks + flushSync(() => { + onRecordingStarted() + }) onStarted && onStarted() }, onCompleted: () => { - onRecordingCompleted() + flushSync(() => { + onRecordingCompleted() + }) onCompleted && onCompleted() }, onError: (error) => { - onRecordingError(error) + flushSync(() => { + onRecordingError(error) + }) onError && onError(error) }, })