Only refetch changed resources when receive a "refreshResource" event
This commit is contained in:
@@ -1,23 +1,36 @@
|
||||
import { useSelector } from 'react-redux'
|
||||
import { useState } from 'react'
|
||||
import { useRefresh } from 'react-admin'
|
||||
import { useRefresh, useDataProvider } from 'react-admin'
|
||||
|
||||
export const useResourceRefresh = (...resources) => {
|
||||
export const useResourceRefresh = (...visibleResources) => {
|
||||
const [lastTime, setLastTime] = useState(Date.now())
|
||||
const refreshData = useSelector(
|
||||
(state) => state.activity?.refresh || { lastTime }
|
||||
)
|
||||
const refresh = useRefresh()
|
||||
const dataProvider = useDataProvider()
|
||||
const refreshData = useSelector(
|
||||
(state) => state.activity?.refresh || { lastReceived: lastTime }
|
||||
)
|
||||
const { resources, lastReceived } = refreshData
|
||||
|
||||
const resource = refreshData.resource
|
||||
if (refreshData.lastTime > lastTime) {
|
||||
if (
|
||||
resource === '' ||
|
||||
resources.length === 0 ||
|
||||
resources.includes(resource)
|
||||
) {
|
||||
refresh()
|
||||
}
|
||||
setLastTime(refreshData.lastTime)
|
||||
if (lastReceived <= lastTime) {
|
||||
return
|
||||
}
|
||||
setLastTime(lastReceived)
|
||||
|
||||
if (
|
||||
resources &&
|
||||
(resources['*'] === '*' ||
|
||||
Object.values(resources).find((v) => v.find((v2) => v2 === '*')))
|
||||
) {
|
||||
refresh()
|
||||
return
|
||||
}
|
||||
if (resources) {
|
||||
Object.keys(resources).forEach((r) => {
|
||||
if (visibleResources.length === 0 || visibleResources?.includes(r)) {
|
||||
resources[r]?.forEach((id) => {
|
||||
dataProvider.getOne(r, { id })
|
||||
})
|
||||
}
|
||||
})
|
||||
}
|
||||
}
|
||||
|
||||
@@ -0,0 +1,135 @@
|
||||
import * as React from 'react'
|
||||
import * as Redux from 'react-redux'
|
||||
import * as RA from 'react-admin'
|
||||
import { useResourceRefresh } from './useResourceRefresh'
|
||||
|
||||
jest.mock('react', () => ({
|
||||
...jest.requireActual('react'),
|
||||
useState: jest.fn(),
|
||||
}))
|
||||
|
||||
jest.mock('react-redux', () => ({
|
||||
...jest.requireActual('react-redux'),
|
||||
useSelector: jest.fn(),
|
||||
}))
|
||||
|
||||
jest.mock('react-admin', () => ({
|
||||
...jest.requireActual('react-admin'),
|
||||
useRefresh: jest.fn(),
|
||||
useDataProvider: jest.fn(),
|
||||
}))
|
||||
|
||||
describe('useResourceRefresh', () => {
|
||||
const setState = jest.fn()
|
||||
const useStateMock = (initState) => [initState, setState]
|
||||
const refresh = jest.fn()
|
||||
const useRefreshMock = () => refresh
|
||||
const getOne = jest.fn()
|
||||
const useDataProviderMock = () => ({ getOne })
|
||||
let lastTime
|
||||
|
||||
beforeEach(() => {
|
||||
jest.spyOn(React, 'useState').mockImplementation(useStateMock)
|
||||
jest.spyOn(RA, 'useRefresh').mockImplementation(useRefreshMock)
|
||||
jest.spyOn(RA, 'useDataProvider').mockImplementation(useDataProviderMock)
|
||||
lastTime = new Date(new Date().valueOf() + 1000)
|
||||
})
|
||||
|
||||
afterEach(() => {
|
||||
jest.clearAllMocks()
|
||||
})
|
||||
|
||||
it('stores last time checked, to avoid redundant runs', () => {
|
||||
const useSelectorMock = () => ({ lastReceived: lastTime })
|
||||
jest.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
|
||||
|
||||
useResourceRefresh()
|
||||
|
||||
expect(setState).toHaveBeenCalledWith(lastTime)
|
||||
})
|
||||
|
||||
it("does not run again if lastTime didn't change", () => {
|
||||
jest.spyOn(React, 'useState').mockImplementation(() => [lastTime, setState])
|
||||
const useSelectorMock = () => ({ lastReceived: lastTime })
|
||||
jest.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
|
||||
|
||||
useResourceRefresh()
|
||||
|
||||
expect(setState).not.toHaveBeenCalled()
|
||||
})
|
||||
|
||||
describe('No visible resources specified', () => {
|
||||
it('triggers a UI refresh when received a "any" resource refresh', () => {
|
||||
const useSelectorMock = () => ({
|
||||
lastReceived: lastTime,
|
||||
resources: { '*': '*' },
|
||||
})
|
||||
jest.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
|
||||
|
||||
useResourceRefresh()
|
||||
|
||||
expect(refresh).toHaveBeenCalledTimes(1)
|
||||
expect(getOne).not.toHaveBeenCalled()
|
||||
})
|
||||
|
||||
it('triggers a UI refresh when received an "any" id', () => {
|
||||
const useSelectorMock = () => ({
|
||||
lastReceived: lastTime,
|
||||
resources: { album: ['*'] },
|
||||
})
|
||||
jest.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
|
||||
|
||||
useResourceRefresh()
|
||||
|
||||
expect(refresh).toHaveBeenCalledTimes(1)
|
||||
expect(getOne).not.toHaveBeenCalled()
|
||||
})
|
||||
|
||||
it('triggers a refetch of the resources received', () => {
|
||||
const useSelectorMock = () => ({
|
||||
lastReceived: lastTime,
|
||||
resources: { album: ['al-1', 'al-2'], song: ['sg-1', 'sg-2'] },
|
||||
})
|
||||
jest.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
|
||||
|
||||
useResourceRefresh()
|
||||
|
||||
expect(refresh).not.toHaveBeenCalled()
|
||||
expect(getOne).toHaveBeenCalledTimes(4)
|
||||
expect(getOne).toHaveBeenCalledWith('album', { id: 'al-1' })
|
||||
expect(getOne).toHaveBeenCalledWith('album', { id: 'al-2' })
|
||||
expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
|
||||
expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-2' })
|
||||
})
|
||||
})
|
||||
|
||||
describe('Visible resources specified', () => {
|
||||
it('triggers a UI refresh when received a "any" resource refresh', () => {
|
||||
const useSelectorMock = () => ({
|
||||
lastReceived: lastTime,
|
||||
resources: { '*': '*' },
|
||||
})
|
||||
jest.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
|
||||
|
||||
useResourceRefresh('album')
|
||||
|
||||
expect(refresh).toHaveBeenCalledTimes(1)
|
||||
expect(getOne).not.toHaveBeenCalled()
|
||||
})
|
||||
|
||||
it('triggers a refetch of the resources received if they are visible', () => {
|
||||
const useSelectorMock = () => ({
|
||||
lastReceived: lastTime,
|
||||
resources: { album: ['al-1', 'al-2'], song: ['sg-1', 'sg-2'] },
|
||||
})
|
||||
jest.spyOn(Redux, 'useSelector').mockImplementation(useSelectorMock)
|
||||
|
||||
useResourceRefresh('song')
|
||||
|
||||
expect(refresh).not.toHaveBeenCalled()
|
||||
expect(getOne).toHaveBeenCalledTimes(2)
|
||||
expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-1' })
|
||||
expect(getOne).toHaveBeenCalledWith('song', { id: 'sg-2' })
|
||||
})
|
||||
})
|
||||
})
|
||||
Reference in New Issue
Block a user