Skip to content

Commit 62de3b4

Browse files
author
Alex Lohr
committed
align functionality with original:
- add wrapper support - add asFragment method - add renderHook function improve documentation + tests
1 parent 3f12511 commit 62de3b4

10 files changed

+102
-41
lines changed

.babelrc

-7
This file was deleted.

README.md

+36-8
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
1+
<h1>
2+
<img width="100%" src="https://assets.solidjs.com/banner?type=core&project=solid-testing-library" alt="Solid Testing Library">
3+
</h1>
14
<div align="center">
2-
<h1>Solid Testing Library</h1>
3-
45
<p>Simple and complete Solid DOM testing utilities that encourage good testing
56
practices.</p>
67

@@ -13,14 +14,18 @@ practices.</p>
1314

1415
</div>
1516

16-
<hr />
17+
---
1718

1819
## Table of Contents
1920

2021
- [The Problem](#the-problem)
2122
- [The Solution](#the-solution)
2223
- [Installation](#installation)
2324
- [Docs](#docs)
25+
- [Issues](#issues)
26+
- [Acknowledgement](#acknowledgment)
27+
28+
---
2429

2530
## The Problem
2631

@@ -38,16 +43,39 @@ The Solid Testing Library is a very lightweight solution for testing Solid compo
3843
This module is distributed via npm which is bundled with node and should be installed
3944
as one of your project's `devDependencies`:
4045

41-
```
46+
```sh
4247
npm install --save-dev solid-testing-library
4348
```
49+
4450
If you using Jest we recommend using [solid-jest](https://github.com/solidjs/solid-jest) to properly resolve the browser version of Solid as Jest will default to the server version when run in Node.
4551

46-
💡 You may also be interested in installing `@testing-library/jest-dom` so you can use
52+
💡 If you are using Jest or vitest, you may also be interested in installing `@testing-library/jest-dom` so you can use
4753
[the custom jest matchers](https://github.com/testing-library/jest-dom).
4854

4955
## Docs
5056

51-
TODO
52-
<!-- See the [docs](https://testing-library.com/docs/preact-testing-library/intro) over at the Testing
53-
Library website. -->
57+
See the [docs](https://testing-library.com/docs/preact-testing-library/intro) over at the Testing Library website.
58+
59+
There are two key difference, though:
60+
61+
⚠️ Solid.js does *not* rerender, it merely executes side effects triggered by reactive state that change the DOM, therefore there is no rerender method. You can use global signals to manipulate your test component in a way that causes it to update.
62+
63+
Solid.js' reactive changes are pretty instantaneous, so there is rarely need to use `waitFor(…)` or `await findByRole(…)` - with the exception of transitions and Suspense.
64+
65+
⚠️ Solid.js external reactive state does not require any DOM elements to run in, so our `renderHook` call has no `container`, `baseElement` or queries in its options or return value. Instead, it has an `owner` to be used with [`runWithOwner`](https://www.solidjs.com/docs/latest/api#runwithowner) if required. It also exposes a `cleanup` function, though this is already automatically called after the test is finished.
66+
67+
## Issues
68+
69+
If you find any issues, please [check on the issues page](https://github.com/solidjs/solid-testing-library/issues) if they are already known. If not, opening an issue will be much appreciated, even more so if it contains a
70+
71+
- short description
72+
- minimal reproduction code
73+
- list of possible workarounds, if there are any
74+
75+
If you think you can fix an issue yourself, feel free to [open a pull-request](https://github.com/solidjs/solid-testing-library/pulls). If functionality changes, please don't forget to add or adapt tests.
76+
77+
## Acknowledgment
78+
79+
Thanks goes to [Kent C. Dodds](https://kentcdodds.com/) and his colleagues for creating testing-library and to the creators of [preact-testing-library](https://github.com/testing-library/preact-testing-library).
80+
81+
This library has been created by Ryan Carniato and is currently maintained by Alex Lohr.

package.json

+5-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
{
22
"name": "solid-testing-library",
3-
"version": "0.4.0",
3+
"version": "0.5.0",
44
"description": "Simple and complete Solid testing utilities that encourage good testing practices.",
55
"type": "module",
66
"main": "./dist/index.js",
@@ -14,7 +14,9 @@
1414
},
1515
"license": "MIT",
1616
"author": "Ryan Carniato",
17-
"maintainers": ["Alex Lohr"],
17+
"maintainers": [
18+
"Alex Lohr"
19+
],
1820
"homepage": "https://github.com/solidjs/solid-testing-library#readme",
1921
"repository": {
2022
"type": "git",
@@ -49,6 +51,7 @@
4951
"test:watch": "npm test --watch",
5052
"test:coverage": "npm test -- --coverage",
5153
"setup": "npm install && npm run validate",
54+
"prettier": "prettier -w src/**/* ./*.json ./vitest.config.js",
5255
"validate": "npm run typecheck && npm run test:coverage && npm run build",
5356
"report:coverage": "cat ./coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js && rm -rf ./coverage"
5457
},

src/__tests__/basic.tsx

+20-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import "@testing-library/jest-dom/extend-expect";
22
import { createSignal, createEffect } from "solid-js";
3-
import { render, screen } from "..";
3+
import { render, renderHook, screen } from "..";
44
import userEvent from "@testing-library/user-event";
55

66
declare global {
@@ -71,3 +71,22 @@ test("queries should not return elements outside of the container", () => {
7171
container.parentNode!.insertBefore(falseContainer, getAllByText("Some text...")[0].parentNode);
7272
expect(getAllByText("Some text...")[0] === container.childNodes[0]).toBe(true);
7373
});
74+
75+
test("wrapper option works correctly", () => {
76+
const { asFragment } = render(() => <div>Component</div>, {
77+
wrapper: props => <div>Wrapper {props.children}</div>
78+
});
79+
expect(asFragment()).toBe("<div>Wrapper <div>Component</div></div>");
80+
});
81+
82+
test("renderHook works correctly", () => {
83+
const createDate = () => {
84+
const [date, setDate] = createSignal(new Date());
85+
return [date, (d: Date) => d ? setDate(d) : setDate(new Date())] as const;
86+
}
87+
const { result: [ date, setDate ] } = renderHook(createDate);
88+
expect(date()).toBeInstanceOf(Date);
89+
const newDate = new Date();
90+
setDate(newDate);
91+
expect(date()).toBe(newDate);
92+
});

src/__tests__/debug.tsx

+3-3
Original file line numberDiff line numberDiff line change
@@ -29,9 +29,9 @@ test("debug pretty prints multiple containers", () => {
2929
</>
3030
);
3131

32-
render(() => <HelloWorld />);
33-
const multipleElements = screen.getAllByTestId("testId");
34-
screen.debug(multipleElements);
32+
const { debug, getAllByTestId } = render(() => <HelloWorld />);
33+
const multipleElements = getAllByTestId("testId");
34+
debug(multipleElements);
3535
expect(console.log).toHaveBeenCalledTimes(2);
3636
expect(console.log).toHaveBeenCalledWith(expect.stringContaining("Hello World"));
3737
});

src/__tests__/tsconfig.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -4,4 +4,4 @@
44
"types": ["vitest/globals", "@testing-library/jest-dom"]
55
},
66
"include": ["./*.tsx"]
7-
}
7+
}

src/index.ts

+20-14
Original file line numberDiff line numberDiff line change
@@ -1,30 +1,22 @@
11
import { getQueriesForElement, prettyDOM } from "@testing-library/dom";
2+
import { createRoot, getOwner } from "solid-js";
23
import { hydrate as solidHydrate, render as solidRender } from "solid-js/web";
34

4-
import type { Ui, Result, Options, Ref } from "./types";
5-
6-
/**
7-
* Shim for the teardown function
8-
*/
9-
declare var teardown: Function | undefined;
5+
import type { Ui, Result, Options, Ref, RenderHookResult, RenderHookOptions } from "./types";
106

117
/* istanbul ignore next */
128
if (!process.env.STL_SKIP_AUTO_CLEANUP) {
139
if (typeof afterEach === "function") {
1410
afterEach(async () => {
1511
await cleanup();
1612
});
17-
} else if (typeof teardown === "function") {
18-
teardown(async () => {
19-
await cleanup();
20-
});
2113
}
2214
}
2315

2416
const mountedContainers = new Set<Ref>();
2517

2618
function render(ui: Ui, options: Options = {}): Result {
27-
let { container, baseElement = container, queries, hydrate = false } = options;
19+
let { container, baseElement = container, queries, hydrate = false, wrapper } = options;
2820

2921
if (!baseElement) {
3022
// Default to document.body instead of documentElement to avoid output of potentially-large
@@ -36,9 +28,13 @@ function render(ui: Ui, options: Options = {}): Result {
3628
container = baseElement.appendChild(document.createElement("div"));
3729
}
3830

31+
const wrappedUi: Ui = typeof wrapper === 'function'
32+
? () => wrapper!({ children: ui() })
33+
: ui;
34+
3935
const dispose = hydrate
40-
? ((solidHydrate(ui, container) as unknown) as () => void)
41-
: solidRender(ui, container);
36+
? ((solidHydrate(wrappedUi, container) as unknown) as () => void)
37+
: solidRender(wrappedUi, container);
4238

4339
// We'll add it to the mounted containers regardless of whether it's actually
4440
// added to document.body so the cleanup method works regardless of whether
@@ -48,6 +44,7 @@ function render(ui: Ui, options: Options = {}): Result {
4844
const queryHelpers = getQueriesForElement(container, queries)
4945

5046
return {
47+
asFragment: () => container?.innerHTML as string,
5148
container,
5249
baseElement,
5350
debug: (el = baseElement, maxLength, options) =>
@@ -59,11 +56,20 @@ function render(ui: Ui, options: Options = {}): Result {
5956
} as Result;
6057
}
6158

59+
export function renderHook<A extends any[], R>(hook: (...args: A) => R, options?: RenderHookOptions<A>): RenderHookResult<R> {
60+
const initialProps: A | [] = Array.isArray(options) ? options : options?.initialProps || [];
61+
const [dispose, owner, result] = createRoot((dispose) => [dispose, getOwner(), hook(...initialProps as A)]);
62+
63+
mountedContainers.add({ dispose });
64+
65+
return { result, cleanup: dispose, owner };
66+
}
67+
6268
function cleanupAtContainer(ref: Ref) {
6369
const { container, dispose } = ref;
6470
dispose();
6571

66-
if (container.parentNode === document.body) {
72+
if (container?.parentNode === document.body) {
6773
document.body.removeChild(container);
6874
}
6975

src/types.ts

+15-3
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
1-
import type { JSX } from "solid-js";
1+
import type { Component, JSX, Owner } from "solid-js";
22
import { queries } from "@testing-library/dom";
33
import type { Queries, BoundFunctions, prettyFormat } from "@testing-library/dom";
44

55
export interface Ref {
6-
container: HTMLElement;
6+
container?: HTMLElement;
77
dispose: () => void;
88
}
99

@@ -14,6 +14,7 @@ export interface Options {
1414
baseElement?: HTMLElement;
1515
queries?: Queries & typeof queries;
1616
hydrate?: boolean;
17+
wrapper?: Component<{ children: JSX.Element }>;
1718
}
1819

1920
export type DebugFn = (
@@ -23,8 +24,19 @@ export type DebugFn = (
2324
) => void;
2425

2526
export type Result = BoundFunctions<typeof queries> & {
27+
asFragment: () => string;
2628
container: HTMLElement;
2729
baseElement: HTMLElement;
2830
debug: DebugFn;
2931
unmount: () => void;
30-
} ;
32+
};
33+
34+
export type RenderHookOptions<A extends any[]> = {
35+
initialProps?: A
36+
} | A;
37+
38+
export type RenderHookResult<R> = {
39+
result: R;
40+
owner: Owner | null;
41+
cleanup: () => void;
42+
};

tsconfig.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -13,5 +13,5 @@
1313
"outDir": "./dist",
1414
"module": "commonjs"
1515
},
16-
"include": ["src/index.ts", "src/types.ts"],
16+
"include": ["src/index.ts", "src/types.ts"]
1717
}

vitest.config.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ export default defineConfig({
1414
transformMode: {
1515
web: [/\.[jt]sx?$/]
1616
},
17-
include: 'src/__tests__/*.tsx'
17+
include: "src/__tests__/*.tsx"
1818
},
1919
resolve: {
2020
conditions: ["browser", "development"]

0 commit comments

Comments
 (0)