Only refetch changed resources when receive a "refreshResource" event

This commit is contained in:
Deluan
2021-06-15 16:09:01 -04:00
parent 8a56584aed
commit 8383527aab
6 changed files with 235 additions and 39 deletions
+28 -15
View File
@@ -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 })
})
}
})
}
}
+135
View File
@@ -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' })
})
})
})