-
Notifications
You must be signed in to change notification settings - Fork 4
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
David Maskasky
committed
Sep 10, 2024
1 parent
20e1c63
commit 212a4bf
Showing
38 changed files
with
9,898 additions
and
2 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
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,12 @@ | ||
import { atom } from '../../jotai'; | ||
import { createStore } from './derivedStore'; | ||
|
||
it('calls subscribe callback when state changes', () => { | ||
const countAtom = atom(1); | ||
const store = createStore(); | ||
const updateSpy = jest.fn((num) => num); | ||
store.sub(countAtom, () => updateSpy(store.get(countAtom))); | ||
store.set(countAtom, 2); | ||
expect(updateSpy).toHaveBeenCalledTimes(1); | ||
expect(updateSpy).toHaveBeenLastCalledWith(2); | ||
}); |
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,223 @@ | ||
import { StrictMode, Suspense, useState } from 'react'; | ||
import { render, waitFor } from '@testing-library/react'; | ||
import userEvent from '@testing-library/user-event'; | ||
import { useAtomValue, useSetAtom } from '../../../../jotai/react'; | ||
import { atom } from '../../../../jotai/vanilla'; | ||
|
||
describe('abortable atom test', () => { | ||
it('can abort with signal.aborted', async () => { | ||
const countAtom = atom(0); | ||
let abortedCount = 0; | ||
const resolve: (() => void)[] = []; | ||
const derivedAtom = atom(async (get, { signal }) => { | ||
const count = get(countAtom); | ||
await new Promise<void>((r) => resolve.push(r)); | ||
if (signal.aborted) { | ||
++abortedCount; | ||
} | ||
return count; | ||
}); | ||
|
||
const Component = () => { | ||
const count = useAtomValue(derivedAtom); | ||
return <div>count: {count}</div>; | ||
}; | ||
|
||
const Controls = () => { | ||
const setCount = useSetAtom(countAtom); | ||
return ( | ||
<> | ||
<button onClick={() => setCount((c) => c + 1)}>button</button> | ||
</> | ||
); | ||
}; | ||
|
||
const { findByText, getByText } = render( | ||
<StrictMode> | ||
<Suspense fallback="loading"> | ||
<Component /> | ||
<Controls /> | ||
</Suspense> | ||
</StrictMode>, | ||
); | ||
|
||
await findByText('loading'); | ||
|
||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 0'); | ||
expect(abortedCount).toBe(0); | ||
|
||
await userEvent.click(getByText('button')); | ||
await userEvent.click(getByText('button')); | ||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 2'); | ||
|
||
expect(abortedCount).toBe(1); | ||
|
||
await userEvent.click(getByText('button')); | ||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 3'); | ||
expect(abortedCount).toBe(1); | ||
}); | ||
|
||
it('can abort with event listener', async () => { | ||
const countAtom = atom(0); | ||
let abortedCount = 0; | ||
const resolve: (() => void)[] = []; | ||
const derivedAtom = atom(async (get, { signal }) => { | ||
const count = get(countAtom); | ||
const callback = () => { | ||
++abortedCount; | ||
}; | ||
signal.addEventListener('abort', callback); | ||
await new Promise<void>((r) => resolve.push(r)); | ||
signal.removeEventListener('abort', callback); | ||
return count; | ||
}); | ||
|
||
const Component = () => { | ||
const count = useAtomValue(derivedAtom); | ||
return <div>count: {count}</div>; | ||
}; | ||
|
||
const Controls = () => { | ||
const setCount = useSetAtom(countAtom); | ||
return ( | ||
<> | ||
<button onClick={() => setCount((c) => c + 1)}>button</button> | ||
</> | ||
); | ||
}; | ||
|
||
const { findByText, getByText } = render( | ||
<StrictMode> | ||
<Suspense fallback="loading"> | ||
<Component /> | ||
<Controls /> | ||
</Suspense> | ||
</StrictMode>, | ||
); | ||
|
||
await findByText('loading'); | ||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 0'); | ||
|
||
expect(abortedCount).toBe(0); | ||
|
||
await userEvent.click(getByText('button')); | ||
await userEvent.click(getByText('button')); | ||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 2'); | ||
|
||
expect(abortedCount).toBe(1); | ||
|
||
await userEvent.click(getByText('button')); | ||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 3'); | ||
|
||
expect(abortedCount).toBe(1); | ||
}); | ||
|
||
it('does not abort on unmount', async () => { | ||
const countAtom = atom(0); | ||
let abortedCount = 0; | ||
const resolve: (() => void)[] = []; | ||
const derivedAtom = atom(async (get, { signal }) => { | ||
const count = get(countAtom); | ||
await new Promise<void>((r) => resolve.push(r)); | ||
if (signal.aborted) { | ||
++abortedCount; | ||
} | ||
return count; | ||
}); | ||
|
||
const Component = () => { | ||
const count = useAtomValue(derivedAtom); | ||
return <div>count: {count}</div>; | ||
}; | ||
|
||
const Parent = () => { | ||
const setCount = useSetAtom(countAtom); | ||
const [show, setShow] = useState(true); | ||
return ( | ||
<> | ||
{show ? <Component /> : 'hidden'} | ||
<button onClick={() => setCount((c) => c + 1)}>button</button> | ||
<button onClick={() => setShow((x) => !x)}>toggle</button> | ||
</> | ||
); | ||
}; | ||
|
||
const { findByText, getByText } = render( | ||
<StrictMode> | ||
<Suspense fallback="loading"> | ||
<Parent /> | ||
</Suspense> | ||
</StrictMode>, | ||
); | ||
|
||
await findByText('loading'); | ||
|
||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 0'); | ||
expect(abortedCount).toBe(0); | ||
|
||
await userEvent.click(getByText('button')); | ||
await userEvent.click(getByText('toggle')); | ||
|
||
await findByText('hidden'); | ||
|
||
resolve.splice(0).forEach((fn) => fn()); | ||
await waitFor(() => expect(abortedCount).toBe(0)); | ||
}); | ||
|
||
it('throws aborted error (like fetch)', async () => { | ||
const countAtom = atom(0); | ||
const resolve: (() => void)[] = []; | ||
const derivedAtom = atom(async (get, { signal }) => { | ||
const count = get(countAtom); | ||
await new Promise<void>((r) => resolve.push(r)); | ||
if (signal.aborted) { | ||
throw new Error('aborted'); | ||
} | ||
return count; | ||
}); | ||
|
||
const Component = () => { | ||
const count = useAtomValue(derivedAtom); | ||
return <div>count: {count}</div>; | ||
}; | ||
|
||
const Controls = () => { | ||
const setCount = useSetAtom(countAtom); | ||
return ( | ||
<> | ||
<button onClick={() => setCount((c) => c + 1)}>button</button> | ||
</> | ||
); | ||
}; | ||
|
||
const { findByText, getByText } = render( | ||
<StrictMode> | ||
<Suspense fallback="loading"> | ||
<Component /> | ||
<Controls /> | ||
</Suspense> | ||
</StrictMode>, | ||
); | ||
|
||
await findByText('loading'); | ||
|
||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 0'); | ||
|
||
await userEvent.click(getByText('button')); | ||
await userEvent.click(getByText('button')); | ||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 2'); | ||
|
||
await userEvent.click(getByText('button')); | ||
resolve.splice(0).forEach((fn) => fn()); | ||
await findByText('count: 3'); | ||
}); | ||
}); |
Oops, something went wrong.