Skip to content

Commit

Permalink
fix(cacheable-section): synchronously flush recording state for UI co…
Browse files Browse the repository at this point in the history
…nsistency
  • Loading branch information
KaiVandivier committed Nov 25, 2024
1 parent de28ecc commit 04bc604
Show file tree
Hide file tree
Showing 7 changed files with 67 additions and 53 deletions.
11 changes: 5 additions & 6 deletions services/data/src/react/hooks/useDataQuery.ts
Original file line number Diff line number Diff line change
@@ -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'
Expand Down
20 changes: 10 additions & 10 deletions services/offline/src/__tests__/integration.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ const TestControls = ({
/>
<div data-testid={`is-cached-${id}`}>{isCached ? 'yes' : 'no'}</div>
<div data-testid={`last-updated-${id}`}>
{lastUpdated || 'never'}
{lastUpdated?.toISOString() ?? 'never'}
</div>
<div data-testid={`recording-state-${id}`}>{recordingState}</div>
</>
Expand Down Expand Up @@ -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 = () => {
Expand Down Expand Up @@ -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 =
Expand Down Expand Up @@ -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 = () => {
Expand Down Expand Up @@ -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()
Expand All @@ -326,9 +326,9 @@ describe('useCacheableSection can be used inside a child of CacheableSection', (

render(<ChildTest makeRecordingHandler={makeRecordingHandler} />)

// 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
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,10 @@
import { renderHook } from '@testing-library/react-hooks'
import React, { FC } from 'react'
import { renderHook } from '@testing-library/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<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={mockOfflineInterface}>
{children}
</OfflineProvider>
Expand Down
50 changes: 29 additions & 21 deletions services/offline/src/lib/__tests__/use-cacheable-section.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -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<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={mockOfflineInterface}>
{children}
Expand All @@ -42,7 +42,7 @@ it.skip('renders in the default state initially', () => {
})

it('has stable references', () => {
const wrapper: FC = ({ children }) => (
const wrapper: FC<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={mockOfflineInterface}>
{children}
</OfflineProvider>
Expand All @@ -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()
Expand All @@ -78,7 +78,7 @@ it.skip('handles a successful recording', async (done) => {
]),
}
const wrapper: FC<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={testOfflineInterface}>
<OfflineProvider offlineInterface={recordingSuccessOfflineInterface}>
{children}
</OfflineProvider>
)
Expand All @@ -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)

Expand Down Expand Up @@ -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 =
Expand All @@ -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<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={testOfflineInterface}>
<OfflineProvider offlineInterface={recordingErrorOfflineInterface}>
{children}
</OfflineProvider>
)
Expand Down Expand Up @@ -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<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={testOfflineInterface}>
<OfflineProvider offlineInterface={messageErrorOfflineInterface}>
{children}
</OfflineProvider>
)
Expand All @@ -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()
Expand All @@ -210,7 +218,7 @@ it.skip('handles remove and updates sections', async () => {
.mockResolvedValueOnce([]),
}
const wrapper: FC<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={testOfflineInterface}>
<OfflineProvider offlineInterface={sectionOpsOfflineInterface}>
{children}
</OfflineProvider>
)
Expand All @@ -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()
Expand All @@ -244,7 +252,7 @@ it.skip('handles a change in ID', async () => {
]),
}
const wrapper: FC<PropsWithChildren> = ({ children }) => (
<OfflineProvider offlineInterface={testOfflineInterface}>
<OfflineProvider offlineInterface={idChangeOfflineInterface}>
{children}
</OfflineProvider>
)
Expand All @@ -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()
Expand Down
11 changes: 8 additions & 3 deletions services/offline/src/lib/cacheable-section-state.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@ import PropTypes from 'prop-types'
import React, { useEffect, useCallback, useMemo } from 'react'
import {
GlobalStateStore,
GlobalStateStoreMutationCreator,
IndexedDBCachedSection,
RecordingState,
} from '../types'
Expand Down Expand Up @@ -108,12 +109,14 @@ interface RecordingStateControls {
*/
export function useRecordingState(id: string): RecordingStateControls {
const recordingStateSelector = useCallback(
(state) => state.recordingStates[id],
(state: any) => state.recordingStates[id],
[id]
)
const [recordingState] = useGlobalState(recordingStateSelector)

const setRecordingStateMutationCreator = useCallback(
const setRecordingStateMutationCreator = useCallback<
GlobalStateStoreMutationCreator<RecordingState>
>(
(newState) => (state: any) => ({
...state,
recordingStates: { ...state.recordingStates, [id]: newState },
Expand Down Expand Up @@ -155,7 +158,9 @@ export function useRecordingState(id: string): RecordingStateControls {
function useSyncCachedSections() {
const offlineInterface = useOfflineInterface()

const setCachedSectionsMutationCreator = useCallback(
const setCachedSectionsMutationCreator = useCallback<
GlobalStateStoreMutationCreator<CachedSectionsById>
>(
(cachedSections) => (state: any) => ({
...state,
cachedSections,
Expand Down
15 changes: 12 additions & 3 deletions services/offline/src/lib/cacheable-section.tsx
Original file line number Diff line number Diff line change
@@ -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'
Expand Down Expand Up @@ -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)
},
})
Expand Down
7 changes: 0 additions & 7 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -1780,13 +1780,6 @@
"@dhis2/pwa" "12.0.0-alpha.25"
moment "^2.24.0"

"@dhis2/[email protected]":
version "10.2.0"
resolved "https://registry.yarnpkg.com/@dhis2/app-shell/-/app-shell-10.2.0.tgz#2f69aa047dedb6545c75052d8969a80502f0d4a6"
integrity sha512-AqI3TQ53eKsxOMNAPlUpDNUFxqy3tDNZsk8NU4v+PL5nUMEi9dDUtI12emi7sk3UbSjSPEwDR2xDKWzWvTnTZQ==
dependencies:
post-robot "^10.0.46"

"@dhis2/[email protected]":
version "12.0.0-alpha.25"
resolved "https://registry.yarnpkg.com/@dhis2/app-shell/-/app-shell-12.0.0-alpha.25.tgz#61ce08c51a43f887275c4eceb93a05404a64dcad"
Expand Down

0 comments on commit 04bc604

Please sign in to comment.