diff --git a/jest.config.js b/jest.config.js index cb0c20a71..27c4f089a 100644 --- a/jest.config.js +++ b/jest.config.js @@ -6,6 +6,7 @@ const esModules = [ '@jupyterlab/', 'lib0', 'nanoid', + 'nbdime', 'vscode-ws-jsonrpc', 'y-protocols', 'y-websocket', @@ -30,6 +31,8 @@ module.exports = { '/jupyter-config', '/ui-tests' ], + reporters: ['default', 'github-actions'], + setupFiles: ['/testutils/jest-setup-files.js'], testRegex: 'src/.*/.*.spec.ts[x]?$', transformIgnorePatterns: [`/node_modules/(?!${esModules}).+`] }; diff --git a/jupyterlab_git/tests/test_handlers.py b/jupyterlab_git/tests/test_handlers.py index 350250109..07b7d80ac 100644 --- a/jupyterlab_git/tests/test_handlers.py +++ b/jupyterlab_git/tests/test_handlers.py @@ -51,6 +51,7 @@ async def test_all_history_handler_localbranch(mock_git, jp_fetch, jp_root_dir): mock_git.status.return_value = maybe_future(status) # When + body = {"history_count": 25} response = await jp_fetch( NAMESPACE, local_path.name, "all_history", body=json.dumps(body), method="POST" ) diff --git a/package.json b/package.json index d295a5282..1538c77cb 100644 --- a/package.json +++ b/package.json @@ -102,8 +102,10 @@ "@babel/preset-env": "^7.0.0", "@jupyterlab/builder": "^4.0.0", "@jupyterlab/testutils": "^4.0.0", + "@testing-library/jest-dom": "^6.1.4", + "@testing-library/react": "^14.0.0", + "@testing-library/user-event": "^14.5.1", "@types/diff-match-patch": "^1.0.32", - "@types/enzyme": "^3.1.15", "@types/jest": "^29.2.0", "@types/json-schema": "^7.0.11", "@types/react": "^18.0.26", @@ -114,10 +116,8 @@ "@types/resize-observer-browser": "^0.1.7", "@typescript-eslint/eslint-plugin": "^6.1.0", "@typescript-eslint/parser": "^6.1.0", - "@wojtekmaj/enzyme-adapter-react-17": "^0.8.0", "all-contributors-cli": "^6.14.0", "css-loader": "^6.7.1", - "enzyme": "^3.7.0", "eslint": "^8.36.0", "eslint-config-prettier": "^8.8.0", "eslint-plugin-prettier": "^5.0.0", @@ -126,11 +126,11 @@ "husky": "^8.0.3", "identity-obj-proxy": "^3.0.0", "jest": "^29.2.0", - "jest-fetch-mock": "^3.0.0", "lint-staged": "^13.2.3", "mkdirp": "^1.0.3", "npm-run-all": "^4.1.5", "prettier": "^3.0.0", + "resize-observer-polyfill": "^1.5.1", "rimraf": "^5.0.1", "source-map-loader": "^1.0.2", "style-loader": "^3.3.1", @@ -139,7 +139,7 @@ "stylelint-config-standard": "^34.0.0", "stylelint-csstree-validator": "^3.0.0", "stylelint-prettier": "^4.0.0", - "ts-jest": "^26.0.0", + "ts-jest": "^29.1.0", "typescript": "~5.0.2", "yjs": "^13.5.40" }, diff --git a/src/__tests__/commands.spec.tsx b/src/__tests__/commands.spec.tsx index 4620fb7a7..ccbf4cf26 100644 --- a/src/__tests__/commands.spec.tsx +++ b/src/__tests__/commands.spec.tsx @@ -14,6 +14,10 @@ import { IMockedResponse, mockedRequestAPI } from './utils'; +import { + CodeMirrorEditorFactory, + EditorLanguageRegistry +} from '@jupyterlab/codemirror'; jest.mock('../git'); jest.mock('@jupyterlab/apputils'); @@ -43,7 +47,7 @@ describe('git-commands', () => { }; mockGit.requestAPI.mockImplementation( - mockedRequestAPI({ responses: mockResponses }) + mockedRequestAPI({ responses: mockResponses }) as any ); commands = new CommandRegistry(); @@ -56,8 +60,10 @@ describe('git-commands', () => { addCommands( app as JupyterFrontEnd, model, + new CodeMirrorEditorFactory().newDocumentEditor, + new EditorLanguageRegistry(), mockedFileBrowserModel, - null, + null as any, nullTranslator ); }); @@ -73,6 +79,7 @@ describe('git-commands', () => { button: { accept: true, actions: [], + ariaLabel: '', caption: '', className: '', displayType: 'default', @@ -86,7 +93,7 @@ describe('git-commands', () => { const spyReset = jest.spyOn(model, 'reset'); spyReset.mockResolvedValueOnce(undefined); const spyCheckout = jest.spyOn(model, 'checkout'); - spyCheckout.mockResolvedValueOnce(undefined); + spyCheckout.mockResolvedValueOnce({} as any); const path = 'file/path.ext'; model.pathRepository = DEFAULT_REPOSITORY_PATH; @@ -140,6 +147,7 @@ describe('git-commands', () => { button: { accept: true, actions: [], + ariaLabel: '', caption: '', className: '', displayType: 'default', @@ -169,7 +177,7 @@ describe('git-commands', () => { } } } - }) + }) as any ); const path = DEFAULT_REPOSITORY_PATH; diff --git a/src/__tests__/model.spec.tsx b/src/__tests__/model.spec.tsx index 608e0896a..88bb59917 100644 --- a/src/__tests__/model.spec.tsx +++ b/src/__tests__/model.spec.tsx @@ -7,7 +7,8 @@ import { defaultMockedResponses, DEFAULT_REPOSITORY_PATH, IMockedResponses, - mockedRequestAPI + mockedRequestAPI, + IMockedResponse } from './utils'; jest.mock('../git'); @@ -16,7 +17,7 @@ describe('IGitExtension', () => { const mockGit = git as jest.Mocked; let model: IGitExtension; const docmanager = jest.mock('@jupyterlab/docmanager') as any; - let mockResponses: IMockedResponses; + let mockResponses: IMockedResponses & {responses: {[endpoint: string]: IMockedResponse}}; beforeEach(async () => { jest.restoreAllMocks(); @@ -29,7 +30,9 @@ describe('IGitExtension', () => { responses: { ...defaultMockedResponses } }; - mockGit.requestAPI.mockImplementation(mockedRequestAPI(mockResponses)); + mockGit.requestAPI.mockImplementation( + mockedRequestAPI(mockResponses) as any + ); model = new GitExtension(docmanager as any, docregistry as any); }); @@ -94,7 +97,7 @@ describe('IGitExtension', () => { }); it('should emit repositoryChanged signal and request a refresh', async () => { - let path = DEFAULT_REPOSITORY_PATH; + let path: string | null = DEFAULT_REPOSITORY_PATH; const testPathSignal = testEmission(model.repositoryChanged, { test: (model, change) => { @@ -198,7 +201,7 @@ describe('IGitExtension', () => { describe('#status', () => { it('should be clear if not in a git repository', async () => { let status: Partial = { - branch: null, + branch: undefined, remote: null, ahead: 0, behind: 0, @@ -254,9 +257,9 @@ describe('IGitExtension', () => { const testSignal = testEmission(model.statusChanged, { test: (model, receivedStatus) => { expect(receivedStatus.branch).toEqual(branch); - expect(receivedStatus.files).toHaveLength(status.files.length); + expect(receivedStatus.files).toHaveLength(status.files?.length ?? 0); expect(receivedStatus.files[0]).toMatchObject({ - ...status.files[0] + ...status.files![0] }); } }); @@ -280,7 +283,7 @@ describe('IGitExtension', () => { ['other_repo/file', null, 'repo'], ['root/other_repo/file', null, 'root/repo'] ])( - '%s should return unmodified status with path relative to the repository', + '%s should return unmodified status with path %s to the repository %s', async (path, expected, repo) => { // Given mockResponses.path = repo; @@ -292,8 +295,8 @@ describe('IGitExtension', () => { if (expected === null) { expect(status).toBeNull(); } else { - expect(status.status).toEqual('unmodified'); - expect(status.to).toEqual(expected); + expect(status!.status).toEqual('unmodified'); + expect(status!.to).toEqual(expected); } } ); diff --git a/src/__tests__/plugin.spec.ts b/src/__tests__/plugin.spec.ts index 6718b3551..6be159a9d 100644 --- a/src/__tests__/plugin.spec.ts +++ b/src/__tests__/plugin.spec.ts @@ -36,21 +36,23 @@ describe('plugin', () => { } } as unknown as jest.Mocked; settingRegistry = new SettingRegistry({ - connector: null - }) as jest.Mocked; + connector: null as any + }) as any; }); beforeEach(() => { jest.resetAllMocks(); mockResponses = { responses: { ...defaultMockedResponses } }; - mockGit.requestAPI.mockImplementation(mockedRequestAPI(mockResponses)); + mockGit.requestAPI.mockImplementation( + mockedRequestAPI(mockResponses) as any + ); }); describe('#activate', () => { it('should fail if no git is installed', async () => { // Given const endpoint = 'settings' + URLExt.objectToQueryString({ version }); - mockResponses.responses[endpoint] = { + mockResponses.responses![endpoint] = { body: request => { return { gitVersion: null, @@ -85,7 +87,7 @@ describe('plugin', () => { it('should fail if git version is < 2', async () => { // Given const endpoint = 'settings' + URLExt.objectToQueryString({ version }); - mockResponses.responses[endpoint] = { + mockResponses.responses![endpoint] = { body: request => { return { gitVersion: '1.8.7', @@ -119,7 +121,7 @@ describe('plugin', () => { it('should fail if server and extension version do not match', async () => { // Given const endpoint = 'settings' + URLExt.objectToQueryString({ version }); - mockResponses.responses[endpoint] = { + mockResponses.responses![endpoint] = { body: request => { return { gitVersion: '2.22.0', @@ -156,7 +158,7 @@ describe('plugin', () => { it('should fail if the server extension is not installed', async () => { // Given const endpoint = 'settings' + URLExt.objectToQueryString({ version }); - mockResponses.responses[endpoint] = { + mockResponses.responses![endpoint] = { status: 404 }; const mockedErrorMessage = showErrorMessage as jest.MockedFunction< diff --git a/src/__tests__/test-components/BranchMenu.spec.tsx b/src/__tests__/test-components/BranchMenu.spec.tsx index 9953b107b..f911f827c 100644 --- a/src/__tests__/test-components/BranchMenu.spec.tsx +++ b/src/__tests__/test-components/BranchMenu.spec.tsx @@ -1,19 +1,18 @@ -import { mount, render, shallow } from 'enzyme'; import { showDialog } from '@jupyterlab/apputils'; +import { nullTranslator } from '@jupyterlab/translation'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; import 'jest'; import * as React from 'react'; -import { ActionButton } from '../../components/ActionButton'; import { BranchMenu, IBranchMenuProps } from '../../components/BranchMenu'; import * as git from '../../git'; import { GitExtension } from '../../model'; -import { listItemClass, nameClass } from '../../style/BranchMenu'; import { - mockedRequestAPI, + DEFAULT_REPOSITORY_PATH, defaultMockedResponses, - DEFAULT_REPOSITORY_PATH + mockedRequestAPI } from '../utils'; -import ClearIcon from '@mui/icons-material/Clear'; -import { nullTranslator } from '@jupyterlab/translation'; jest.mock('../../git'); jest.mock('@jupyterlab/apputils'); @@ -86,7 +85,7 @@ describe('BranchMenu', () => { } } } - }) + }) as any ); model = await createModel(); @@ -107,76 +106,61 @@ describe('BranchMenu', () => { } describe('constructor', () => { - it('should return a new instance', () => { - const menu = shallow(); - expect(menu.instance()).toBeInstanceOf(BranchMenu); - }); - it('should set the default menu filter to an empty string', () => { - const menu = shallow(); - expect(menu.state('filter')).toEqual(''); - }); - - it('should set the default flag indicating whether to show a dialog to create a new branch to `false`', () => { - const menu = shallow(); - expect(menu.state('branchDialog')).toEqual(false); + render(); + expect( + screen.getByRole('textbox', { name: 'Filter branch menu' }) + ).toHaveValue(''); }); }); describe('render', () => { it('should display placeholder text for the menu filter', () => { - const component = shallow(); - const node = component.find('input[type="text"]').first(); - expect(node.prop('placeholder')).toEqual('Filter'); + render(); + expect( + screen.getByRole('textbox', { name: 'Filter branch menu' }) + ).toHaveAttribute('placeholder', 'Filter'); }); it('should set a `title` attribute on the input element to filter a branch menu', () => { - const component = shallow(); - const node = component.find('input[type="text"]').first(); - expect(node.prop('title').length > 0).toEqual(true); - }); - - it('should display a button to clear the menu filter once a filter is provided', () => { - const component = shallow(); - component.setState({ - filter: 'foo' - }); - const nodes = component.find(ClearIcon); - expect(nodes.length).toEqual(1); + render(); + expect(screen.getByRole('textbox')).toHaveAttribute( + 'title', + 'Filter branch menu' + ); }); - it('should set a `title` on the button to clear the menu filter', () => { - const component = shallow(); - component.setState({ - filter: 'foo' - }); - const html = component.find(ClearIcon).first().html(); - expect(html.includes('')).toEqual(true); + it('should display a button to clear the menu filter once a filter is provided', async () => { + render(<BranchMenu {...createProps()} />); + await userEvent.type(screen.getByRole('textbox'), 'foo'); + expect(screen.getAllByTitle('Clear the current filter').length).toEqual( + 1 + ); }); it('should display a button to create a new branch', () => { - const component = shallow(<BranchMenu {...createProps()} />); - const node = component.find('input[type="button"]').first(); - expect(node.prop('value')).toEqual('New Branch'); + render(<BranchMenu {...createProps()} />); + expect(screen.getByRole('button', { name: 'New Branch' })).toBeDefined(); }); it('should set a `title` attribute on the button to create a new branch', () => { - const component = shallow(<BranchMenu {...createProps()} />); - const node = component.find('input[type="button"]').first(); - expect(node.prop('title').length > 0).toEqual(true); + render(<BranchMenu {...createProps()} />); + expect( + screen.getByRole('button', { name: 'New Branch' }) + ).toHaveAttribute('title', 'Create a new branch'); }); it('should display a list of branches', () => { - const component = render(<BranchMenu {...createProps()} />); - const nodes = component.find(`.${nameClass}`); + render(<BranchMenu {...createProps()} />); const branches = BRANCHES; - expect(nodes.length).toEqual(branches.length); + expect(screen.getAllByRole('listitem').length).toEqual(branches.length); // Should contain the branch names... for (let i = 0; i < branches.length; i++) { - // @ts-expect-error lastChild not always defined - expect(nodes[i].lastChild.data).toEqual(branches[i].name); + expect( + screen.getByText(branches[i].name, { exact: true }) + ).toBeDefined(); } }); @@ -210,7 +194,7 @@ describe('BranchMenu', () => { it(`should${ display ? ' ' : 'not ' }display delete and merge buttons for ${JSON.stringify(branch)}`, () => { - const menu = mount( + render( <BranchMenu {...createProps({ currentBranch: 'current', @@ -219,9 +203,9 @@ describe('BranchMenu', () => { /> ); - const item = menu.find(`.${listItemClass}`); - - expect(item.find(ActionButton).length).toEqual(display ? 2 : 0); + expect( + screen.getByRole('listitem').querySelectorAll('button').length + ).toEqual(display ? 2 : 0); }); }); @@ -246,13 +230,13 @@ describe('BranchMenu', () => { }, isChecked: null, value: undefined - }); + }) as any; }); const spy = jest.spyOn(GitExtension.prototype, 'deleteBranch'); const branchName = 'main'; - const menu = mount( + render( <BranchMenu {...createProps({ currentBranch: 'current', @@ -270,9 +254,9 @@ describe('BranchMenu', () => { /> ); - const item = menu.find(`.${listItemClass}`); - const button = item.find(ActionButton); - button.at(0).simulate('click'); + await userEvent.click( + screen.getByRole('button', { name: 'Delete this branch locally' }) + ); // Need to wait that the dialog is resolve so 'deleteBranch' is called before // this test ends. @@ -287,7 +271,7 @@ describe('BranchMenu', () => { const branchName = 'main'; const fakeExecutioner = jest.fn(); - const menu = mount( + render( <BranchMenu {...createProps({ currentBranch: 'current', @@ -308,9 +292,11 @@ describe('BranchMenu', () => { /> ); - const item = menu.find(`.${listItemClass}`); - const button = item.find(ActionButton); - button.at(1).simulate('click'); + await userEvent.click( + screen.getByRole('button', { + name: 'Merge this branch into the current one' + }) + ); expect(fakeExecutioner).toHaveBeenCalledTimes(1); expect(fakeExecutioner).toHaveBeenCalledWith('git:merge', { @@ -318,59 +304,40 @@ describe('BranchMenu', () => { }); }); - it('should set a `title` attribute for each displayed branch', () => { - const component = render(<BranchMenu {...createProps()} />); - const nodes = component.find(`.${listItemClass}`); - - const branches = BRANCHES; - expect(nodes.length).toEqual(branches.length); + it('should show a dialog to create a new branch when the flag indicating whether to show the dialog is `true`', async () => { + render(<BranchMenu {...createProps({ branching: true })} />); - for (let i = 0; i < branches.length; i++) { - // @ts-expect-error attribs not always defined - expect(nodes[i].attribs['title'].length).toBeGreaterThanOrEqual(0); - } - }); + await userEvent.click(screen.getByRole('button', { name: 'New Branch' })); - it('should not, by default, show a dialog to create a new branch', () => { - const component = shallow(<BranchMenu {...createProps()} />); - const node = component.find('NewBranchDialog').first(); - expect(node.prop('open')).toEqual(false); - }); - - it('should show a dialog to create a new branch when the flag indicating whether to show the dialog is `true`', () => { - const component = shallow(<BranchMenu {...createProps()} />); - component.setState({ - branchDialog: true - }); - const node = component.find('NewBranchDialog').first(); - expect(node.prop('open')).toEqual(true); + expect(screen.getByRole('dialog')).toBeDefined(); }); }); describe('switch branch', () => { - it('should not switch to a specified branch upon clicking its corresponding element when branching is disabled', () => { + it('should not switch to a specified branch upon clicking its corresponding element when branching is disabled', async () => { const spy = jest.spyOn(GitExtension.prototype, 'checkout'); - const component = mount(<BranchMenu {...createProps()} />); - const nodes = component.find( - `.${listItemClass}[title*="${BRANCHES[1].name}"]` + render(<BranchMenu {...createProps()} />); + + await userEvent.click( + screen.getByRole('listitem', { + name: `Switch to branch: ${BRANCHES[1].name}` + }) ); - nodes.at(0).simulate('click'); expect(spy).toHaveBeenCalledTimes(0); spy.mockRestore(); }); - it('should switch to a specified branch upon clicking its corresponding element when branching is enabled', () => { + it('should switch to a specified branch upon clicking its corresponding element when branching is enabled', async () => { const spy = jest.spyOn(GitExtension.prototype, 'checkout'); - const component = mount( - <BranchMenu {...createProps({ branching: true })} /> - ); - const nodes = component.find( - `.${listItemClass}[title*="${BRANCHES[1].name}"]` + render(<BranchMenu {...createProps({ branching: true })} />); + await userEvent.click( + screen.getByRole('listitem', { + name: `Switch to branch: ${BRANCHES[1].name}` + }) ); - nodes.at(0).simulate('click'); expect(spy).toHaveBeenCalledTimes(1); expect(spy).toHaveBeenCalledWith({ @@ -382,30 +349,25 @@ describe('BranchMenu', () => { }); describe('create branch', () => { - it('should not allow creating a new branch when branching is disabled', () => { + it('should not allow creating a new branch when branching is disabled', async () => { const spy = jest.spyOn(GitExtension.prototype, 'checkout'); - const component = shallow(<BranchMenu {...createProps()} />); + render(<BranchMenu {...createProps()} />); - const node = component.find('input[type="button"]').first(); - node.simulate('click'); + await userEvent.click(screen.getByRole('button', { name: 'New Branch' })); - expect(component.state('branchDialog')).toEqual(false); expect(spy).toHaveBeenCalledTimes(0); spy.mockRestore(); }); - it('should display a dialog to create a new branch when branching is enabled and the new branch button is clicked', () => { + it('should display a dialog to create a new branch when branching is enabled and the new branch button is clicked', async () => { const spy = jest.spyOn(GitExtension.prototype, 'checkout'); - const component = shallow( - <BranchMenu {...createProps({ branching: true })} /> - ); + render(<BranchMenu {...createProps({ branching: true })} />); - const node = component.find('input[type="button"]').first(); - node.simulate('click'); + await userEvent.click(screen.getByRole('button', { name: 'New Branch' })); - expect(component.state('branchDialog')).toEqual(true); + expect(screen.getByRole('dialog')).toBeDefined(); expect(spy).toHaveBeenCalledTimes(0); spy.mockRestore(); }); diff --git a/src/__tests__/test-components/CommitBox.spec.tsx b/src/__tests__/test-components/CommitBox.spec.tsx index 26baf7463..1414792c5 100644 --- a/src/__tests__/test-components/CommitBox.spec.tsx +++ b/src/__tests__/test-components/CommitBox.spec.tsx @@ -1,12 +1,11 @@ /* eslint-disable @typescript-eslint/no-empty-function */ import { nullTranslator } from '@jupyterlab/translation'; import { CommandRegistry } from '@lumino/commands'; -import Button from '@mui/material/Button'; -import { shallow } from 'enzyme'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; import 'jest'; import * as React from 'react'; import { CommitBox, ICommitBoxProps } from '../../components/CommitBox'; -import { CommitMessage } from '../../components/CommitMessage'; import { WarningBox } from '../../components/WarningBox'; import { CommandIDs } from '../../tokens'; @@ -44,9 +43,10 @@ describe('CommitBox', () => { describe('#render()', () => { it('should display placeholder text for the commit message summary', () => { const props = defaultProps; - const component = shallow(<CommitBox {...props} />); - const node = component.find(CommitMessage); - expect(node.prop('summaryPlaceholder')).toEqual( + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('textbox')[0]).toHaveAttribute( + 'placeholder', 'Summary (Ctrl+Enter to commit)' ); }); @@ -62,33 +62,34 @@ describe('CommitBox', () => { ...defaultProps, commands: adjustedCommands }; - const component = shallow(<CommitBox {...props} />); - const node = component.find(CommitMessage); - expect(node.prop('summaryPlaceholder')).toEqual( + + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('textbox')[0]).toHaveAttribute( + 'placeholder', 'Summary (Shift+Enter to commit)' ); }); it('should display a button to commit changes', () => { const props = defaultProps; - const component = shallow(<CommitBox {...props} />); - const node = component.find(Button).first(); - expect(node.text()).toEqual('Commit'); + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('button')[0]).toHaveTextContent('Commit'); }); it('should set a `title` attribute on the button to commit changes', () => { const props = defaultProps; - const component = shallow(<CommitBox {...props} />); - const node = component.find(Button).first(); - expect(node.prop('title').length > 0).toEqual(true); + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('button')[0]).toHaveAttribute('title'); }); it('should apply a class to disable the commit button when no files have changes to commit', () => { const props = defaultProps; - const component = shallow(<CommitBox {...props} />); - const node = component.find(Button).first(); - const prop = node.prop('disabled'); - expect(prop).toEqual(true); + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('button')[0]).toHaveAttribute('disabled'); }); it('should apply a class to disable the commit button when files have changes to commit, but the user has not entered a commit message summary', () => { @@ -96,10 +97,9 @@ describe('CommitBox', () => { ...defaultProps, hasFiles: true }; - const component = shallow(<CommitBox {...props} />); - const node = component.find(Button).first(); - const prop = node.prop('disabled'); - expect(prop).toEqual(true); + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('button')[0]).toHaveAttribute('disabled'); }); it('should not apply a class to disable the commit button when files have changes to commit and the user has entered a commit message summary', () => { @@ -108,10 +108,10 @@ describe('CommitBox', () => { summary: 'beep boop', hasFiles: true }; - const component = shallow(<CommitBox {...props} />); - const node = component.find(Button).first(); - const prop = node.prop('disabled'); - expect(prop).toEqual(false); + + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('button')[0]).not.toHaveAttribute('disabled'); }); it('should apply a class to disable the commit input fields in amend mode', () => { @@ -120,8 +120,9 @@ describe('CommitBox', () => { summary: 'beep boop', amend: true }; - const component = shallow(<CommitBox {...props} />); - expect(component.find(CommitMessage).length).toEqual(0); + + render(<CommitBox {...props} />); + expect(screen.queryByRole('textbox')).toBeNull(); }); it('should not apply a class to disable the commit button in amend mode', () => { @@ -130,10 +131,9 @@ describe('CommitBox', () => { hasFiles: true, amend: true }; - const component = shallow(<CommitBox {...props} />); - const node = component.find(Button).first(); - const prop = node.prop('disabled'); - expect(prop).toEqual(false); + render(<CommitBox {...props} />); + + expect(screen.getAllByRole('button')[0]).not.toHaveAttribute('disabled'); }); it('should render a warning box when there are dirty staged files', () => { @@ -143,8 +143,9 @@ describe('CommitBox', () => { <WarningBox title="Warning" content="Warning content."></WarningBox> ) }; - const component = shallow(<CommitBox {...props} />); - expect(component.find(WarningBox).length).toEqual(1); + render(<CommitBox {...props} />); + + expect(screen.getByText('Warning content.')).toBeDefined(); }); }); }); diff --git a/src/__tests__/test-components/CommitMessage.spec.tsx b/src/__tests__/test-components/CommitMessage.spec.tsx index d884cac19..71104126a 100644 --- a/src/__tests__/test-components/CommitMessage.spec.tsx +++ b/src/__tests__/test-components/CommitMessage.spec.tsx @@ -1,7 +1,7 @@ /* eslint-disable @typescript-eslint/no-empty-function */ import { nullTranslator } from '@jupyterlab/translation'; -import Input from '@mui/material/Input'; -import { shallow } from 'enzyme'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; import 'jest'; import * as React from 'react'; import { @@ -22,36 +22,43 @@ describe('CommitMessage', () => { it('should set a `title` attribute on the input element to provide a commit message summary', () => { const props = defaultProps; - const component = shallow(<CommitMessage {...props} />); - const node = component.find(Input).first(); - expect(node.prop('title').length > 0).toEqual(true); + render(<CommitMessage {...props} />); + + expect(screen.getAllByRole('textbox')[0].parentElement).toHaveAttribute( + 'title' + ); }); it('should display placeholder text for the commit message description', () => { const props = defaultProps; - const component = shallow(<CommitMessage {...props} />); - const node = component.find(Input).last(); - expect(node.prop('placeholder')).toEqual('Description (optional)'); + render(<CommitMessage {...props} />); + + expect(screen.getAllByRole('textbox')[1]).toHaveAttribute( + 'placeholder', + 'Description (optional)' + ); }); it('should set a `title` attribute on the input element to provide a commit message description', () => { const props = defaultProps; - const component = shallow(<CommitMessage {...props} />); - const node = component.find(Input).last(); - expect(node.prop('title').length > 0).toEqual(true); + render(<CommitMessage {...props} />); + + expect(screen.getAllByRole('textbox')[1].parentElement).toHaveAttribute( + 'title' + ); }); it('should disable summary input if disabled is true', () => { const props = { ...defaultProps, disabled: true }; - const component = shallow(<CommitMessage {...props} />); - const node = component.find(Input).first(); - expect(node.prop('disabled')).toEqual(true); + render(<CommitMessage {...props} />); + + expect(screen.getAllByRole('textbox')[0]).toHaveAttribute('disabled'); }); it('should disable description input if disabled is true', () => { const props = { ...defaultProps, disabled: true }; - const component = shallow(<CommitMessage {...props} />); - const node = component.find(Input).last(); - expect(node.prop('disabled')).toEqual(true); + render(<CommitMessage {...props} />); + + expect(screen.getAllByRole('textbox')[1]).toHaveAttribute('disabled'); }); }); diff --git a/src/__tests__/test-components/FileItem.spec.tsx b/src/__tests__/test-components/FileItem.spec.tsx index f6969a9aa..a3f6a0cb9 100644 --- a/src/__tests__/test-components/FileItem.spec.tsx +++ b/src/__tests__/test-components/FileItem.spec.tsx @@ -1,8 +1,9 @@ -import { FileItem, IFileItemProps } from '../../src/components/FileItem'; -import * as React from 'react'; -import 'jest'; -import { shallow } from 'enzyme'; import { nullTranslator } from '@jupyterlab/translation'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; +import 'jest'; +import * as React from 'react'; +import { FileItem, IFileItemProps } from '../../components/FileItem'; describe('FileItem', () => { const trans = nullTranslator.load('jupyterlab_git'); @@ -17,7 +18,7 @@ describe('FileItem', () => { is_binary: null, status: null }, - model: null, + model: null as any, onDoubleClick: () => {}, selected: false, setSelection: file => {}, @@ -26,10 +27,10 @@ describe('FileItem', () => { }; describe('#render()', () => { - const component = shallow(<FileItem {...props} />); it('should display the full path on hover', () => { + render(<FileItem {...props} />); expect( - component.find('[title="some/file/path/file-name • Modified"]') + screen.getAllByTitle('some/file/path/file-name • Modified') ).toHaveLength(1); }); }); diff --git a/src/__tests__/test-components/GitPanel.spec.tsx b/src/__tests__/test-components/GitPanel.spec.tsx index 115060f5e..05dbd1b3c 100644 --- a/src/__tests__/test-components/GitPanel.spec.tsx +++ b/src/__tests__/test-components/GitPanel.spec.tsx @@ -1,10 +1,11 @@ import * as apputils from '@jupyterlab/apputils'; import { nullTranslator } from '@jupyterlab/translation'; import { JSONObject } from '@lumino/coreutils'; -import { shallow } from 'enzyme'; +import '@testing-library/jest-dom'; +import { render, screen, waitFor } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; import 'jest'; import React from 'react'; -import { CommitBox } from '../../components/CommitBox'; import { GitPanel, IGitPanelProps } from '../../components/GitPanel'; import * as git from '../../git'; import { GitExtension as GitModel } from '../../model'; @@ -14,6 +15,7 @@ import { IMockedResponse, mockedRequestAPI } from '../utils'; +import { CommandRegistry } from '@lumino/commands'; jest.mock('../../git'); jest.mock('@jupyterlab/apputils'); @@ -56,9 +58,9 @@ describe('GitPanel', () => { const trans = nullTranslator.load('jupyterlab_git'); const props: IGitPanelProps = { - model: null, - commands: null, - settings: null, + model: null as any, + commands: new CommandRegistry(), + settings: null as any, filebrowser: { path: '/dummy/path' } as any, @@ -69,7 +71,9 @@ describe('GitPanel', () => { jest.restoreAllMocks(); const mock = git as jest.Mocked<typeof git>; - mock.requestAPI.mockImplementation(mockedRequestAPI(mockedResponses)); + mock.requestAPI.mockImplementation( + mockedRequestAPI(mockedResponses) as any + ); props.model = new GitModel(); props.model.pathRepository = DEFAULT_REPOSITORY_PATH; @@ -97,7 +101,6 @@ describe('GitPanel', () => { }); describe('#commitFiles()', () => { - let panel: GitPanel; let commitSpy: jest.SpyInstance<Promise<void>>; let configSpy: jest.SpyInstance<Promise<void | JSONObject>>; @@ -119,7 +122,7 @@ describe('GitPanel', () => { iconClass: '', iconLabel: '', label: '' - }, + } as any, isChecked: null, value: { name: commitUser['user.name'], @@ -138,26 +141,59 @@ describe('GitPanel', () => { } }; return Promise.resolve<JSONObject>( - options === undefined + (options === undefined ? response // When getting config options - : null // When setting config options + : null) as any // When setting config options ); }; }; beforeEach(() => { - commitSpy = jest.spyOn(GitModel.prototype, 'commit').mockResolvedValue(); - configSpy = jest.spyOn(GitModel.prototype, 'config'); + configSpy = props.model.config = jest.fn(); + commitSpy = props.model.commit = jest.fn(); + // @ts-expect-error turn off set status + props.model._setStatus = jest.fn(); + + // @ts-expect-error set a private prop + props.model._status = { + branch: 'master', + remote: 'origin/master', + ahead: 0, + behind: 0, + files: [ + { + x: 'M', + y: ' ', + to: 'packages/jupyterlab_toastify/README.md', + from: 'packages/jupyterlab_toastify/README.md', + is_binary: false, + status: 'staged' + } + ], + state: 0 + }; - const panelWrapper = shallow<GitPanel>(<GitPanel {...props} />); - panel = panelWrapper.instance(); + render(<GitPanel {...props} />); }); - it('should commit when commit message is provided', async () => { + it.skip('should commit when commit message is provided', async () => { configSpy.mockResolvedValue({ options: commitUser }); - panel.setState({ commitSummary, commitDescription }); - await panel.commitFiles(); - expect(configSpy).toHaveBeenCalledTimes(1); + + await userEvent.type(screen.getAllByRole('textbox')[0], commitSummary); + await userEvent.type( + screen.getAllByRole('textbox')[1], + commitDescription + ); + + await userEvent.click(screen.getByRole('button', { name: 'Commit' })); + // console.log( + // screen.getByRole('button', { name: 'Commit' }).parentElement!.innerHTML + // ); + + await waitFor(() => { + expect(configSpy).toHaveBeenCalledTimes(1); + }); + expect(commitSpy).toHaveBeenCalledTimes(1); expect(commitSpy).toHaveBeenCalledWith( commitSummary + '\n\n' + commitDescription + '\n', @@ -166,28 +202,27 @@ describe('GitPanel', () => { ); // Only erase commit message upon success - expect(panel.state.commitSummary).toEqual(''); - expect(panel.state.commitDescription).toEqual(''); + expect(screen.getAllByRole('textbox')[0]).toHaveValue(''); + expect(screen.getAllByRole('textbox')[1]).toHaveValue(''); }); it('should not commit without a commit message', async () => { - await panel.commitFiles(); + await userEvent.click(screen.getByRole('button', { name: 'Commit' })); expect(configSpy).not.toHaveBeenCalled(); expect(commitSpy).not.toHaveBeenCalled(); }); - it('should prompt for user identity if explicitly configured', async () => { + it.skip('should prompt for user identity if explicitly configured', async () => { configSpy.mockResolvedValue({ options: commitUser }); props.settings = MockSettings(false, true) as any; - const panelWrapper = shallow<GitPanel>(<GitPanel {...props} />); - panel = panelWrapper.instance(); + render(<GitPanel {...props} />); mockUtils.showDialog.mockResolvedValue(dialogValue); - panel.setState({ commitSummary }); + await userEvent.type(screen.getAllByRole('textbox')[0], commitSummary); + await userEvent.click(screen.getByRole('button', { name: 'Commit' })); - await panel.commitFiles(); expect(configSpy).toHaveBeenCalledTimes(1); expect(configSpy.mock.calls[0]).toHaveLength(0); @@ -196,34 +231,39 @@ describe('GitPanel', () => { expect(commitSpy).toHaveBeenCalledWith(commitSummary, false, author); }); - it('should prompt for user identity if user.name is not set', async () => { + it.skip('should prompt for user identity if user.name is not set', async () => { configSpy.mockImplementation(mockConfigImplementation('user.email')); mockUtils.showDialog.mockResolvedValue(dialogValue); - panel.setState({ commitSummary }); + await userEvent.type(screen.getAllByRole('textbox')[0], commitSummary); + await userEvent.click(screen.getByRole('button', { name: 'Commit' })); - await panel.commitFiles(); - expect(configSpy).toHaveBeenCalledTimes(2); + await waitFor(() => { + expect(configSpy).toHaveBeenCalledTimes(2); + }); expect(configSpy.mock.calls[0]).toHaveLength(0); expect(configSpy.mock.calls[1]).toEqual([commitUser]); expect(commitSpy).toHaveBeenCalledTimes(1); expect(commitSpy).toHaveBeenCalledWith(commitSummary, false, null); }); - it('should prompt for user identity if user.email is not set', async () => { + it.skip('should prompt for user identity if user.email is not set', async () => { configSpy.mockImplementation(mockConfigImplementation('user.name')); mockUtils.showDialog.mockResolvedValue(dialogValue); - panel.setState({ commitSummary }); - await panel.commitFiles(); - expect(configSpy).toHaveBeenCalledTimes(2); + await userEvent.type(screen.getAllByRole('textbox')[0], commitSummary); + await userEvent.click(screen.getByRole('button', { name: 'Commit' })); + + await waitFor(() => { + expect(configSpy).toHaveBeenCalledTimes(2); + }); expect(configSpy.mock.calls[0]).toHaveLength(0); expect(configSpy.mock.calls[1]).toEqual([commitUser]); expect(commitSpy).toHaveBeenCalledTimes(1); expect(commitSpy).toHaveBeenCalledWith(commitSummary, false, null); }); - it('should not commit if no user identity is set and the user rejects the dialog', async () => { + it.skip('should not commit if no user identity is set and the user rejects the dialog', async () => { configSpy.mockResolvedValue({ options: {} }); mockUtils.showDialog.mockResolvedValue({ button: { @@ -234,29 +274,25 @@ describe('GitPanel', () => { value: null }); - panel.setState({ commitSummary, commitDescription }); - try { - await panel.commitFiles(); - } catch (error) { - expect(error.message).toEqual( - 'Failed to set your identity. User refused to set identity.' - ); - } - expect(configSpy).toHaveBeenCalledTimes(1); + await userEvent.type(screen.getAllByRole('textbox')[0], commitSummary); + await userEvent.type( + screen.getAllByRole('textbox')[1], + commitDescription + ); + await userEvent.click(screen.getByRole('button', { name: 'Commit' })); + + await waitFor(() => expect(configSpy).toHaveBeenCalledTimes(1)); expect(configSpy).toHaveBeenCalledWith(); expect(commitSpy).not.toHaveBeenCalled(); // Should not erase commit message - expect(panel.state.commitSummary).toEqual(commitSummary); - expect(panel.state.commitDescription).toEqual(commitDescription); + expect(screen.getAllByRole('textbox')[0]).toHaveValue(commitSummary); + expect(screen.getAllByRole('textbox')[1]).toHaveValue(commitDescription); }); }); describe('#render()', () => { beforeEach(() => { - props.commands = { - keyBindings: { find: jest.fn() } - } as any; props.model = { branches: [], status: {}, @@ -289,6 +325,9 @@ describe('GitPanel', () => { }, dirtyFilesStatusChanged: { connect: jest.fn() + }, + remoteChanged: { + connect: jest.fn() } } as any; @@ -306,12 +345,12 @@ describe('GitPanel', () => { upstream: 'origin' } ]; + (props.model as any).pathRepository = '/path'; - const panel = shallow(<GitPanel {...props} />); - panel.setState({ - repository: '/path' - }); - expect(panel.find(CommitBox).prop('label')).toEqual('Commit and Push'); + render(<GitPanel {...props} />); + + const buttons = screen.getAllByRole('button'); + expect(buttons[buttons.length - 2]).toHaveTextContent('Commit and Push'); }); it('should render Commit if there is no remote branch', () => { @@ -325,12 +364,12 @@ describe('GitPanel', () => { upstream: null } ]; + (props.model as any).pathRepository = '/path'; - const panel = shallow(<GitPanel {...props} />); - panel.setState({ - repository: '/path' - }); - expect(panel.find(CommitBox).prop('label')).toEqual('Commit'); + render(<GitPanel {...props} />); + + const buttons = screen.getAllByRole('button'); + expect(buttons[buttons.length - 2]).toHaveTextContent('Commit'); }); it('should render Commit if there is a remote branch but commitAndPush is false', () => { @@ -344,13 +383,13 @@ describe('GitPanel', () => { upstream: 'origin' } ]; + (props.model as any).pathRepository = '/path'; props.settings = MockSettings(false) as any; - const panel = shallow(<GitPanel {...props} />); - panel.setState({ - repository: '/path' - }); - expect(panel.find(CommitBox).prop('label')).toEqual('Commit'); + render(<GitPanel {...props} />); + + const buttons = screen.getAllByRole('button'); + expect(buttons[buttons.length - 2]).toHaveTextContent('Commit'); }); }); }); diff --git a/src/__tests__/test-components/HistorySideBar.spec.tsx b/src/__tests__/test-components/HistorySideBar.spec.tsx index f140f06fa..e6f9dfe1b 100644 --- a/src/__tests__/test-components/HistorySideBar.spec.tsx +++ b/src/__tests__/test-components/HistorySideBar.spec.tsx @@ -1,17 +1,20 @@ +import { DocumentRegistry } from '@jupyterlab/docregistry'; +import { nullTranslator } from '@jupyterlab/translation'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; +import 'jest'; import * as React from 'react'; -import { shallow } from 'enzyme'; import { HistorySideBar, IHistorySideBarProps } from '../../components/HistorySideBar'; -import 'jest'; - -import { PastCommitNode } from '../../components/PastCommitNode'; import { GitExtension } from '../../model'; -import { nullTranslator } from '@jupyterlab/translation'; -import { FileItem } from '../../components/FileItem'; -import { DocumentRegistry } from '@jupyterlab/docregistry'; -import { SinglePastCommitInfo } from '../../components/SinglePastCommitInfo'; +import { fileStyle } from '../../style/FileItemStyle'; +import { + commitBodyClass, + commitWrapperClass +} from '../../style/PastCommitNode'; +import { selectedHistoryFileStyle } from '../../style/HistorySideBarStyle'; describe('HistorySideBar', () => { const trans = nullTranslator.load('jupyterlab-git'); @@ -19,11 +22,11 @@ describe('HistorySideBar', () => { const props: IHistorySideBarProps = { commits: [ { - commit: null, - author: null, - date: null, - commit_msg: null, - pre_commits: [null] + commit: '1234567890', + author: null as any, + date: null as any, + commit_msg: null as any, + pre_commits: [null as any] } ], branches: [], @@ -31,12 +34,12 @@ describe('HistorySideBar', () => { model: { selectedHistoryFile: null } as GitExtension, - commands: null, + commands: null as any, trans, referenceCommit: null, challengerCommit: null, - onSelectForCompare: _ => async _ => null, - onCompareWithSelected: _ => async _ => null + onSelectForCompare: (() => async () => null) as any, + onCompareWithSelected: (() => async () => null) as any }; beforeEach(() => { @@ -51,21 +54,34 @@ describe('HistorySideBar', () => { }); it('renders the commit nodes', () => { - const historySideBar = shallow(<HistorySideBar {...props} />); - expect(historySideBar.find(PastCommitNode)).toHaveLength(1); - expect(historySideBar.find(SinglePastCommitInfo)).toHaveLength(1); + render(<HistorySideBar {...props} />); + + const pastCommitNodes = screen + .getAllByRole('listitem') + .filter(el => el.classList.contains(commitWrapperClass)); + expect(pastCommitNodes).toHaveLength(1); + const singlePastCommit = Array.from( + pastCommitNodes[0].querySelectorAll(`.${commitBodyClass}`) + ); + expect(singlePastCommit).toHaveLength(1); // Selected history file element - expect(historySideBar.find(FileItem)).toHaveLength(0); + expect( + Array.from(singlePastCommit[0].querySelectorAll(`.${fileStyle}`)) + ).toHaveLength(0); }); it('shows a message if no commits are found', () => { const propsWithoutCommits: IHistorySideBarProps = { ...props, commits: [] }; - const historySideBar = shallow(<HistorySideBar {...propsWithoutCommits} />); - expect(historySideBar.find(PastCommitNode)).toHaveLength(0); + render(<HistorySideBar {...propsWithoutCommits} />); + + expect( + screen + .getAllByRole('listitem') + .filter(el => el.classList.contains(commitWrapperClass)) + ).toHaveLength(0); - const noHistoryFound = historySideBar.find('li'); - expect(noHistoryFound).toHaveLength(1); - expect(noHistoryFound.text()).toEqual('No history found.'); + expect(screen.getAllByRole('listitem')).toHaveLength(1); + expect(screen.getByRole('listitem')).toHaveTextContent('No history found.'); }); it('correctly shows the selected history file', () => { @@ -84,15 +100,21 @@ describe('HistorySideBar', () => { } as GitExtension }; - const historySideBar = shallow( - <HistorySideBar {...propsWithSelectedFile} /> + render(<HistorySideBar {...propsWithSelectedFile} />); + + const selectedHistoryFile = Array.from( + screen + .getAllByRole('list')[0] + .querySelectorAll(`.${selectedHistoryFileStyle}`) ); - const selectedHistoryFile = historySideBar.find(FileItem); expect(selectedHistoryFile).toHaveLength(1); - expect(selectedHistoryFile.prop('file')).toEqual( - propsWithSelectedFile.model.selectedHistoryFile - ); + expect(selectedHistoryFile[0]).toHaveTextContent('filepath/to'); // Only renders with repository history - expect(historySideBar.find(SinglePastCommitInfo)).toHaveLength(0); + const singlePastCommit = Array.from( + screen + .getAllByRole('listitem')[0] + .querySelectorAll(`.${commitBodyClass}`)[0].children + ); + expect(singlePastCommit).toHaveLength(0); }); }); diff --git a/src/__tests__/test-components/ManageRemoteDialogue.spec.tsx b/src/__tests__/test-components/ManageRemoteDialogue.spec.tsx index 84b446883..06d3e83ff 100644 --- a/src/__tests__/test-components/ManageRemoteDialogue.spec.tsx +++ b/src/__tests__/test-components/ManageRemoteDialogue.spec.tsx @@ -1,22 +1,20 @@ -import { shallow } from 'enzyme'; +import { nullTranslator } from '@jupyterlab/translation'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; import 'jest'; import * as React from 'react'; -import { ActionButton } from '../../components/ActionButton'; import { - ManageRemoteDialogue, IManageRemoteDialogueProps, - IManageRemoteDialogueState + ManageRemoteDialogue } from '../../components/ManageRemoteDialogue'; import * as git from '../../git'; import { GitExtension } from '../../model'; -import { createButtonClass } from '../../style/NewBranchDialog'; import { - mockedRequestAPI, + DEFAULT_REPOSITORY_PATH, defaultMockedResponses, - DEFAULT_REPOSITORY_PATH + mockedRequestAPI } from '../utils'; -import ClearIcon from '@mui/icons-material/Clear'; -import { nullTranslator } from '@jupyterlab/translation'; jest.mock('../../git'); jest.mock('@jupyterlab/apputils'); @@ -63,7 +61,7 @@ describe('ManageRemoteDialogue', () => { } } } - }) + }) as any ); model = await createModel(); @@ -81,108 +79,69 @@ describe('ManageRemoteDialogue', () => { } describe('constructor', () => { - it('should return a new instance with initial state', () => { - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); - expect(remoteDialogue.instance()).toBeInstanceOf(ManageRemoteDialogue); - const initialState: IManageRemoteDialogueState = { - newRemote: { - name: '', - url: '' - }, - existingRemotes: null - }; - expect(remoteDialogue.state()).toEqual(initialState); - }); - it('should set the correct state after mounting', async () => { const spyGitGetRemotes = jest.spyOn(GitExtension.prototype, 'getRemotes'); const spyComponentDidMount = jest.spyOn( ManageRemoteDialogue.prototype, 'componentDidMount' ); - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); - await remoteDialogue.instance().componentDidMount(); - expect(remoteDialogue.state()).toEqual({ - newRemote: { - name: '', - url: '' - }, - existingRemotes: REMOTES - }); - expect(spyGitGetRemotes).toHaveBeenCalledTimes(2); - expect(spyComponentDidMount).toHaveBeenCalledTimes(2); + render(<ManageRemoteDialogue {...createProps()} />); + expect(spyGitGetRemotes).toHaveBeenCalledTimes(1); + expect(spyComponentDidMount).toHaveBeenCalledTimes(1); }); }); describe('render', () => { it('should display a title for the dialogue "Manage Remotes"', () => { - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); - const node = remoteDialogue.find('p').first(); - expect(node.text()).toEqual('Manage Remotes'); + render(<ManageRemoteDialogue {...createProps()} />); + const node = screen.getByRole('dialog').querySelector('p'); + expect(node?.textContent).toEqual('Manage Remotes'); }); it('should display a button to close the dialogue', () => { - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); - const nodes = remoteDialogue.find(ClearIcon); - expect(nodes.length).toEqual(1); + render(<ManageRemoteDialogue {...createProps()} />); + expect(screen.getAllByTitle('Close this dialog')).toHaveLength(1); }); it('should display two input boxes for entering new remote name and url', () => { - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); - const nameInput = remoteDialogue.find('input[placeholder="name"]'); - const urlInput = remoteDialogue.find( - 'input[placeholder="Remote Git repository URL"]' - ); - expect(nameInput.length).toEqual(1); - expect(urlInput.length).toEqual(1); + render(<ManageRemoteDialogue {...createProps()} />); + const nameInput = screen.getByPlaceholderText('name'); + const urlInput = screen.getByPlaceholderText('Remote Git repository URL'); + expect(nameInput).toBeDefined(); + expect(urlInput).toBeDefined(); }); it('should display a button to add a new remote', () => { - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); - const node = remoteDialogue.find(`.${createButtonClass}`).first(); - expect(node.prop('value')).toEqual('Add'); + render(<ManageRemoteDialogue {...createProps()} />); + + expect(screen.getByRole('button', { name: 'Add' })).toBeDefined(); }); it('should display buttons to remove existing remotes', async () => { - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); - await remoteDialogue.instance().componentDidMount(); - const nodes = remoteDialogue.find(ActionButton); - expect(nodes.length).toEqual(REMOTES.length); + render(<ManageRemoteDialogue {...createProps()} />); + await screen.findByText(REMOTES[0].name); + expect( + screen.getAllByRole('button', { name: 'Remove this remote' }) + ).toHaveLength(REMOTES.length); }); }); describe('functionality', () => { it('should add a new remote', async () => { - const remoteDialogue = shallow( - <ManageRemoteDialogue {...createProps()} /> - ); + render(<ManageRemoteDialogue {...createProps()} />); const newRemote = { name: 'newRemote', url: 'newremote.com' }; - await remoteDialogue.setState({ - newRemote - }); + + await userEvent.type(screen.getByPlaceholderText('name'), newRemote.name); + await userEvent.type( + screen.getByPlaceholderText('Remote Git repository URL'), + newRemote.url + ); const spyGitAddRemote = jest.spyOn(GitExtension.prototype, 'addRemote'); - const addRemoteButton = remoteDialogue - .find(`.${createButtonClass}`) - .first(); - addRemoteButton.simulate('click'); + await userEvent.click(screen.getByRole('button', { name: 'Add' })); expect(spyGitAddRemote).toHaveBeenCalledTimes(1); expect(spyGitAddRemote).toHaveBeenCalledWith( newRemote.url, diff --git a/src/__tests__/test-components/NotebookDiff.spec.tsx b/src/__tests__/test-components/NotebookDiff.spec.tsx index 02f4d4386..98a5a5477 100644 --- a/src/__tests__/test-components/NotebookDiff.spec.tsx +++ b/src/__tests__/test-components/NotebookDiff.spec.tsx @@ -4,6 +4,7 @@ import { NotebookDiff, ROOT_CLASS } from '../../components/diff/NotebookDiff'; import { requestAPI } from '../../git'; import { Git } from '../../tokens'; import * as diffResponse from './data/nbDiffResponse.json'; +import { RenderMimeRegistry } from '@jupyterlab/rendermime'; jest.mock('../../git'); @@ -28,7 +29,7 @@ describe('NotebookDiff', () => { (requestAPI as jest.Mock).mockResolvedValueOnce(diffResponse); // When - const widget = new NotebookDiff(model, null); + const widget = new NotebookDiff(model, new RenderMimeRegistry()); await widget.ready; // Then @@ -36,7 +37,7 @@ describe('NotebookDiff', () => { const terminateTest = new Promise(resolve => { resolveTest = resolve; }); - setImmediate(() => { + setTimeout(() => { expect(requestAPI).toHaveBeenCalled(); expect(requestAPI).toBeCalledWith('diffnotebook', 'POST', { currentContent: 'challenger', @@ -48,7 +49,7 @@ describe('NotebookDiff', () => { expect(widget.node.querySelectorAll(`.${ROOT_CLASS}`)).toHaveLength(1); expect(widget.node.querySelectorAll('.jp-Notebook-diff')).toHaveLength(1); resolveTest(); - }); + }, 1); await terminateTest; }); @@ -77,7 +78,7 @@ describe('NotebookDiff', () => { ); // When - const widget = new NotebookDiff(model, null); + const widget = new NotebookDiff(model, new RenderMimeRegistry()); await widget.ready; // Then @@ -85,17 +86,17 @@ describe('NotebookDiff', () => { const terminateTest = new Promise(resolve => { resolveTest = resolve; }); - setImmediate(() => { + setTimeout(() => { expect(requestAPI).toHaveBeenCalled(); expect(requestAPI).toBeCalledWith('diffnotebook', 'POST', { currentContent: 'challenger', previousContent: 'reference' }); expect( - widget.node.querySelector('.jp-git-diff-error').innerHTML + widget.node.querySelector('.jp-git-diff-error')!.innerHTML ).toContain('TEST_ERROR_MESSAGE'); resolveTest(); - }); + }, 1); await terminateTest; }); }); diff --git a/src/__tests__/test-components/PastCommitNode.spec.tsx b/src/__tests__/test-components/PastCommitNode.spec.tsx index 9956f1fa9..8cdaaa495 100644 --- a/src/__tests__/test-components/PastCommitNode.spec.tsx +++ b/src/__tests__/test-components/PastCommitNode.spec.tsx @@ -1,5 +1,7 @@ import { nullTranslator } from '@jupyterlab/translation'; -import { shallow } from 'enzyme'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; import 'jest'; import * as React from 'react'; import { @@ -71,7 +73,7 @@ describe('PastCommitNode', () => { const tags: Git.ITag[] = notMatchingTags.concat(matchingTags); const toggleCommitExpansion = jest.fn(); const props: IPastCommitNodeProps = { - model: null, + model: null as any, commit: { commit: '2414721b194453f058079d897d13c4e377f92dc6', author: 'author', @@ -81,47 +83,47 @@ describe('PastCommitNode', () => { }, branches: branches, tagsList: tags, - commands: null, + commands: null as any, trans, - onCompareWithSelected: null, - onSelectForCompare: null, + onCompareWithSelected: null as any, + onSelectForCompare: null as any, expanded: false, toggleCommitExpansion, setRef: () => null }; test('Includes commit info', () => { - const node = shallow(<PastCommitNode {...props} />); - expect(node.text()).toMatch(props.commit.author); - expect(node.text()).toMatch(props.commit.commit.slice(0, 7)); - expect(node.text()).toMatch(props.commit.date); - expect(node.text()).toMatch(props.commit.commit_msg); + render(<PastCommitNode {...props} />); + expect(screen.getByText(props.commit.author)).toBeDefined(); + expect(screen.getByText(props.commit.commit.slice(0, 7))).toBeDefined(); + expect(screen.getByText(props.commit.date)).toBeDefined(); + expect(screen.getByText(props.commit.commit_msg)).toBeDefined(); }); test('Includes only relevant branch info', () => { - const node = shallow(<PastCommitNode {...props} />); - expect(node.text()).toMatch('name3'); - expect(node.text()).toMatch('name4'); - expect(node.text()).not.toMatch('name1'); - expect(node.text()).not.toMatch('name2'); + render(<PastCommitNode {...props} />); + expect(screen.getByText('name3')).toBeDefined(); + expect(screen.getByText('name4')).toBeDefined(); + expect(screen.queryByText('name1')).toBeNull(); + expect(screen.queryByText('name2')).toBeNull(); }); test('Includes only relevant tag info', () => { - const node = shallow(<PastCommitNode {...props} />); - expect(node.text()).toMatch('1.0.0'); - expect(node.text()).toMatch('feature-1'); - expect(node.text()).not.toMatch('feature-2'); - expect(node.text()).not.toMatch('patch-007'); + render(<PastCommitNode {...props} />); + expect(screen.getByText('1.0.0')).toBeDefined(); + expect(screen.getByText('feature-1')).toBeDefined(); + expect(screen.queryByText('feature-2')).toBeNull(); + expect(screen.queryByText('patch-007')).toBeNull(); }); - test('Toggle show details', () => { + test('Toggle show details', async () => { // simulates SinglePastCommitInfo child - const node = shallow( + render( <PastCommitNode {...props}> <div id="singlePastCommitInfo"></div> </PastCommitNode> ); - node.simulate('click'); + await userEvent.click(screen.getByRole('listitem')); expect(toggleCommitExpansion).toBeCalledTimes(1); expect(toggleCommitExpansion).toHaveBeenCalledWith(props.commit.commit); }); diff --git a/src/__tests__/test-components/PlainTextDiff.spec.tsx b/src/__tests__/test-components/PlainTextDiff.spec.tsx index a155c20bb..5d226f04c 100644 --- a/src/__tests__/test-components/PlainTextDiff.spec.tsx +++ b/src/__tests__/test-components/PlainTextDiff.spec.tsx @@ -1,11 +1,9 @@ import 'jest'; -import { mergeView } from '../../components/diff/mergeview'; import { DiffModel } from '../../components/diff/model'; import { PlainTextDiff } from '../../components/diff/PlainTextDiff'; import { Git } from '../../tokens'; jest.mock('../../git'); -jest.mock('../../components/diff/mergeview'); describe('PlainTextDiff', () => { it('should render file diff', async () => { @@ -25,10 +23,8 @@ describe('PlainTextDiff', () => { repositoryPath: 'path' }); - const mockMergeView = mergeView as jest.Mocked<typeof mergeView>; - // When - const widget = new PlainTextDiff(model); + const widget = new PlainTextDiff({ model }); await widget.ready; // Then @@ -36,14 +32,12 @@ describe('PlainTextDiff', () => { const terminateTest = new Promise(resolve => { resolveTest = resolve; }); - setImmediate(() => { + setTimeout(() => { expect(widget.node.querySelectorAll('.jp-git-diff-error')).toHaveLength( 0 ); - // merge view was not called as it happens when the widget got attach - expect(mockMergeView).not.toHaveBeenCalled(); resolveTest(); - }); + }, 0); await terminateTest; }); @@ -64,10 +58,8 @@ describe('PlainTextDiff', () => { repositoryPath: 'path' }); - const mockMergeView = mergeView as jest.Mocked<typeof mergeView>; - // When - const widget = new PlainTextDiff(model); + const widget = new PlainTextDiff({ model }); await widget.ready; // Then @@ -75,13 +67,12 @@ describe('PlainTextDiff', () => { const terminateTest = new Promise(resolve => { resolveTest = resolve; }); - setImmediate(() => { + setTimeout(() => { expect( - widget.node.querySelector('.jp-git-diff-error').innerHTML + widget.node.querySelector('.jp-git-diff-error')!.innerHTML ).toContain('TEST_ERROR_MESSAGE'); - expect(mockMergeView).not.toHaveBeenCalled(); resolveTest(); - }); + }, 0); await terminateTest; }); }); diff --git a/src/__tests__/test-components/TagMenu.spec.tsx b/src/__tests__/test-components/TagMenu.spec.tsx index 596601726..edf220451 100644 --- a/src/__tests__/test-components/TagMenu.spec.tsx +++ b/src/__tests__/test-components/TagMenu.spec.tsx @@ -1,12 +1,12 @@ import { nullTranslator } from '@jupyterlab/translation'; -import ClearIcon from '@mui/icons-material/Clear'; -import { mount, shallow } from 'enzyme'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; import 'jest'; import * as React from 'react'; import { ITagMenuProps, TagMenu } from '../../components/TagMenu'; import * as git from '../../git'; import { GitExtension } from '../../model'; -import { listItemClass } from '../../style/BranchMenu'; import { IGitExtension } from '../../tokens'; import { DEFAULT_REPOSITORY_PATH, @@ -69,7 +69,7 @@ describe('TagMenu', () => { } } } - }) + }) as any ); model = await createModel(); @@ -87,105 +87,84 @@ describe('TagMenu', () => { } describe('constructor', () => { - it('should return a new instance', () => { - const menu = shallow(<TagMenu {...createProps()} />); - expect(menu.instance()).toBeInstanceOf(TagMenu); - }); - it('should set the default menu filter to an empty string', () => { - const menu = shallow(<TagMenu {...createProps()} />); - expect(menu.state('filter')).toEqual(''); - }); - - it('should set the default flag indicating whether to show a dialog to create a new tag to `false`', () => { - const menu = shallow(<TagMenu {...createProps()} />); - expect(menu.state('tagDialog')).toEqual(false); + render(<TagMenu {...createProps()} />); + expect( + screen.getByRole('textbox', { name: 'Filter tag menu' }) + ).toHaveValue(''); }); }); describe('render', () => { it('should display placeholder text for the menu filter', () => { - const component = shallow(<TagMenu {...createProps()} />); - const node = component.find('input[type="text"]').first(); - expect(node.prop('placeholder')).toEqual('Filter'); + render(<TagMenu {...createProps()} />); + expect( + screen.getByRole('textbox', { name: 'Filter tag menu' }) + ).toHaveAttribute('placeholder', 'Filter'); }); it('should set a `title` attribute on the input element to filter a tag menu', () => { - const component = shallow(<TagMenu {...createProps()} />); - const node = component.find('input[type="text"]').first(); - expect(node.prop('title').length > 0).toEqual(true); - }); - - it('should display a button to clear the menu filter once a filter is provided', () => { - const component = shallow(<TagMenu {...createProps()} />); - component.setState({ - filter: 'foo' - }); - const nodes = component.find(ClearIcon); - expect(nodes.length).toEqual(1); + render(<TagMenu {...createProps()} />); + expect(screen.getByRole('textbox')).toHaveAttribute( + 'title', + 'Filter tag menu' + ); }); - it('should set a `title` on the button to clear the menu filter', () => { - const component = shallow(<TagMenu {...createProps()} />); - component.setState({ - filter: 'foo' - }); - const html = component.find(ClearIcon).first().html(); - expect(html.includes('<title>')).toEqual(true); + it('should display a button to clear the menu filter once a filter is provided', async () => { + render(<TagMenu {...createProps()} />); + await userEvent.type(screen.getByRole('textbox'), 'foo'); + expect(screen.getAllByTitle('Clear the current filter').length).toEqual( + 1 + ); }); it('should display a button to create a new tag', () => { - const component = shallow(<TagMenu {...createProps()} />); - const node = component.find('input[type="button"]').first(); - expect(node.prop('value')).toEqual('New Tag'); + render(<TagMenu {...createProps()} />); + expect(screen.getByRole('button', { name: 'New Tag' })).toBeDefined(); }); it('should set a `title` attribute on the button to create a new tag', () => { - const component = shallow(<TagMenu {...createProps()} />); - const node = component.find('input[type="button"]').first(); - expect(node.prop('title').length > 0).toEqual(true); + render(<TagMenu {...createProps()} />); + expect(screen.getByRole('button', { name: 'New Tag' })).toHaveAttribute( + 'title', + 'Create a new tag' + ); }); it('should not, by default, show a dialog to create a new tag', () => { - const component = shallow(<TagMenu {...createProps()} />); - const node = component.find('NewTagDialogBox').first(); - expect(node.prop('open')).toEqual(false); + render(<TagMenu {...createProps()} />); + + expect(screen.queryByRole('dialog')).toBeNull(); }); - it('should show a dialog to create a new tag when the flag indicating whether to show the dialog is `true`', () => { - const component = shallow(<TagMenu {...createProps()} />); - component.setState({ - tagDialog: true - }); - const node = component.find('NewTagDialogBox').first(); - expect(node.prop('open')).toEqual(true); + it('should show a dialog to create a new tag when the flag indicating whether to show the dialog is `true`', async () => { + render(<TagMenu {...createProps({ branching: true })} />); + + await userEvent.click(screen.getByRole('button', { name: 'New Tag' })); + + expect(screen.getByRole('dialog')).toBeDefined(); }); }); describe('switch tag', () => { - it('should not switch to a specified tag upon clicking its corresponding element when branching is disabled', () => { + it('should not switch to a specified tag upon clicking its corresponding element when branching is disabled', async () => { const spy = jest.spyOn(GitExtension.prototype, 'checkoutTag'); - const component = mount(<TagMenu {...createProps()} />); - const nodes = component.find( - `.${listItemClass}[title*="${TAGS[1].name}"]` - ); - nodes.at(0).simulate('click'); + render(<TagMenu {...createProps()} />); + + await userEvent.click(screen.getByTitle(new RegExp(TAGS[1].name))); expect(spy).toHaveBeenCalledTimes(0); spy.mockRestore(); }); - it('should switch to a specified tag upon clicking its corresponding element when branching is enabled', () => { + it('should switch to a specified tag upon clicking its corresponding element when branching is enabled', async () => { const spy = jest.spyOn(GitExtension.prototype, 'checkoutTag'); - const component = mount( - <TagMenu {...createProps({ branching: true })} /> - ); - const nodes = component.find( - `.${listItemClass}[title*="${TAGS[1].name}"]` - ); - nodes.at(0).simulate('click'); + render(<TagMenu {...createProps({ branching: true })} />); + + await userEvent.click(screen.getByTitle(new RegExp(TAGS[1].name))); expect(spy).toHaveBeenCalledTimes(1); expect(spy).toHaveBeenCalledWith(TAGS[1].name); @@ -195,30 +174,26 @@ describe('TagMenu', () => { }); describe('create tag', () => { - it('should not allow creating a new tag when branching is disabled', () => { + it('should not allow creating a new tag when branching is disabled', async () => { const spy = jest.spyOn(GitExtension.prototype, 'setTag'); - const component = shallow(<TagMenu {...createProps()} />); + render(<TagMenu {...createProps()} />); - const node = component.find('input[type="button"]').first(); - node.simulate('click'); + await userEvent.click(screen.getByRole('button', { name: 'New Tag' })); - expect(component.state('tagDialog')).toEqual(false); + expect(screen.queryByRole('dialog')).toBeNull(); expect(spy).toHaveBeenCalledTimes(0); spy.mockRestore(); }); - it('should display a dialog to create a new tag when branching is enabled and the new tag button is clicked', () => { + it('should display a dialog to create a new tag when branching is enabled and the new tag button is clicked', async () => { const spy = jest.spyOn(GitExtension.prototype, 'setTag'); - const component = shallow( - <TagMenu {...createProps({ branching: true })} /> - ); + render(<TagMenu {...createProps({ branching: true })} />); - const node = component.find('input[type="button"]').first(); - node.simulate('click'); + await userEvent.click(screen.getByRole('button', { name: 'New Tag' })); - expect(component.state('tagDialog')).toEqual(true); + expect(screen.getByRole('dialog')).toBeDefined(); expect(spy).toHaveBeenCalledTimes(0); spy.mockRestore(); }); diff --git a/src/__tests__/test-components/Toolbar.spec.tsx b/src/__tests__/test-components/Toolbar.spec.tsx index 01137ae81..9bd0b85c7 100644 --- a/src/__tests__/test-components/Toolbar.spec.tsx +++ b/src/__tests__/test-components/Toolbar.spec.tsx @@ -1,14 +1,13 @@ import { nullTranslator } from '@jupyterlab/translation'; -import { refreshIcon } from '@jupyterlab/ui-components'; -import { shallow } from 'enzyme'; +import '@testing-library/jest-dom'; +import { render, screen } from '@testing-library/react'; +import userEvent from '@testing-library/user-event'; import 'jest'; import * as React from 'react'; -import { ActionButton } from '../../components/ActionButton'; import { IToolbarProps, Toolbar } from '../../components/Toolbar'; import * as git from '../../git'; import { GitExtension } from '../../model'; -import { pullIcon, pushIcon } from '../../style/icons'; -import { toolbarMenuButtonClass } from '../../style/Toolbar'; +import { badgeClass } from '../../style/Toolbar'; import { DEFAULT_REPOSITORY_PATH, mockedRequestAPI } from '../utils'; import { CommandIDs } from '../../tokens'; @@ -49,7 +48,7 @@ describe('Toolbar', () => { ], tagsList: model.tagsList, pastCommits: [], - repository: model.pathRepository, + repository: model.pathRepository!, model: model, branching: false, nCommitsAhead: 0, @@ -66,164 +65,122 @@ describe('Toolbar', () => { jest.restoreAllMocks(); const mock = git as jest.Mocked<typeof git>; - mock.requestAPI.mockImplementation(mockedRequestAPI()); + mock.requestAPI.mockImplementation(mockedRequestAPI() as any); model = await createModel(); }); describe('constructor', () => { - it('should return a new instance', () => { - const el = shallow(<Toolbar {...createProps()} />); - expect(el.instance()).toBeInstanceOf(Toolbar); - }); - it('should set the default flag indicating whether to show a branch menu to `false`', () => { - const el = shallow<Toolbar>(<Toolbar {...createProps()} />); - expect(el.state().branchMenu).toEqual(false); + render(<Toolbar {...createProps()} />); + + expect(screen.queryByText('Branches')).toBeNull(); }); }); describe('render', () => { it('should display a button to pull the latest changes', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const nodes = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pullIcon); - expect(nodes.length).toEqual(1); + render(<Toolbar {...createProps()} />); expect( - toolbar.find('[data-test-id="pull-badge"]').prop('invisible') - ).toEqual(true); - }); - - it('should set the `title` attribute on the button to pull the latest changes', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pullIcon); + screen.getAllByRole('button', { name: 'Pull latest changes' }) + ).toBeDefined(); - expect(button.prop('title')).toEqual('Pull latest changes'); + expect( + screen + .getByRole('button', { name: 'Pull latest changes' }) + .parentElement?.querySelector(`.${badgeClass} > .MuiBadge-badge`) + ).toHaveClass('MuiBadge-invisible'); }); it('should display a badge on pull icon if behind', () => { - const toolbar = shallow<Toolbar>( - <Toolbar {...createProps({ nCommitsBehind: 1 })} /> - ); + render(<Toolbar {...createProps({ nCommitsBehind: 1 })} />); expect( - toolbar.find('[data-test-id="pull-badge"]').prop('invisible') - ).toEqual(false); + screen + .getByRole('button', { name: /^Pull latest changes/ }) + .parentElement?.querySelector(`.${badgeClass} > .MuiBadge-badge`) + ).not.toHaveClass('MuiBadge-invisible'); }); it('should display a button to push the latest changes', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const nodes = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pushIcon); - expect(nodes.length).toEqual(1); + render(<Toolbar {...createProps()} />); expect( - toolbar.find('[data-test-id="push-badge"]').prop('invisible') - ).toEqual(true); - }); - - it('should set the `title` attribute on the button to push the latest changes', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pushIcon) - .first(); + screen.getAllByRole('button', { name: 'Push committed changes' }) + ).toBeDefined(); - expect(button.prop('title')).toEqual('Push committed changes'); + expect( + screen + .getByRole('button', { name: 'Push committed changes' }) + .parentElement?.querySelector(`.${badgeClass} > .MuiBadge-badge`) + ).toHaveClass('MuiBadge-invisible'); }); - it('should display a badge on pull icon if behind', () => { - const toolbar = shallow<Toolbar>( - <Toolbar {...createProps({ nCommitsAhead: 1 })} /> - ); + it('should display a badge on push icon if behind', () => { + render(<Toolbar {...createProps({ nCommitsAhead: 1 })} />); expect( - toolbar.find('[data-test-id="push-badge"]').prop('invisible') - ).toEqual(false); + screen + .getByRole('button', { name: /^Push committed changes/ }) + .parentElement?.querySelector(`.${badgeClass} > .MuiBadge-badge`) + ).not.toHaveClass('MuiBadge-invisible'); }); it('should display a button to refresh the current repository', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const nodes = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === refreshIcon); - - expect(nodes.length).toEqual(1); - }); - - it('should set the `title` attribute on the button to refresh the current repository', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === refreshIcon) - .first(); + render(<Toolbar {...createProps()} />); - expect(button.prop('title')).toEqual( - 'Refresh the repository to detect local and remote changes' - ); + expect( + screen.getAllByRole('button', { + name: 'Refresh the repository to detect local and remote changes' + }) + ).toBeDefined(); }); it('should display a button to toggle a repository menu', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar.find(`.${toolbarMenuButtonClass}`).first(); + render(<Toolbar {...createProps()} />); - const text = button.text(); - expect(text.includes('Current Repository')).toEqual(true); - }); - - it('should set the `title` attribute on the button to toggle a repository menu', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar.find(`.${toolbarMenuButtonClass}`).first(); - - const bool = button.prop('title').includes('Current repository: '); - expect(bool).toEqual(true); + expect(screen.getByText('Current Repository')).toBeDefined(); }); it('should display a button to toggle a branch menu', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar.find(`.${toolbarMenuButtonClass}`).at(1); + render(<Toolbar {...createProps()} />); - const text = button.text(); - expect(text.includes('Current Branch')).toEqual(true); + expect(screen.getByText('Current Branch')).toBeDefined(); }); it('should set the `title` attribute on the button to toggle a branch menu', () => { const currentBranch = 'main'; - const toolbar = shallow<Toolbar>( - <Toolbar {...createProps({ currentBranch })} /> - ); - const button = toolbar.find(`.${toolbarMenuButtonClass}`).at(1); + render(<Toolbar {...createProps({ currentBranch })} />); - expect(button.prop('title')).toEqual('Manage branches and tags'); + expect( + screen.getByRole('button', { name: /^Current Branch/ }) + ).toHaveAttribute('title', 'Manage branches and tags'); }); }); describe('branch menu', () => { it('should not, by default, display a branch menu', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const nodes = toolbar.find('BranchMenu'); + render(<Toolbar {...createProps()} />); - expect(nodes.length).toEqual(0); + expect(screen.queryByText('Branches')).toBeNull(); }); - it('should display a branch menu when the button to display a branch menu is clicked', () => { - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar.find(`.${toolbarMenuButtonClass}`).at(1); + it('should display a branch menu when the button to display a branch menu is clicked', async () => { + render(<Toolbar {...createProps()} />); - button.simulate('click'); - expect(toolbar.find('BranchMenu').length).toEqual(1); + await userEvent.click( + screen.getByRole('button', { name: /^Current Branch/ }) + ); + + expect(screen.getByText('Branches')).toBeDefined(); }); }); describe('pull changes', () => { - it('should pull changes when the button to pull the latest changes is clicked', () => { + it('should pull changes when the button to pull the latest changes is clicked', async () => { const mockedExecute = jest.fn(); - const toolbar = shallow<Toolbar>( + render( <Toolbar {...createProps({ commands: { @@ -232,19 +189,18 @@ describe('Toolbar', () => { })} /> ); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pullIcon) - .first(); - button.simulate('click'); + await userEvent.click( + screen.getByRole('button', { name: 'Pull latest changes' }) + ); + expect(mockedExecute).toHaveBeenCalledTimes(1); expect(mockedExecute).toHaveBeenCalledWith(CommandIDs.gitPull); }); - it('should not pull changes when the pull button is clicked but there is no remote branch', () => { + it('should not pull changes when the pull button is clicked but there is no remote branch', async () => { const mockedExecute = jest.fn(); - const toolbar = shallow<Toolbar>( + render( <Toolbar {...createProps({ branches: [ @@ -263,20 +219,21 @@ describe('Toolbar', () => { })} /> ); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pullIcon) - .first(); - button.simulate('click'); + await userEvent.click( + screen.getAllByRole('button', { + name: 'No remote repository defined' + })[0] + ); + expect(mockedExecute).toHaveBeenCalledTimes(0); }); }); describe('push changes', () => { - it('should push changes when the button to push the latest changes is clicked', () => { + it('should push changes when the button to push the latest changes is clicked', async () => { const mockedExecute = jest.fn(); - const toolbar = shallow<Toolbar>( + render( <Toolbar {...createProps({ commands: { @@ -285,19 +242,16 @@ describe('Toolbar', () => { })} /> ); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pushIcon) - .first(); - - button.simulate('click'); + await userEvent.click( + screen.getByRole('button', { name: 'Push committed changes' }) + ); expect(mockedExecute).toHaveBeenCalledTimes(1); expect(mockedExecute).toHaveBeenCalledWith(CommandIDs.gitPush); }); - it('should not push changes when the push button is clicked but there is no remote branch', () => { + it('should not push changes when the push button is clicked but there is no remote branch', async () => { const mockedExecute = jest.fn(); - const toolbar = shallow<Toolbar>( + render( <Toolbar {...createProps({ branches: [ @@ -316,26 +270,24 @@ describe('Toolbar', () => { })} /> ); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === pushIcon) - .first(); - - button.simulate('click'); + await userEvent.click( + screen.getAllByRole('button', { + name: 'No remote repository defined' + })[1] + ); expect(mockedExecute).toHaveBeenCalledTimes(0); }); }); describe('refresh repository', () => { - it('should refresh the repository when the button to refresh the repository is clicked', () => { + it('should refresh the repository when the button to refresh the repository is clicked', async () => { const spy = jest.spyOn(model, 'refresh'); - const toolbar = shallow<Toolbar>(<Toolbar {...createProps()} />); - const button = toolbar - .find(ActionButton) - .findWhere(n => n.prop('icon') === refreshIcon) - .first(); - - button.simulate('click'); + render(<Toolbar {...createProps()} />); + await userEvent.click( + screen.getByRole('button', { + name: 'Refresh the repository to detect local and remote changes' + }) + ); expect(spy).toHaveBeenCalledTimes(1); spy.mockReset(); diff --git a/src/__tests__/utils.ts b/src/__tests__/utils.ts index 693ea5a59..428475086 100644 --- a/src/__tests__/utils.ts +++ b/src/__tests__/utils.ts @@ -10,7 +10,7 @@ export interface IMockedResponse { export interface IMockedResponses { // Folder path in URI; default = DEFAULT_REPOSITORY_PATH - path?: string; + path?: string | null; // Endpoint responses?: { [endpoint: string]: IMockedResponse; @@ -23,37 +23,42 @@ export const defaultMockedResponses: { [endpoint: string]: IMockedResponse; } = { branch: { - body: () => { - return { - code: 0, - branches: [], - current_branch: { name: '' } - }; - } + body: () => ({ + code: 0, + branches: [], + current_branch: { name: '' } + }) }, changed_files: { - body: () => { - return { - code: 0, - files: [] - }; - } + body: () => ({ + code: 0, + files: [] + }) }, show_prefix: { - body: () => { - return { - code: 0, - path: '' - }; - } + body: () => ({ + code: 0, + path: '' + }) + }, + stash: { + body: () => ({ + code: 0, + message: '', + command: '' + }) }, status: { - body: () => { - return { - code: 0, - files: [] - }; - } + body: () => ({ + code: 0, + files: [] + }) + }, + tags: { + body: () => ({ + code: 0, + tags: [] + }) } }; diff --git a/src/commandsAndMenu.tsx b/src/commandsAndMenu.tsx index 64c35cb1b..61ffbeefd 100644 --- a/src/commandsAndMenu.tsx +++ b/src/commandsAndMenu.tsx @@ -1622,7 +1622,6 @@ export function addCommands( const dialog = ReactWidget.create( <NewTagDialogBox pastCommits={[commit.commit]} - logger={logger} model={gitModel} trans={trans} open={tagDialog} @@ -1639,35 +1638,34 @@ export function addCommands( const tagName = await waitForDialog.promise; if (tagName) { - logger.log({ - level: Level.RUNNING, - message: trans.__( - "Create tag pointing to '%1'...", - commit.commit.commit_msg - ) - }); + const id = Notification.emit( + trans.__("Create tag pointing to '%1'...", commit.commit.commit_msg), + 'in-progress' + ); try { await gitModel.setTag(tagName, commit.commit.commit); } catch (err) { - logger.log({ - level: Level.ERROR, + Notification.update({ + id, message: trans.__( "Failed to create tag '%1' poining to '%2'.", tagName, commit ), - error: err as Error + type: 'error', + ...showError(err as any, trans) }); return; } - logger.log({ - level: Level.SUCCESS, + Notification.update({ + id, message: trans.__( "Created tag '%1' pointing to '%2'.", tagName, commit - ) + ), + type: 'success' }); } }, diff --git a/src/components/BranchMenu.tsx b/src/components/BranchMenu.tsx index ba2222b85..d33b2901b 100644 --- a/src/components/BranchMenu.tsx +++ b/src/components/BranchMenu.tsx @@ -275,6 +275,7 @@ export class BranchMenu extends React.Component< isActive ? activeListItemClass : null )} onClick={this._onBranchClickFactory(branch.name)} + role="listitem" style={style} > <branchIcon.react className={listItemIconClass} tag="span" /> diff --git a/src/components/FileItem.tsx b/src/components/FileItem.tsx index 9bab17ab4..832fdec96 100644 --- a/src/components/FileItem.tsx +++ b/src/components/FileItem.tsx @@ -19,19 +19,6 @@ import { fileLabelStyle } from '../style/FilePathStyle'; import { Git } from '../tokens'; import { FilePath } from './FilePath'; -// Git status codes https://git-scm.com/docs/git-status -export const STATUS_CODES = { - M: 'Modified', - A: 'Added', - D: 'Deleted', - R: 'Renamed', - C: 'Copied', - U: 'Updated', - B: 'Behind', - '?': 'Untracked', - '!': 'Ignored' -}; - /** * File marker properties */ @@ -168,8 +155,11 @@ export class FileItem extends React.PureComponent<IFileItemProps> { } }; - protected _getFileChangedLabel(change: keyof typeof STATUS_CODES): string { - return STATUS_CODES[change] || 'Unmodified'; + protected _getFileChangedLabel( + change: string, + trans: TranslationBundle + ): string { + return Private.get_status(change, trans) || trans.__('Unmodified'); } protected _getFileChangedLabelClass(change: string): string { @@ -215,8 +205,8 @@ export class FileItem extends React.PureComponent<IFileItemProps> { const status_code = file.status === 'staged' ? file.x : file.y; const status = file.status === 'unmerged' - ? 'Conflicted' - : this._getFileChangedLabel(status_code as any); + ? this.props.trans.__('Conflicted') + : this._getFileChangedLabel(status_code as any, this.props.trans); return ( <div @@ -268,3 +258,24 @@ export class FileItem extends React.PureComponent<IFileItemProps> { ); } } + +namespace Private { + let i18nCodes: Record<string, string> | null = null; + export function get_status(code: string, trans: TranslationBundle): string { + if (!i18nCodes) { + // Git status codes https://git-scm.com/docs/git-status + i18nCodes = { + M: trans.__('Modified'), + A: trans.__('Added'), + D: trans.__('Deleted'), + R: trans.__('Renamed'), + C: trans.__('Copied'), + U: trans.__('Updated'), + B: trans.__('Behind'), + '?': trans.__('Untracked'), + '!': trans.__('Ignored') + }; + } + return i18nCodes[code]; + } +} diff --git a/src/components/GitCommitGraph.tsx b/src/components/GitCommitGraph.tsx index bbec73182..4a2ac196b 100644 --- a/src/components/GitCommitGraph.tsx +++ b/src/components/GitCommitGraph.tsx @@ -114,7 +114,7 @@ export class GitCommitGraph extends React.Component<IGitCommitGraphProps> { const colour = getColour(branch); const style = { stroke: colour, - 'stroke-width': this._lineWidth, + strokeWidth: this._lineWidth, fill: 'none' }; @@ -165,7 +165,7 @@ export class GitCommitGraph extends React.Component<IGitCommitGraphProps> { const strokeWidth = 1; const style = { stroke: colour, - 'stroke-width': strokeWidth, + strokeWidth: strokeWidth, fill: colour }; @@ -222,7 +222,7 @@ export class GitCommitGraph extends React.Component<IGitCommitGraphProps> { const height = this.getHeight(); const width = this.getWidth(); - const style = { height, width, 'flex-shrink': 0 }; + const style = { height, width, flexShrink: 0 }; return ( <svg height={height} width={width} style={style}> diff --git a/src/components/Toolbar.tsx b/src/components/Toolbar.tsx index 2e0694415..6ee37d175 100644 --- a/src/components/Toolbar.tsx +++ b/src/components/Toolbar.tsx @@ -167,7 +167,6 @@ export class Toolbar extends React.Component<IToolbarProps, IToolbarState> { className={badgeClass} variant="dot" invisible={!hasRemote || this.props.nCommitsBehind === 0} - data-test-id="pull-badge" > <ActionButton className={toolbarButtonClass} @@ -193,7 +192,6 @@ export class Toolbar extends React.Component<IToolbarProps, IToolbarState> { invisible={ !hasRemote || (this.props.nCommitsAhead === 0 && hasUpstream) } - data-test-id="push-badge" > <ActionButton className={toolbarButtonClass} diff --git a/src/components/diff/PlainTextDiff.ts b/src/components/diff/PlainTextDiff.ts index a37cb1a83..cd068eb08 100644 --- a/src/components/diff/PlainTextDiff.ts +++ b/src/components/diff/PlainTextDiff.ts @@ -62,7 +62,7 @@ export class PlainTextDiff extends Panel implements Git.Diff.IDiffWidget { editorFactory }: { model: Git.Diff.IModel; - languageRegistry: IEditorLanguageRegistry; + languageRegistry?: IEditorLanguageRegistry; editorFactory?: CodeEditor.Factory; trans?: TranslationBundle; }) { @@ -80,8 +80,9 @@ export class PlainTextDiff extends Panel implements Git.Diff.IDiffWidget { ); const getReady = new PromiseDelegate<void>(); this._isReady = getReady.promise; - this._editorFactory = editorFactory ?? createEditorFactory(); - this._languageRegistry = languageRegistry; + this._languageRegistry = languageRegistry ?? new EditorLanguageRegistry(); + this._editorFactory = + editorFactory ?? createEditorFactory(this._languageRegistry); this._model = model; this._trans = trans ?? nullTranslator.load('jupyterlab_git'); @@ -399,10 +400,12 @@ function createStringDiffModel( * * @returns Editor factory */ -function createEditorFactory(): CodeEditor.Factory { +function createEditorFactory( + languages: IEditorLanguageRegistry +): CodeEditor.Factory { const factory = new CodeMirrorEditorFactory({ extensions: new EditorExtensionRegistry(), - languages: new EditorLanguageRegistry() + languages }); return factory.newInlineEditor.bind(factory); diff --git a/src/model.ts b/src/model.ts index e15939ffb..368f5a0a3 100644 --- a/src/model.ts +++ b/src/model.ts @@ -406,12 +406,14 @@ export class GitExtension implements IGitExtension { * @returns The file status or null if path repository is null or path not in repository */ getFile(path: string): Git.IStatusFile | null { - if (!this.pathRepository || this._status === null) { + if (this.pathRepository === null) { return null; } - const fileStatus = this._status.files.find(status => { - return this.getRelativeFilePath(status.to) === path; - }); + const fileStatus = this._status?.files + ? this._status.files.find(status => { + return this.getRelativeFilePath(status.to) === path; + }) + : null; if (!fileStatus) { const relativePath = PathExt.relative( diff --git a/src/style/Toolbar.ts b/src/style/Toolbar.ts index 36f95aacf..01b61bb91 100644 --- a/src/style/Toolbar.ts +++ b/src/style/Toolbar.ts @@ -117,7 +117,7 @@ export const badgeClass = style({ $nest: { '& > .MuiBadge-badge': { top: 12, - right: 15, + right: 5, backgroundColor: 'var(--jp-warn-color1)' } } diff --git a/style/diff-common.css b/style/diff-common.css index ea90b3b54..a67f49058 100644 --- a/style/diff-common.css +++ b/style/diff-common.css @@ -60,7 +60,13 @@ button.jp-git-diff-resolve .jp-ToolbarButtonComponent-label { .jp-git-diff-banner span { padding: 0 4px; font-weight: bold; + overflow: hidden; + text-overflow: ellipsis; +} +.jp-git-merge-banner span { + padding: 0 4px; + font-weight: bold; overflow: hidden; text-overflow: ellipsis; } @@ -86,14 +92,6 @@ button.jp-git-diff-resolve .jp-ToolbarButtonComponent-label { gap: 10px; } -.jp-git-merge-banner span { - padding: 0 4px; - font-weight: bold; - - overflow: hidden; - text-overflow: ellipsis; -} - .jp-git-merge-banner span:first-of-type { background-color: var(--jp-merge-local-color1); } @@ -149,14 +147,12 @@ button.jp-git-diff-resolve .jp-ToolbarButtonComponent-label { background-color: var(--jp-git-diff-deleted-color); } -.jp-git-diff-root .cm-merge-left-editor .cm-merge-l-chunk:not(.jp-Merge-conflict) { +.jp-git-diff-root + .cm-merge-left-editor + .cm-merge-l-chunk:not(.jp-Merge-conflict) { background-color: var(--jp-merge-local-color); } -.jp-git-diff-root .cm-merge-left-editor .cm-merge-l-inserted { - background-color: var(--jp-merge-local-color1); -} - .jp-git-diff-root .cm-merge-r-inserted, .jp-git-diff-root .cm-merge-l-inserted, .jp-git-diff-root .cm-diff-right-editor .cm-merge-r-chunk, @@ -164,6 +160,10 @@ button.jp-git-diff-resolve .jp-ToolbarButtonComponent-label { background-color: var(--jp-git-diff-added-color); } +.jp-git-diff-root .cm-merge-left-editor .cm-merge-l-inserted { + background-color: var(--jp-merge-local-color1); +} + .jp-git-diff-root .cm-merge-spacer { background-image: repeating-linear-gradient( 145deg, diff --git a/style/variables.css b/style/variables.css index b941bb77f..bb2cdfb03 100644 --- a/style/variables.css +++ b/style/variables.css @@ -10,7 +10,6 @@ --jp-git-diff-deleted-color1: rgb(255 0 0 / 40%); --jp-git-diff-output-border-color: rgb(0 141 255 / 70%); --jp-git-diff-output-color: rgb(0 141 255 / 30%); - --jp-merge-local-color: rgb(31 31 224 / 20%); --jp-merge-local-color1: rgb(31 31 224 / 40%); } diff --git a/testutils/jest-file-mock.js b/testutils/jest-file-mock.js deleted file mode 100644 index 86059f362..000000000 --- a/testutils/jest-file-mock.js +++ /dev/null @@ -1 +0,0 @@ -module.exports = 'test-file-stub'; diff --git a/testutils/jest-setup-files.js b/testutils/jest-setup-files.js index 9c71967dd..b365082fc 100644 --- a/testutils/jest-setup-files.js +++ b/testutils/jest-setup-files.js @@ -1,7 +1,92 @@ -globalThis.fetch = require('jest-fetch-mock'); -// Use node crypto for crypto -globalThis.crypto = require('crypto'); - -require('enzyme').configure({ - adapter: new (require('@wojtekmaj/enzyme-adapter-react-17'))() +/* global globalThis */ +globalThis.DragEvent = class DragEvent {}; +if ( + typeof globalThis.TextDecoder === 'undefined' || + typeof globalThis.TextEncoder === 'undefined' +) { + const util = require('util'); + globalThis.TextDecoder = util.TextDecoder; + globalThis.TextEncoder = util.TextEncoder; +} +const fetchMod = (window.fetch = require('node-fetch')); +window.Request = fetchMod.Request; +window.Headers = fetchMod.Headers; +window.Response = fetchMod.Response; +globalThis.Image = window.Image; +window.focus = () => { + /* JSDom throws "Not Implemented" */ +}; +window.document.elementFromPoint = (left, top) => document.body; +if (!window.hasOwnProperty('getSelection')) { + // Minimal getSelection() that supports a fake selection + window.getSelection = function getSelection() { + return { + _selection: '', + selectAllChildren: () => { + this._selection = 'foo'; + }, + toString: () => { + const val = this._selection; + this._selection = ''; + return val; + } + }; + }; +} +// Used by xterm.js +window.matchMedia = function (media) { + return { + matches: false, + media, + onchange: () => { + /* empty */ + }, + addEventListener: () => { + /* empty */ + }, + removeEventListener: () => { + /* empty */ + }, + dispatchEvent: () => { + return true; + }, + addListener: () => { + /* empty */ + }, + removeListener: () => { + /* empty */ + } + }; +}; +process.on('unhandledRejection', (error, promise) => { + console.error('Unhandled promise rejection somewhere in tests'); + if (error) { + console.error(error); + const stack = error.stack; + if (stack) { + console.error(stack); + } + } + promise.catch(err => console.error('promise rejected', err)); }); +if (window.requestIdleCallback === undefined) { + // On Safari, requestIdleCallback is not available, so we use replacement functions for `idleCallbacks` + // See: https://developer.mozilla.org/en-US/docs/Web/API/Background_Tasks_API#falling_back_to_settimeout + // eslint-disable-next-line @typescript-eslint/ban-types + window.requestIdleCallback = function (handler) { + let startTime = Date.now(); + return setTimeout(function () { + handler({ + didTimeout: false, + timeRemaining: function () { + return Math.max(0, 50.0 - (Date.now() - startTime)); + } + }); + }, 1); + }; + window.cancelIdleCallback = function (id) { + clearTimeout(id); + }; +} + +globalThis.ResizeObserver = require('resize-observer-polyfill'); diff --git a/yarn.lock b/yarn.lock index 2bb49aa22..fb9d63c49 100644 --- a/yarn.lock +++ b/yarn.lock @@ -12,6 +12,13 @@ __metadata: languageName: node linkType: hard +"@adobe/css-tools@npm:^4.3.1": + version: 4.3.1 + resolution: "@adobe/css-tools@npm:4.3.1" + checksum: ad43456379ff391132aff687ece190cb23ea69395e23c9b96690eeabe2468da89a4aaf266e4f8b6eaab53db3d1064107ce0f63c3a974e864f4a04affc768da3f + languageName: node + linkType: hard + "@ampproject/remapping@npm:^2.2.0": version: 2.2.1 resolution: "@ampproject/remapping@npm:2.2.1" @@ -22,7 +29,7 @@ __metadata: languageName: node linkType: hard -"@babel/code-frame@npm:^7.0.0, @babel/code-frame@npm:^7.12.13, @babel/code-frame@npm:^7.22.13": +"@babel/code-frame@npm:^7.0.0, @babel/code-frame@npm:^7.10.4, @babel/code-frame@npm:^7.12.13, @babel/code-frame@npm:^7.22.13": version: 7.22.13 resolution: "@babel/code-frame@npm:7.22.13" dependencies: @@ -1307,6 +1314,15 @@ __metadata: languageName: node linkType: hard +"@babel/runtime@npm:^7.9.2": + version: 7.23.2 + resolution: "@babel/runtime@npm:7.23.2" + dependencies: + regenerator-runtime: ^0.14.0 + checksum: 6c4df4839ec75ca10175f636d6362f91df8a3137f86b38f6cd3a4c90668a0fe8e9281d320958f4fbd43b394988958585a17c3aab2a4ea6bf7316b22916a371fb + languageName: node + linkType: hard + "@babel/template@npm:^7.22.15, @babel/template@npm:^7.22.5, @babel/template@npm:^7.3.3": version: 7.22.15 resolution: "@babel/template@npm:7.22.15" @@ -2157,19 +2173,6 @@ __metadata: languageName: node linkType: hard -"@jest/types@npm:^26.6.2": - version: 26.6.2 - resolution: "@jest/types@npm:26.6.2" - dependencies: - "@types/istanbul-lib-coverage": ^2.0.0 - "@types/istanbul-reports": ^3.0.0 - "@types/node": "*" - "@types/yargs": ^15.0.0 - chalk: ^4.0.0 - checksum: a0bd3d2f22f26ddb23f41fddf6e6a30bf4fab2ce79ec1cb6ce6fdfaf90a72e00f4c71da91ec61e13db3b10c41de22cf49d07c57ff2b59171d64b29f909c1d8d6 - languageName: node - linkType: hard - "@jest/types@npm:^29.6.3": version: 29.6.3 resolution: "@jest/types@npm:29.6.3" @@ -2636,8 +2639,10 @@ __metadata: "@mui/lab": ^5.0.0-alpha.127 "@mui/material": ^5.12.1 "@mui/styles": ^5.12.0 + "@testing-library/jest-dom": ^6.1.4 + "@testing-library/react": ^14.0.0 + "@testing-library/user-event": ^14.5.1 "@types/diff-match-patch": ^1.0.32 - "@types/enzyme": ^3.1.15 "@types/jest": ^29.2.0 "@types/json-schema": ^7.0.11 "@types/react": ^18.0.26 @@ -2648,11 +2653,9 @@ __metadata: "@types/resize-observer-browser": ^0.1.7 "@typescript-eslint/eslint-plugin": ^6.1.0 "@typescript-eslint/parser": ^6.1.0 - "@wojtekmaj/enzyme-adapter-react-17": ^0.8.0 all-contributors-cli: ^6.14.0 css-loader: ^6.7.1 diff-match-patch: ^1.0.4 - enzyme: ^3.7.0 eslint: ^8.36.0 eslint-config-prettier: ^8.8.0 eslint-plugin-prettier: ^5.0.0 @@ -2662,7 +2665,6 @@ __metadata: husky: ^8.0.3 identity-obj-proxy: ^3.0.0 jest: ^29.2.0 - jest-fetch-mock: ^3.0.0 lint-staged: ^13.2.3 mkdirp: ^1.0.3 nbdime: ^7.0.0-alpha.2 @@ -2673,6 +2675,7 @@ __metadata: react-dom: ^18.2.0 react-virtualized-auto-sizer: ^1.0.2 react-window: ^1.8.5 + resize-observer-polyfill: ^1.5.1 rimraf: ^5.0.1 source-map-loader: ^1.0.2 style-loader: ^3.3.1 @@ -2681,7 +2684,7 @@ __metadata: stylelint-config-standard: ^34.0.0 stylelint-csstree-validator: ^3.0.0 stylelint-prettier: ^4.0.0 - ts-jest: ^26.0.0 + ts-jest: ^29.1.0 typescript: ~5.0.2 typestyle: ^2.0.1 yjs: ^13.5.40 @@ -3695,6 +3698,75 @@ __metadata: languageName: node linkType: hard +"@testing-library/dom@npm:^9.0.0": + version: 9.3.3 + resolution: "@testing-library/dom@npm:9.3.3" + dependencies: + "@babel/code-frame": ^7.10.4 + "@babel/runtime": ^7.12.5 + "@types/aria-query": ^5.0.1 + aria-query: 5.1.3 + chalk: ^4.1.0 + dom-accessibility-api: ^0.5.9 + lz-string: ^1.5.0 + pretty-format: ^27.0.2 + checksum: 34e0a564da7beb92aa9cc44a9080221e2412b1a132eb37be3d513fe6c58027674868deb9f86195756d98d15ba969a30fe00632a4e26e25df2a5a4f6ac0686e37 + languageName: node + linkType: hard + +"@testing-library/jest-dom@npm:^6.1.4": + version: 6.1.4 + resolution: "@testing-library/jest-dom@npm:6.1.4" + dependencies: + "@adobe/css-tools": ^4.3.1 + "@babel/runtime": ^7.9.2 + aria-query: ^5.0.0 + chalk: ^3.0.0 + css.escape: ^1.5.1 + dom-accessibility-api: ^0.5.6 + lodash: ^4.17.15 + redent: ^3.0.0 + peerDependencies: + "@jest/globals": ">= 28" + "@types/jest": ">= 28" + jest: ">= 28" + vitest: ">= 0.32" + peerDependenciesMeta: + "@jest/globals": + optional: true + "@types/jest": + optional: true + jest: + optional: true + vitest: + optional: true + checksum: c6bd9469554136a25d94b55ea16736d56b8c5d200526023774dbf35ca35551a721257e6734f1b404bbd07ae0a1950f1912b5be60e113db2ff2ff50af14f7085c + languageName: node + linkType: hard + +"@testing-library/react@npm:^14.0.0": + version: 14.0.0 + resolution: "@testing-library/react@npm:14.0.0" + dependencies: + "@babel/runtime": ^7.12.5 + "@testing-library/dom": ^9.0.0 + "@types/react-dom": ^18.0.0 + peerDependencies: + react: ^18.0.0 + react-dom: ^18.0.0 + checksum: 4a54c8f56cc4a39b50803205f84f06280bb76521d6d5d4b3b36651d760c7c7752ef142d857d52aaf4fad4848ed7a8be49afc793a5dda105955d2f8bef24901ac + languageName: node + linkType: hard + +"@testing-library/user-event@npm:^14.5.1": + version: 14.5.1 + resolution: "@testing-library/user-event@npm:14.5.1" + peerDependencies: + "@testing-library/dom": ">=7.21.4" + checksum: 3e6bc9fd53dfe2f3648190193ed2fd4bca2a1bfb47f68810df3b33f05412526e5fd5c4ef9dc5375635e0f4cdf1859916867b597eed22bda1321e04242ea6c519 + languageName: node + linkType: hard + "@tootallnate/once@npm:2": version: 2.0.0 resolution: "@tootallnate/once@npm:2.0.0" @@ -3702,6 +3774,13 @@ __metadata: languageName: node linkType: hard +"@types/aria-query@npm:^5.0.1": + version: 5.0.2 + resolution: "@types/aria-query@npm:5.0.2" + checksum: 19394fea016e72da39dd5ef1cf1643e3252b7ee99d8f0b3a8740d3b72f874443fc1e00a41935b36fdfaf92cd735d4ae10dc5d6ab8f1192527d4c0471bb8ff8e4 + languageName: node + linkType: hard + "@types/babel__core@npm:^7.1.14": version: 7.20.2 resolution: "@types/babel__core@npm:7.20.2" @@ -3743,15 +3822,6 @@ __metadata: languageName: node linkType: hard -"@types/cheerio@npm:*": - version: 0.22.31 - resolution: "@types/cheerio@npm:0.22.31" - dependencies: - "@types/node": "*" - checksum: 8d73d22fdd384c290514dad6f9f4a436f5a90bc836bbe9b46fe4f557041a03484f0547291d0347185a806149f465355fc225dc87477b8cf5af5be307bb75e6a4 - languageName: node - linkType: hard - "@types/diff-match-patch@npm:^1.0.32": version: 1.0.32 resolution: "@types/diff-match-patch@npm:1.0.32" @@ -3759,16 +3829,6 @@ __metadata: languageName: node linkType: hard -"@types/enzyme@npm:^3.1.15": - version: 3.10.12 - resolution: "@types/enzyme@npm:3.10.12" - dependencies: - "@types/cheerio": "*" - "@types/react": "*" - checksum: 356e9142566b68c9b324ae71a7b93f03512a1c009a1d337a25ce4f495590f3e79de08aa4a0016d6224cb228c27832d92b6d7d3276ba5962302c41d0577e8a912 - languageName: node - linkType: hard - "@types/eslint-scope@npm:^3.7.3": version: 3.7.4 resolution: "@types/eslint-scope@npm:3.7.4" @@ -3902,6 +3962,15 @@ __metadata: languageName: node linkType: hard +"@types/react-dom@npm:^18.0.0": + version: 18.2.13 + resolution: "@types/react-dom@npm:18.2.13" + dependencies: + "@types/react": "*" + checksum: 22ba066b141dca5a5a9227fae0afc7c94b470fff8e8a38ade72649da57a8ea04d0cb2ba3e22005e7d8e772d49bddd28855b1dd98e6defd033bba6afb6edff883 + languageName: node + linkType: hard + "@types/react-dom@npm:^18.2.0": version: 18.2.10 resolution: "@types/react-dom@npm:18.2.10" @@ -4009,15 +4078,6 @@ __metadata: languageName: node linkType: hard -"@types/yargs@npm:^15.0.0": - version: 15.0.15 - resolution: "@types/yargs@npm:15.0.15" - dependencies: - "@types/yargs-parser": "*" - checksum: 3420f6bcc508a895ef91858f8e6de975c710e4498cf6ed293f1174d3f1ad56edb4ab8481219bf6190f64a3d4115fab1d13ab3edc90acd54fba7983144040e446 - languageName: node - linkType: hard - "@types/yargs@npm:^17.0.8": version: 17.0.26 resolution: "@types/yargs@npm:17.0.26" @@ -4334,38 +4394,6 @@ __metadata: languageName: node linkType: hard -"@wojtekmaj/enzyme-adapter-react-17@npm:^0.8.0": - version: 0.8.0 - resolution: "@wojtekmaj/enzyme-adapter-react-17@npm:0.8.0" - dependencies: - "@wojtekmaj/enzyme-adapter-utils": ^0.2.0 - enzyme-shallow-equal: ^1.0.0 - has: ^1.0.0 - prop-types: ^15.7.0 - react-is: ^17.0.0 - react-test-renderer: ^17.0.0 - peerDependencies: - enzyme: ^3.0.0 - react: ^17.0.0-0 - react-dom: ^17.0.0-0 - checksum: aa9674f06f6db269b72168ebf46c4513938993479eb60bac30cb6183b5aca6108ade3d08af4f56c142cb219415480d0c4b454ba9452b85c32f711c806b39cd8c - languageName: node - linkType: hard - -"@wojtekmaj/enzyme-adapter-utils@npm:^0.2.0": - version: 0.2.0 - resolution: "@wojtekmaj/enzyme-adapter-utils@npm:0.2.0" - dependencies: - function.prototype.name: ^1.1.0 - has: ^1.0.0 - object.fromentries: ^2.0.0 - prop-types: ^15.7.0 - peerDependencies: - react: ^17.0.0-0 - checksum: 837741f1382acdb02ce304745eccfdcff03f1cae2a4fb833056a7a753308cd1182b0b32a10a04be6bfedaaab8f4acd5b458bfe0b9ebaa6119c4aaaba74a14ae4 - languageName: node - linkType: hard - "@xtuc/ieee754@npm:^1.2.0": version: 1.2.0 resolution: "@xtuc/ieee754@npm:1.2.0" @@ -4642,6 +4670,24 @@ __metadata: languageName: node linkType: hard +"aria-query@npm:5.1.3": + version: 5.1.3 + resolution: "aria-query@npm:5.1.3" + dependencies: + deep-equal: ^2.0.5 + checksum: 929ff95f02857b650fb4cbcd2f41072eee2f46159a6605ea03bf63aa572e35ffdff43d69e815ddc462e16e07de8faba3978afc2813650b4448ee18c9895d982b + languageName: node + linkType: hard + +"aria-query@npm:^5.0.0": + version: 5.3.0 + resolution: "aria-query@npm:5.3.0" + dependencies: + dequal: ^2.0.3 + checksum: 305bd73c76756117b59aba121d08f413c7ff5e80fa1b98e217a3443fcddb9a232ee790e24e432b59ae7625aebcf4c47cb01c2cac872994f0b426f5bdfcd96ba9 + languageName: node + linkType: hard + "array-buffer-byte-length@npm:^1.0.0": version: 1.0.0 resolution: "array-buffer-byte-length@npm:1.0.0" @@ -4672,31 +4718,6 @@ __metadata: languageName: node linkType: hard -"array.prototype.filter@npm:^1.0.0": - version: 1.0.1 - resolution: "array.prototype.filter@npm:1.0.1" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.3 - es-abstract: ^1.19.0 - es-array-method-boxes-properly: ^1.0.0 - is-string: ^1.0.7 - checksum: 574b52dcebf2def7bedb05449b60e5e3819093fa77f88c3f87a9611361d2745c7aacde01cd3ed7accafd632ee1e0340b655dd26dc7c060429cb4566058e63134 - languageName: node - linkType: hard - -"array.prototype.flat@npm:^1.2.3": - version: 1.3.0 - resolution: "array.prototype.flat@npm:1.3.0" - dependencies: - call-bind: ^1.0.2 - define-properties: ^1.1.3 - es-abstract: ^1.19.2 - es-shim-unscopables: ^1.0.0 - checksum: 2a652b3e8dc0bebb6117e42a5ab5738af0203a14c27341d7bb2431467bdb4b348e2c5dc555dfcda8af0a5e4075c400b85311ded73861c87290a71a17c3e0a257 - languageName: node - linkType: hard - "array.prototype.flatmap@npm:^1.3.1": version: 1.3.1 resolution: "array.prototype.flatmap@npm:1.3.1" @@ -4908,13 +4929,6 @@ __metadata: languageName: node linkType: hard -"boolbase@npm:^1.0.0": - version: 1.0.0 - resolution: "boolbase@npm:1.0.0" - checksum: 3e25c80ef626c3a3487c73dbfc70ac322ec830666c9ad915d11b701142fab25ec1e63eff2c450c74347acfd2de854ccde865cd79ef4db1683f7c7b046ea43bb0 - languageName: node - linkType: hard - "bplist-parser@npm:^0.2.0": version: 0.2.0 resolution: "bplist-parser@npm:0.2.0" @@ -4984,7 +4998,7 @@ __metadata: languageName: node linkType: hard -"buffer-from@npm:1.x, buffer-from@npm:^1.0.0": +"buffer-from@npm:^1.0.0": version: 1.1.2 resolution: "buffer-from@npm:1.1.2" checksum: 0448524a562b37d4d7ed9efd91685a5b77a50672c556ea254ac9a6d30e3403a517d8981f10e565db24e8339413b43c97ca2951f10e399c6125a0d8911f5679bb @@ -5088,6 +5102,16 @@ __metadata: languageName: node linkType: hard +"chalk@npm:^3.0.0": + version: 3.0.0 + resolution: "chalk@npm:3.0.0" + dependencies: + ansi-styles: ^4.1.0 + supports-color: ^7.1.0 + checksum: 8e3ddf3981c4da405ddbd7d9c8d91944ddf6e33d6837756979f7840a29272a69a5189ecae0ff84006750d6d1e92368d413335eab4db5476db6e6703a1d1e0505 + languageName: node + linkType: hard + "chalk@npm:^4.0.0, chalk@npm:^4.1.0, chalk@npm:^4.1.2": version: 4.1.2 resolution: "chalk@npm:4.1.2" @@ -5112,35 +5136,6 @@ __metadata: languageName: node linkType: hard -"cheerio-select@npm:^2.1.0": - version: 2.1.0 - resolution: "cheerio-select@npm:2.1.0" - dependencies: - boolbase: ^1.0.0 - css-select: ^5.1.0 - css-what: ^6.1.0 - domelementtype: ^2.3.0 - domhandler: ^5.0.3 - domutils: ^3.0.1 - checksum: 843d6d479922f28a6c5342c935aff1347491156814de63c585a6eb73baf7bb4185c1b4383a1195dca0f12e3946d737c7763bcef0b9544c515d905c5c44c5308b - languageName: node - linkType: hard - -"cheerio@npm:^1.0.0-rc.3": - version: 1.0.0-rc.12 - resolution: "cheerio@npm:1.0.0-rc.12" - dependencies: - cheerio-select: ^2.1.0 - dom-serializer: ^2.0.0 - domhandler: ^5.0.3 - domutils: ^3.0.1 - htmlparser2: ^8.0.1 - parse5: ^7.0.0 - parse5-htmlparser2-tree-adapter: ^7.0.0 - checksum: 5d4c1b7a53cf22d3a2eddc0aff70cf23cbb30d01a4c79013e703a012475c02461aa1fcd99127e8d83a02216386ed6942b2c8103845fd0812300dd199e6e7e054 - languageName: node - linkType: hard - "child_process@npm:~1.0.2": version: 1.0.2 resolution: "child_process@npm:1.0.2" @@ -5162,13 +5157,6 @@ __metadata: languageName: node linkType: hard -"ci-info@npm:^2.0.0": - version: 2.0.0 - resolution: "ci-info@npm:2.0.0" - checksum: 3b374666a85ea3ca43fa49aa3a048d21c9b475c96eb13c133505d2324e7ae5efd6a454f41efe46a152269e9b6a00c9edbe63ec7fa1921957165aae16625acd67 - languageName: node - linkType: hard - "ci-info@npm:^3.2.0": version: 3.9.0 resolution: "ci-info@npm:3.9.0" @@ -5351,7 +5339,7 @@ __metadata: languageName: node linkType: hard -"commander@npm:^2.11.0, commander@npm:^2.19.0, commander@npm:^2.20.0": +"commander@npm:^2.11.0, commander@npm:^2.20.0": version: 2.20.3 resolution: "commander@npm:2.20.3" checksum: ab8c07884e42c3a8dbc5dd9592c606176c7eb5c1ca5ff274bcf907039b2c41de3626f684ea75ccf4d361ba004bbaff1f577d5384c155f3871e456bdf27becf9e @@ -5474,15 +5462,6 @@ __metadata: languageName: node linkType: hard -"cross-fetch@npm:^3.0.4": - version: 3.1.5 - resolution: "cross-fetch@npm:3.1.5" - dependencies: - node-fetch: 2.6.7 - checksum: f6b8c6ee3ef993ace6277fd789c71b6acf1b504fd5f5c7128df4ef2f125a429e29cd62dc8c127523f04a5f2fa4771ed80e3f3d9695617f441425045f505cf3bb - languageName: node - linkType: hard - "cross-spawn@npm:^6.0.5": version: 6.0.5 resolution: "cross-spawn@npm:6.0.5" @@ -5532,19 +5511,6 @@ __metadata: languageName: node linkType: hard -"css-select@npm:^5.1.0": - version: 5.1.0 - resolution: "css-select@npm:5.1.0" - dependencies: - boolbase: ^1.0.0 - css-what: ^6.1.0 - domhandler: ^5.0.2 - domutils: ^3.0.1 - nth-check: ^2.0.1 - checksum: 2772c049b188d3b8a8159907192e926e11824aea525b8282981f72ba3f349cf9ecd523fdf7734875ee2cb772246c22117fc062da105b6d59afe8dcd5c99c9bda - languageName: node - linkType: hard - "css-tree@npm:^2.3.1": version: 2.3.1 resolution: "css-tree@npm:2.3.1" @@ -5565,10 +5531,10 @@ __metadata: languageName: node linkType: hard -"css-what@npm:^6.1.0": - version: 6.1.0 - resolution: "css-what@npm:6.1.0" - checksum: b975e547e1e90b79625918f84e67db5d33d896e6de846c9b584094e529f0c63e2ab85ee33b9daffd05bff3a146a1916bec664e18bb76dd5f66cbff9fc13b2bbe +"css.escape@npm:^1.5.1": + version: 1.5.1 + resolution: "css.escape@npm:1.5.1" + checksum: f6d38088d870a961794a2580b2b2af1027731bb43261cfdce14f19238a88664b351cc8978abc20f06cc6bbde725699dec8deb6fe9816b139fc3f2af28719e774 languageName: node linkType: hard @@ -5695,6 +5661,32 @@ __metadata: languageName: node linkType: hard +"deep-equal@npm:^2.0.5": + version: 2.2.2 + resolution: "deep-equal@npm:2.2.2" + dependencies: + array-buffer-byte-length: ^1.0.0 + call-bind: ^1.0.2 + es-get-iterator: ^1.1.3 + get-intrinsic: ^1.2.1 + is-arguments: ^1.1.1 + is-array-buffer: ^3.0.2 + is-date-object: ^1.0.5 + is-regex: ^1.1.4 + is-shared-array-buffer: ^1.0.2 + isarray: ^2.0.5 + object-is: ^1.1.5 + object-keys: ^1.1.1 + object.assign: ^4.1.4 + regexp.prototype.flags: ^1.5.0 + side-channel: ^1.0.4 + which-boxed-primitive: ^1.0.2 + which-collection: ^1.0.1 + which-typed-array: ^1.1.9 + checksum: eb61c35157b6ecb96a5359b507b083fbff8ddb4c86a78a781ee38485f77a667465e45d63ee2ebd8a00e86d94c80e499906900cd82c2debb400237e1662cd5397 + languageName: node + linkType: hard + "deep-is@npm:^0.1.3": version: 0.1.4 resolution: "deep-is@npm:0.1.4" @@ -5731,6 +5723,17 @@ __metadata: languageName: node linkType: hard +"define-data-property@npm:^1.0.1": + version: 1.1.1 + resolution: "define-data-property@npm:1.1.1" + dependencies: + get-intrinsic: ^1.2.1 + gopd: ^1.0.1 + has-property-descriptors: ^1.0.0 + checksum: a29855ad3f0630ea82e3c5012c812efa6ca3078d5c2aa8df06b5f597c1cde6f7254692df41945851d903e05a1668607b6d34e778f402b9ff9ffb38111f1a3f0d + languageName: node + linkType: hard + "define-lazy-prop@npm:^3.0.0": version: 3.0.0 resolution: "define-lazy-prop@npm:3.0.0" @@ -5762,6 +5765,13 @@ __metadata: languageName: node linkType: hard +"dequal@npm:^2.0.3": + version: 2.0.3 + resolution: "dequal@npm:2.0.3" + checksum: 8679b850e1a3d0ebbc46ee780d5df7b478c23f335887464023a631d1b9af051ad4a6595a44220f9ff8ff95a8ddccf019b5ad778a976fd7bbf77383d36f412f90 + languageName: node + linkType: hard + "detect-newline@npm:^3.0.0": version: 3.1.0 resolution: "detect-newline@npm:3.1.0" @@ -5799,13 +5809,6 @@ __metadata: languageName: node linkType: hard -"discontinuous-range@npm:1.0.0": - version: 1.0.0 - resolution: "discontinuous-range@npm:1.0.0" - checksum: 8ee88d7082445b6eadc7c03bebe6dc978f96760c45e9f65d16ca66174d9e086a9e3855ee16acf65625e1a07a846a17de674f02a5964a6aebe5963662baf8b5c8 - languageName: node - linkType: hard - "doctrine@npm:^2.1.0": version: 2.1.0 resolution: "doctrine@npm:2.1.0" @@ -5824,6 +5827,13 @@ __metadata: languageName: node linkType: hard +"dom-accessibility-api@npm:^0.5.6, dom-accessibility-api@npm:^0.5.9": + version: 0.5.16 + resolution: "dom-accessibility-api@npm:0.5.16" + checksum: 005eb283caef57fc1adec4d5df4dd49189b628f2f575af45decb210e04d634459e3f1ee64f18b41e2dcf200c844bc1d9279d80807e686a30d69a4756151ad248 + languageName: node + linkType: hard + "dom-helpers@npm:^5.0.1": version: 5.2.1 resolution: "dom-helpers@npm:5.2.1" @@ -5845,18 +5855,7 @@ __metadata: languageName: node linkType: hard -"dom-serializer@npm:^2.0.0": - version: 2.0.0 - resolution: "dom-serializer@npm:2.0.0" - dependencies: - domelementtype: ^2.3.0 - domhandler: ^5.0.2 - entities: ^4.2.0 - checksum: cd1810544fd8cdfbd51fa2c0c1128ec3a13ba92f14e61b7650b5de421b88205fd2e3f0cc6ace82f13334114addb90ed1c2f23074a51770a8e9c1273acbc7f3e6 - languageName: node - linkType: hard - -"domelementtype@npm:^2.0.1, domelementtype@npm:^2.2.0, domelementtype@npm:^2.3.0": +"domelementtype@npm:^2.0.1, domelementtype@npm:^2.2.0": version: 2.3.0 resolution: "domelementtype@npm:2.3.0" checksum: ee837a318ff702622f383409d1f5b25dd1024b692ef64d3096ff702e26339f8e345820f29a68bcdcea8cfee3531776b3382651232fbeae95612d6f0a75efb4f6 @@ -5881,15 +5880,6 @@ __metadata: languageName: node linkType: hard -"domhandler@npm:^5.0.1, domhandler@npm:^5.0.2, domhandler@npm:^5.0.3": - version: 5.0.3 - resolution: "domhandler@npm:5.0.3" - dependencies: - domelementtype: ^2.3.0 - checksum: 0f58f4a6af63e6f3a4320aa446d28b5790a009018707bce2859dcb1d21144c7876482b5188395a188dfa974238c019e0a1e610d2fc269a12b2c192ea2b0b131c - languageName: node - linkType: hard - "domutils@npm:^2.5.2": version: 2.8.0 resolution: "domutils@npm:2.8.0" @@ -5901,17 +5891,6 @@ __metadata: languageName: node linkType: hard -"domutils@npm:^3.0.1": - version: 3.0.1 - resolution: "domutils@npm:3.0.1" - dependencies: - dom-serializer: ^2.0.0 - domelementtype: ^2.3.0 - domhandler: ^5.0.1 - checksum: 23aa7a840572d395220e173cb6263b0d028596e3950100520870a125af33ff819e6f609e1606d6f7d73bd9e7feb03bb404286e57a39063b5384c62b724d987b3 - languageName: node - linkType: hard - "duplicate-package-checker-webpack-plugin@npm:^3.0.0": version: 3.0.0 resolution: "duplicate-package-checker-webpack-plugin@npm:3.0.0" @@ -5992,7 +5971,7 @@ __metadata: languageName: node linkType: hard -"entities@npm:^4.2.0, entities@npm:^4.3.0, entities@npm:^4.4.0": +"entities@npm:^4.4.0": version: 4.4.0 resolution: "entities@npm:4.4.0" checksum: 84d250329f4b56b40fa93ed067b194db21e8815e4eb9b59f43a086f0ecd342814f6bc483de8a77da5d64e0f626033192b1b4f1792232a7ea6b970ebe0f3187c2 @@ -6015,46 +5994,6 @@ __metadata: languageName: node linkType: hard -"enzyme-shallow-equal@npm:^1.0.0, enzyme-shallow-equal@npm:^1.0.1": - version: 1.0.5 - resolution: "enzyme-shallow-equal@npm:1.0.5" - dependencies: - has: ^1.0.3 - object-is: ^1.1.5 - checksum: e18a728225b3ef501a223608955e2c8e915adf24dfe4d778bdbc89e4ecd80384723e9d44780176be1529f6b642e7837211f502bff89f62833d8f9cae027997e0 - languageName: node - linkType: hard - -"enzyme@npm:^3.7.0": - version: 3.11.0 - resolution: "enzyme@npm:3.11.0" - dependencies: - array.prototype.flat: ^1.2.3 - cheerio: ^1.0.0-rc.3 - enzyme-shallow-equal: ^1.0.1 - function.prototype.name: ^1.1.2 - has: ^1.0.3 - html-element-map: ^1.2.0 - is-boolean-object: ^1.0.1 - is-callable: ^1.1.5 - is-number-object: ^1.0.4 - is-regex: ^1.0.5 - is-string: ^1.0.5 - is-subset: ^0.1.1 - lodash.escape: ^4.0.1 - lodash.isequal: ^4.5.0 - object-inspect: ^1.7.0 - object-is: ^1.0.2 - object.assign: ^4.1.0 - object.entries: ^1.1.1 - object.values: ^1.1.1 - raf: ^3.4.1 - rst-selector-parser: ^2.2.3 - string.prototype.trim: ^1.2.1 - checksum: 69ae80049c3f405122b8e619f1cf8b04f32b3cc2b6134c29ed8c0f05e87a0b15080f1121096ec211954a710f4787300af9157078c863012de87eee16e98e64ea - languageName: node - linkType: hard - "err-code@npm:^2.0.2": version: 2.0.3 resolution: "err-code@npm:2.0.3" @@ -6071,7 +6010,7 @@ __metadata: languageName: node linkType: hard -"es-abstract@npm:^1.19.0, es-abstract@npm:^1.19.1, es-abstract@npm:^1.19.2, es-abstract@npm:^1.20.4": +"es-abstract@npm:^1.19.0, es-abstract@npm:^1.19.1, es-abstract@npm:^1.20.4": version: 1.21.2 resolution: "es-abstract@npm:1.21.2" dependencies: @@ -6113,10 +6052,20 @@ __metadata: languageName: node linkType: hard -"es-array-method-boxes-properly@npm:^1.0.0": - version: 1.0.0 - resolution: "es-array-method-boxes-properly@npm:1.0.0" - checksum: 2537fcd1cecf187083890bc6f5236d3a26bf39237433587e5bf63392e88faae929dbba78ff0120681a3f6f81c23fe3816122982c160d63b38c95c830b633b826 +"es-get-iterator@npm:^1.1.3": + version: 1.1.3 + resolution: "es-get-iterator@npm:1.1.3" + dependencies: + call-bind: ^1.0.2 + get-intrinsic: ^1.1.3 + has-symbols: ^1.0.3 + is-arguments: ^1.1.1 + is-map: ^2.0.2 + is-set: ^2.0.2 + is-string: ^1.0.7 + isarray: ^2.0.5 + stop-iteration-iterator: ^1.0.0 + checksum: 8fa118da42667a01a7c7529f8a8cca514feeff243feec1ce0bb73baaa3514560bd09d2b3438873cf8a5aaec5d52da248131de153b28e2638a061b6e4df13267d languageName: node linkType: hard @@ -6724,7 +6673,7 @@ __metadata: languageName: node linkType: hard -"function.prototype.name@npm:^1.1.0, function.prototype.name@npm:^1.1.2, function.prototype.name@npm:^1.1.5": +"function.prototype.name@npm:^1.1.5": version: 1.1.5 resolution: "function.prototype.name@npm:1.1.5" dependencies: @@ -6786,7 +6735,7 @@ __metadata: languageName: node linkType: hard -"get-intrinsic@npm:^1.0.2, get-intrinsic@npm:^1.1.1, get-intrinsic@npm:^1.1.3, get-intrinsic@npm:^1.2.0": +"get-intrinsic@npm:^1.0.2, get-intrinsic@npm:^1.1.1, get-intrinsic@npm:^1.1.3, get-intrinsic@npm:^1.2.0, get-intrinsic@npm:^1.2.1": version: 1.2.1 resolution: "get-intrinsic@npm:1.2.1" dependencies: @@ -7039,7 +6988,7 @@ __metadata: languageName: node linkType: hard -"has@npm:^1.0.0, has@npm:^1.0.3": +"has@npm:^1.0.3": version: 1.0.3 resolution: "has@npm:1.0.3" dependencies: @@ -7073,16 +7022,6 @@ __metadata: languageName: node linkType: hard -"html-element-map@npm:^1.2.0": - version: 1.3.1 - resolution: "html-element-map@npm:1.3.1" - dependencies: - array.prototype.filter: ^1.0.0 - call-bind: ^1.0.2 - checksum: 7408da008d37bfa76b597e298ae0ed530258065deb29fbd73d40f7cbd123b654d1022a7a8cfbe713e57d90c5bef844399f5c8a46cde7d55c91d305024c921d08 - languageName: node - linkType: hard - "html-encoding-sniffer@npm:^3.0.0": version: 3.0.0 resolution: "html-encoding-sniffer@npm:3.0.0" @@ -7118,18 +7057,6 @@ __metadata: languageName: node linkType: hard -"htmlparser2@npm:^8.0.1": - version: 8.0.1 - resolution: "htmlparser2@npm:8.0.1" - dependencies: - domelementtype: ^2.3.0 - domhandler: ^5.0.2 - domutils: ^3.0.1 - entities: ^4.3.0 - checksum: 06d5c71e8313597722bc429ae2a7a8333d77bd3ab07ccb916628384b37332027b047f8619448d8f4a3312b6609c6ea3302a4e77435d859e9e686999e6699ca39 - languageName: node - linkType: hard - "http-cache-semantics@npm:^4.1.1": version: 4.1.1 resolution: "http-cache-semantics@npm:4.1.1" @@ -7335,7 +7262,7 @@ __metadata: languageName: node linkType: hard -"internal-slot@npm:^1.0.3, internal-slot@npm:^1.0.5": +"internal-slot@npm:^1.0.3, internal-slot@npm:^1.0.4, internal-slot@npm:^1.0.5": version: 1.0.5 resolution: "internal-slot@npm:1.0.5" dependencies: @@ -7360,6 +7287,16 @@ __metadata: languageName: node linkType: hard +"is-arguments@npm:^1.1.1": + version: 1.1.1 + resolution: "is-arguments@npm:1.1.1" + dependencies: + call-bind: ^1.0.2 + has-tostringtag: ^1.0.0 + checksum: 7f02700ec2171b691ef3e4d0e3e6c0ba408e8434368504bb593d0d7c891c0dbfda6d19d30808b904a6cb1929bca648c061ba438c39f296c2a8ca083229c49f27 + languageName: node + linkType: hard + "is-array-buffer@npm:^3.0.1, is-array-buffer@npm:^3.0.2": version: 3.0.2 resolution: "is-array-buffer@npm:3.0.2" @@ -7387,7 +7324,7 @@ __metadata: languageName: node linkType: hard -"is-boolean-object@npm:^1.0.1, is-boolean-object@npm:^1.1.0": +"is-boolean-object@npm:^1.1.0": version: 1.1.2 resolution: "is-boolean-object@npm:1.1.2" dependencies: @@ -7397,24 +7334,13 @@ __metadata: languageName: node linkType: hard -"is-callable@npm:^1.1.3, is-callable@npm:^1.1.4, is-callable@npm:^1.1.5, is-callable@npm:^1.2.7": +"is-callable@npm:^1.1.3, is-callable@npm:^1.1.4, is-callable@npm:^1.2.7": version: 1.2.7 resolution: "is-callable@npm:1.2.7" checksum: 61fd57d03b0d984e2ed3720fb1c7a897827ea174bd44402878e059542ea8c4aeedee0ea0985998aa5cc2736b2fa6e271c08587addb5b3959ac52cf665173d1ac languageName: node linkType: hard -"is-ci@npm:^2.0.0": - version: 2.0.0 - resolution: "is-ci@npm:2.0.0" - dependencies: - ci-info: ^2.0.0 - bin: - is-ci: bin.js - checksum: 77b869057510f3efa439bbb36e9be429d53b3f51abd4776eeea79ab3b221337fe1753d1e50058a9e2c650d38246108beffb15ccfd443929d77748d8c0cc90144 - languageName: node - linkType: hard - "is-core-module@npm:^2.13.0, is-core-module@npm:^2.5.0, is-core-module@npm:^2.9.0": version: 2.13.0 resolution: "is-core-module@npm:2.13.0" @@ -7424,7 +7350,7 @@ __metadata: languageName: node linkType: hard -"is-date-object@npm:^1.0.1": +"is-date-object@npm:^1.0.1, is-date-object@npm:^1.0.5": version: 1.0.5 resolution: "is-date-object@npm:1.0.5" dependencies: @@ -7513,6 +7439,13 @@ __metadata: languageName: node linkType: hard +"is-map@npm:^2.0.1, is-map@npm:^2.0.2": + version: 2.0.2 + resolution: "is-map@npm:2.0.2" + checksum: ace3d0ecd667bbdefdb1852de601268f67f2db725624b1958f279316e13fecb8fa7df91fd60f690d7417b4ec180712f5a7ee967008e27c65cfd475cc84337728 + languageName: node + linkType: hard + "is-negative-zero@npm:^2.0.2": version: 2.0.2 resolution: "is-negative-zero@npm:2.0.2" @@ -7573,7 +7506,7 @@ __metadata: languageName: node linkType: hard -"is-regex@npm:^1.0.5, is-regex@npm:^1.1.4": +"is-regex@npm:^1.1.4": version: 1.1.4 resolution: "is-regex@npm:1.1.4" dependencies: @@ -7583,6 +7516,13 @@ __metadata: languageName: node linkType: hard +"is-set@npm:^2.0.1, is-set@npm:^2.0.2": + version: 2.0.2 + resolution: "is-set@npm:2.0.2" + checksum: b64343faf45e9387b97a6fd32be632ee7b269bd8183701f3b3f5b71a7cf00d04450ed8669d0bd08753e08b968beda96fca73a10fd0ff56a32603f64deba55a57 + languageName: node + linkType: hard + "is-shared-array-buffer@npm:^1.0.2": version: 1.0.2 resolution: "is-shared-array-buffer@npm:1.0.2" @@ -7615,13 +7555,6 @@ __metadata: languageName: node linkType: hard -"is-subset@npm:^0.1.1": - version: 0.1.1 - resolution: "is-subset@npm:0.1.1" - checksum: 97b8d7852af165269b7495095691a6ce6cf20bdfa1f846f97b4560ee190069686107af4e277fbd93aa0845c4d5db704391460ff6e9014aeb73264ba87893df44 - languageName: node - linkType: hard - "is-symbol@npm:^1.0.2, is-symbol@npm:^1.0.3": version: 1.0.4 resolution: "is-symbol@npm:1.0.4" @@ -7644,6 +7577,13 @@ __metadata: languageName: node linkType: hard +"is-weakmap@npm:^2.0.1": + version: 2.0.1 + resolution: "is-weakmap@npm:2.0.1" + checksum: 1222bb7e90c32bdb949226e66d26cb7bce12e1e28e3e1b40bfa6b390ba3e08192a8664a703dff2a00a84825f4e022f9cd58c4599ff9981ab72b1d69479f4f7f6 + languageName: node + linkType: hard + "is-weakref@npm:^1.0.2": version: 1.0.2 resolution: "is-weakref@npm:1.0.2" @@ -7653,6 +7593,16 @@ __metadata: languageName: node linkType: hard +"is-weakset@npm:^2.0.1": + version: 2.0.2 + resolution: "is-weakset@npm:2.0.2" + dependencies: + call-bind: ^1.0.2 + get-intrinsic: ^1.1.1 + checksum: 5d8698d1fa599a0635d7ca85be9c26d547b317ed8fd83fc75f03efbe75d50001b5eececb1e9971de85fcde84f69ae6f8346bc92d20d55d46201d328e4c74a367 + languageName: node + linkType: hard + "is-wsl@npm:^2.2.0": version: 2.2.0 resolution: "is-wsl@npm:2.2.0" @@ -7662,6 +7612,13 @@ __metadata: languageName: node linkType: hard +"isarray@npm:^2.0.5": + version: 2.0.5 + resolution: "isarray@npm:2.0.5" + checksum: bd5bbe4104438c4196ba58a54650116007fa0262eccef13a4c55b2e09a5b36b59f1e75b9fcc49883dd9d4953892e6fc007eef9e9155648ceea036e184b0f930a + languageName: node + linkType: hard + "isexe@npm:^2.0.0": version: 2.0.0 resolution: "isexe@npm:2.0.0" @@ -7933,16 +7890,6 @@ __metadata: languageName: node linkType: hard -"jest-fetch-mock@npm:^3.0.0": - version: 3.0.3 - resolution: "jest-fetch-mock@npm:3.0.3" - dependencies: - cross-fetch: ^3.0.4 - promise-polyfill: ^8.1.3 - checksum: fb052f7e0ef1c8192a9c15efdd1b18d281ab68fc6b1648b30bff8880fe24418bdf12190ea79b1996932dc15417c3c01f5b2d77ef7104a7e7943e7cbe8d61071d - languageName: node - linkType: hard - "jest-get-type@npm:^29.6.3": version: 29.6.3 resolution: "jest-get-type@npm:29.6.3" @@ -8168,20 +8115,6 @@ __metadata: languageName: node linkType: hard -"jest-util@npm:^26.1.0": - version: 26.6.2 - resolution: "jest-util@npm:26.6.2" - dependencies: - "@jest/types": ^26.6.2 - "@types/node": "*" - chalk: ^4.0.0 - graceful-fs: ^4.2.4 - is-ci: ^2.0.0 - micromatch: ^4.0.2 - checksum: 3c6a5fba05c4c6892cd3a9f66196ea8867087b77a5aa1a3f6cd349c785c3f1ca24abfd454664983aed1a165cab7846688e44fe8630652d666ba326b08625bc3d - languageName: node - linkType: hard - "jest-util@npm:^29.0.0, jest-util@npm:^29.7.0": version: 29.7.0 resolution: "jest-util@npm:29.7.0" @@ -8430,7 +8363,7 @@ __metadata: languageName: node linkType: hard -"json5@npm:2.x, json5@npm:^2.1.2, json5@npm:^2.2.3": +"json5@npm:^2.1.2, json5@npm:^2.2.3": version: 2.2.3 resolution: "json5@npm:2.2.3" bin: @@ -8752,20 +8685,6 @@ __metadata: languageName: node linkType: hard -"lodash.flattendeep@npm:^4.4.0": - version: 4.4.0 - resolution: "lodash.flattendeep@npm:4.4.0" - checksum: 8521c919acac3d4bcf0aaf040c1ca9cb35d6c617e2d72e9b4d51c9a58b4366622cd6077441a18be626c3f7b28227502b3bf042903d447b056ee7e0b11d45c722 - languageName: node - linkType: hard - -"lodash.isequal@npm:^4.5.0": - version: 4.5.0 - resolution: "lodash.isequal@npm:4.5.0" - checksum: da27515dc5230eb1140ba65ff8de3613649620e8656b19a6270afe4866b7bd461d9ba2ac8a48dcc57f7adac4ee80e1de9f965d89d4d81a0ad52bb3eec2609644 - languageName: node - linkType: hard - "lodash.memoize@npm:4.x": version: 4.1.2 resolution: "lodash.memoize@npm:4.1.2" @@ -8794,7 +8713,7 @@ __metadata: languageName: node linkType: hard -"lodash@npm:4.x, lodash@npm:^4.11.2, lodash@npm:^4.17.19, lodash@npm:^4.17.20, lodash@npm:^4.17.21, lodash@npm:^4.17.4, lodash@npm:^4.7.0": +"lodash@npm:^4.11.2, lodash@npm:^4.17.15, lodash@npm:^4.17.19, lodash@npm:^4.17.20, lodash@npm:^4.17.21, lodash@npm:^4.17.4, lodash@npm:^4.7.0": version: 4.17.21 resolution: "lodash@npm:4.17.21" checksum: eb835a2e51d381e561e508ce932ea50a8e5a68f4ebdd771ea240d3048244a8d13658acbd502cd4829768c56f2e16bdd4340b9ea141297d472517b83868e677f7 @@ -8856,6 +8775,15 @@ __metadata: languageName: node linkType: hard +"lz-string@npm:^1.5.0": + version: 1.5.0 + resolution: "lz-string@npm:1.5.0" + bin: + lz-string: bin/bin.js + checksum: 1ee98b4580246fd90dd54da6e346fb1caefcf05f677c686d9af237a157fdea3fd7c83a4bc58f858cd5b10a34d27afe0fdcbd0505a47e0590726a873dc8b8f65d + languageName: node + linkType: hard + "make-dir@npm:^4.0.0": version: 4.0.0 resolution: "make-dir@npm:4.0.0" @@ -8989,7 +8917,7 @@ __metadata: languageName: node linkType: hard -"micromatch@npm:^4.0.2, micromatch@npm:^4.0.4, micromatch@npm:^4.0.5": +"micromatch@npm:^4.0.4, micromatch@npm:^4.0.5": version: 4.0.5 resolution: "micromatch@npm:4.0.5" dependencies: @@ -9029,7 +8957,7 @@ __metadata: languageName: node linkType: hard -"min-indent@npm:^1.0.1": +"min-indent@npm:^1.0.0, min-indent@npm:^1.0.1": version: 1.0.1 resolution: "min-indent@npm:1.0.1" checksum: bfc6dd03c5eaf623a4963ebd94d087f6f4bbbfd8c41329a7f09706b0cb66969c4ddd336abeb587bc44bc6f08e13bf90f0b374f9d71f9f01e04adc2cd6f083ef1 @@ -9176,15 +9104,6 @@ __metadata: languageName: node linkType: hard -"mkdirp@npm:1.x, mkdirp@npm:^1.0.3, mkdirp@npm:^1.0.4": - version: 1.0.4 - resolution: "mkdirp@npm:1.0.4" - bin: - mkdirp: bin/cmd.js - checksum: a96865108c6c3b1b8e1d5e9f11843de1e077e57737602de1b82030815f311be11f96f09cce59bd5b903d0b29834733e5313f9301e3ed6d6f6fba2eae0df4298f - languageName: node - linkType: hard - "mkdirp@npm:^0.5.1": version: 0.5.6 resolution: "mkdirp@npm:0.5.6" @@ -9196,10 +9115,12 @@ __metadata: languageName: node linkType: hard -"moo@npm:^0.5.0": - version: 0.5.1 - resolution: "moo@npm:0.5.1" - checksum: 2d8c013f1f9aad8e5c7a9d4a03dbb4eecd91b9fe5e9446fbc7561fd38d4d161c742434acff385722542fe7b360fce9c586da62442379e62e4158ad49c7e1a6b7 +"mkdirp@npm:^1.0.3, mkdirp@npm:^1.0.4": + version: 1.0.4 + resolution: "mkdirp@npm:1.0.4" + bin: + mkdirp: bin/cmd.js + checksum: a96865108c6c3b1b8e1d5e9f11843de1e077e57737602de1b82030815f311be11f96f09cce59bd5b903d0b29834733e5313f9301e3ed6d6f6fba2eae0df4298f languageName: node linkType: hard @@ -9288,23 +9209,6 @@ __metadata: languageName: node linkType: hard -"nearley@npm:^2.7.10": - version: 2.20.1 - resolution: "nearley@npm:2.20.1" - dependencies: - commander: ^2.19.0 - moo: ^0.5.0 - railroad-diagrams: ^1.0.0 - randexp: 0.4.6 - bin: - nearley-railroad: bin/nearley-railroad.js - nearley-test: bin/nearley-test.js - nearley-unparse: bin/nearley-unparse.js - nearleyc: bin/nearleyc.js - checksum: 42c2c330c13c7991b48221c5df00f4352c2f8851636ae4d1f8ca3c8e193fc1b7668c78011d1cad88cca4c1c4dc087425420629c19cc286d7598ec15533aaef26 - languageName: node - linkType: hard - "negotiator@npm:^0.6.3": version: 0.6.3 resolution: "negotiator@npm:0.6.3" @@ -9326,7 +9230,7 @@ __metadata: languageName: node linkType: hard -"node-fetch@npm:2.6.7, node-fetch@npm:^2.6.0": +"node-fetch@npm:^2.6.0": version: 2.6.7 resolution: "node-fetch@npm:2.6.7" dependencies: @@ -9468,15 +9372,6 @@ __metadata: languageName: node linkType: hard -"nth-check@npm:^2.0.1": - version: 2.1.1 - resolution: "nth-check@npm:2.1.1" - dependencies: - boolbase: ^1.0.0 - checksum: 5afc3dafcd1573b08877ca8e6148c52abd565f1d06b1eb08caf982e3fa289a82f2cae697ffb55b5021e146d60443f1590a5d6b944844e944714a5b549675bcd3 - languageName: node - linkType: hard - "nwsapi@npm:^2.2.2": version: 2.2.7 resolution: "nwsapi@npm:2.2.7" @@ -9491,14 +9386,14 @@ __metadata: languageName: node linkType: hard -"object-inspect@npm:^1.12.3, object-inspect@npm:^1.7.0, object-inspect@npm:^1.9.0": +"object-inspect@npm:^1.12.3, object-inspect@npm:^1.9.0": version: 1.12.3 resolution: "object-inspect@npm:1.12.3" checksum: dabfd824d97a5f407e6d5d24810d888859f6be394d8b733a77442b277e0808860555176719c5905e765e3743a7cada6b8b0a3b85e5331c530fd418cc8ae991db languageName: node linkType: hard -"object-is@npm:^1.0.2, object-is@npm:^1.1.5": +"object-is@npm:^1.1.5": version: 1.1.5 resolution: "object-is@npm:1.1.5" dependencies: @@ -9515,7 +9410,7 @@ __metadata: languageName: node linkType: hard -"object.assign@npm:^4.1.0, object.assign@npm:^4.1.3, object.assign@npm:^4.1.4": +"object.assign@npm:^4.1.3, object.assign@npm:^4.1.4": version: 4.1.4 resolution: "object.assign@npm:4.1.4" dependencies: @@ -9527,7 +9422,7 @@ __metadata: languageName: node linkType: hard -"object.entries@npm:^1.1.1, object.entries@npm:^1.1.6": +"object.entries@npm:^1.1.6": version: 1.1.6 resolution: "object.entries@npm:1.1.6" dependencies: @@ -9538,7 +9433,7 @@ __metadata: languageName: node linkType: hard -"object.fromentries@npm:^2.0.0, object.fromentries@npm:^2.0.6": +"object.fromentries@npm:^2.0.6": version: 2.0.6 resolution: "object.fromentries@npm:2.0.6" dependencies: @@ -9559,7 +9454,7 @@ __metadata: languageName: node linkType: hard -"object.values@npm:^1.1.1, object.values@npm:^1.1.6": +"object.values@npm:^1.1.6": version: 1.1.6 resolution: "object.values@npm:1.1.6" dependencies: @@ -9729,16 +9624,6 @@ __metadata: languageName: node linkType: hard -"parse5-htmlparser2-tree-adapter@npm:^7.0.0": - version: 7.0.0 - resolution: "parse5-htmlparser2-tree-adapter@npm:7.0.0" - dependencies: - domhandler: ^5.0.2 - parse5: ^7.0.0 - checksum: fc5d01e07733142a1baf81de5c2a9c41426c04b7ab29dd218acb80cd34a63177c90aff4a4aee66cf9f1d0aeecff1389adb7452ad6f8af0a5888e3e9ad6ef733d - languageName: node - linkType: hard - "parse5@npm:^7.0.0, parse5@npm:^7.1.1": version: 7.1.2 resolution: "parse5@npm:7.1.2" @@ -9839,13 +9724,6 @@ __metadata: languageName: node linkType: hard -"performance-now@npm:^2.1.0": - version: 2.1.0 - resolution: "performance-now@npm:2.1.0" - checksum: 534e641aa8f7cba160f0afec0599b6cecefbb516a2e837b512be0adbe6c1da5550e89c78059c7fabc5c9ffdf6627edabe23eb7c518c4500067a898fa65c2b550 - languageName: node - linkType: hard - "picocolors@npm:^1.0.0": version: 1.0.0 resolution: "picocolors@npm:1.0.0" @@ -10021,6 +9899,17 @@ __metadata: languageName: node linkType: hard +"pretty-format@npm:^27.0.2": + version: 27.5.1 + resolution: "pretty-format@npm:27.5.1" + dependencies: + ansi-regex: ^5.0.1 + ansi-styles: ^5.0.0 + react-is: ^17.0.1 + checksum: cf610cffcb793885d16f184a62162f2dd0df31642d9a18edf4ca298e909a8fe80bdbf556d5c9573992c102ce8bf948691da91bf9739bee0ffb6e79c8a8a6e088 + languageName: node + linkType: hard + "pretty-format@npm:^29.0.0, pretty-format@npm:^29.7.0": version: 29.7.0 resolution: "pretty-format@npm:29.7.0" @@ -10039,13 +9928,6 @@ __metadata: languageName: node linkType: hard -"promise-polyfill@npm:^8.1.3": - version: 8.2.3 - resolution: "promise-polyfill@npm:8.2.3" - checksum: f320278bab8b8ce32f0e2f377d75aabe35c90a79f92b3e001db084d635da1fb8740ba8a26b5c1c2b4cdca5a4c988f9b6b3eb30f0b151d0362d3d9c1675024a8f - languageName: node - linkType: hard - "promise-retry@npm:^2.0.1": version: 2.0.1 resolution: "promise-retry@npm:2.0.1" @@ -10066,7 +9948,7 @@ __metadata: languageName: node linkType: hard -"prop-types@npm:^15.6.2, prop-types@npm:^15.7.0, prop-types@npm:^15.8.1": +"prop-types@npm:^15.6.2, prop-types@npm:^15.8.1": version: 15.8.1 resolution: "prop-types@npm:15.8.1" dependencies: @@ -10119,32 +10001,6 @@ __metadata: languageName: node linkType: hard -"raf@npm:^3.4.1": - version: 3.4.1 - resolution: "raf@npm:3.4.1" - dependencies: - performance-now: ^2.1.0 - checksum: 50ba284e481c8185dbcf45fc4618ba3aec580bb50c9121385d5698cb6012fe516d2015b1df6dd407a7b7c58d44be8086108236affbce1861edd6b44637c8cd52 - languageName: node - linkType: hard - -"railroad-diagrams@npm:^1.0.0": - version: 1.0.0 - resolution: "railroad-diagrams@npm:1.0.0" - checksum: 9e312af352b5ed89c2118edc0c06cef2cc039681817f65266719606e4e91ff6ae5374c707cc9033fe29a82c2703edf3c63471664f97f0167c85daf6f93496319 - languageName: node - linkType: hard - -"randexp@npm:0.4.6": - version: 0.4.6 - resolution: "randexp@npm:0.4.6" - dependencies: - discontinuous-range: 1.0.0 - ret: ~0.1.10 - checksum: 3c0d440a3f89d6d36844aa4dd57b5cdb0cab938a41956a16da743d3a3578ab32538fc41c16cc0984b6938f2ae4cbc0216967e9829e52191f70e32690d8e3445d - languageName: node - linkType: hard - "randombytes@npm:^2.1.0": version: 2.1.0 resolution: "randombytes@npm:2.1.0" @@ -10166,13 +10022,6 @@ __metadata: languageName: node linkType: hard -"react-is@npm:^16.12.0 || ^17.0.0 || ^18.0.0, react-is@npm:^18.0.0, react-is@npm:^18.2.0": - version: 18.2.0 - resolution: "react-is@npm:18.2.0" - checksum: e72d0ba81b5922759e4aff17e0252bd29988f9642ed817f56b25a3e217e13eea8a7f2322af99a06edb779da12d5d636e9fda473d620df9a3da0df2a74141d53e - languageName: node - linkType: hard - "react-is@npm:^16.13.1, react-is@npm:^16.7.0": version: 16.13.1 resolution: "react-is@npm:16.13.1" @@ -10180,36 +10029,17 @@ __metadata: languageName: node linkType: hard -"react-is@npm:^17.0.0, react-is@npm:^17.0.2": +"react-is@npm:^17.0.1": version: 17.0.2 resolution: "react-is@npm:17.0.2" checksum: 9d6d111d8990dc98bc5402c1266a808b0459b5d54830bbea24c12d908b536df7883f268a7868cfaedde3dd9d4e0d574db456f84d2e6df9c4526f99bb4b5344d8 languageName: node linkType: hard -"react-shallow-renderer@npm:^16.13.1": - version: 16.15.0 - resolution: "react-shallow-renderer@npm:16.15.0" - dependencies: - object-assign: ^4.1.1 - react-is: ^16.12.0 || ^17.0.0 || ^18.0.0 - peerDependencies: - react: ^16.0.0 || ^17.0.0 || ^18.0.0 - checksum: 6052c7e3e9627485120ebd8257f128aad8f56386fe8d42374b7743eac1be457c33506d153c7886b4e32923c0c352d402ab805ef9ca02dbcd8393b2bdeb6e5af8 - languageName: node - linkType: hard - -"react-test-renderer@npm:^17.0.0": - version: 17.0.2 - resolution: "react-test-renderer@npm:17.0.2" - dependencies: - object-assign: ^4.1.1 - react-is: ^17.0.2 - react-shallow-renderer: ^16.13.1 - scheduler: ^0.20.2 - peerDependencies: - react: 17.0.2 - checksum: e6b5c6ed2a0bde2c34f1ab9523ff9bc4c141a271daf730d6b852374e83acc0155d58ab71a318251e953ebfa65b8bebb9c5dce3eba1ccfcbef7cc4e1e8261c401 +"react-is@npm:^18.0.0, react-is@npm:^18.2.0": + version: 18.2.0 + resolution: "react-is@npm:18.2.0" + checksum: e72d0ba81b5922759e4aff17e0252bd29988f9642ed817f56b25a3e217e13eea8a7f2322af99a06edb779da12d5d636e9fda473d620df9a3da0df2a74141d53e languageName: node linkType: hard @@ -10314,6 +10144,16 @@ __metadata: languageName: node linkType: hard +"redent@npm:^3.0.0": + version: 3.0.0 + resolution: "redent@npm:3.0.0" + dependencies: + indent-string: ^4.0.0 + strip-indent: ^3.0.0 + checksum: fa1ef20404a2d399235e83cc80bd55a956642e37dd197b4b612ba7327bf87fa32745aeb4a1634b2bab25467164ab4ed9c15be2c307923dd08b0fe7c52431ae6b + languageName: node + linkType: hard + "redent@npm:^4.0.0": version: 4.0.0 resolution: "redent@npm:4.0.0" @@ -10367,6 +10207,17 @@ __metadata: languageName: node linkType: hard +"regexp.prototype.flags@npm:^1.5.0": + version: 1.5.1 + resolution: "regexp.prototype.flags@npm:1.5.1" + dependencies: + call-bind: ^1.0.2 + define-properties: ^1.2.0 + set-function-name: ^2.0.0 + checksum: 869edff00288442f8d7fa4c9327f91d85f3b3acf8cbbef9ea7a220345cf23e9241b6def9263d2c1ebcf3a316b0aa52ad26a43a84aa02baca3381717b3e307f47 + languageName: node + linkType: hard + "regexpu-core@npm:^5.3.1": version: 5.3.2 resolution: "regexpu-core@npm:5.3.2" @@ -10420,6 +10271,13 @@ __metadata: languageName: node linkType: hard +"resize-observer-polyfill@npm:^1.5.1": + version: 1.5.1 + resolution: "resize-observer-polyfill@npm:1.5.1" + checksum: 57e7f79489867b00ba43c9c051524a5c8f162a61d5547e99333549afc23e15c44fd43f2f318ea0261ea98c0eb3158cca261e6f48d66e1ed1cd1f340a43977094 + languageName: node + linkType: hard + "resolve-cwd@npm:^3.0.0": version: 3.0.0 resolution: "resolve-cwd@npm:3.0.0" @@ -10512,13 +10370,6 @@ __metadata: languageName: node linkType: hard -"ret@npm:~0.1.10": - version: 0.1.15 - resolution: "ret@npm:0.1.15" - checksum: d76a9159eb8c946586567bd934358dfc08a36367b3257f7a3d7255fdd7b56597235af23c6afa0d7f0254159e8051f93c918809962ebd6df24ca2a83dbe4d4151 - languageName: node - linkType: hard - "retry@npm:^0.12.0": version: 0.12.0 resolution: "retry@npm:0.12.0" @@ -10562,16 +10413,6 @@ __metadata: languageName: node linkType: hard -"rst-selector-parser@npm:^2.2.3": - version: 2.2.3 - resolution: "rst-selector-parser@npm:2.2.3" - dependencies: - lodash.flattendeep: ^4.4.0 - nearley: ^2.7.10 - checksum: fbfb2f6a7d4c9b3e013ef555ac06e5dba444e0d37dc959b94c507b6c34093ef10fe98141338d9cac58e5ae0f9453a5ef7f85af3d5e6386b237c1b3552debe4a0 - languageName: node - linkType: hard - "run-applescript@npm:^5.0.0": version: 5.0.0 resolution: "run-applescript@npm:5.0.0" @@ -10663,16 +10504,6 @@ __metadata: languageName: node linkType: hard -"scheduler@npm:^0.20.2": - version: 0.20.2 - resolution: "scheduler@npm:0.20.2" - dependencies: - loose-envify: ^1.1.0 - object-assign: ^4.1.1 - checksum: c4b35cf967c8f0d3e65753252d0f260271f81a81e427241295c5a7b783abf4ea9e905f22f815ab66676f5313be0a25f47be582254db8f9241b259213e999b8fc - languageName: node - linkType: hard - "scheduler@npm:^0.23.0": version: 0.23.0 resolution: "scheduler@npm:0.23.0" @@ -10725,23 +10556,23 @@ __metadata: languageName: node linkType: hard -"semver@npm:7.x, semver@npm:^7.3.4, semver@npm:^7.3.5, semver@npm:^7.3.8, semver@npm:^7.5.3, semver@npm:^7.5.4": - version: 7.5.4 - resolution: "semver@npm:7.5.4" - dependencies: - lru-cache: ^6.0.0 +"semver@npm:^6.3.0, semver@npm:^6.3.1": + version: 6.3.1 + resolution: "semver@npm:6.3.1" bin: semver: bin/semver.js - checksum: 12d8ad952fa353b0995bf180cdac205a4068b759a140e5d3c608317098b3575ac2f1e09182206bf2eb26120e1c0ed8fb92c48c592f6099680de56bb071423ca3 + checksum: ae47d06de28836adb9d3e25f22a92943477371292d9b665fb023fae278d345d508ca1958232af086d85e0155aee22e313e100971898bbb8d5d89b8b1d4054ca2 languageName: node linkType: hard -"semver@npm:^6.3.0, semver@npm:^6.3.1": - version: 6.3.1 - resolution: "semver@npm:6.3.1" +"semver@npm:^7.3.4, semver@npm:^7.3.5, semver@npm:^7.3.8, semver@npm:^7.5.3, semver@npm:^7.5.4": + version: 7.5.4 + resolution: "semver@npm:7.5.4" + dependencies: + lru-cache: ^6.0.0 bin: semver: bin/semver.js - checksum: ae47d06de28836adb9d3e25f22a92943477371292d9b665fb023fae278d345d508ca1958232af086d85e0155aee22e313e100971898bbb8d5d89b8b1d4054ca2 + checksum: 12d8ad952fa353b0995bf180cdac205a4068b759a140e5d3c608317098b3575ac2f1e09182206bf2eb26120e1c0ed8fb92c48c592f6099680de56bb071423ca3 languageName: node linkType: hard @@ -10761,6 +10592,17 @@ __metadata: languageName: node linkType: hard +"set-function-name@npm:^2.0.0": + version: 2.0.1 + resolution: "set-function-name@npm:2.0.1" + dependencies: + define-data-property: ^1.0.1 + functions-have-names: ^1.2.3 + has-property-descriptors: ^1.0.0 + checksum: 4975d17d90c40168eee2c7c9c59d023429f0a1690a89d75656306481ece0c3c1fb1ebcc0150ea546d1913e35fbd037bace91372c69e543e51fc5d1f31a9fa126 + languageName: node + linkType: hard + "shallow-clone@npm:^3.0.0": version: 3.0.1 resolution: "shallow-clone@npm:3.0.1" @@ -11039,6 +10881,15 @@ __metadata: languageName: node linkType: hard +"stop-iteration-iterator@npm:^1.0.0": + version: 1.0.0 + resolution: "stop-iteration-iterator@npm:1.0.0" + dependencies: + internal-slot: ^1.0.4 + checksum: d04173690b2efa40e24ab70e5e51a3ff31d56d699550cfad084104ab3381390daccb36652b25755e420245f3b0737de66c1879eaa2a8d4fc0a78f9bf892fcb42 + languageName: node + linkType: hard + "string-argv@npm:^0.3.1": version: 0.3.2 resolution: "string-argv@npm:0.3.2" @@ -11105,7 +10956,7 @@ __metadata: languageName: node linkType: hard -"string.prototype.trim@npm:^1.2.1, string.prototype.trim@npm:^1.2.7": +"string.prototype.trim@npm:^1.2.7": version: 1.2.7 resolution: "string.prototype.trim@npm:1.2.7" dependencies: @@ -11193,6 +11044,15 @@ __metadata: languageName: node linkType: hard +"strip-indent@npm:^3.0.0": + version: 3.0.0 + resolution: "strip-indent@npm:3.0.0" + dependencies: + min-indent: ^1.0.0 + checksum: 18f045d57d9d0d90cd16f72b2313d6364fd2cb4bf85b9f593523ad431c8720011a4d5f08b6591c9d580f446e78855c5334a30fb91aa1560f5d9f95ed1b4a0530 + languageName: node + linkType: hard + "strip-indent@npm:^4.0.0": version: 4.0.0 resolution: "strip-indent@npm:4.0.0" @@ -11594,29 +11454,6 @@ __metadata: languageName: node linkType: hard -"ts-jest@npm:^26.0.0": - version: 26.5.6 - resolution: "ts-jest@npm:26.5.6" - dependencies: - bs-logger: 0.x - buffer-from: 1.x - fast-json-stable-stringify: 2.x - jest-util: ^26.1.0 - json5: 2.x - lodash: 4.x - make-error: 1.x - mkdirp: 1.x - semver: 7.x - yargs-parser: 20.x - peerDependencies: - jest: ">=26 <27" - typescript: ">=3.8 <5.0" - bin: - ts-jest: cli.js - checksum: 6f65ad4fe67ab3f0fd4c7f9954acbee863af05b2b3f88dd0f490bbcdc58002960fac908b2cb9f009ec14da6fe13cb00a39e291260d6e555abe72448d1c0a017f - languageName: node - linkType: hard - "ts-jest@npm:^29.1.0": version: 29.1.1 resolution: "ts-jest@npm:29.1.1" @@ -12193,6 +12030,18 @@ __metadata: languageName: node linkType: hard +"which-collection@npm:^1.0.1": + version: 1.0.1 + resolution: "which-collection@npm:1.0.1" + dependencies: + is-map: ^2.0.1 + is-set: ^2.0.1 + is-weakmap: ^2.0.1 + is-weakset: ^2.0.1 + checksum: c815bbd163107ef9cb84f135e6f34453eaf4cca994e7ba85ddb0d27cea724c623fae2a473ceccfd5549c53cc65a5d82692de418166df3f858e1e5dc60818581c + languageName: node + linkType: hard + "which-module@npm:^2.0.0": version: 2.0.1 resolution: "which-module@npm:2.0.1" @@ -12461,13 +12310,6 @@ __metadata: languageName: node linkType: hard -"yargs-parser@npm:20.x, yargs-parser@npm:^20.2.9": - version: 20.2.9 - resolution: "yargs-parser@npm:20.2.9" - checksum: 8bb69015f2b0ff9e17b2c8e6bfe224ab463dd00ca211eece72a4cd8a906224d2703fb8a326d36fdd0e68701e201b2a60ed7cf81ce0fd9b3799f9fe7745977ae3 - languageName: node - linkType: hard - "yargs-parser@npm:^18.1.2": version: 18.1.3 resolution: "yargs-parser@npm:18.1.3" @@ -12478,6 +12320,13 @@ __metadata: languageName: node linkType: hard +"yargs-parser@npm:^20.2.9": + version: 20.2.9 + resolution: "yargs-parser@npm:20.2.9" + checksum: 8bb69015f2b0ff9e17b2c8e6bfe224ab463dd00ca211eece72a4cd8a906224d2703fb8a326d36fdd0e68701e201b2a60ed7cf81ce0fd9b3799f9fe7745977ae3 + languageName: node + linkType: hard + "yargs-parser@npm:^21.0.1, yargs-parser@npm:^21.1.1": version: 21.1.1 resolution: "yargs-parser@npm:21.1.1"