mirror of
https://github.com/pmndrs/zustand.git
synced 2025-12-08 19:45:52 +00:00
test: incorporate eslint-plugin-testing-library and fix reported issues (#2844)
This commit is contained in:
parent
9782058029
commit
f689d78a31
@ -21,6 +21,7 @@
|
||||
"react-hooks",
|
||||
"import",
|
||||
"vitest",
|
||||
"testing-library",
|
||||
"eslint-plugin-react-compiler"
|
||||
],
|
||||
"parser": "@typescript-eslint/parser",
|
||||
@ -109,8 +110,10 @@
|
||||
},
|
||||
"overrides": [
|
||||
{
|
||||
"extends": ["plugin:testing-library/react"],
|
||||
"files": ["tests/**/*.ts", "tests/**/*.tsx"],
|
||||
"rules": {
|
||||
"testing-library/no-node-access": "off",
|
||||
"import/extensions": ["error", "never"],
|
||||
"@typescript-eslint/no-unused-vars": "off"
|
||||
}
|
||||
|
||||
@ -130,6 +130,7 @@
|
||||
"eslint-plugin-react": "^7.37.2",
|
||||
"eslint-plugin-react-compiler": "19.0.0-beta-6fc168f-20241025",
|
||||
"eslint-plugin-react-hooks": "^5.0.0",
|
||||
"eslint-plugin-testing-library": "^6.4.0",
|
||||
"eslint-plugin-vitest": "^0.5.4",
|
||||
"immer": "^10.1.1",
|
||||
"jsdom": "^25.0.1",
|
||||
|
||||
113
pnpm-lock.yaml
generated
113
pnpm-lock.yaml
generated
@ -74,6 +74,9 @@ devDependencies:
|
||||
eslint-plugin-react-hooks:
|
||||
specifier: ^5.0.0
|
||||
version: 5.0.0(eslint@8.57.0)
|
||||
eslint-plugin-testing-library:
|
||||
specifier: ^6.4.0
|
||||
version: 6.4.0(eslint@8.57.0)(typescript@5.6.3)
|
||||
eslint-plugin-vitest:
|
||||
specifier: ^0.5.4
|
||||
version: 0.5.4(@typescript-eslint/eslint-plugin@8.12.2)(eslint@8.57.0)(typescript@5.6.3)(vitest@2.1.4)
|
||||
@ -1231,6 +1234,10 @@ packages:
|
||||
resolution: {integrity: sha512-AYnb1nQyY49te+VRAVgmzfcgjYS91mY5P0TKUDCLEM+gNnA+3T6rWITXRLYCpahpqSQbN5cE+gHpnPyXjHWxcw==}
|
||||
dev: true
|
||||
|
||||
/@types/json-schema@7.0.15:
|
||||
resolution: {integrity: sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA==}
|
||||
dev: true
|
||||
|
||||
/@types/json5@0.0.29:
|
||||
resolution: {integrity: sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ==}
|
||||
dev: true
|
||||
@ -1262,6 +1269,10 @@ packages:
|
||||
resolution: {integrity: sha512-60BCwRFOZCQhDncwQdxxeOEEkbc5dIMccYLwbxsS4TUNeVECQ/pBJ0j09mrHOl/JJvpRPGwO9SvE4nR2Nb/a4Q==}
|
||||
dev: true
|
||||
|
||||
/@types/semver@7.5.8:
|
||||
resolution: {integrity: sha512-I8EUhyrgfLrcTkzV3TSsGyl1tSuPrEDzr0yd5m90UgNxQkyDXULk3b6MlQqTCpZpNtWe1K0hzclnZkTcLBe2UQ==}
|
||||
dev: true
|
||||
|
||||
/@types/use-sync-external-store@0.0.6:
|
||||
resolution: {integrity: sha512-zFDAD+tlpf2r4asuHEj0XH6pY6i0g5NeAHPn+15wk3BV6JA69eERFXC1gyGThDkVa1zCyKr5jox1+2LbV/AMLg==}
|
||||
dev: true
|
||||
@ -1314,6 +1325,14 @@ packages:
|
||||
- supports-color
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/scope-manager@5.62.0:
|
||||
resolution: {integrity: sha512-VXuvVvZeQCQb5Zgf4HAxc04q5j+WrNAtNh9OwCsCgpKqESMTu3tF/jhZ3xG6T4NZwWl65Bg8KuS2uEvhSfLl0w==}
|
||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||
dependencies:
|
||||
'@typescript-eslint/types': 5.62.0
|
||||
'@typescript-eslint/visitor-keys': 5.62.0
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/scope-manager@7.18.0:
|
||||
resolution: {integrity: sha512-jjhdIE/FPF2B7Z1uzc6i3oWKbGcHb87Qw7AWj6jmEqNOfDFbJWtjt/XfwCpvNkpGWlcJaog5vTR+VV8+w9JflA==}
|
||||
engines: {node: ^18.18.0 || >=20.0.0}
|
||||
@ -1349,6 +1368,11 @@ packages:
|
||||
- supports-color
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/types@5.62.0:
|
||||
resolution: {integrity: sha512-87NVngcbVXUahrRTqIK27gD2t5Cu1yuCXxbLcFtCzZGlfyVWWh8mLHkoxzjsB6DDNnvdL+fW8MiwPEJyGJQDgQ==}
|
||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/types@7.18.0:
|
||||
resolution: {integrity: sha512-iZqi+Ds1y4EDYUtlOOC+aUmxnE9xS/yCigkjA7XpTKV6nCBd3Hp/PRGGmdwnfkV2ThMyYldP1wRpm/id99spTQ==}
|
||||
engines: {node: ^18.18.0 || >=20.0.0}
|
||||
@ -1359,6 +1383,27 @@ packages:
|
||||
engines: {node: ^18.18.0 || ^20.9.0 || >=21.1.0}
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/typescript-estree@5.62.0(typescript@5.6.3):
|
||||
resolution: {integrity: sha512-CmcQ6uY7b9y694lKdRB8FEel7JbU/40iSAPomu++SjLMntB+2Leay2LO6i8VnJk58MtE9/nQSFIH6jpyRWyYzA==}
|
||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||
peerDependencies:
|
||||
typescript: '*'
|
||||
peerDependenciesMeta:
|
||||
typescript:
|
||||
optional: true
|
||||
dependencies:
|
||||
'@typescript-eslint/types': 5.62.0
|
||||
'@typescript-eslint/visitor-keys': 5.62.0
|
||||
debug: 4.3.7
|
||||
globby: 11.1.0
|
||||
is-glob: 4.0.3
|
||||
semver: 7.6.3
|
||||
tsutils: 3.21.0(typescript@5.6.3)
|
||||
typescript: 5.6.3
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/typescript-estree@7.18.0(typescript@5.6.3):
|
||||
resolution: {integrity: sha512-aP1v/BSPnnyhMHts8cf1qQ6Q1IFwwRvAQGRvBFkWlo3/lH29OXA3Pts+c10nxRxIBrDnoMqzhgdwVe5f2D6OzA==}
|
||||
engines: {node: ^18.18.0 || >=20.0.0}
|
||||
@ -1403,6 +1448,26 @@ packages:
|
||||
- supports-color
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/utils@5.62.0(eslint@8.57.0)(typescript@5.6.3):
|
||||
resolution: {integrity: sha512-n8oxjeb5aIbPFEtmQxQYOLI0i9n5ySBEY/ZEHHZqKQSFnxio1rv6dthascc9dLuwrL0RC5mPCxB7vnAVGAYWAQ==}
|
||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||
peerDependencies:
|
||||
eslint: ^6.0.0 || ^7.0.0 || ^8.0.0
|
||||
dependencies:
|
||||
'@eslint-community/eslint-utils': 4.4.1(eslint@8.57.0)
|
||||
'@types/json-schema': 7.0.15
|
||||
'@types/semver': 7.5.8
|
||||
'@typescript-eslint/scope-manager': 5.62.0
|
||||
'@typescript-eslint/types': 5.62.0
|
||||
'@typescript-eslint/typescript-estree': 5.62.0(typescript@5.6.3)
|
||||
eslint: 8.57.0
|
||||
eslint-scope: 5.1.1
|
||||
semver: 7.6.3
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
- typescript
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/utils@7.18.0(eslint@8.57.0)(typescript@5.6.3):
|
||||
resolution: {integrity: sha512-kK0/rNa2j74XuHVcoCZxdFBMF+aq/vH83CXAOHieC+2Gis4mF8jJXT5eAfyD3K0sAxtPuwxaIOIOvhwzVDt/kw==}
|
||||
engines: {node: ^18.18.0 || >=20.0.0}
|
||||
@ -1435,6 +1500,14 @@ packages:
|
||||
- typescript
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/visitor-keys@5.62.0:
|
||||
resolution: {integrity: sha512-07ny+LHRzQXepkGg6w0mFY41fVUNBrL2Roj/++7V1txKugfjm/Ci/qSND03r2RhlJhJYMcTn9AhhSSqQp0Ysyw==}
|
||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||
dependencies:
|
||||
'@typescript-eslint/types': 5.62.0
|
||||
eslint-visitor-keys: 3.4.3
|
||||
dev: true
|
||||
|
||||
/@typescript-eslint/visitor-keys@7.18.0:
|
||||
resolution: {integrity: sha512-cDF0/Gf81QpY3xYyJKDV14Zwdmid5+uuENhjH2EqFaF0ni+yAyq/LzMaIJdhNJXZI7uLzwIlA+V7oWoyn6Curg==}
|
||||
engines: {node: ^18.18.0 || >=20.0.0}
|
||||
@ -2388,6 +2461,19 @@ packages:
|
||||
string.prototype.repeat: 1.0.0
|
||||
dev: true
|
||||
|
||||
/eslint-plugin-testing-library@6.4.0(eslint@8.57.0)(typescript@5.6.3):
|
||||
resolution: {integrity: sha512-yeWF+YgCgvNyPNI9UKnG0FjeE2sk93N/3lsKqcmR8dSfeXJwFT5irnWo7NjLf152HkRzfoFjh3LsBUrhvFz4eA==}
|
||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0, npm: '>=6'}
|
||||
peerDependencies:
|
||||
eslint: ^7.5.0 || ^8.0.0 || ^9.0.0
|
||||
dependencies:
|
||||
'@typescript-eslint/utils': 5.62.0(eslint@8.57.0)(typescript@5.6.3)
|
||||
eslint: 8.57.0
|
||||
transitivePeerDependencies:
|
||||
- supports-color
|
||||
- typescript
|
||||
dev: true
|
||||
|
||||
/eslint-plugin-vitest@0.5.4(@typescript-eslint/eslint-plugin@8.12.2)(eslint@8.57.0)(typescript@5.6.3)(vitest@2.1.4):
|
||||
resolution: {integrity: sha512-um+odCkccAHU53WdKAw39MY61+1x990uXjSPguUCq3VcEHdqJrOb8OTMrbYlY6f9jAKx7x98kLVlIe3RJeJqoQ==}
|
||||
engines: {node: ^18.0.0 || >= 20.0.0}
|
||||
@ -2410,6 +2496,14 @@ packages:
|
||||
- typescript
|
||||
dev: true
|
||||
|
||||
/eslint-scope@5.1.1:
|
||||
resolution: {integrity: sha512-2NxwbF/hZ0KpepYN0cNbo+FN6XoK7GaHlQhgx/hIZl6Va0bF45RQOOwhLIy8lQDbuCiadSLCBnH2CFYquit5bw==}
|
||||
engines: {node: '>=8.0.0'}
|
||||
dependencies:
|
||||
esrecurse: 4.3.0
|
||||
estraverse: 4.3.0
|
||||
dev: true
|
||||
|
||||
/eslint-scope@7.2.2:
|
||||
resolution: {integrity: sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg==}
|
||||
engines: {node: ^12.22.0 || ^14.17.0 || >=16.0.0}
|
||||
@ -2493,6 +2587,11 @@ packages:
|
||||
estraverse: 5.3.0
|
||||
dev: true
|
||||
|
||||
/estraverse@4.3.0:
|
||||
resolution: {integrity: sha512-39nnKffWz8xN1BU/2c79n9nB9HDzo0niYUqx6xyqUnyoAnQyyWpOTdZEeiCch8BBu515t4wp9ZmgVfVhn9EBpw==}
|
||||
engines: {node: '>=4.0'}
|
||||
dev: true
|
||||
|
||||
/estraverse@5.3.0:
|
||||
resolution: {integrity: sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA==}
|
||||
engines: {node: '>=4.0'}
|
||||
@ -4130,10 +4229,24 @@ packages:
|
||||
strip-bom: 3.0.0
|
||||
dev: true
|
||||
|
||||
/tslib@1.14.1:
|
||||
resolution: {integrity: sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg==}
|
||||
dev: true
|
||||
|
||||
/tslib@2.8.0:
|
||||
resolution: {integrity: sha512-jWVzBLplnCmoaTr13V9dYbiQ99wvZRd0vNWaDRg+aVYRcjDF3nDksxFDE/+fkXnKhpnUUkmx5pK/v8mCtLVqZA==}
|
||||
dev: true
|
||||
|
||||
/tsutils@3.21.0(typescript@5.6.3):
|
||||
resolution: {integrity: sha512-mHKK3iUXL+3UF6xL5k0PEhKRUBKPBCv/+RkEOpjRWxxx27KKRBmmA60A9pgOUvMi8GKhRMPEmjBRPzs2W7O1OA==}
|
||||
engines: {node: '>= 6'}
|
||||
peerDependencies:
|
||||
typescript: '>=2.8.0 || >= 3.2.0-dev || >= 3.3.0-dev || >= 3.4.0-dev || >= 3.5.0-dev || >= 3.6.0-dev || >= 3.6.0-beta || >= 3.7.0-dev || >= 3.7.0-beta'
|
||||
dependencies:
|
||||
tslib: 1.14.1
|
||||
typescript: 5.6.3
|
||||
dev: true
|
||||
|
||||
/type-check@0.4.0:
|
||||
resolution: {integrity: sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew==}
|
||||
engines: {node: '>= 0.8.0'}
|
||||
|
||||
@ -729,7 +729,7 @@ it('works with "undefined" state', async () => {
|
||||
return <div>str: {str}</div>
|
||||
}
|
||||
|
||||
const { findByText } = render(
|
||||
render(
|
||||
<StrictMode>
|
||||
<Component />
|
||||
</StrictMode>,
|
||||
|
||||
@ -336,8 +336,12 @@ describe('persist middleware with async configuration', () => {
|
||||
)
|
||||
|
||||
await screen.findByText('count: 0')
|
||||
|
||||
await waitFor(() => {
|
||||
expect(console.error).toHaveBeenCalled()
|
||||
})
|
||||
|
||||
await waitFor(() => {
|
||||
expect(onRehydrateStorageSpy).toBeCalledWith({ count: 0 }, undefined)
|
||||
})
|
||||
})
|
||||
|
||||
@ -1,5 +1,5 @@
|
||||
import { useState } from 'react'
|
||||
import { act, fireEvent, render } from '@testing-library/react'
|
||||
import { act, fireEvent, render, screen } from '@testing-library/react'
|
||||
import { beforeEach, describe, expect, it, vi } from 'vitest'
|
||||
import { create } from 'zustand'
|
||||
import { useShallow } from 'zustand/react/shallow'
|
||||
@ -60,12 +60,12 @@ describe('useShallow', () => {
|
||||
})
|
||||
|
||||
it('input and output selectors always return shallow equal values', () => {
|
||||
const res = render(
|
||||
const { rerender } = render(
|
||||
<TestUseShallowSimple state={{ a: 1, b: 2 }} selector={Object.keys} />,
|
||||
)
|
||||
|
||||
expect(testUseShallowSimpleCallback).toHaveBeenCalledTimes(0)
|
||||
fireEvent.click(res.getByTestId('test-shallow'))
|
||||
fireEvent.click(screen.getByTestId('test-shallow'))
|
||||
|
||||
const firstRender = testUseShallowSimpleCallback.mock.lastCall?.[0]
|
||||
|
||||
@ -73,14 +73,14 @@ describe('useShallow', () => {
|
||||
expect(firstRender).toBeTruthy()
|
||||
expect(firstRender?.selectorOutput).toEqual(firstRender?.useShallowOutput)
|
||||
|
||||
res.rerender(
|
||||
rerender(
|
||||
<TestUseShallowSimple
|
||||
state={{ a: 1, b: 2, c: 3 }}
|
||||
selector={Object.keys}
|
||||
/>,
|
||||
)
|
||||
|
||||
fireEvent.click(res.getByTestId('test-shallow'))
|
||||
fireEvent.click(screen.getByTestId('test-shallow'))
|
||||
expect(testUseShallowSimpleCallback).toHaveBeenCalledTimes(2)
|
||||
|
||||
const secondRender = testUseShallowSimpleCallback.mock.lastCall?.[0]
|
||||
@ -91,25 +91,25 @@ describe('useShallow', () => {
|
||||
|
||||
it('returns the previously computed instance when possible', () => {
|
||||
const state = { a: 1, b: 2 }
|
||||
const res = render(
|
||||
const { rerender } = render(
|
||||
<TestUseShallowSimple state={state} selector={Object.keys} />,
|
||||
)
|
||||
|
||||
fireEvent.click(res.getByTestId('test-shallow'))
|
||||
fireEvent.click(screen.getByTestId('test-shallow'))
|
||||
expect(testUseShallowSimpleCallback).toHaveBeenCalledTimes(1)
|
||||
const output1 =
|
||||
testUseShallowSimpleCallback.mock.lastCall?.[0]?.useShallowOutput
|
||||
expect(output1).toBeTruthy()
|
||||
|
||||
// Change selector, same output
|
||||
res.rerender(
|
||||
rerender(
|
||||
<TestUseShallowSimple
|
||||
state={state}
|
||||
selector={(state) => Object.keys(state)}
|
||||
/>,
|
||||
)
|
||||
|
||||
fireEvent.click(res.getByTestId('test-shallow'))
|
||||
fireEvent.click(screen.getByTestId('test-shallow'))
|
||||
expect(testUseShallowSimpleCallback).toHaveBeenCalledTimes(2)
|
||||
|
||||
const output2 =
|
||||
@ -137,10 +137,10 @@ describe('useShallow', () => {
|
||||
}
|
||||
|
||||
expect(countRenders).toBe(0)
|
||||
const res = render(<TestShallow />)
|
||||
render(<TestShallow />)
|
||||
|
||||
expect(countRenders).toBe(1)
|
||||
expect(res.getByTestId('test-shallow').textContent).toBe('a,b,c')
|
||||
expect(screen.getByTestId('test-shallow').textContent).toBe('a,b,c')
|
||||
|
||||
act(() => {
|
||||
useMyStore.setState({ a: 4 }) // This will not cause a re-render.
|
||||
@ -153,7 +153,7 @@ describe('useShallow', () => {
|
||||
})
|
||||
|
||||
expect(countRenders).toBe(2)
|
||||
expect(res.getByTestId('test-shallow').textContent).toBe('a,b,c,d')
|
||||
expect(screen.getByTestId('test-shallow').textContent).toBe('a,b,c,d')
|
||||
})
|
||||
|
||||
it('does not cause stale closure issues', () => {
|
||||
@ -176,12 +176,12 @@ describe('useShallow', () => {
|
||||
)
|
||||
}
|
||||
|
||||
const res = render(<TestShallowWithState />)
|
||||
render(<TestShallowWithState />)
|
||||
|
||||
expect(res.getByTestId('test-shallow').textContent).toBe('a,b,c,0')
|
||||
expect(screen.getByTestId('test-shallow').textContent).toBe('a,b,c,0')
|
||||
|
||||
fireEvent.click(res.getByTestId('test-shallow'))
|
||||
fireEvent.click(screen.getByTestId('test-shallow'))
|
||||
|
||||
expect(res.getByTestId('test-shallow').textContent).toBe('a,b,c,1')
|
||||
expect(screen.getByTestId('test-shallow').textContent).toBe('a,b,c,1')
|
||||
})
|
||||
})
|
||||
|
||||
@ -40,7 +40,7 @@ describe.skipIf(!React.version.startsWith('18'))(
|
||||
'react-dom/client',
|
||||
)
|
||||
|
||||
const markup = renderToString(
|
||||
const view = renderToString(
|
||||
<React.Suspense fallback={<div>Loading...</div>}>
|
||||
<Counter />
|
||||
</React.Suspense>,
|
||||
@ -48,7 +48,7 @@ describe.skipIf(!React.version.startsWith('18'))(
|
||||
|
||||
const container = document.createElement('div')
|
||||
document.body.appendChild(container)
|
||||
container.innerHTML = markup
|
||||
container.innerHTML = view
|
||||
|
||||
expect(container.textContent).toContain('bears: 0')
|
||||
|
||||
@ -80,7 +80,7 @@ describe.skipIf(!React.version.startsWith('18'))(
|
||||
return <div>bears: {bears}</div>
|
||||
}
|
||||
|
||||
const markup = renderToString(
|
||||
const view = renderToString(
|
||||
<React.Suspense fallback={<div>Loading...</div>}>
|
||||
<Component />
|
||||
</React.Suspense>,
|
||||
@ -88,7 +88,7 @@ describe.skipIf(!React.version.startsWith('18'))(
|
||||
|
||||
const container = document.createElement('div')
|
||||
document.body.appendChild(container)
|
||||
container.innerHTML = markup
|
||||
container.innerHTML = view
|
||||
|
||||
expect(container.textContent).toContain('bears: 0')
|
||||
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user