-
Notifications
You must be signed in to change notification settings - Fork 3.8k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
97516: ui: refine hook to schedule data refreshes r=xinhaoz a=xinhaoz This commit refines the logic in the `useFetchDataWithPolling` hook. Namely, it has been renamed to `useScheduleFunction` to be more generic. It also fixes a bug where the first call would not occur if `shouldPoll` was false. This mainly affects the insights pages where if the page was loaded with a custom time interval, the request would not be made. Epic: none Release note: None Co-authored-by: Xin Hao Zhang <[email protected]>
- Loading branch information
Showing
4 changed files
with
220 additions
and
55 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,142 @@ | ||
// Copyright 2023 The Cockroach Authors. | ||
// | ||
// Use of this software is governed by the Business Source License | ||
// included in the file licenses/BSL.txt. | ||
// | ||
// As of the Change Date specified in that file, in accordance with | ||
// the Business Source License, use of this software will be governed | ||
// by the Apache License, Version 2.0, included in the file | ||
// licenses/APL.txt. | ||
|
||
import React from "react"; | ||
import { useScheduleFunction } from "./hooks"; | ||
import { | ||
render, | ||
cleanup, | ||
screen, | ||
fireEvent, | ||
waitFor, | ||
} from "@testing-library/react"; | ||
import moment from "moment"; | ||
|
||
describe("useScheduleFunction", () => { | ||
let mockFn: jest.Mock; | ||
|
||
beforeEach(() => { | ||
mockFn = jest.fn(); | ||
}); | ||
|
||
afterEach(() => { | ||
cleanup(); | ||
}); | ||
|
||
it("should schedule the function according to the lastCompleted time and interval provided", async () => { | ||
const timeoutMs = 1500; | ||
const TestScheduleHook = () => { | ||
const [lastUpdated, setLastUpdated] = React.useState<moment.Moment>(null); | ||
|
||
const setLastUpdatedToNow = () => { | ||
mockFn(); | ||
setLastUpdated(moment.utc()); | ||
}; | ||
|
||
useScheduleFunction(setLastUpdatedToNow, true, timeoutMs, lastUpdated); | ||
return <div />; | ||
}; | ||
|
||
const { unmount } = render(<TestScheduleHook />); | ||
await waitFor( | ||
() => new Promise(res => setTimeout(res, timeoutMs * 3 + 1000)), // Add 0.5s of buffer. | ||
{ timeout: timeoutMs * 3 + 1500 }, | ||
); | ||
// 3 intervals and the initial call on mount, since last completed was initially null. | ||
expect(mockFn).toBeCalledTimes(4); | ||
|
||
unmount(); | ||
// Verify scheduling is stopped on unmount. | ||
await waitFor( | ||
() => new Promise(res => setTimeout(res, timeoutMs + 500)), // Add 0.5s of buffer. | ||
{ timeout: timeoutMs + 550 }, | ||
); | ||
expect(mockFn).toBeCalledTimes(4); | ||
// }); | ||
}, 20000); | ||
|
||
it("should schedule the function immediately if returned scheduleNow is used", async () => { | ||
const TestScheduleHook = () => { | ||
const [lastUpdated, setLastUpdated] = React.useState<moment.Moment>( | ||
moment.utc(), | ||
); | ||
|
||
const setLastUpdatedToNow = () => { | ||
mockFn(); | ||
setLastUpdated(moment.utc()); | ||
}; | ||
|
||
const [scheduleNow] = useScheduleFunction( | ||
setLastUpdatedToNow, | ||
true, | ||
3000, // Longer timeout. | ||
lastUpdated, | ||
); | ||
|
||
const onClick = () => { | ||
scheduleNow(); | ||
}; | ||
|
||
return ( | ||
<div> | ||
<button onClick={onClick}></button> | ||
</div> | ||
); | ||
}; | ||
|
||
render(<TestScheduleHook />); | ||
const button = await screen.getByRole("button"); | ||
|
||
fireEvent.click(button); | ||
await waitFor(() => expect(mockFn).toBeCalledTimes(1), { timeout: 1500 }); | ||
|
||
fireEvent.click(button); | ||
await waitFor(() => expect(mockFn).toBeCalledTimes(2), { timeout: 1500 }); | ||
|
||
// Verify that the schedule is set up correctly after by waiting the interval length. | ||
await waitFor(() => new Promise(res => setTimeout(res, 3100)), { | ||
timeout: 4000, | ||
}); | ||
expect(mockFn).toBeCalledTimes(3); | ||
}, 10000); | ||
|
||
it("should clear the scheduled func immediately if clear is used", async () => { | ||
const TestScheduleHook = () => { | ||
// eslint-disable-next-line @typescript-eslint/no-unused-vars | ||
const [_, clear] = useScheduleFunction(mockFn, true, 1000, moment.utc()); | ||
|
||
React.useEffect(() => { | ||
clear(); | ||
}); | ||
|
||
return <div />; | ||
}; | ||
|
||
render(<TestScheduleHook />); | ||
await waitFor(() => new Promise(res => setTimeout(res, 5000)), { | ||
timeout: 5500, | ||
}); | ||
expect(mockFn).toBeCalledTimes(0); | ||
}, 10000); | ||
|
||
it("should not reschedule the func if shouldReschedule=false", async () => { | ||
const TestScheduleHook = () => { | ||
// Since we pass in a last completed time here, we should expect no calls. | ||
useScheduleFunction(mockFn, false, 100, moment.utc()); | ||
return <div />; | ||
}; | ||
|
||
render(<TestScheduleHook />); | ||
await waitFor(() => new Promise(res => setTimeout(res, 3000)), { | ||
timeout: 3500, | ||
}); | ||
expect(mockFn).toBeCalledTimes(0); | ||
}); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters