toiljs 0.0.15 → 0.0.19
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.babelrc +13 -13
- package/.gitattributes +2 -2
- package/.github/ISSUE_TEMPLATE/bug_report.md +38 -38
- package/.github/ISSUE_TEMPLATE/bug_report.yml +90 -90
- package/.github/ISSUE_TEMPLATE/config.yml +8 -8
- package/.github/ISSUE_TEMPLATE/feature_request.md +20 -20
- package/.github/PULL_REQUEST_TEMPLATE.md +43 -43
- package/.github/changelog-config.json +45 -45
- package/.github/dependabot.yml +27 -27
- package/.github/workflows/ci.yml +191 -191
- package/.prettierrc.json +11 -11
- package/.vscode/settings.json +9 -9
- package/CHANGELOG.md +116 -5
- package/LICENSE +187 -187
- package/README.md +524 -315
- package/as-pect.asconfig.json +34 -34
- package/as-pect.config.js +65 -65
- package/assets/logo.svg +36 -36
- package/build/backend/.tsbuildinfo +1 -1
- package/build/backend/index.d.ts +1 -0
- package/build/backend/index.js +20 -1
- package/build/cli/.tsbuildinfo +1 -1
- package/build/cli/index.js +1320 -696
- package/build/client/.tsbuildinfo +1 -1
- package/build/client/dev/devtools.d.ts +6 -0
- package/build/client/dev/devtools.js +479 -0
- package/build/client/dev/error-overlay.d.ts +9 -0
- package/build/client/dev/error-overlay.js +19 -4
- package/build/client/errors.d.ts +1 -0
- package/build/client/errors.js +3 -0
- package/build/client/index.d.ts +2 -0
- package/build/client/index.js +2 -0
- package/build/client/navigation/prefetch.d.ts +1 -0
- package/build/client/navigation/prefetch.js +35 -0
- package/build/client/routing/Router.js +1 -1
- package/build/client/routing/hooks.js +6 -2
- package/build/client/routing/loader.d.ts +23 -0
- package/build/client/routing/loader.js +53 -7
- package/build/client/routing/mount.js +4 -3
- package/build/client/rpc.d.ts +1 -0
- package/build/client/rpc.js +37 -0
- package/build/compiler/.tsbuildinfo +1 -1
- package/build/compiler/config.d.ts +16 -0
- package/build/compiler/config.js +9 -0
- package/build/compiler/docs.js +78 -21
- package/build/compiler/generate.js +5 -4
- package/build/compiler/index.d.ts +3 -2
- package/build/compiler/index.js +2 -2
- package/build/compiler/plugin.js +228 -0
- package/build/compiler/prerender.d.ts +1 -0
- package/build/compiler/prerender.js +1 -1
- package/build/compiler/seo.d.ts +1 -1
- package/build/compiler/seo.js +20 -5
- package/build/compiler/ssg.js +39 -2
- package/build/compiler/vite.js +25 -0
- package/build/io/.tsbuildinfo +1 -1
- package/build/io/codec.d.ts +54 -0
- package/build/io/codec.js +143 -0
- package/build/io/index.d.ts +1 -2
- package/build/io/index.js +1 -2
- package/build/logger/.tsbuildinfo +1 -1
- package/build/shared/.tsbuildinfo +1 -1
- package/eslint.config.js +48 -48
- package/examples/basic/client/404.tsx +11 -11
- package/examples/basic/client/components/.gitkeep +1 -1
- package/examples/basic/client/global-error.tsx +13 -13
- package/examples/basic/client/layout.tsx +25 -25
- package/examples/basic/client/public/images/.gitkeep +1 -1
- package/examples/basic/client/public/images/logo.svg +36 -36
- package/examples/basic/client/public/robots.txt +2 -2
- package/examples/basic/client/routes/docs/[...slug].tsx +12 -12
- package/examples/basic/client/routes/features/error/error.tsx +16 -16
- package/examples/basic/client/routes/features/index.tsx +1 -1
- package/examples/basic/client/routes/features/template/b.tsx +14 -14
- package/examples/basic/client/routes/files/[[...slug]].tsx +21 -21
- package/examples/basic/client/routes/gallery/layout.tsx +13 -13
- package/examples/basic/client/routes/io.tsx +23 -24
- package/examples/basic/client/routes/loader-demo/loading.tsx +13 -13
- package/examples/basic/client/routes/rest.tsx +74 -0
- package/examples/basic/client/routes/rpc.tsx +43 -0
- package/examples/basic/client/routes/search.tsx +61 -61
- package/examples/basic/client/toil.tsx +5 -5
- package/package.json +167 -148
- package/presets/eslint.js +88 -88
- package/presets/no-uint8array-tostring.js +200 -200
- package/presets/prettier-plugin.js +51 -0
- package/presets/prettier.json +19 -18
- package/presets/tsconfig.json +37 -37
- package/server/runtime/README.md +97 -0
- package/server/runtime/abort/abort.ts +27 -0
- package/server/runtime/env/Server.ts +61 -0
- package/server/runtime/envelope.ts +191 -0
- package/server/runtime/exports/index.ts +52 -0
- package/server/runtime/handlers/ToilHandler.ts +34 -0
- package/server/runtime/index.ts +26 -0
- package/server/runtime/lang/Potential.ts +5 -0
- package/server/runtime/memory.ts +81 -0
- package/server/runtime/request.ts +55 -0
- package/server/runtime/response.ts +86 -0
- package/server/runtime/rest/Rest.ts +39 -0
- package/server/runtime/rest/RestHandler.ts +20 -0
- package/server/runtime/rest/RouteContext.ts +82 -0
- package/server/runtime/rest/match.ts +48 -0
- package/server/runtime/tsconfig.json +7 -0
- package/src/backend/index.ts +202 -160
- package/src/cli/create.ts +15 -5
- package/src/cli/diagnostics.ts +81 -0
- package/src/cli/doctor.ts +384 -7
- package/src/cli/index.ts +11 -2
- package/src/cli/proc.ts +50 -50
- package/src/cli/updates.ts +69 -69
- package/src/cli/validate.ts +31 -31
- package/src/client/channel/channel.ts +146 -146
- package/src/client/components/Form.tsx +65 -65
- package/src/client/components/Script.tsx +113 -113
- package/src/client/components/Slot.tsx +21 -21
- package/src/client/dev/devtools.tsx +1018 -0
- package/src/client/dev/error-overlay.tsx +30 -4
- package/src/client/errors.ts +11 -0
- package/src/client/head/head.ts +167 -167
- package/src/client/head/metadata.ts +112 -112
- package/src/client/index.ts +91 -89
- package/src/client/navigation/NavLink.tsx +86 -86
- package/src/client/navigation/navigation.ts +235 -235
- package/src/client/navigation/prefetch.ts +169 -130
- package/src/client/navigation/scroll.ts +53 -53
- package/src/client/routing/Router.tsx +8 -2
- package/src/client/routing/action.ts +122 -122
- package/src/client/routing/error-boundary.tsx +43 -43
- package/src/client/routing/hooks.ts +21 -6
- package/src/client/routing/loader.ts +325 -235
- package/src/client/routing/match.ts +47 -47
- package/src/client/routing/mount.tsx +54 -52
- package/src/client/routing/params-context.ts +10 -10
- package/src/client/routing/slot-context.ts +7 -7
- package/src/client/rpc.ts +64 -0
- package/src/client/search/search.ts +189 -189
- package/src/client/search/use-page-search.ts +73 -73
- package/src/client/types.ts +73 -73
- package/src/compiler/config.ts +221 -182
- package/src/compiler/docs.ts +285 -228
- package/src/compiler/generate.ts +395 -394
- package/src/compiler/index.ts +66 -57
- package/src/compiler/pages.ts +70 -70
- package/src/compiler/plugin.ts +258 -2
- package/src/compiler/prerender.ts +156 -156
- package/src/compiler/seo.ts +417 -390
- package/src/compiler/ssg.ts +171 -126
- package/src/compiler/vite.ts +34 -0
- package/src/io/FastMap.ts +151 -127
- package/src/io/FastSet.ts +15 -1
- package/src/io/codec.ts +217 -0
- package/src/io/index.ts +10 -11
- package/src/io/lengths.ts +14 -14
- package/src/io/types.ts +19 -18
- package/src/logger/index.ts +22 -22
- package/src/shared/index.ts +10 -10
- package/std/client/index.d.ts +15 -15
- package/std/client/package.json +3 -3
- package/test/assembly/example.spec.ts +17 -7
- package/test/channel.test.ts +21 -21
- package/test/doctor.test.ts +65 -0
- package/test/dom/Link.test.tsx +47 -47
- package/test/dom/NavLink.test.tsx +37 -37
- package/test/dom/error-overlay.test.tsx +44 -44
- package/test/dom/loader.test.tsx +121 -121
- package/test/dom/navigation.test.ts +59 -59
- package/test/dom/revalidate.test.tsx +38 -38
- package/test/dom/route-head.test.tsx +78 -78
- package/test/dom/router-loading.test.tsx +44 -44
- package/test/dom/scroll.test.ts +56 -56
- package/test/dom/use-metadata.test.tsx +58 -58
- package/test/errors.test.ts +21 -0
- package/test/io.test.ts +117 -93
- package/test/navlink.test.ts +28 -28
- package/test/placeholder.test.ts +9 -9
- package/test/prettier-plugin.test.ts +46 -0
- package/test/routes.test.ts +76 -76
- package/test/rpc.test.ts +50 -0
- package/test/seo.test.ts +175 -164
- package/test/slot-layouts.test.ts +69 -69
- package/test/ssg.test.ts +36 -36
- package/test/update.test.ts +44 -44
- package/test/validate.test.ts +42 -42
- package/tests/data-parity/generated-parity.ts +99 -0
- package/tests/data-parity/parity.ts +80 -0
- package/tests/data-parity/spec.ts +46 -0
- package/toil-routes.d.ts +7 -0
- package/tsconfig.backend.json +13 -13
- package/tsconfig.base.json +35 -35
- package/tsconfig.cli.json +13 -13
- package/tsconfig.client.json +14 -14
- package/tsconfig.compiler.json +13 -13
- package/tsconfig.io.json +12 -12
- package/tsconfig.json +22 -22
- package/tsconfig.logger.json +12 -12
- package/tsconfig.server.json +10 -10
- package/tsconfig.shared.json +12 -12
- package/vitest.config.ts +26 -26
- package/.idea/codeStyles/Project.xml +0 -54
- package/.idea/codeStyles/codeStyleConfig.xml +0 -5
- package/.idea/inspectionProfiles/Project_Default.xml +0 -6
- package/.idea/modules.xml +0 -8
- package/.idea/prettier.xml +0 -7
- package/.idea/toiljs.iml +0 -8
- package/.idea/vcs.xml +0 -6
- package/.toil/entry.tsx +0 -9
- package/.toil/index.html +0 -12
- package/.toil/routes.ts +0 -9
- package/build/cli/configure.d.ts +0 -16
- package/build/cli/configure.js +0 -272
- package/build/cli/create.d.ts +0 -16
- package/build/cli/create.js +0 -420
- package/build/cli/diagnostics.d.ts +0 -55
- package/build/cli/diagnostics.js +0 -333
- package/build/cli/doctor.d.ts +0 -6
- package/build/cli/doctor.js +0 -249
- package/build/cli/features.d.ts +0 -25
- package/build/cli/features.js +0 -107
- package/build/cli/index.d.ts +0 -2
- package/build/cli/proc.d.ts +0 -6
- package/build/cli/proc.js +0 -31
- package/build/cli/ui.d.ts +0 -9
- package/build/cli/ui.js +0 -75
- package/build/cli/update.d.ts +0 -7
- package/build/cli/update.js +0 -117
- package/build/cli/updates.d.ts +0 -10
- package/build/cli/updates.js +0 -45
- package/build/cli/validate.d.ts +0 -4
- package/build/cli/validate.js +0 -19
- package/build/client/Link.d.ts +0 -8
- package/build/client/Link.js +0 -44
- package/build/client/NavLink.d.ts +0 -14
- package/build/client/NavLink.js +0 -37
- package/build/client/Router.d.ts +0 -7
- package/build/client/Router.js +0 -55
- package/build/client/channel.d.ts +0 -23
- package/build/client/channel.js +0 -94
- package/build/client/error-boundary.d.ts +0 -16
- package/build/client/error-boundary.js +0 -19
- package/build/client/head.d.ts +0 -26
- package/build/client/head.js +0 -87
- package/build/client/hooks.d.ts +0 -17
- package/build/client/hooks.js +0 -48
- package/build/client/lazy.d.ts +0 -16
- package/build/client/lazy.js +0 -53
- package/build/client/match.d.ts +0 -2
- package/build/client/match.js +0 -32
- package/build/client/mount.d.ts +0 -2
- package/build/client/mount.js +0 -13
- package/build/client/navigation.d.ts +0 -13
- package/build/client/navigation.js +0 -97
- package/build/client/params-context.d.ts +0 -2
- package/build/client/params-context.js +0 -2
- package/build/client/prefetch.d.ts +0 -11
- package/build/client/prefetch.js +0 -100
- package/build/client/runtime.d.ts +0 -31
- package/build/client/runtime.js +0 -112
- package/build/client/scroll.d.ts +0 -8
- package/build/client/scroll.js +0 -36
- package/build/io/BinaryReader.d.ts +0 -44
- package/build/io/BinaryReader.js +0 -244
- package/build/io/BinaryWriter.d.ts +0 -44
- package/build/io/BinaryWriter.js +0 -297
- package/build/server/release.wasm +0 -0
- package/build/server/release.wat +0 -9
- package/src/io/BinaryReader.ts +0 -340
- package/src/io/BinaryWriter.ts +0 -385
- package/src/server/index.ts +0 -10
- package/src/server/main.ts +0 -13
- package/src/server/tsconfig.json +0 -4
- package/toil-env.d.ts +0 -16
- package/toilconfig.json +0 -30
package/test/dom/Link.test.tsx
CHANGED
|
@@ -1,47 +1,47 @@
|
|
|
1
|
-
// @vitest-environment jsdom
|
|
2
|
-
import { cleanup, fireEvent, render } from '@testing-library/react';
|
|
3
|
-
import { afterEach, beforeEach, describe, expect, it } from 'vitest';
|
|
4
|
-
|
|
5
|
-
import { Link } from '../../src/client/navigation/Link';
|
|
6
|
-
|
|
7
|
-
afterEach(cleanup);
|
|
8
|
-
beforeEach(() => {
|
|
9
|
-
window.history.replaceState({}, '', '/');
|
|
10
|
-
});
|
|
11
|
-
|
|
12
|
-
describe('Link', () => {
|
|
13
|
-
it('intercepts a plain internal click (client nav, default prevented)', () => {
|
|
14
|
-
const { getByText } = render(<Link href="/about">go</Link>);
|
|
15
|
-
const prevented = !fireEvent.click(getByText('go'));
|
|
16
|
-
expect(prevented).toBe(true);
|
|
17
|
-
expect(window.location.pathname).toBe('/about');
|
|
18
|
-
});
|
|
19
|
-
|
|
20
|
-
it('does not intercept external links', () => {
|
|
21
|
-
const { getByText } = render(<Link href="https://example.com">ext</Link>);
|
|
22
|
-
const notPrevented = fireEvent.click(getByText('ext'));
|
|
23
|
-
expect(notPrevented).toBe(true);
|
|
24
|
-
expect(window.location.pathname).toBe('/');
|
|
25
|
-
});
|
|
26
|
-
|
|
27
|
-
it('does not intercept modified (cmd/ctrl) clicks', () => {
|
|
28
|
-
const { getByText } = render(<Link href="/about">go</Link>);
|
|
29
|
-
const notPrevented = fireEvent.click(getByText('go'), { metaKey: true });
|
|
30
|
-
expect(notPrevented).toBe(true);
|
|
31
|
-
expect(window.location.pathname).toBe('/');
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
it('forwards anchor attributes (rel, target)', () => {
|
|
35
|
-
const { getByText } = render(
|
|
36
|
-
<Link
|
|
37
|
-
href="https://x.com"
|
|
38
|
-
target="_blank"
|
|
39
|
-
rel="noopener noreferrer">
|
|
40
|
-
x
|
|
41
|
-
</Link>,
|
|
42
|
-
);
|
|
43
|
-
const a = getByText('x') as HTMLAnchorElement;
|
|
44
|
-
expect(a.getAttribute('rel')).toBe('noopener noreferrer');
|
|
45
|
-
expect(a.target).toBe('_blank');
|
|
46
|
-
});
|
|
47
|
-
});
|
|
1
|
+
// @vitest-environment jsdom
|
|
2
|
+
import { cleanup, fireEvent, render } from '@testing-library/react';
|
|
3
|
+
import { afterEach, beforeEach, describe, expect, it } from 'vitest';
|
|
4
|
+
|
|
5
|
+
import { Link } from '../../src/client/navigation/Link';
|
|
6
|
+
|
|
7
|
+
afterEach(cleanup);
|
|
8
|
+
beforeEach(() => {
|
|
9
|
+
window.history.replaceState({}, '', '/');
|
|
10
|
+
});
|
|
11
|
+
|
|
12
|
+
describe('Link', () => {
|
|
13
|
+
it('intercepts a plain internal click (client nav, default prevented)', () => {
|
|
14
|
+
const { getByText } = render(<Link href="/about">go</Link>);
|
|
15
|
+
const prevented = !fireEvent.click(getByText('go'));
|
|
16
|
+
expect(prevented).toBe(true);
|
|
17
|
+
expect(window.location.pathname).toBe('/about');
|
|
18
|
+
});
|
|
19
|
+
|
|
20
|
+
it('does not intercept external links', () => {
|
|
21
|
+
const { getByText } = render(<Link href="https://example.com">ext</Link>);
|
|
22
|
+
const notPrevented = fireEvent.click(getByText('ext'));
|
|
23
|
+
expect(notPrevented).toBe(true);
|
|
24
|
+
expect(window.location.pathname).toBe('/');
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
it('does not intercept modified (cmd/ctrl) clicks', () => {
|
|
28
|
+
const { getByText } = render(<Link href="/about">go</Link>);
|
|
29
|
+
const notPrevented = fireEvent.click(getByText('go'), { metaKey: true });
|
|
30
|
+
expect(notPrevented).toBe(true);
|
|
31
|
+
expect(window.location.pathname).toBe('/');
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('forwards anchor attributes (rel, target)', () => {
|
|
35
|
+
const { getByText } = render(
|
|
36
|
+
<Link
|
|
37
|
+
href="https://x.com"
|
|
38
|
+
target="_blank"
|
|
39
|
+
rel="noopener noreferrer">
|
|
40
|
+
x
|
|
41
|
+
</Link>,
|
|
42
|
+
);
|
|
43
|
+
const a = getByText('x') as HTMLAnchorElement;
|
|
44
|
+
expect(a.getAttribute('rel')).toBe('noopener noreferrer');
|
|
45
|
+
expect(a.target).toBe('_blank');
|
|
46
|
+
});
|
|
47
|
+
});
|
|
@@ -1,37 +1,37 @@
|
|
|
1
|
-
// @vitest-environment jsdom
|
|
2
|
-
import { cleanup, render } from '@testing-library/react';
|
|
3
|
-
import { afterEach, describe, expect, it } from 'vitest';
|
|
4
|
-
|
|
5
|
-
import { NavLink } from '../../src/client/navigation/NavLink';
|
|
6
|
-
|
|
7
|
-
afterEach(cleanup);
|
|
8
|
-
|
|
9
|
-
describe('NavLink', () => {
|
|
10
|
-
it('adds the active class + aria-current on the current route', () => {
|
|
11
|
-
window.history.replaceState({}, '', '/about');
|
|
12
|
-
const { getByText } = render(<NavLink href="/about">about</NavLink>);
|
|
13
|
-
const a = getByText('about') as HTMLAnchorElement;
|
|
14
|
-
expect(a.className).toContain('active');
|
|
15
|
-
expect(a.getAttribute('aria-current')).toBe('page');
|
|
16
|
-
});
|
|
17
|
-
|
|
18
|
-
it('is not active on a different route', () => {
|
|
19
|
-
window.history.replaceState({}, '', '/home');
|
|
20
|
-
const { getByText } = render(<NavLink href="/about">about</NavLink>);
|
|
21
|
-
const a = getByText('about') as HTMLAnchorElement;
|
|
22
|
-
expect(a.className).not.toContain('active');
|
|
23
|
-
expect(a.getAttribute('aria-current')).toBeNull();
|
|
24
|
-
});
|
|
25
|
-
|
|
26
|
-
it('supports a function className', () => {
|
|
27
|
-
window.history.replaceState({}, '', '/about');
|
|
28
|
-
const { getByText } = render(
|
|
29
|
-
<NavLink
|
|
30
|
-
href="/about"
|
|
31
|
-
className={({ isActive }) => (isActive ? 'on' : 'off')}>
|
|
32
|
-
about
|
|
33
|
-
</NavLink>,
|
|
34
|
-
);
|
|
35
|
-
expect((getByText('about') as HTMLAnchorElement).className).toBe('on');
|
|
36
|
-
});
|
|
37
|
-
});
|
|
1
|
+
// @vitest-environment jsdom
|
|
2
|
+
import { cleanup, render } from '@testing-library/react';
|
|
3
|
+
import { afterEach, describe, expect, it } from 'vitest';
|
|
4
|
+
|
|
5
|
+
import { NavLink } from '../../src/client/navigation/NavLink';
|
|
6
|
+
|
|
7
|
+
afterEach(cleanup);
|
|
8
|
+
|
|
9
|
+
describe('NavLink', () => {
|
|
10
|
+
it('adds the active class + aria-current on the current route', () => {
|
|
11
|
+
window.history.replaceState({}, '', '/about');
|
|
12
|
+
const { getByText } = render(<NavLink href="/about">about</NavLink>);
|
|
13
|
+
const a = getByText('about') as HTMLAnchorElement;
|
|
14
|
+
expect(a.className).toContain('active');
|
|
15
|
+
expect(a.getAttribute('aria-current')).toBe('page');
|
|
16
|
+
});
|
|
17
|
+
|
|
18
|
+
it('is not active on a different route', () => {
|
|
19
|
+
window.history.replaceState({}, '', '/home');
|
|
20
|
+
const { getByText } = render(<NavLink href="/about">about</NavLink>);
|
|
21
|
+
const a = getByText('about') as HTMLAnchorElement;
|
|
22
|
+
expect(a.className).not.toContain('active');
|
|
23
|
+
expect(a.getAttribute('aria-current')).toBeNull();
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
it('supports a function className', () => {
|
|
27
|
+
window.history.replaceState({}, '', '/about');
|
|
28
|
+
const { getByText } = render(
|
|
29
|
+
<NavLink
|
|
30
|
+
href="/about"
|
|
31
|
+
className={({ isActive }) => (isActive ? 'on' : 'off')}>
|
|
32
|
+
about
|
|
33
|
+
</NavLink>,
|
|
34
|
+
);
|
|
35
|
+
expect((getByText('about') as HTMLAnchorElement).className).toBe('on');
|
|
36
|
+
});
|
|
37
|
+
});
|
|
@@ -1,44 +1,44 @@
|
|
|
1
|
-
// @vitest-environment jsdom
|
|
2
|
-
import { act, cleanup, fireEvent, render } from '@testing-library/react';
|
|
3
|
-
import { afterEach, describe, expect, it, vi } from 'vitest';
|
|
4
|
-
|
|
5
|
-
import {
|
|
6
|
-
DevErrorBoundary,
|
|
7
|
-
DevErrorOverlay,
|
|
8
|
-
initDevErrorOverlay,
|
|
9
|
-
} from '../../src/client/dev/error-overlay';
|
|
10
|
-
|
|
11
|
-
afterEach(cleanup);
|
|
12
|
-
|
|
13
|
-
function Boom(): never {
|
|
14
|
-
throw new Error('render boom');
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
describe('dev error overlay', () => {
|
|
18
|
-
it('surfaces an uncaught render error', () => {
|
|
19
|
-
// React logs caught boundary errors to console.error, silence it for a clean test run.
|
|
20
|
-
const spy = vi.spyOn(console, 'error').mockImplementation(() => undefined);
|
|
21
|
-
const { getByRole } = render(
|
|
22
|
-
<>
|
|
23
|
-
<DevErrorBoundary>
|
|
24
|
-
<Boom />
|
|
25
|
-
</DevErrorBoundary>
|
|
26
|
-
<DevErrorOverlay />
|
|
27
|
-
</>,
|
|
28
|
-
);
|
|
29
|
-
expect(getByRole('alert').textContent).toContain('render boom');
|
|
30
|
-
spy.mockRestore();
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
it('surfaces an unhandled window error and dismisses it', async () => {
|
|
34
|
-
initDevErrorOverlay();
|
|
35
|
-
const { findByRole, queryByRole, getByText } = render(<DevErrorOverlay />);
|
|
36
|
-
act(() => {
|
|
37
|
-
window.dispatchEvent(new ErrorEvent('error', { error: new Error('async boom') }));
|
|
38
|
-
});
|
|
39
|
-
const alert = await findByRole('alert');
|
|
40
|
-
expect(alert.textContent).toContain('async boom');
|
|
41
|
-
fireEvent.click(getByText('Dismiss'));
|
|
42
|
-
expect(queryByRole('alert')).toBeNull();
|
|
43
|
-
});
|
|
44
|
-
});
|
|
1
|
+
// @vitest-environment jsdom
|
|
2
|
+
import { act, cleanup, fireEvent, render } from '@testing-library/react';
|
|
3
|
+
import { afterEach, describe, expect, it, vi } from 'vitest';
|
|
4
|
+
|
|
5
|
+
import {
|
|
6
|
+
DevErrorBoundary,
|
|
7
|
+
DevErrorOverlay,
|
|
8
|
+
initDevErrorOverlay,
|
|
9
|
+
} from '../../src/client/dev/error-overlay';
|
|
10
|
+
|
|
11
|
+
afterEach(cleanup);
|
|
12
|
+
|
|
13
|
+
function Boom(): never {
|
|
14
|
+
throw new Error('render boom');
|
|
15
|
+
}
|
|
16
|
+
|
|
17
|
+
describe('dev error overlay', () => {
|
|
18
|
+
it('surfaces an uncaught render error', () => {
|
|
19
|
+
// React logs caught boundary errors to console.error, silence it for a clean test run.
|
|
20
|
+
const spy = vi.spyOn(console, 'error').mockImplementation(() => undefined);
|
|
21
|
+
const { getByRole } = render(
|
|
22
|
+
<>
|
|
23
|
+
<DevErrorBoundary>
|
|
24
|
+
<Boom />
|
|
25
|
+
</DevErrorBoundary>
|
|
26
|
+
<DevErrorOverlay />
|
|
27
|
+
</>,
|
|
28
|
+
);
|
|
29
|
+
expect(getByRole('alert').textContent).toContain('render boom');
|
|
30
|
+
spy.mockRestore();
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
it('surfaces an unhandled window error and dismisses it', async () => {
|
|
34
|
+
initDevErrorOverlay();
|
|
35
|
+
const { findByRole, queryByRole, getByText } = render(<DevErrorOverlay />);
|
|
36
|
+
act(() => {
|
|
37
|
+
window.dispatchEvent(new ErrorEvent('error', { error: new Error('async boom') }));
|
|
38
|
+
});
|
|
39
|
+
const alert = await findByRole('alert');
|
|
40
|
+
expect(alert.textContent).toContain('async boom');
|
|
41
|
+
fireEvent.click(getByText('Dismiss'));
|
|
42
|
+
expect(queryByRole('alert')).toBeNull();
|
|
43
|
+
});
|
|
44
|
+
});
|
package/test/dom/loader.test.tsx
CHANGED
|
@@ -1,121 +1,121 @@
|
|
|
1
|
-
// @vitest-environment jsdom
|
|
2
|
-
import { afterEach, beforeEach, describe, expect, expectTypeOf, it, vi } from 'vitest';
|
|
3
|
-
|
|
4
|
-
import type { Revalidate } from '../../src/client/routing/loader';
|
|
5
|
-
import {
|
|
6
|
-
clearLoaderData,
|
|
7
|
-
invalidateLoaderData,
|
|
8
|
-
type LoaderData,
|
|
9
|
-
loaderKey,
|
|
10
|
-
readRouteData,
|
|
11
|
-
} from '../../src/client/routing/loader';
|
|
12
|
-
import type { RouteDef } from '../../src/client/types';
|
|
13
|
-
|
|
14
|
-
/** Reads route data, awaiting the suspending promise once if it's pending. */
|
|
15
|
-
async function read(route: RouteDef, key: string, epoch: number): Promise<unknown> {
|
|
16
|
-
try {
|
|
17
|
-
return readRouteData(route, {}, key, epoch).data;
|
|
18
|
-
} catch (thrown) {
|
|
19
|
-
if (thrown instanceof Promise) {
|
|
20
|
-
await thrown;
|
|
21
|
-
return readRouteData(route, {}, key, epoch).data;
|
|
22
|
-
}
|
|
23
|
-
throw thrown;
|
|
24
|
-
}
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
/** A route whose `load()` count tells us how many times the loader actually ran. */
|
|
28
|
-
function makeRoute(revalidate?: Revalidate): { route: RouteDef; loads: () => number } {
|
|
29
|
-
let loads = 0;
|
|
30
|
-
const route: RouteDef = {
|
|
31
|
-
pattern: '/x',
|
|
32
|
-
load: () => {
|
|
33
|
-
loads += 1;
|
|
34
|
-
return Promise.resolve({
|
|
35
|
-
default: () => null,
|
|
36
|
-
loader: () => ({ n: loads }),
|
|
37
|
-
revalidate,
|
|
38
|
-
});
|
|
39
|
-
},
|
|
40
|
-
};
|
|
41
|
-
return { route, loads: () => loads };
|
|
42
|
-
}
|
|
43
|
-
|
|
44
|
-
beforeEach(() => {
|
|
45
|
-
clearLoaderData();
|
|
46
|
-
});
|
|
47
|
-
afterEach(() => {
|
|
48
|
-
vi.useRealTimers();
|
|
49
|
-
});
|
|
50
|
-
|
|
51
|
-
describe('loader caching', () => {
|
|
52
|
-
it('reuses cached data on re-read within the same navigation (loader runs once)', async () => {
|
|
53
|
-
const { route, loads } = makeRoute();
|
|
54
|
-
const key = loaderKey('/x', '');
|
|
55
|
-
await read(route, key, 1);
|
|
56
|
-
await read(route, key, 1);
|
|
57
|
-
expect(loads()).toBe(1);
|
|
58
|
-
});
|
|
59
|
-
|
|
60
|
-
it('a route with no loader stays cached across navigations (never re-suspends)', async () => {
|
|
61
|
-
let loads = 0;
|
|
62
|
-
const route: RouteDef = {
|
|
63
|
-
pattern: '/p',
|
|
64
|
-
load: () => {
|
|
65
|
-
loads += 1;
|
|
66
|
-
return Promise.resolve({ default: () => null });
|
|
67
|
-
},
|
|
68
|
-
};
|
|
69
|
-
const key = loaderKey('/p', '');
|
|
70
|
-
await read(route, key, 1);
|
|
71
|
-
await read(route, key, 2);
|
|
72
|
-
await read(route, key, 3);
|
|
73
|
-
expect(loads).toBe(1);
|
|
74
|
-
});
|
|
75
|
-
|
|
76
|
-
it('refetches on a new navigation under the default policy', async () => {
|
|
77
|
-
const { route, loads } = makeRoute();
|
|
78
|
-
const key = loaderKey('/x', '');
|
|
79
|
-
await read(route, key, 1);
|
|
80
|
-
await read(route, key, 2);
|
|
81
|
-
expect(loads()).toBe(2);
|
|
82
|
-
});
|
|
83
|
-
|
|
84
|
-
it('revalidate=false caches across navigations', async () => {
|
|
85
|
-
const { route, loads } = makeRoute(false);
|
|
86
|
-
const key = loaderKey('/x', '');
|
|
87
|
-
await read(route, key, 1);
|
|
88
|
-
await read(route, key, 2);
|
|
89
|
-
expect(loads()).toBe(1);
|
|
90
|
-
});
|
|
91
|
-
|
|
92
|
-
it('numeric revalidate caches until the staleTime elapses', async () => {
|
|
93
|
-
vi.useFakeTimers();
|
|
94
|
-
const { route, loads } = makeRoute(1); // 1 second
|
|
95
|
-
const key = loaderKey('/x', '');
|
|
96
|
-
await read(route, key, 1);
|
|
97
|
-
await read(route, key, 2); // still fresh
|
|
98
|
-
expect(loads()).toBe(1);
|
|
99
|
-
vi.advanceTimersByTime(1500); // now stale
|
|
100
|
-
await read(route, key, 3);
|
|
101
|
-
expect(loads()).toBe(2);
|
|
102
|
-
});
|
|
103
|
-
|
|
104
|
-
it('invalidateLoaderData(href) forces a refetch of that route', async () => {
|
|
105
|
-
const { route, loads } = makeRoute(false);
|
|
106
|
-
const key = loaderKey('/x', '');
|
|
107
|
-
await read(route, key, 1);
|
|
108
|
-
invalidateLoaderData('/x');
|
|
109
|
-
await read(route, key, 1);
|
|
110
|
-
expect(loads()).toBe(2);
|
|
111
|
-
});
|
|
112
|
-
});
|
|
113
|
-
|
|
114
|
-
describe('useLoaderData type inference', () => {
|
|
115
|
-
it('LoaderData<typeof loader> resolves to the loader return type', () => {
|
|
116
|
-
const loader = async () => Promise.resolve({ a: 1, b: 'x' as string | null });
|
|
117
|
-
expectTypeOf<LoaderData<typeof loader>>().toEqualTypeOf<{ a: number; b: string | null }>();
|
|
118
|
-
// An explicit type still passes straight through.
|
|
119
|
-
expectTypeOf<LoaderData<{ id: number }>>().toEqualTypeOf<{ id: number }>();
|
|
120
|
-
});
|
|
121
|
-
});
|
|
1
|
+
// @vitest-environment jsdom
|
|
2
|
+
import { afterEach, beforeEach, describe, expect, expectTypeOf, it, vi } from 'vitest';
|
|
3
|
+
|
|
4
|
+
import type { Revalidate } from '../../src/client/routing/loader';
|
|
5
|
+
import {
|
|
6
|
+
clearLoaderData,
|
|
7
|
+
invalidateLoaderData,
|
|
8
|
+
type LoaderData,
|
|
9
|
+
loaderKey,
|
|
10
|
+
readRouteData,
|
|
11
|
+
} from '../../src/client/routing/loader';
|
|
12
|
+
import type { RouteDef } from '../../src/client/types';
|
|
13
|
+
|
|
14
|
+
/** Reads route data, awaiting the suspending promise once if it's pending. */
|
|
15
|
+
async function read(route: RouteDef, key: string, epoch: number): Promise<unknown> {
|
|
16
|
+
try {
|
|
17
|
+
return readRouteData(route, {}, key, epoch).data;
|
|
18
|
+
} catch (thrown) {
|
|
19
|
+
if (thrown instanceof Promise) {
|
|
20
|
+
await thrown;
|
|
21
|
+
return readRouteData(route, {}, key, epoch).data;
|
|
22
|
+
}
|
|
23
|
+
throw thrown;
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
/** A route whose `load()` count tells us how many times the loader actually ran. */
|
|
28
|
+
function makeRoute(revalidate?: Revalidate): { route: RouteDef; loads: () => number } {
|
|
29
|
+
let loads = 0;
|
|
30
|
+
const route: RouteDef = {
|
|
31
|
+
pattern: '/x',
|
|
32
|
+
load: () => {
|
|
33
|
+
loads += 1;
|
|
34
|
+
return Promise.resolve({
|
|
35
|
+
default: () => null,
|
|
36
|
+
loader: () => ({ n: loads }),
|
|
37
|
+
revalidate,
|
|
38
|
+
});
|
|
39
|
+
},
|
|
40
|
+
};
|
|
41
|
+
return { route, loads: () => loads };
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
beforeEach(() => {
|
|
45
|
+
clearLoaderData();
|
|
46
|
+
});
|
|
47
|
+
afterEach(() => {
|
|
48
|
+
vi.useRealTimers();
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
describe('loader caching', () => {
|
|
52
|
+
it('reuses cached data on re-read within the same navigation (loader runs once)', async () => {
|
|
53
|
+
const { route, loads } = makeRoute();
|
|
54
|
+
const key = loaderKey('/x', '');
|
|
55
|
+
await read(route, key, 1);
|
|
56
|
+
await read(route, key, 1);
|
|
57
|
+
expect(loads()).toBe(1);
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
it('a route with no loader stays cached across navigations (never re-suspends)', async () => {
|
|
61
|
+
let loads = 0;
|
|
62
|
+
const route: RouteDef = {
|
|
63
|
+
pattern: '/p',
|
|
64
|
+
load: () => {
|
|
65
|
+
loads += 1;
|
|
66
|
+
return Promise.resolve({ default: () => null });
|
|
67
|
+
},
|
|
68
|
+
};
|
|
69
|
+
const key = loaderKey('/p', '');
|
|
70
|
+
await read(route, key, 1);
|
|
71
|
+
await read(route, key, 2);
|
|
72
|
+
await read(route, key, 3);
|
|
73
|
+
expect(loads).toBe(1);
|
|
74
|
+
});
|
|
75
|
+
|
|
76
|
+
it('refetches on a new navigation under the default policy', async () => {
|
|
77
|
+
const { route, loads } = makeRoute();
|
|
78
|
+
const key = loaderKey('/x', '');
|
|
79
|
+
await read(route, key, 1);
|
|
80
|
+
await read(route, key, 2);
|
|
81
|
+
expect(loads()).toBe(2);
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
it('revalidate=false caches across navigations', async () => {
|
|
85
|
+
const { route, loads } = makeRoute(false);
|
|
86
|
+
const key = loaderKey('/x', '');
|
|
87
|
+
await read(route, key, 1);
|
|
88
|
+
await read(route, key, 2);
|
|
89
|
+
expect(loads()).toBe(1);
|
|
90
|
+
});
|
|
91
|
+
|
|
92
|
+
it('numeric revalidate caches until the staleTime elapses', async () => {
|
|
93
|
+
vi.useFakeTimers();
|
|
94
|
+
const { route, loads } = makeRoute(1); // 1 second
|
|
95
|
+
const key = loaderKey('/x', '');
|
|
96
|
+
await read(route, key, 1);
|
|
97
|
+
await read(route, key, 2); // still fresh
|
|
98
|
+
expect(loads()).toBe(1);
|
|
99
|
+
vi.advanceTimersByTime(1500); // now stale
|
|
100
|
+
await read(route, key, 3);
|
|
101
|
+
expect(loads()).toBe(2);
|
|
102
|
+
});
|
|
103
|
+
|
|
104
|
+
it('invalidateLoaderData(href) forces a refetch of that route', async () => {
|
|
105
|
+
const { route, loads } = makeRoute(false);
|
|
106
|
+
const key = loaderKey('/x', '');
|
|
107
|
+
await read(route, key, 1);
|
|
108
|
+
invalidateLoaderData('/x');
|
|
109
|
+
await read(route, key, 1);
|
|
110
|
+
expect(loads()).toBe(2);
|
|
111
|
+
});
|
|
112
|
+
});
|
|
113
|
+
|
|
114
|
+
describe('useLoaderData type inference', () => {
|
|
115
|
+
it('LoaderData<typeof loader> resolves to the loader return type', () => {
|
|
116
|
+
const loader = async () => Promise.resolve({ a: 1, b: 'x' as string | null });
|
|
117
|
+
expectTypeOf<LoaderData<typeof loader>>().toEqualTypeOf<{ a: number; b: string | null }>();
|
|
118
|
+
// An explicit type still passes straight through.
|
|
119
|
+
expectTypeOf<LoaderData<{ id: number }>>().toEqualTypeOf<{ id: number }>();
|
|
120
|
+
});
|
|
121
|
+
});
|
|
@@ -1,59 +1,59 @@
|
|
|
1
|
-
// @vitest-environment jsdom
|
|
2
|
-
import { beforeEach, describe, expect, it } from 'vitest';
|
|
3
|
-
|
|
4
|
-
import {
|
|
5
|
-
back,
|
|
6
|
-
isNavigationPending,
|
|
7
|
-
navigate,
|
|
8
|
-
navigationEpoch,
|
|
9
|
-
settleNavigation,
|
|
10
|
-
subscribeLocation,
|
|
11
|
-
} from '../../src/client/navigation/navigation';
|
|
12
|
-
|
|
13
|
-
beforeEach(() => {
|
|
14
|
-
window.history.replaceState({}, '', '/');
|
|
15
|
-
});
|
|
16
|
-
|
|
17
|
-
describe('navigate', () => {
|
|
18
|
-
it('updates the location and notifies subscribers', () => {
|
|
19
|
-
let calls = 0;
|
|
20
|
-
const off = subscribeLocation(() => {
|
|
21
|
-
calls += 1;
|
|
22
|
-
});
|
|
23
|
-
navigate('/about');
|
|
24
|
-
expect(window.location.pathname).toBe('/about');
|
|
25
|
-
expect(calls).toBe(1);
|
|
26
|
-
off();
|
|
27
|
-
});
|
|
28
|
-
|
|
29
|
-
it('replace updates the location too', () => {
|
|
30
|
-
navigate('/replaced', { replace: true });
|
|
31
|
-
expect(window.location.pathname).toBe('/replaced');
|
|
32
|
-
});
|
|
33
|
-
|
|
34
|
-
it('increments the navigation epoch', () => {
|
|
35
|
-
const before = navigationEpoch();
|
|
36
|
-
navigate('/a');
|
|
37
|
-
expect(navigationEpoch()).toBe(before + 1);
|
|
38
|
-
});
|
|
39
|
-
|
|
40
|
-
it('is pending after navigate, settled after settleNavigation', () => {
|
|
41
|
-
navigate('/pending');
|
|
42
|
-
expect(isNavigationPending()).toBe(true);
|
|
43
|
-
settleNavigation();
|
|
44
|
-
expect(isNavigationPending()).toBe(false);
|
|
45
|
-
});
|
|
46
|
-
|
|
47
|
-
it('back() triggers a notification', () => {
|
|
48
|
-
navigate('/one');
|
|
49
|
-
navigate('/two');
|
|
50
|
-
let popped = 0;
|
|
51
|
-
const off = subscribeLocation(() => {
|
|
52
|
-
popped += 1;
|
|
53
|
-
});
|
|
54
|
-
back();
|
|
55
|
-
// jsdom fires popstate synchronously for history.back within the same task.
|
|
56
|
-
expect(popped).toBeGreaterThanOrEqual(0);
|
|
57
|
-
off();
|
|
58
|
-
});
|
|
59
|
-
});
|
|
1
|
+
// @vitest-environment jsdom
|
|
2
|
+
import { beforeEach, describe, expect, it } from 'vitest';
|
|
3
|
+
|
|
4
|
+
import {
|
|
5
|
+
back,
|
|
6
|
+
isNavigationPending,
|
|
7
|
+
navigate,
|
|
8
|
+
navigationEpoch,
|
|
9
|
+
settleNavigation,
|
|
10
|
+
subscribeLocation,
|
|
11
|
+
} from '../../src/client/navigation/navigation';
|
|
12
|
+
|
|
13
|
+
beforeEach(() => {
|
|
14
|
+
window.history.replaceState({}, '', '/');
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
describe('navigate', () => {
|
|
18
|
+
it('updates the location and notifies subscribers', () => {
|
|
19
|
+
let calls = 0;
|
|
20
|
+
const off = subscribeLocation(() => {
|
|
21
|
+
calls += 1;
|
|
22
|
+
});
|
|
23
|
+
navigate('/about');
|
|
24
|
+
expect(window.location.pathname).toBe('/about');
|
|
25
|
+
expect(calls).toBe(1);
|
|
26
|
+
off();
|
|
27
|
+
});
|
|
28
|
+
|
|
29
|
+
it('replace updates the location too', () => {
|
|
30
|
+
navigate('/replaced', { replace: true });
|
|
31
|
+
expect(window.location.pathname).toBe('/replaced');
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
it('increments the navigation epoch', () => {
|
|
35
|
+
const before = navigationEpoch();
|
|
36
|
+
navigate('/a');
|
|
37
|
+
expect(navigationEpoch()).toBe(before + 1);
|
|
38
|
+
});
|
|
39
|
+
|
|
40
|
+
it('is pending after navigate, settled after settleNavigation', () => {
|
|
41
|
+
navigate('/pending');
|
|
42
|
+
expect(isNavigationPending()).toBe(true);
|
|
43
|
+
settleNavigation();
|
|
44
|
+
expect(isNavigationPending()).toBe(false);
|
|
45
|
+
});
|
|
46
|
+
|
|
47
|
+
it('back() triggers a notification', () => {
|
|
48
|
+
navigate('/one');
|
|
49
|
+
navigate('/two');
|
|
50
|
+
let popped = 0;
|
|
51
|
+
const off = subscribeLocation(() => {
|
|
52
|
+
popped += 1;
|
|
53
|
+
});
|
|
54
|
+
back();
|
|
55
|
+
// jsdom fires popstate synchronously for history.back within the same task.
|
|
56
|
+
expect(popped).toBeGreaterThanOrEqual(0);
|
|
57
|
+
off();
|
|
58
|
+
});
|
|
59
|
+
});
|