-
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.
fix: full rewrite of ScopeProvider to address known issues
- Loading branch information
David Maskasky
committed
Sep 25, 2024
1 parent
3198ee1
commit 45b1c02
Showing
50 changed files
with
10,924 additions
and
10 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,213 @@ | ||
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 | ||
}) | ||
|
||
function Component() { | ||
const count = useAtomValue(derivedAtom) | ||
return <div>count: {count}</div> | ||
} | ||
|
||
function 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 | ||
}) | ||
|
||
function Component() { | ||
const count = useAtomValue(derivedAtom) | ||
return <div>count: {count}</div> | ||
} | ||
|
||
function 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 | ||
}) | ||
|
||
function Component() { | ||
const count = useAtomValue(derivedAtom) | ||
return <div>count: {count}</div> | ||
} | ||
|
||
function 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 | ||
}) | ||
|
||
function Component() { | ||
const count = useAtomValue(derivedAtom) | ||
return <div>count: {count}</div> | ||
} | ||
|
||
function 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.