diff --git a/.github/workflows/e2e-tests.yml b/.github/workflows/e2e-tests.yml index 928baecde..00d9834bb 100644 --- a/.github/workflows/e2e-tests.yml +++ b/.github/workflows/e2e-tests.yml @@ -289,17 +289,6 @@ jobs: - windows-latest-8-cores shardIndex: [1, 2, 3, 4] shardTotal: [4] - # Disable macos and windows tests on hourly e2e tests since we only care - # about server side changes. - # Technique from https://github.com/joaomcteixeira/python-project-skeleton/pull/31/files - isScheduled: - - ${{ github.event_name == 'schedule' }} - exclude: - - os: namespace-profile-macos-8-cores - isScheduled: true - - os: windows-latest-8-cores - isScheduled: true - # TODO: add ref here for main and latest release tag runs-on: ${{ matrix.os }} steps: - uses: actions/checkout@v4 diff --git a/.husky/pre-commit b/.husky/pre-commit new file mode 100755 index 000000000..4aae0f234 --- /dev/null +++ b/.husky/pre-commit @@ -0,0 +1 @@ +npm run fmt diff --git a/.husky/pre-push b/.husky/pre-push deleted file mode 100755 index 6ca0c7637..000000000 --- a/.husky/pre-push +++ /dev/null @@ -1,4 +0,0 @@ -#!/usr/bin/env sh -. "$(dirname -- "$0")/_/husky.sh" - -npm run fmt-check diff --git a/docs/kcl/getNextAdjacentEdge.md b/docs/kcl/getNextAdjacentEdge.md index 764de41ed..f09e1bff0 100644 --- a/docs/kcl/getNextAdjacentEdge.md +++ b/docs/kcl/getNextAdjacentEdge.md @@ -9,7 +9,7 @@ Get the next adjacent edge to the edge given. ```js -getNextAdjacentEdge(tag: TagIdentifier): Uuid +getNextAdjacentEdge(edge: TagIdentifier): Uuid ``` @@ -17,7 +17,7 @@ getNextAdjacentEdge(tag: TagIdentifier): Uuid | Name | Type | Description | Required | |----------|------|-------------|----------| -| [`tag`](/docs/kcl/types/tag) | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | +| `edge` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | The tag of the edge you want to find the next adjacent edge of. | Yes | ### Returns diff --git a/docs/kcl/getOppositeEdge.md b/docs/kcl/getOppositeEdge.md index 8ff4b784a..f879fdf13 100644 --- a/docs/kcl/getOppositeEdge.md +++ b/docs/kcl/getOppositeEdge.md @@ -9,7 +9,7 @@ Get the opposite edge to the edge given. ```js -getOppositeEdge(tag: TagIdentifier): Uuid +getOppositeEdge(edge: TagIdentifier): Uuid ``` @@ -17,7 +17,7 @@ getOppositeEdge(tag: TagIdentifier): Uuid | Name | Type | Description | Required | |----------|------|-------------|----------| -| [`tag`](/docs/kcl/types/tag) | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | +| `edge` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | The tag of the edge you want to find the opposite edge of. | Yes | ### Returns diff --git a/docs/kcl/getPreviousAdjacentEdge.md b/docs/kcl/getPreviousAdjacentEdge.md index a2ad65740..75d28bcb6 100644 --- a/docs/kcl/getPreviousAdjacentEdge.md +++ b/docs/kcl/getPreviousAdjacentEdge.md @@ -9,7 +9,7 @@ Get the previous adjacent edge to the edge given. ```js -getPreviousAdjacentEdge(tag: TagIdentifier): Uuid +getPreviousAdjacentEdge(edge: TagIdentifier): Uuid ``` @@ -17,7 +17,7 @@ getPreviousAdjacentEdge(tag: TagIdentifier): Uuid | Name | Type | Description | Required | |----------|------|-------------|----------| -| [`tag`](/docs/kcl/types/tag) | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | +| `edge` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | The tag of the edge you want to find the previous adjacent edge of. | Yes | ### Returns diff --git a/docs/kcl/hole.md b/docs/kcl/hole.md index 45cd795e2..7dd404d78 100644 --- a/docs/kcl/hole.md +++ b/docs/kcl/hole.md @@ -62,6 +62,6 @@ exampleSketch = startSketchOn(-XZ) example = extrude(exampleSketch, length = 1) ``` -![Rendered example of hole 1]() +![Rendered example of hole 1]() diff --git a/docs/kcl/hollow.md b/docs/kcl/hollow.md index 11dcff38d..e39a6f118 100644 --- a/docs/kcl/hollow.md +++ b/docs/kcl/hollow.md @@ -56,7 +56,7 @@ firstSketch = startSketchOn(-XZ) |> hollow(thickness = 0.5) ``` -![Rendered example of hollow 1]() +![Rendered example of hollow 1]() ```js // Hollow a sketch on face object. @@ -80,6 +80,6 @@ thing2 = startSketchOn(case, face = END) hollow(case, thickness = 0.5) ``` -![Rendered example of hollow 2]() +![Rendered example of hollow 2]() diff --git a/docs/kcl/modules.md b/docs/kcl/modules.md index 39d2844be..313996aec 100644 --- a/docs/kcl/modules.md +++ b/docs/kcl/modules.md @@ -10,7 +10,7 @@ isolated from other files as a separate module. When you define a function, you can use `export` before it to make it available to other modules. -``` +```kcl // util.kcl export fn increment(x) { return x + 1 @@ -31,11 +31,11 @@ Imported files _must_ be in the same project so that units are uniform across modules. This means that it must be in the same directory. Import statements must be at the top-level of a file. It is not allowed to have -an `import` statement inside a function or in the body of an if-else. +an `import` statement inside a function or in the body of an if‑else. Multiple functions can be exported in a file. -``` +```kcl // util.kcl export fn increment(x) { return x + 1 @@ -58,6 +58,211 @@ Imported symbols can be renamed for convenience or to avoid name collisions. import increment as inc, decrement as dec from "util.kcl" ``` +--- + +## Functions vs `clone` + +There are two common patterns for re‑using geometry: + +1. **Wrap the construction in a function** – flexible and fully parametric. +2. **Duplicate an existing object with `clone`** – lightning‑fast, but an exact + duplicate. + +### Parametric function example + +```kcl +fn cube(center) { + return startSketchOn(XY) + |> startProfileAt([center[0] - 10, center[1] - 10], %) + |> line(endAbsolute = [center[0] + 10, center[1] - 10]) + |> line(endAbsolute = [center[0] + 10, center[1] + 10]) + |> line(endAbsolute = [center[0] - 10, center[1] + 10]) + |> close() + |> extrude(length = 10) +} + +myCube = cube([0, 0]) +``` + +*Pros* +- Any argument can be a parameter – size, position, appearance, etc. +- Works great inside loops, arrays, or optimisation sweeps. + +*Cons* +- Every invocation rebuilds the entire feature tree. +- **Slower** than a straight duplicate – each call is its own render job. + +### `clone` example + +```kcl +sketch001 = startSketchOn(-XZ) + |> circle(center = [0, 0], radius = 10) + |> extrude(length = 5) + |> appearance(color = "#ff0000", metalness = 90, roughness = 90) + +sketch002 = clone(sketch001) // ✓ instant copy +``` + +*Pros* +- Roughly an O(1) operation – we just duplicate the underlying engine handle. +- Perfect when you need ten identical bolts or two copies of the same imported STEP file. + +*Cons* +- **Not parametric** – the clone is exactly the same shape as the source. +- If you need to tweak dimensions per‑instance, you’re back to a function. + +> **Rule of thumb** – Reach for `clone` when the geometry is already what you want. Reach for a function when you need customisation. + +--- + +## Module‑level parallelism + +Under the hood, the Design Studio runs **every module in parallel** where it can. This means: + +- The top‑level code of `foo.kcl`, `bar.kcl`, and `baz.kcl` all start executing immediately and concurrently. +- Imports that read foreign files (STEP/OBJ/…) overlap their I/O and background render. +- CPU‑bound calculations in separate modules get their own worker threads. + +### Why modules beat one‑big‑file + +If you shoe‑horn everything into `main.kcl`, each statement runs sequentially: + +```norun +import "big.step" as gizmo // blocks main while reading + +gizmo |> translate(x=50) // blocks again while waiting for render +``` + +Split `gizmo` into its own file and the read/render can overlap whatever else `main.kcl` is doing. + +```norun +// gizmo.kcl (worker A) +import "big.step" + +// main.kcl (worker B) +import "gizmo.kcl" as gizmo // non‑blocking + +// ... other setup ... + +gizmo |> translate(x=50) // only blocks here +``` + +### Gotcha: defining but **not** calling functions + +Defining a function inside a module is instantaneous – we just record the byte‑code. The heavy lifting happens when the function is **called**. So: + +```norun +// util.kcl +export fn makeBolt(size) { /* … expensive CAD … */ } +``` + +If `main.kcl` waits until the very end to call `makeBolt`, *none* of that work was parallelised – you’ve pushed the cost back onto the serial tail of your script. + +**Better:** call it early or move the invocation into another module. + +```norun +// bolt_instance.kcl +import makeBolt from "util.kcl" +bolt = makeBolt(5) // executed in parallel +bolt +``` + +Now `main.kcl` can `import "bolt_instance.kcl" as bolt` and get the result that was rendered while it was busy doing other things. + +--- + +## Whole module import + +You can also import the whole module. This is useful if you want to use the +result of a module as a variable, like a part. + +```norun +import "tests/inputs/cube.kcl" as cube +cube + |> translate(x=10) +``` + +This imports the whole module and makes it available as `cube`. You can then +use it like any other object. The `cube` variable is now a reference to the +result of the module. This means that if you change the module, the `cube` +variable will change as well. + +In `cube.kcl`, you cannot have multiple objects. It has to be a single part. If +you have multiple objects, you will get an error. This is because the module is +expected to return a single object that can be used as a variable. + +You also cannot assign that object to a variable. This is because the module is +expected to return a single object that can be used as a variable. + +So for example, this is not allowed: + +```norun +... a bunch of code to create cube and cube2 ... + +myUnion = union([cube, cube2]) +``` + +What you need to do instead is: + +```norun +... a bunch of code to create cube and cube2 ... + +union([cube, cube2]) +``` + +That way the last line will return the union of the two objects. + +Or what you could do instead is: + +```norun +... a bunch of code to create cube and cube2 ... + +myUnion = union([cube, cube2]) +myUnion +``` + +This will return the union of the two objects, but it will not be assigned to a +variable. This is because the module is expected to return a single object that +can be used as a variable. + +--- + +## Multiple instances of the same import + +Whether you are importing a file from another CAD system or a KCL file, that +file represents object(s) in memory. If you import the same file multiple times, +it will only be rendered once. + +If you want to have multiple instances of the same object, you can use the +[`clone`](/docs/kcl/clone) function. This will render a new instance of the object in memory. + +```norun +import cube from "tests/inputs/cube.kcl" + +cube + |> translate(x=10) +clone(cube) + |> translate(x=20) +``` + +In the sample above, the `cube` object is imported from a KCL file. The first +instance is translated 10 units in the x direction. The second instance is +cloned and translated 20 units in the x direction. The two instances are now +separate objects in memory, and can be manipulated independently. + +Here is an example with a file from another CAD system: + +```kcl +import "tests/inputs/cube.step" as cube + +cube + |> translate(x=10) +clone(cube) + |> translate(x=20) +``` + +--- + ## Importing files from other CAD systems `import` can also be used to import files from other CAD systems. The format of the statement is the @@ -69,25 +274,17 @@ import "tests/inputs/cube.obj" // Use `cube` just like a KCL object. ``` -```norun -import "tests/inputs/cube-2.sldprt" as cube +```kcl +import "tests/inputs/cube.sldprt" as cube // Use `cube` just like a KCL object. ``` -You can make the file format explicit using a format attribute (useful if using a different -extension), e.g., - -```norun -@(format = obj) -import "tests/inputs/cube" -``` - For formats lacking unit data (such as STL, OBJ, or PLY files), the default unit of measurement is millimeters. Alternatively you may specify the unit -by using an attirbute. Likewise, you can also specify a coordinate system. E.g., +by using an attribute. Likewise, you can also specify a coordinate system. E.g., -```norun +```kcl @(unitLength = ft, coords = opengl) import "tests/inputs/cube.obj" ``` @@ -110,97 +307,55 @@ Coordinate systems: - `opengl`, forward: +Z, up: +Y, handedness: right - `vulkan`, forward: +Z, up: -Y, handedness: left -### Performance +--- -Parallelized foreign-file imports now let you overlap file reads, initialization, +## Performance deep‑dive for foreign‑file imports + +Parallelized foreign‑file imports now let you overlap file reads, initialization, and rendering. To maximize throughput, you need to understand the three distinct stages—reading, initializing (background render start), and invocation (blocking) —and structure your code to defer blocking operations until the end. -#### Foreign Import Execution Stages +### Foreign import execution stages -1. **Import (Read) Stage** - ```norun +1. **Import (Read / Initialization) Stage** + ```kcl import "tests/inputs/cube.step" as cube - ``` - - Reads the file from disk and makes its API available. - - **Does _not_** start Engine rendering or block your script. + ``` + - Reads the file from disk and makes its API available. + - Starts engine rendering but **does not block** your script. + - This kick‑starts the render pipeline while you keep executing other code. -2. **Initialization (Background Render) Stage** - ```norun +2. **Invocation (Blocking) Stage** + ```kcl import "tests/inputs/cube.step" as cube - myCube = cube // <- This line starts background rendering - ``` - - Invoking the imported symbol (assignment or plain call) triggers Engine rendering _in the background_. - - This kick‑starts the render pipeline but doesn’t block—you can continue other work while the Engine processes the model. + cube + |> translate(z=10) // ← blocks here only + ``` + - Any method call (e.g., `translate`, `scale`, `rotate`) waits for the background render to finish before applying transformations. -3. **Invocation (Blocking) Stage** - ```norun - import "tests/inputs/cube.step" as cube +### Best practices - myCube = cube +#### 1. Defer blocking calls - myCube - |> translate(z=10) // <- This line blocks - ``` - - Any method call (e.g., `translate`, `scale`, `rotate`) waits for the background render to finish before applying transformations. - - This is the only point where your script will block. - -> **Nuance:** Foreign imports differ from pure KCL modules—calling the same import symbol multiple times (e.g., `screw` twice) starts background rendering twice. - -#### Best Practices - -##### 1. Defer Blocking Calls -Initialize early but delay all transformations until after your heavy computation: -```norun -import "tests/inputs/cube.step" as cube // 1) Read - -myCube = cube // 2) Background render starts +```kcl +import "tests/inputs/cube.step" as cube // 1) Read / Background render starts -// --- perform other operations and calculations or setup here --- +// --- perform other operations and calculations here --- -myCube - |> translate(z=10) // 3) Blocks only here -``` - -##### 2. Encapsulate Imports in Modules -Keep `main.kcl` free of reads and initialization; wrap them: - -```norun -// imports.kcl -import "tests/inputs/cube.step" as cube // Read only - - -export myCube = cube // Kick off rendering -``` - -```norun -// main.kcl -import myCube from "imports.kcl" // Import the initialized object - - -// ... computations ... - - -myCube - |> translate(z=10) // Blocking call at the end -``` - -##### 3. Avoid Immediate Method Calls - -```norun -import "tests/inputs/cube.step" as cube - cube - |> translate(z=10) // Blocks immediately, negating parallelism + |> translate(z=10) // 2) Blocks only here ``` -Both calling methods right on `cube` immediately or leaving an implicit import without assignment introduce blocking. +#### 2. Split heavy work into separate modules -#### Future Improvements +Place computationally expensive or IO‑heavy work into its own module so it can render in parallel while `main.kcl` continues. + +#### Future improvements + +Upcoming releases will auto‑analyse dependencies and only block when truly necessary. Until then, explicit deferral will give you the best performance. -Upcoming releases will auto‑analyze dependencies and only block when truly necessary. Until then, explicit deferral and modular wrapping give you the best performance. diff --git a/docs/kcl/shell.md b/docs/kcl/shell.md index 4ba3eeadc..f0e999664 100644 --- a/docs/kcl/shell.md +++ b/docs/kcl/shell.md @@ -62,7 +62,7 @@ firstSketch = startSketchOn(-XZ) shell(firstSketch, faces = [START], thickness = 0.25) ``` -![Rendered example of shell 1]() +![Rendered example of shell 1]() ```js // Remove a tagged face and the end face for the extrusion. @@ -119,7 +119,7 @@ thing2 = startSketchOn(case, face = END) shell(case, faces = [START], thickness = 5) ``` -![Rendered example of shell 4]() +![Rendered example of shell 4]() ```js // Shell a sketch on face object on the end face. diff --git a/docs/kcl/std.json b/docs/kcl/std.json index 9d39e9361..91986165e 100644 --- a/docs/kcl/std.json +++ b/docs/kcl/std.json @@ -111819,10 +111819,10 @@ "summary": "Get the next adjacent edge to the edge given.", "description": "", "tags": [], - "keywordArguments": false, + "keywordArguments": true, "args": [ { - "name": "tag", + "name": "edge", "type": "TagIdentifier", "schema": { "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", @@ -111839,7 +111839,8 @@ }, "required": true, "includeInSnippet": true, - "labelRequired": true + "description": "The tag of the edge you want to find the next adjacent edge of.", + "labelRequired": false } ], "returnValue": { @@ -111866,10 +111867,10 @@ "summary": "Get the opposite edge to the edge given.", "description": "", "tags": [], - "keywordArguments": false, + "keywordArguments": true, "args": [ { - "name": "tag", + "name": "edge", "type": "TagIdentifier", "schema": { "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", @@ -111886,7 +111887,8 @@ }, "required": true, "includeInSnippet": true, - "labelRequired": true + "description": "The tag of the edge you want to find the opposite edge of.", + "labelRequired": false } ], "returnValue": { @@ -111913,10 +111915,10 @@ "summary": "Get the previous adjacent edge to the edge given.", "description": "", "tags": [], - "keywordArguments": false, + "keywordArguments": true, "args": [ { - "name": "tag", + "name": "edge", "type": "TagIdentifier", "schema": { "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", @@ -111933,7 +111935,8 @@ }, "required": true, "includeInSnippet": true, - "labelRequired": true + "description": "The tag of the edge you want to find the previous adjacent edge of.", + "labelRequired": false } ], "returnValue": { diff --git a/e2e/playwright/auth.spec.ts b/e2e/playwright/auth.spec.ts index eae4051dd..f9c290964 100644 --- a/e2e/playwright/auth.spec.ts +++ b/e2e/playwright/auth.spec.ts @@ -13,12 +13,22 @@ test.describe('Authentication tests', () => { await page.setBodyDimensions({ width: 1000, height: 500 }) await homePage.projectSection.waitFor() + // This is only needed as an override to test-utils' setup() for this test + await page.addInitScript(() => { + localStorage.setItem('TOKEN_PERSIST_KEY', '') + }) + await test.step('Click on sign out and expect sign in page', async () => { await toolbar.userSidebarButton.click() await toolbar.signOutButton.click() await expect(signInPage.signInButton).toBeVisible() }) + await test.step("Refresh doesn't log the user back in", async () => { + await page.reload() + await expect(signInPage.signInButton).toBeVisible() + }) + await test.step('Click on sign in and cancel, click again and expect different code', async () => { await signInPage.signInButton.click() await expect(signInPage.userCode).toBeVisible() @@ -30,6 +40,7 @@ test.describe('Authentication tests', () => { await expect(signInPage.userCode).toBeVisible() const secondUserCode = await signInPage.userCode.textContent() expect(secondUserCode).not.toEqual(firstUserCode) + await signInPage.cancelSignInButton.click() }) await test.step('Press back button and remain on home page', async () => { @@ -48,6 +59,12 @@ test.describe('Authentication tests', () => { // Longer timeout than usual here for the wait on home page await expect(homePage.projectSection).toBeVisible({ timeout: 10000 }) }) + + await test.step('Click on sign out and expect sign in page', async () => { + await toolbar.userSidebarButton.click() + await toolbar.signOutButton.click() + await expect(signInPage.signInButton).toBeVisible() + }) } ) }) diff --git a/e2e/playwright/basic-sketch.spec.ts b/e2e/playwright/basic-sketch.spec.ts index d3a0da867..22f638bcb 100644 --- a/e2e/playwright/basic-sketch.spec.ts +++ b/e2e/playwright/basic-sketch.spec.ts @@ -155,7 +155,7 @@ async function doBasicSketch( |> xLine(length = -segLen(seg01))`) } -test.describe('Basic sketch', { tag: ['@skipWin'] }, () => { +test.describe('Basic sketch', () => { test('code pane open at start', async ({ page, homePage }) => { test.fixme(orRunWhenFullSuiteEnabled()) await doBasicSketch(page, homePage, ['code']) diff --git a/e2e/playwright/can-create-sketches-on-all-planes-and-their-back-sides.spec.ts b/e2e/playwright/can-create-sketches-on-all-planes-and-their-back-sides.spec.ts index 9e2fee320..bda3c3b71 100644 --- a/e2e/playwright/can-create-sketches-on-all-planes-and-their-back-sides.spec.ts +++ b/e2e/playwright/can-create-sketches-on-all-planes-and-their-back-sides.spec.ts @@ -8,130 +8,126 @@ import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' import { getUtils } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe( - 'Can create sketches on all planes and their back sides', - { tag: ['@skipWin'] }, - () => { - const sketchOnPlaneAndBackSideTest = async ( - page: Page, - homePage: HomePageFixture, - scene: SceneFixture, - toolbar: ToolbarFixture, - plane: string, - clickCoords: { x: number; y: number } - ) => { - const u = await getUtils(page) - await page.setBodyDimensions({ width: 1200, height: 500 }) +test.describe('Can create sketches on all planes and their back sides', () => { + const sketchOnPlaneAndBackSideTest = async ( + page: Page, + homePage: HomePageFixture, + scene: SceneFixture, + toolbar: ToolbarFixture, + plane: string, + clickCoords: { x: number; y: number } + ) => { + const u = await getUtils(page) + await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() - const XYPlanRed: [number, number, number] = [98, 50, 51] - await scene.expectPixelColor(XYPlanRed, { x: 700, y: 300 }, 15) + await homePage.goToModelingScene() + const XYPlanRed: [number, number, number] = [98, 50, 51] + await scene.expectPixelColor(XYPlanRed, { x: 700, y: 300 }, 15) - await u.openDebugPanel() + await u.openDebugPanel() - const coord = - plane === '-XY' || plane === '-YZ' || plane === 'XZ' ? -100 : 100 - const camCommand: EngineCommand = { - type: 'modeling_cmd_req', - cmd_id: uuidv4(), - cmd: { - type: 'default_camera_look_at', - center: { x: 0, y: 0, z: 0 }, - vantage: { x: coord, y: coord, z: coord }, - up: { x: 0, y: 0, z: 1 }, - }, - } - const updateCamCommand: EngineCommand = { - type: 'modeling_cmd_req', - cmd_id: uuidv4(), - cmd: { - type: 'default_camera_get_settings', - }, - } - - const code = `@settings(defaultLengthUnit = in)sketch001 = startSketchOn(${plane})profile001 = startProfileAt([0.91, -1.22], sketch001)` - - await u.openDebugPanel() - - await u.clearCommandLogs() - await page.getByRole('button', { name: 'Start Sketch' }).click() - - await u.sendCustomCmd(camCommand) - await page.waitForTimeout(100) - await u.sendCustomCmd(updateCamCommand) - - await u.closeDebugPanel() - - await page.mouse.click(clickCoords.x, clickCoords.y) - await page.waitForTimeout(600) // wait for animation - - await toolbar.waitUntilSketchingReady() - - await expect( - page.getByRole('button', { name: 'line Line', exact: true }) - ).toBeVisible() - - await u.closeDebugPanel() - await page.mouse.click(707, 393) - - await expect(page.locator('.cm-content')).toHaveText(code) - - await page - .getByRole('button', { name: 'line Line', exact: true }) - .first() - .click() - await u.openAndClearDebugPanel() - await page.getByRole('button', { name: 'Exit Sketch' }).click() - await u.expectCmdLog('[data-message-type="execution-done"]') - - await u.clearCommandLogs() - await u.removeCurrentCode() + const coord = + plane === '-XY' || plane === '-YZ' || plane === 'XZ' ? -100 : 100 + const camCommand: EngineCommand = { + type: 'modeling_cmd_req', + cmd_id: uuidv4(), + cmd: { + type: 'default_camera_look_at', + center: { x: 0, y: 0, z: 0 }, + vantage: { x: coord, y: coord, z: coord }, + up: { x: 0, y: 0, z: 1 }, + }, + } + const updateCamCommand: EngineCommand = { + type: 'modeling_cmd_req', + cmd_id: uuidv4(), + cmd: { + type: 'default_camera_get_settings', + }, } - const planeConfigs = [ - { - plane: 'XY', - coords: { x: 600, y: 388 }, - description: 'red plane', - }, - { - plane: 'YZ', - coords: { x: 700, y: 250 }, - description: 'green plane', - }, - { - plane: 'XZ', - coords: { x: 684, y: 427 }, - description: 'blue plane', - }, - { - plane: '-XY', - coords: { x: 600, y: 118 }, - description: 'back of red plane', - }, - { - plane: '-YZ', - coords: { x: 700, y: 219 }, - description: 'back of green plane', - }, - { - plane: '-XZ', - coords: { x: 700, y: 80 }, - description: 'back of blue plane', - }, - ] + const code = `@settings(defaultLengthUnit = in)sketch001 = startSketchOn(${plane})profile001 = startProfileAt([0.91, -1.22], sketch001)` - for (const config of planeConfigs) { - test(config.plane, async ({ page, homePage, scene, toolbar }) => { - await sketchOnPlaneAndBackSideTest( - page, - homePage, - scene, - toolbar, - config.plane, - config.coords - ) - }) - } + await u.openDebugPanel() + + await u.clearCommandLogs() + await page.getByRole('button', { name: 'Start Sketch' }).click() + + await u.sendCustomCmd(camCommand) + await page.waitForTimeout(100) + await u.sendCustomCmd(updateCamCommand) + + await u.closeDebugPanel() + + await page.mouse.click(clickCoords.x, clickCoords.y) + await page.waitForTimeout(600) // wait for animation + + await toolbar.waitUntilSketchingReady() + + await expect( + page.getByRole('button', { name: 'line Line', exact: true }) + ).toBeVisible() + + await u.closeDebugPanel() + await page.mouse.click(707, 393) + + await expect(page.locator('.cm-content')).toHaveText(code) + + await page + .getByRole('button', { name: 'line Line', exact: true }) + .first() + .click() + await u.openAndClearDebugPanel() + await page.getByRole('button', { name: 'Exit Sketch' }).click() + await u.expectCmdLog('[data-message-type="execution-done"]') + + await u.clearCommandLogs() + await u.removeCurrentCode() } -) + + const planeConfigs = [ + { + plane: 'XY', + coords: { x: 600, y: 388 }, + description: 'red plane', + }, + { + plane: 'YZ', + coords: { x: 700, y: 250 }, + description: 'green plane', + }, + { + plane: 'XZ', + coords: { x: 684, y: 427 }, + description: 'blue plane', + }, + { + plane: '-XY', + coords: { x: 600, y: 118 }, + description: 'back of red plane', + }, + { + plane: '-YZ', + coords: { x: 700, y: 219 }, + description: 'back of green plane', + }, + { + plane: '-XZ', + coords: { x: 700, y: 80 }, + description: 'back of blue plane', + }, + ] + + for (const config of planeConfigs) { + test(config.plane, async ({ page, homePage, scene, toolbar }) => { + await sketchOnPlaneAndBackSideTest( + page, + homePage, + scene, + toolbar, + config.plane, + config.coords + ) + }) + } +}) diff --git a/e2e/playwright/code-pane-and-errors.spec.ts b/e2e/playwright/code-pane-and-errors.spec.ts index 2ef0ff990..fc8115705 100644 --- a/e2e/playwright/code-pane-and-errors.spec.ts +++ b/e2e/playwright/code-pane-and-errors.spec.ts @@ -10,7 +10,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Code pane and errors', { tag: ['@skipWin'] }, () => { +test.describe('Code pane and errors', () => { test('Typing KCL errors induces a badge on the code pane button', async ({ page, homePage, diff --git a/e2e/playwright/command-bar-tests.spec.ts b/e2e/playwright/command-bar-tests.spec.ts index ecb1043d4..2b660467a 100644 --- a/e2e/playwright/command-bar-tests.spec.ts +++ b/e2e/playwright/command-bar-tests.spec.ts @@ -9,7 +9,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Command bar tests', { tag: ['@skipWin'] }, () => { +test.describe('Command bar tests', () => { test('Extrude from command bar selects extrude line after', async ({ page, homePage, @@ -179,57 +179,57 @@ test.describe('Command bar tests', { tag: ['@skipWin'] }, () => { await expect(commandLevelArgButton).toHaveText('level: project') }) - test( - 'Command bar keybinding works from code editor and can change a setting', - { tag: ['@skipWin'] }, - async ({ page, homePage }) => { - await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() + test('Command bar keybinding works from code editor and can change a setting', async ({ + page, + homePage, + }) => { + await page.setBodyDimensions({ width: 1200, height: 500 }) + await homePage.goToModelingScene() - // FIXME: No KCL code, unable to wait for engine execution - await page.waitForTimeout(10000) + // FIXME: No KCL code, unable to wait for engine execution + await page.waitForTimeout(10000) - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - // Put the cursor in the code editor - await page.locator('.cm-content').click() + // Put the cursor in the code editor + await page.locator('.cm-content').click() - // Now try the same, but with the keyboard shortcut, check focus - await page.keyboard.press('ControlOrMeta+K') + // Now try the same, but with the keyboard shortcut, check focus + await page.keyboard.press('ControlOrMeta+K') - let cmdSearchBar = page.getByPlaceholder('Search commands') - await expect(cmdSearchBar).toBeVisible() - await expect(cmdSearchBar).toBeFocused() + let cmdSearchBar = page.getByPlaceholder('Search commands') + await expect(cmdSearchBar).toBeVisible() + await expect(cmdSearchBar).toBeFocused() - // Try typing in the command bar - await cmdSearchBar.fill('theme') - const themeOption = page.getByRole('option', { - name: 'Settings · app · theme', - }) - await expect(themeOption).toBeVisible() - await themeOption.click() - const themeInput = page.getByPlaceholder('dark') - await expect(themeInput).toBeVisible() - await expect(themeInput).toBeFocused() - // Select dark theme - await page.keyboard.press('ArrowDown') - await page.keyboard.press('ArrowDown') - await page.keyboard.press('ArrowDown') - await expect( - page.getByRole('option', { name: 'system' }) - ).toHaveAttribute('data-headlessui-state', 'active') - await page.keyboard.press('Enter') + // Try typing in the command bar + await cmdSearchBar.fill('theme') + const themeOption = page.getByRole('option', { + name: 'Settings · app · theme', + }) + await expect(themeOption).toBeVisible() + await themeOption.click() + const themeInput = page.getByPlaceholder('dark') + await expect(themeInput).toBeVisible() + await expect(themeInput).toBeFocused() + // Select dark theme + await page.keyboard.press('ArrowDown') + await page.keyboard.press('ArrowDown') + await page.keyboard.press('ArrowDown') + await expect(page.getByRole('option', { name: 'system' })).toHaveAttribute( + 'data-headlessui-state', + 'active' + ) + await page.keyboard.press('Enter') - // Check the toast appeared - await expect( - page.getByText(`Set theme to "system" as a user default`) - ).toBeVisible() - // Check that the theme changed - await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) - } - ) + // Check the toast appeared + await expect( + page.getByText(`Set theme to "system" as a user default`) + ).toBeVisible() + // Check that the theme changed + await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) + }) test('Can extrude from the command bar', async ({ page, diff --git a/e2e/playwright/desktop-export.spec.ts b/e2e/playwright/desktop-export.spec.ts index 36a7c2194..aa5e27ec7 100644 --- a/e2e/playwright/desktop-export.spec.ts +++ b/e2e/playwright/desktop-export.spec.ts @@ -10,7 +10,7 @@ import { expect, test } from '@e2e/playwright/zoo-test' test( 'export works on the first try', - { tag: ['@electron', '@skipLocalEngine'] }, + { tag: ['@electron', '@macos', '@windows', '@skipLocalEngine'] }, async ({ page, context, scene, tronApp, cmdBar }, testInfo) => { if (!tronApp) { fail() diff --git a/e2e/playwright/editor-tests.spec.ts b/e2e/playwright/editor-tests.spec.ts index 5ba90f185..653e18b99 100644 --- a/e2e/playwright/editor-tests.spec.ts +++ b/e2e/playwright/editor-tests.spec.ts @@ -10,7 +10,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Editor tests', { tag: ['@skipWin'] }, () => { +test.describe('Editor tests', () => { test('can comment out code with ctrl+/', async ({ page, homePage }) => { const u = await getUtils(page) await page.setBodyDimensions({ width: 1000, height: 500 }) @@ -989,162 +989,162 @@ sketch001 = startSketchOn(XZ) |> close()`) }) - test( - 'Can undo a sketch modification with ctrl+z', - { tag: ['@skipWin'] }, - async ({ page, homePage, editor }) => { - const u = await getUtils(page) - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit=in) + test('Can undo a sketch modification with ctrl+z', async ({ + page, + homePage, + editor, + }) => { + const u = await getUtils(page) + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit=in) sketch001 = startSketchOn(XZ) |> startProfileAt([4.61, -10.01], %) |> line(end = [12.73, -0.09]) |> tangentialArc(endAbsolute = [24.95, -0.38]) |> close() |> extrude(length = 5)` - ) - }) + ) + }) - await page.setBodyDimensions({ width: 1200, height: 500 }) + await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await homePage.goToModelingScene() + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - await page.waitForTimeout(100) - await u.openAndClearDebugPanel() - await u.sendCustomCmd({ - type: 'modeling_cmd_req', - cmd_id: uuidv4(), - cmd: { - type: 'default_camera_look_at', - vantage: { x: 0, y: -1250, z: 580 }, - center: { x: 0, y: 0, z: 0 }, - up: { x: 0, y: 0, z: 1 }, - }, - }) - await page.waitForTimeout(100) - await u.sendCustomCmd({ - type: 'modeling_cmd_req', - cmd_id: uuidv4(), - cmd: { - type: 'default_camera_get_settings', - }, - }) - await page.waitForTimeout(100) + await page.waitForTimeout(100) + await u.openAndClearDebugPanel() + await u.sendCustomCmd({ + type: 'modeling_cmd_req', + cmd_id: uuidv4(), + cmd: { + type: 'default_camera_look_at', + vantage: { x: 0, y: -1250, z: 580 }, + center: { x: 0, y: 0, z: 0 }, + up: { x: 0, y: 0, z: 1 }, + }, + }) + await page.waitForTimeout(100) + await u.sendCustomCmd({ + type: 'modeling_cmd_req', + cmd_id: uuidv4(), + cmd: { + type: 'default_camera_get_settings', + }, + }) + await page.waitForTimeout(100) - const startPX = [1200 / 2, 500 / 2] + const startPX = [1200 / 2, 500 / 2] - const dragPX = 40 + const dragPX = 40 - await page.getByText('startProfileAt([4.61, -10.01], %)').click() - await expect( - page.getByRole('button', { name: 'Edit Sketch' }) - ).toBeVisible() - await page.getByRole('button', { name: 'Edit Sketch' }).click() - await page.waitForTimeout(400) - let prevContent = await page.locator('.cm-content').innerText() + await page.getByText('startProfileAt([4.61, -10.01], %)').click() + await expect( + page.getByRole('button', { name: 'Edit Sketch' }) + ).toBeVisible() + await page.getByRole('button', { name: 'Edit Sketch' }).click() + await page.waitForTimeout(400) + let prevContent = await page.locator('.cm-content').innerText() - await expect(page.getByTestId('segment-overlay')).toHaveCount(2) + await expect(page.getByTestId('segment-overlay')).toHaveCount(2) - // drag startProfileAt handle - await page.dragAndDrop('#stream', '#stream', { - sourcePosition: { x: startPX[0] + 68, y: startPX[1] + 147 }, - targetPosition: { x: startPX[0] + dragPX, y: startPX[1] + dragPX }, - }) - await page.waitForTimeout(100) - await expect(page.locator('.cm-content')).not.toHaveText(prevContent) - prevContent = await page.locator('.cm-content').innerText() + // drag startProfileAt handle + await page.dragAndDrop('#stream', '#stream', { + sourcePosition: { x: startPX[0] + 68, y: startPX[1] + 147 }, + targetPosition: { x: startPX[0] + dragPX, y: startPX[1] + dragPX }, + }) + await page.waitForTimeout(100) + await expect(page.locator('.cm-content')).not.toHaveText(prevContent) + prevContent = await page.locator('.cm-content').innerText() - // drag line handle - // we wait so it saves the code - await page.waitForTimeout(800) + // drag line handle + // we wait so it saves the code + await page.waitForTimeout(800) - const lineEnd = await u.getBoundingBox('[data-overlay-index="0"]') - await page.waitForTimeout(100) - await page.dragAndDrop('#stream', '#stream', { - sourcePosition: { x: lineEnd.x - 5, y: lineEnd.y }, - targetPosition: { x: lineEnd.x + dragPX, y: lineEnd.y + dragPX }, - }) - await expect(page.locator('.cm-content')).not.toHaveText(prevContent) - prevContent = await page.locator('.cm-content').innerText() + const lineEnd = await u.getBoundingBox('[data-overlay-index="0"]') + await page.waitForTimeout(100) + await page.dragAndDrop('#stream', '#stream', { + sourcePosition: { x: lineEnd.x - 5, y: lineEnd.y }, + targetPosition: { x: lineEnd.x + dragPX, y: lineEnd.y + dragPX }, + }) + await expect(page.locator('.cm-content')).not.toHaveText(prevContent) + prevContent = await page.locator('.cm-content').innerText() - // we wait so it saves the code - await page.waitForTimeout(800) + // we wait so it saves the code + await page.waitForTimeout(800) - // drag tangentialArc handle - const tangentEnd = await u.getBoundingBox('[data-overlay-index="1"]') - await page.dragAndDrop('#stream', '#stream', { - sourcePosition: { x: tangentEnd.x + 10, y: tangentEnd.y - 5 }, - targetPosition: { - x: tangentEnd.x + dragPX, - y: tangentEnd.y + dragPX, - }, - }) - await page.waitForTimeout(100) - await expect(page.locator('.cm-content')).not.toHaveText(prevContent) + // drag tangentialArc handle + const tangentEnd = await u.getBoundingBox('[data-overlay-index="1"]') + await page.dragAndDrop('#stream', '#stream', { + sourcePosition: { x: tangentEnd.x + 10, y: tangentEnd.y - 5 }, + targetPosition: { + x: tangentEnd.x + dragPX, + y: tangentEnd.y + dragPX, + }, + }) + await page.waitForTimeout(100) + await expect(page.locator('.cm-content')).not.toHaveText(prevContent) - // expect the code to have changed - await editor.expectEditor.toContain( - `sketch001 = startSketchOn(XZ) + // expect the code to have changed + await editor.expectEditor.toContain( + `sketch001 = startSketchOn(XZ) |> startProfileAt([2.71, -2.71], %) |> line(end = [15.4, -2.78]) |> tangentialArc(endAbsolute = [27.6, -3.05]) |> close() |> extrude(length = 5)`, - { shouldNormalise: true } - ) + { shouldNormalise: true } + ) - // Hit undo - await page.keyboard.down('Control') - await page.keyboard.press('KeyZ') - await page.keyboard.up('Control') + // Hit undo + await page.keyboard.down('Control') + await page.keyboard.press('KeyZ') + await page.keyboard.up('Control') - await editor.expectEditor.toContain( - `sketch001 = startSketchOn(XZ) + await editor.expectEditor.toContain( + `sketch001 = startSketchOn(XZ) |> startProfileAt([2.71, -2.71], %) |> line(end = [15.4, -2.78]) |> tangentialArc(endAbsolute = [24.95, -0.38]) |> close() |> extrude(length = 5)`, - { shouldNormalise: true } - ) + { shouldNormalise: true } + ) - // Hit undo again. - await page.keyboard.down('Control') - await page.keyboard.press('KeyZ') - await page.keyboard.up('Control') + // Hit undo again. + await page.keyboard.down('Control') + await page.keyboard.press('KeyZ') + await page.keyboard.up('Control') - await editor.expectEditor.toContain( - `sketch001 = startSketchOn(XZ) + await editor.expectEditor.toContain( + `sketch001 = startSketchOn(XZ) |> startProfileAt([2.71, -2.71], %) |> line(end = [12.73, -0.09]) |> tangentialArc(endAbsolute = [24.95, -0.38]) |> close() |> extrude(length = 5)`, - { shouldNormalise: true } - ) + { shouldNormalise: true } + ) - // Hit undo again. - await page.keyboard.down('Control') - await page.keyboard.press('KeyZ') - await page.keyboard.up('Control') + // Hit undo again. + await page.keyboard.down('Control') + await page.keyboard.press('KeyZ') + await page.keyboard.up('Control') - await page.waitForTimeout(100) - await editor.expectEditor.toContain( - `sketch001 = startSketchOn(XZ) + await page.waitForTimeout(100) + await editor.expectEditor.toContain( + `sketch001 = startSketchOn(XZ) |> startProfileAt([4.61, -10.01], %) |> line(end = [12.73, -0.09]) |> tangentialArc(endAbsolute = [24.95, -0.38]) |> close() |> extrude(length = 5)`, - { shouldNormalise: true } - ) - } - ) + { shouldNormalise: true } + ) + }) test( `Can import a local OBJ file`, diff --git a/e2e/playwright/file-tree.spec.ts b/e2e/playwright/file-tree.spec.ts index 75a8e284f..f9c7a75e1 100644 --- a/e2e/playwright/file-tree.spec.ts +++ b/e2e/playwright/file-tree.spec.ts @@ -29,7 +29,7 @@ test.describe('integrations tests', () => { ) }) - const [clickObj] = await scene.makeMouseHelpers(726, 272) + const [clickObj] = scene.makeMouseHelpers(726, 272) await test.step('setup test', async () => { await homePage.expectState({ @@ -73,7 +73,7 @@ test.describe('integrations tests', () => { }) await test.step('setup for next assertion', async () => { await toolbar.openFile('main.kcl') - await page.waitForTimeout(1000) + await page.waitForTimeout(2000) await clickObj() await page.waitForTimeout(1000) await scene.moveNoWhere() diff --git a/e2e/playwright/fixtures/toolbarFixture.ts b/e2e/playwright/fixtures/toolbarFixture.ts index 2542bade9..37bee58c7 100644 --- a/e2e/playwright/fixtures/toolbarFixture.ts +++ b/e2e/playwright/fixtures/toolbarFixture.ts @@ -174,6 +174,13 @@ export class ToolbarFixture { openFile = async (fileName: string) => { await this.filePane.getByText(fileName).click() } + selectTangentialArc = async () => { + await this.page.getByRole('button', { name: 'caret down arcs:' }).click() + await expect( + this.page.getByTestId('dropdown-three-point-arc') + ).toBeVisible() + await this.page.getByTestId('dropdown-tangential-arc').click() + } selectCenterRectangle = async () => { await this.page .getByRole('button', { name: 'caret down rectangles:' }) diff --git a/e2e/playwright/lib/api-reporter.ts b/e2e/playwright/lib/api-reporter.ts index 395761930..f43738e69 100644 --- a/e2e/playwright/lib/api-reporter.ts +++ b/e2e/playwright/lib/api-reporter.ts @@ -41,7 +41,7 @@ class MyAPIReporter implements Reporter { annotations: test.annotations.map((a) => a.type), // e.g. 'fail' or 'fixme' id: test.id, // computed file/test/project ID used for reruns retry: result.retry, - tags: test.tags, // e.g. '@snapshot' or '@skipWin' + tags: test.tags, // e.g. '@snapshot' or '@skipLocalEngine' // Extra environment variables CI_COMMIT_SHA: process.env.CI_COMMIT_SHA || null, CI_PR_NUMBER: process.env.CI_PR_NUMBER || null, diff --git a/e2e/playwright/native-file-menu.spec.ts b/e2e/playwright/native-file-menu.spec.ts index 72f5606cf..c1794e67a 100644 --- a/e2e/playwright/native-file-menu.spec.ts +++ b/e2e/playwright/native-file-menu.spec.ts @@ -6,2352 +6,2372 @@ import { expect, test } from '@e2e/playwright/zoo-test' * Not all menu actions are tested. Some are default electron menu actions. * Test file menu actions that trigger something in the frontend */ -test.describe('Native file menu', { tag: ['@electron'] }, () => { - test.skip() // TODO: Reimplement native file menu tests - test.describe('Home page', () => { - test.describe('File role', () => { - test('Home.File.Create project', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - return false - } - const newProject = app.applicationMenu.getMenuItemById( - 'File.Create project' - ) - if (!newProject) return false - newProject.click() - return true - }) +test.describe( + 'Native file menu', + { tag: ['@electron', '@macos', '@windows'] }, + () => { + test.skip() // TODO: Reimplement native file menu tests + test.describe('Home page', () => { + test.describe('File role', () => { + test('Home.File.Create project', async ({ tronApp, cmdBar, page }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + return false + } + const newProject = app.applicationMenu.getMenuItemById( + 'File.Create project' + ) + if (!newProject) return false + newProject.click() + return true + }) + ) + .toBe(true) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actualArgument = await cmdBar.cmdBarElement + .getByTestId('cmd-bar-arg-value') + .inputValue() + const expectedArgument = 'untitled' + expect(actualArgument).toBe(expectedArgument) + }) + test('Home.File.Open project', async ({ tronApp, cmdBar, page }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const openProject = + app.applicationMenu.getMenuItemById('File.Open project') + if (!openProject) { + return false + } + openProject.click() + return true + }) + ) + .toBe(true) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Open project' + expect(actual).toBe(expected) + }) + test('Home.File.Preferences.User settings', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + console.log(app) + if (!app || !app.applicationMenu) { + return false + } + const userSettings = app.applicationMenu.getMenuItemById( + 'File.Preferences.User settings' + ) + if (!userSettings) return false + userSettings.click() + return true + }) + ) + .toBe(true) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + // You are viewing the user tab + const actualText = settings.getByText( + 'The overall appearance of the app' ) - .toBe(true) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actualArgument = await cmdBar.cmdBarElement - .getByTestId('cmd-bar-arg-value') - .inputValue() - const expectedArgument = 'untitled' - expect(actualArgument).toBe(expectedArgument) + await expect(actualText).toBeVisible() + }) + test('Home.File.Preferences.Keybindings', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) { + fail() + } + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const keybindings = app.applicationMenu.getMenuItemById( + 'File.Preferences.Keybindings' + ) + if (!keybindings) { + return false + } + keybindings.click() + return true + }) + ) + .toBe(true) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + // You are viewing the keybindings tab + const enterSketchMode = settings.locator('#enter-sketch-mode') + await expect(enterSketchMode).toBeVisible() + }) + test('Home.File.Preferences.User default units', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + return false + } + const menu = app.applicationMenu.getMenuItemById( + 'File.Preferences.User default units' + ) + if (!menu) return false + menu.click() + return true + }) + ) + .toBe(true) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + const defaultUnit = settings.locator('#defaultUnit') + await expect(defaultUnit).toBeVisible() + }) + test('Home.File.Preferences.Theme', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const menu = app.applicationMenu.getMenuItemById( + 'File.Preferences.Theme' + ) + if (!menu) { + return false + } + menu.click() + return true + }) + ) + .toBe(true) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Settings · app · theme' + expect(actual).toBe(expected) + }) + test('Home.File.Preferences.Theme color', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + return false + } + const menu = app.applicationMenu.getMenuItemById( + 'File.Preferences.Theme color' + ) + if (!menu) return false + menu.click() + return true + }) + ) + .toBe(true) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + const defaultUnit = settings.locator('#themeColor') + await expect(defaultUnit).toBeVisible() + }) + test('Home.File.Preferences.Sign out', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const menu = + app.applicationMenu.getMenuItemById('File.Sign out') + if (!menu) { + return false + } + // FIXME: Add back when you can actually sign out + // menu.click() + return true + }) + ) + .toBe(true) + // FIXME: When signing out during E2E the page is not bound correctly. + // It cannot find the button + // const signIn = page.getByTestId('sign-in-button') + // await expect(signIn).toBeVisible() + }) }) - test('Home.File.Open project', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const openProject = - app.applicationMenu.getMenuItemById('File.Open project') - if (!openProject) { - return false - } - openProject.click() - return true - }) - ) - .toBe(true) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Open project' - expect(actual).toBe(expected) + + test.describe('Edit role', () => { + test('Home.Edit.Rename project', async ({ tronApp, cmdBar, page }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + return false + } + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Rename project' + ) + if (!menu) return false + menu.click() + return true + }) + ) + .toBe(true) + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Rename project' + expect(actual).toBe(expected) + }) + test('Home.Edit.Delete project', async ({ tronApp, cmdBar, page }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Delete project' + ) + if (!menu) { + return false + } + menu.click() + return true + }) + ) + .toBe(true) + // Check the placeholder project name exists + const actual = async () => + cmdBar.cmdBarElement.getByTestId('command-name').textContent() + const expected = 'Delete project' + await expect.poll(async () => await actual()).toBe(expected) + }) + test('Home.Edit.Change project directory', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + return false + } + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Change project directory' + ) + if (!menu) return false + menu.click() + return true + }) + ) + .toBe(true) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + const projectDirectory = settings.locator('#projectDirectory') + await expect(projectDirectory).toBeVisible() + }) }) - test('Home.File.Preferences.User settings', async ({ - tronApp, - cmdBar, - page, - }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - console.log(app) - if (!app || !app.applicationMenu) { - return false - } - const userSettings = app.applicationMenu.getMenuItemById( - 'File.Preferences.User settings' - ) - if (!userSettings) return false - userSettings.click() - return true - }) - ) - .toBe(true) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - // You are viewing the user tab - const actualText = settings.getByText( - 'The overall appearance of the app' - ) - await expect(actualText).toBeVisible() + test.describe('View role', () => { + test('Home.View.Command Palette...', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const menu = app.applicationMenu.getMenuItemById( + 'View.Command Palette...' + ) + if (!menu) { + return false + } + menu.click() + return true + }) + ) + .toBe(true) + // Check the placeholder project name exists + const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') + await expect(actual).toBeVisible() + }) }) - test('Home.File.Preferences.Keybindings', async ({ - tronApp, - cmdBar, - page, - }) => { - if (!tronApp) { - fail() - } - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const keybindings = app.applicationMenu.getMenuItemById( - 'File.Preferences.Keybindings' - ) - if (!keybindings) { - return false - } - keybindings.click() - return true - }) + test.describe('Help role', () => { + test('Home.Help.Show all commands', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + return false + } + const menu = app.applicationMenu.getMenuItemById( + 'Help.Show all commands' + ) + if (!menu) return false + menu.click() + return true + }) + ) + .toBe(true) + // Check the placeholder project name exists + const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') + await expect(actual).toBeVisible() + }) + test('Home.Help.KCL code samples', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const menu = app.applicationMenu.getMenuItemById( + 'Help.KCL code samples' + ) + if (!menu) { + return false + } + return true + }) + ) + .toBe(true) + }) + test('Home.Help.Report a bug', async ({ tronApp, cmdBar, page }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + return false + } + const menu = + app.applicationMenu.getMenuItemById('Help.Report a bug') + if (!menu) return false + menu.click() + return true + }) + ) + .toBe(true) + // Core dump and refresh magic number timeout + await page.waitForTimeout(7000) + const actual = page.getByText( + 'No Projects found, ready to make your first one?' ) - .toBe(true) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - // You are viewing the keybindings tab - const enterSketchMode = settings.locator('#enter-sketch-mode') - await expect(enterSketchMode).toBeVisible() - }) - test('Home.File.Preferences.User default units', async ({ - tronApp, - cmdBar, - page, - }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - return false - } - const menu = app.applicationMenu.getMenuItemById( - 'File.Preferences.User default units' - ) - if (!menu) return false - menu.click() - return true - }) + await expect(actual).toBeVisible() + }) + test('Home.Help.Reset onboarding', async ({ + tronApp, + cmdBar, + page, + }) => { + if (!tronApp) fail() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const menu = app.applicationMenu.getMenuItemById( + 'Help.Reset onboarding' + ) + if (!menu) { + return false + } + menu.click() + return true + }) + ) + .toBe(true) + + const actual = page.getByText( + `This is a hardware design tool that lets you edit visually, with code, or both. It's powered by the KittyCAD Design API, the first API created for anyone to build hardware design tools.` ) - .toBe(true) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - const defaultUnit = settings.locator('#defaultUnit') - await expect(defaultUnit).toBeVisible() - }) - test('Home.File.Preferences.Theme', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const menu = app.applicationMenu.getMenuItemById( - 'File.Preferences.Theme' - ) - if (!menu) { - return false - } - menu.click() - return true - }) - ) - .toBe(true) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Settings · app · theme' - expect(actual).toBe(expected) - }) - test('Home.File.Preferences.Theme color', async ({ - tronApp, - cmdBar, - page, - }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - return false - } - const menu = app.applicationMenu.getMenuItemById( - 'File.Preferences.Theme color' - ) - if (!menu) return false - menu.click() - return true - }) - ) - .toBe(true) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - const defaultUnit = settings.locator('#themeColor') - await expect(defaultUnit).toBeVisible() - }) - test('Home.File.Preferences.Sign out', async ({ - tronApp, - cmdBar, - page, - }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const menu = - app.applicationMenu.getMenuItemById('File.Sign out') - if (!menu) { - return false - } - // FIXME: Add back when you can actually sign out - // menu.click() - return true - }) - ) - .toBe(true) - // FIXME: When signing out during E2E the page is not bound correctly. - // It cannot find the button - // const signIn = page.getByTestId('sign-in-button') - // await expect(signIn).toBeVisible() + await expect(actual).toBeVisible() + }) }) }) + test.describe('Modeling page', () => { + test.describe('File Role', () => { + test('Modeling.File.Create project', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) fail() + await homePage.goToModelingScene() + await scene.connectionEstablished() - test.describe('Edit role', () => { - test('Home.Edit.Rename project', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - return false - } - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Rename project' - ) - if (!menu) return false - menu.click() - return true - }) + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) fail() + const newProject = app.applicationMenu.getMenuItemById( + 'File.Create project' + ) + if (!newProject) fail() + newProject.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actualArgument = await cmdBar.cmdBarElement + .getByTestId('cmd-bar-arg-value') + .inputValue() + const expectedArgument = 'untitled' + expect(actualArgument).toBe(expectedArgument) + }) + test('Modeling.File.Open project', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const openProject = + app.applicationMenu.getMenuItemById('File.Open project') + if (!openProject) { + throw new Error('File.Open project') + } + openProject.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Open project' + expect(actual).toBe(expected) + }) + test('Modeling.File.Load external model', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const openProject = app.applicationMenu.getMenuItemById( + 'File.Load external model' + ) + if (!openProject) { + throw new Error('File.Load external model') + } + openProject.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Load external model' + expect(actual).toBe(expected) + }) + test('Modeling.File.Export current part', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const openProject = app.applicationMenu.getMenuItemById( + 'File.Export current part' + ) + if (!openProject) { + throw new Error('File.Export current part') + } + openProject.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Export' + expect(actual).toBe(expected) + }) + test('Modeling.File.Share part via Zoo link', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const openProject = app.applicationMenu.getMenuItemById( + 'File.Share part via Zoo link' + ) + if (!openProject) { + throw new Error('File.Share part via Zoo link') + } + openProject.click() + }) + + const textToCheck = + 'Link copied to clipboard. Anyone who clicks this link will get a copy of this file. Share carefully!' + // Check if text appears anywhere in the page + const isTextVisible = page.getByText(textToCheck) + + await expect(isTextVisible).toBeVisible({ timeout: 10000 }) + }) + test('Modeling.File.Preferences.Project settings', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const openProject = app.applicationMenu.getMenuItemById( + 'File.Preferences.Project settings' + ) + if (!openProject) { + throw new Error('File.Preferences.Project settings') + } + openProject.click() + }) + + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + // You are viewing the user tab + const actualText = settings.getByText( + 'The hue of the primary theme color for the app' ) - .toBe(true) - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Rename project' - expect(actual).toBe(expected) + await expect(actualText).toBeVisible() + }) + test('Modeling.File.Preferences.User settings', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const userSettings = app.applicationMenu.getMenuItemById( + 'File.Preferences.User settings' + ) + if (!userSettings) { + throw new Error('File.Preferences.User settings') + } + userSettings.click() + }) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + // You are viewing the user tab + const actualText = settings.getByText( + 'The overall appearance of the app' + ) + await expect(actualText).toBeVisible() + }) + test('Modeling.File.Preferences.Keybindings', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const keybindings = app.applicationMenu.getMenuItemById( + 'File.Preferences.Keybindings' + ) + if (!keybindings) { + throw new Error('File.Preferences.Keybindings') + } + keybindings.click() + }) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + // You are viewing the keybindings tab + const enterSketchMode = settings.locator('#enter-sketch-mode') + await expect(enterSketchMode).toBeVisible() + }) + test('Modeling.File.Preferences.User default units', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'File.Preferences.User default units' + ) + if (!menu) { + throw new Error('File.Preferences.User default units') + } + menu.click() + }) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + const defaultUnit = settings.locator('#defaultUnit') + await expect(defaultUnit).toBeVisible() + }) + test('Modeling.File.Preferences.Theme', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'File.Preferences.Theme' + ) + if (!menu) { + throw new Error('File.Preferences.Theme') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Settings · app · theme' + expect(actual).toBe(expected) + }) + test('Modeling.File.Preferences.Theme color', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'File.Preferences.Theme color' + ) + if (!menu) { + throw new Error('File.Preferences.Theme color') + } + menu.click() + }) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + const defaultUnit = settings.locator('#themeColor') + await expect(defaultUnit).toBeVisible() + }) + test('Modeling.File.Preferences.Sign out', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = + app.applicationMenu.getMenuItemById('File.Sign out') + if (!menu) { + throw new Error('File.Sign out') + } + // FIXME: Add back when you can actually sign out + // menu.click() + return true + }) + ) + .toBe(true) + // FIXME: When signing out during E2E the page is not bound correctly. + // It cannot find the button + // const signIn = page.getByTestId('sign-in-button') + // await expect(signIn).toBeVisible() + }) }) - test('Home.Edit.Delete project', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Delete project' - ) - if (!menu) { - return false - } - menu.click() - return true - }) - ) - .toBe(true) - // Check the placeholder project name exists - const actual = async () => - cmdBar.cmdBarElement.getByTestId('command-name').textContent() - const expected = 'Delete project' - await expect.poll(async () => await actual()).toBe(expected) + test.describe('Edit role', () => { + test('Modeling.Edit.Modify with Zoo Text-To-CAD', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Modify with Zoo Text-To-CAD' + ) + if (!menu) { + throw new Error('Edit.Modify with Zoo Text-To-CAD') + } + menu.click() + }) + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Prompt-to-edit' + expect(actual).toBe(expected) + }) + test('Modeling.Edit.Edit parameter', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Edit parameter' + ) + if (!menu) { + throw new Error('Edit.Edit parameter') + } + menu.click() + }) + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Edit parameter' + expect(actual).toBe(expected) + }) + test('Modeling.Edit.Format code', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById('Edit.Format code') + if (!menu) { + throw new Error('Edit.Format code') + } + // NO OP: Do not test that the code mirror will actually format the code. + // The format code happens, there is no UI. + // The actual business logic to test this feature should be in another E2E test. + // menu.click() + }) + }) + test('Modeling.Edit.Rename project', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Rename project' + ) + if (!menu) { + throw new Error('Edit.Rename project') + } + menu.click() + }) + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Rename project' + expect(actual).toBe(expected) + }) + test('Modeling.Edit.Delete project', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Delete project' + ) + if (!menu) { + throw new Error('Edit.Delete project') + } + menu.click() + }) + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Delete project' + expect(actual).toBe(expected) + }) + test('Modeling.Edit.Change project directory', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Edit.Change project directory' + ) + if (!menu) { + throw new Error('Edit.Change project directory') + } + menu.click() + }) + const settings = page.getByTestId('settings-dialog-panel') + await expect(settings).toBeVisible() + const projectDirectory = settings.locator('#projectDirectory') + await expect(projectDirectory).toBeVisible() + }) }) - test('Home.Edit.Change project directory', async ({ - tronApp, - cmdBar, - page, - }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - return false - } - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Change project directory' - ) - if (!menu) return false - menu.click() - return true - }) + test.describe('View role', () => { + test('Modeling.View.Command Palette...', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Command Palette...' + ) + if (!menu) { + throw new Error('View.Command Palette...') + } + menu.click() + }) + // Check the placeholder project name exists + const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') + await expect(actual).toBeVisible() + }) + test('Modeling.View.Orthographic view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Orthographic view' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + + const textToCheck = + 'Set camera projection to "orthographic" as a user default' + // Check if text appears anywhere in the page + const isTextVisible = page.getByText(textToCheck) + + await expect(isTextVisible).toBeVisible({ timeout: 10000 }) + }) + test('Modeling.View.Perspective view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Perspective view' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + + const textToCheck = + 'Set camera projection to "perspective" as a user default' + // Check if text appears anywhere in the page + const isTextVisible = page.getByText(textToCheck) + + await expect(isTextVisible).toBeVisible({ timeout: 10000 }) + }) + test('Modeling.View.Standard views.Right view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Right view' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + + // TODO: Make all of these screenshot E2E tests. + // Wait for camera to move + // await page.waitForTimeout(5000) + + // const locator = page.getByTestId('gizmo').locator('canvas') + // const image = await locator.screenshot({ path: 'Modeling.View.Standard-views.Right-view.png' }); + // expect(image).toMatchSnapshot('Modeling.View.Standard-views.Right-view') + }) + test('Modeling.View.Standard views.Back view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Back view' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Standard views.Top view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Top view' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Standard views.Left view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Left view' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Standard views.Front view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Front view' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Standard views.Bottom view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Bottom view' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Standard views.Reset view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Reset view' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Standard views.Center view on selection', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Center view on selection' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Standard views.Refresh', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Standard views.Refresh' + ) + if (!menu) { + throw new Error('menu missing') + } + // menu.click() + }) + }) + test('Modeling.View.Named views.Create named view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Named views.Create named view' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Create named view' + expect(actual).toBe(expected) + }) + test('Modeling.View.Named views.Load named view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Named views.Load named view' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Load named view' + expect(actual).toBe(expected) + }) + test('Modeling.View.Named views.Delete named view', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Named views.Delete named view' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Delete named view' + expect(actual).toBe(expected) + }) + test('Modeling.View.Panes.Feature tree', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Panes.Feature tree' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + + const button = page.getByTestId('feature-tree-pane-button') + const isPressed = await button.getAttribute('aria-pressed') + expect(isPressed).toBe('true') + }) + test('Modeling.View.Panes.KCL code', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Panes.KCL code' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + + const button = page.getByTestId('code-pane-button') + const isPressed = await button.getAttribute('aria-pressed') + expect(isPressed).toBe('true') + }) + test('Modeling.View.Panes.Project files', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Panes.Project files' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + + const button = page.getByTestId('files-pane-button') + const isPressed = await button.getAttribute('aria-pressed') + expect(isPressed).toBe('true') + }) + test('Modeling.View.Panes.Variables', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'View.Panes.Variables' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + + const button = page.getByTestId('variables-pane-button') + const isPressed = await button.getAttribute('aria-pressed') + expect(isPressed).toBe('true') + }) + test('Modeling.View.Panes.Logs', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById('View.Panes.Logs') + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + + const button = page.getByTestId('logs-pane-button') + const isPressed = await button.getAttribute('aria-pressed') + expect(isPressed).toBe('true') + }) + }) + test.describe('Design role', () => { + // TODO Start sketch + test('Modeling.Design.Create an offset plane', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create an offset plane' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Offset plane' + expect(actual).toBe(expected) + }) + test('Modeling.Design.Create a helix', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create a helix' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Helix' + expect(actual).toBe(expected) + }) + test('Modeling.Design.Create a parameter', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create a parameter' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Create parameter' + expect(actual).toBe(expected) + }) + + test('Modeling.Design.Create an additive feature.Extrude', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create an additive feature.Extrude' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Extrude' + expect(actual).toBe(expected) + }) + test('Modeling.Design.Create an additive feature.Revolve', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create an additive feature.Revolve' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Revolve' + expect(actual).toBe(expected) + }) + test('Modeling.Design.Create an additive feature.Sweep', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create an additive feature.Sweep' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Sweep' + expect(actual).toBe(expected) + }) + test('Modeling.Design.Create an additive feature.Loft', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create an additive feature.Loft' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Loft' + expect(actual).toBe(expected) + }) + test('Modeling.Design.Apply modification feature.Fillet', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Apply modification feature.Fillet' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Fillet' + expect(actual).toBe(expected) + }) + test('Modeling.Design.Apply modification feature.Chamfer', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Apply modification feature.Chamfer' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Chamfer' + expect(actual).toBe(expected) + }) + + test('Modeling.Design.Apply modification feature.Shell', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Apply modification feature.Shell' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Shell' + expect(actual).toBe(expected) + }) + + test('Modeling.Design.Insert from project file', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.settled(cmdBar) + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const openProject = app.applicationMenu.getMenuItemById( + 'Design.Insert from project file' + ) + if (!openProject) { + throw new Error('Design.Insert from project file') + } + openProject.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Insert' + expect(actual).toBe(expected) + }) + + test('Modeling.Design.Create with Zoo Text-To-CAD', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Create with Zoo Text-To-CAD' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Text-to-CAD' + expect(actual).toBe(expected) + }) + + test('Modeling.Design.Modify with Zoo Text-To-CAD', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) { + throw new Error('app or app.applicationMenu is missing') + } + const menu = app.applicationMenu.getMenuItemById( + 'Design.Modify with Zoo Text-To-CAD' + ) + if (!menu) { + throw new Error('menu missing') + } + menu.click() + }) + // Check that the command bar is opened + await expect(cmdBar.cmdBarElement).toBeVisible() + // Check the placeholder project name exists + const actual = await cmdBar.cmdBarElement + .getByTestId('command-name') + .textContent() + const expected = 'Prompt-to-edit' + expect(actual).toBe(expected) + }) + }) + test.describe('Help role', () => { + test('Modeling.Help.Show all commands', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) fail() + const menu = app.applicationMenu.getMenuItemById( + 'Help.Show all commands' + ) + if (!menu) fail() + menu.click() + }) + // Check the placeholder project name exists + const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') + await expect(actual).toBeVisible() + }) + test('Modeling.Help.KCL code samples', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) fail() + const menu = app.applicationMenu.getMenuItemById( + 'Help.KCL code samples' + ) + if (!menu) fail() + }) + }) + test('Modeling.Help.Report a bug', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + toolbar, + }) => { + // TODO: this test has been dead dead on the idle stream branch + test.fixme(orRunWhenFullSuiteEnabled()) + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await expect + .poll( + async () => + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) return false + const menu = + app.applicationMenu.getMenuItemById('Help.Report a bug') + if (!menu) return false + menu.click() + return true + }) + ) + .toBe(true) + // Core dump and refresh magic number timeout + await scene.connectionEstablished() + await expect(toolbar.startSketchBtn).toBeVisible() + }) + test('Modeling.Help.Reset onboarding', async ({ + tronApp, + cmdBar, + page, + homePage, + scene, + }) => { + if (!tronApp) { + throwTronAppMissing() + return + } + await homePage.goToModelingScene() + await scene.connectionEstablished() + + // Run electron snippet to find the Menu! + await page.waitForTimeout(100) // wait for createModelingPageMenu() to run + await tronApp.electron.evaluate(async ({ app }) => { + if (!app || !app.applicationMenu) fail() + const menu = app.applicationMenu.getMenuItemById( + 'Help.Reset onboarding' + ) + if (!menu) fail() + menu.click() + }) + + const actual = page.getByText( + `This is a hardware design tool that lets you edit visually, with code, or both. It's powered by the KittyCAD Design API, the first API created for anyone to build hardware design tools.` ) - .toBe(true) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - const projectDirectory = settings.locator('#projectDirectory') - await expect(projectDirectory).toBeVisible() + await expect(actual).toBeVisible() + }) }) }) - test.describe('View role', () => { - test('Home.View.Command Palette...', async ({ - tronApp, - cmdBar, - page, - }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const menu = app.applicationMenu.getMenuItemById( - 'View.Command Palette...' - ) - if (!menu) { - return false - } - menu.click() - return true - }) - ) - .toBe(true) - // Check the placeholder project name exists - const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') - await expect(actual).toBeVisible() - }) - }) - test.describe('Help role', () => { - test('Home.Help.Show all commands', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - return false - } - const menu = app.applicationMenu.getMenuItemById( - 'Help.Show all commands' - ) - if (!menu) return false - menu.click() - return true - }) - ) - .toBe(true) - // Check the placeholder project name exists - const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') - await expect(actual).toBeVisible() - }) - test('Home.Help.KCL code samples', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const menu = app.applicationMenu.getMenuItemById( - 'Help.KCL code samples' - ) - if (!menu) { - return false - } - return true - }) - ) - .toBe(true) - }) - test('Home.Help.Report a bug', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - return false - } - const menu = - app.applicationMenu.getMenuItemById('Help.Report a bug') - if (!menu) return false - menu.click() - return true - }) - ) - .toBe(true) - // Core dump and refresh magic number timeout - await page.waitForTimeout(7000) - const actual = page.getByText( - 'No Projects found, ready to make your first one?' - ) - await expect(actual).toBeVisible() - }) - test('Home.Help.Reset onboarding', async ({ tronApp, cmdBar, page }) => { - if (!tronApp) fail() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const menu = app.applicationMenu.getMenuItemById( - 'Help.Reset onboarding' - ) - if (!menu) { - return false - } - menu.click() - return true - }) - ) - .toBe(true) - - const actual = page.getByText( - `This is a hardware design tool that lets you edit visually, with code, or both. It's powered by the KittyCAD Design API, the first API created for anyone to build hardware design tools.` - ) - await expect(actual).toBeVisible() - }) - }) - }) - test.describe('Modeling page', () => { - test.describe('File Role', () => { - test('Modeling.File.Create project', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) fail() - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) fail() - const newProject = app.applicationMenu.getMenuItemById( - 'File.Create project' - ) - if (!newProject) fail() - newProject.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actualArgument = await cmdBar.cmdBarElement - .getByTestId('cmd-bar-arg-value') - .inputValue() - const expectedArgument = 'untitled' - expect(actualArgument).toBe(expectedArgument) - }) - test('Modeling.File.Open project', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const openProject = - app.applicationMenu.getMenuItemById('File.Open project') - if (!openProject) { - throw new Error('File.Open project') - } - openProject.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Open project' - expect(actual).toBe(expected) - }) - test('Modeling.File.Load external model', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const openProject = app.applicationMenu.getMenuItemById( - 'File.Load external model' - ) - if (!openProject) { - throw new Error('File.Load external model') - } - openProject.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Load external model' - expect(actual).toBe(expected) - }) - test('Modeling.File.Export current part', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const openProject = app.applicationMenu.getMenuItemById( - 'File.Export current part' - ) - if (!openProject) { - throw new Error('File.Export current part') - } - openProject.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Export' - expect(actual).toBe(expected) - }) - test('Modeling.File.Share part via Zoo link', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const openProject = app.applicationMenu.getMenuItemById( - 'File.Share part via Zoo link' - ) - if (!openProject) { - throw new Error('File.Share part via Zoo link') - } - openProject.click() - }) - - const textToCheck = - 'Link copied to clipboard. Anyone who clicks this link will get a copy of this file. Share carefully!' - // Check if text appears anywhere in the page - const isTextVisible = page.getByText(textToCheck) - - await expect(isTextVisible).toBeVisible({ timeout: 10000 }) - }) - test('Modeling.File.Preferences.Project settings', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const openProject = app.applicationMenu.getMenuItemById( - 'File.Preferences.Project settings' - ) - if (!openProject) { - throw new Error('File.Preferences.Project settings') - } - openProject.click() - }) - - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - // You are viewing the user tab - const actualText = settings.getByText( - 'The hue of the primary theme color for the app' - ) - await expect(actualText).toBeVisible() - }) - test('Modeling.File.Preferences.User settings', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const userSettings = app.applicationMenu.getMenuItemById( - 'File.Preferences.User settings' - ) - if (!userSettings) { - throw new Error('File.Preferences.User settings') - } - userSettings.click() - }) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - // You are viewing the user tab - const actualText = settings.getByText( - 'The overall appearance of the app' - ) - await expect(actualText).toBeVisible() - }) - test('Modeling.File.Preferences.Keybindings', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const keybindings = app.applicationMenu.getMenuItemById( - 'File.Preferences.Keybindings' - ) - if (!keybindings) { - throw new Error('File.Preferences.Keybindings') - } - keybindings.click() - }) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - // You are viewing the keybindings tab - const enterSketchMode = settings.locator('#enter-sketch-mode') - await expect(enterSketchMode).toBeVisible() - }) - test('Modeling.File.Preferences.User default units', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'File.Preferences.User default units' - ) - if (!menu) { - throw new Error('File.Preferences.User default units') - } - menu.click() - }) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - const defaultUnit = settings.locator('#defaultUnit') - await expect(defaultUnit).toBeVisible() - }) - test('Modeling.File.Preferences.Theme', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'File.Preferences.Theme' - ) - if (!menu) { - throw new Error('File.Preferences.Theme') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Settings · app · theme' - expect(actual).toBe(expected) - }) - test('Modeling.File.Preferences.Theme color', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'File.Preferences.Theme color' - ) - if (!menu) { - throw new Error('File.Preferences.Theme color') - } - menu.click() - }) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - const defaultUnit = settings.locator('#themeColor') - await expect(defaultUnit).toBeVisible() - }) - test('Modeling.File.Preferences.Sign out', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = - app.applicationMenu.getMenuItemById('File.Sign out') - if (!menu) { - throw new Error('File.Sign out') - } - // FIXME: Add back when you can actually sign out - // menu.click() - return true - }) - ) - .toBe(true) - // FIXME: When signing out during E2E the page is not bound correctly. - // It cannot find the button - // const signIn = page.getByTestId('sign-in-button') - // await expect(signIn).toBeVisible() - }) - }) - test.describe('Edit role', () => { - test('Modeling.Edit.Modify with Zoo Text-To-CAD', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Modify with Zoo Text-To-CAD' - ) - if (!menu) { - throw new Error('Edit.Modify with Zoo Text-To-CAD') - } - menu.click() - }) - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Prompt-to-edit' - expect(actual).toBe(expected) - }) - test('Modeling.Edit.Edit parameter', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Edit parameter' - ) - if (!menu) { - throw new Error('Edit.Edit parameter') - } - menu.click() - }) - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Edit parameter' - expect(actual).toBe(expected) - }) - test('Modeling.Edit.Format code', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById('Edit.Format code') - if (!menu) { - throw new Error('Edit.Format code') - } - // NO OP: Do not test that the code mirror will actually format the code. - // The format code happens, there is no UI. - // The actual business logic to test this feature should be in another E2E test. - // menu.click() - }) - }) - test('Modeling.Edit.Rename project', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Rename project' - ) - if (!menu) { - throw new Error('Edit.Rename project') - } - menu.click() - }) - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Rename project' - expect(actual).toBe(expected) - }) - test('Modeling.Edit.Delete project', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Delete project' - ) - if (!menu) { - throw new Error('Edit.Delete project') - } - menu.click() - }) - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Delete project' - expect(actual).toBe(expected) - }) - test('Modeling.Edit.Change project directory', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Edit.Change project directory' - ) - if (!menu) { - throw new Error('Edit.Change project directory') - } - menu.click() - }) - const settings = page.getByTestId('settings-dialog-panel') - await expect(settings).toBeVisible() - const projectDirectory = settings.locator('#projectDirectory') - await expect(projectDirectory).toBeVisible() - }) - }) - test.describe('View role', () => { - test('Modeling.View.Command Palette...', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Command Palette...' - ) - if (!menu) { - throw new Error('View.Command Palette...') - } - menu.click() - }) - // Check the placeholder project name exists - const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') - await expect(actual).toBeVisible() - }) - test('Modeling.View.Orthographic view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Orthographic view' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - - const textToCheck = - 'Set camera projection to "orthographic" as a user default' - // Check if text appears anywhere in the page - const isTextVisible = page.getByText(textToCheck) - - await expect(isTextVisible).toBeVisible({ timeout: 10000 }) - }) - test('Modeling.View.Perspective view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Perspective view' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - - const textToCheck = - 'Set camera projection to "perspective" as a user default' - // Check if text appears anywhere in the page - const isTextVisible = page.getByText(textToCheck) - - await expect(isTextVisible).toBeVisible({ timeout: 10000 }) - }) - test('Modeling.View.Standard views.Right view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Right view' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - - // TODO: Make all of these screenshot E2E tests. - // Wait for camera to move - // await page.waitForTimeout(5000) - - // const locator = page.getByTestId('gizmo').locator('canvas') - // const image = await locator.screenshot({ path: 'Modeling.View.Standard-views.Right-view.png' }); - // expect(image).toMatchSnapshot('Modeling.View.Standard-views.Right-view') - }) - test('Modeling.View.Standard views.Back view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Back view' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Standard views.Top view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Top view' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Standard views.Left view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Left view' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Standard views.Front view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Front view' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Standard views.Bottom view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Bottom view' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Standard views.Reset view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Reset view' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Standard views.Center view on selection', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Center view on selection' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Standard views.Refresh', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Standard views.Refresh' - ) - if (!menu) { - throw new Error('menu missing') - } - // menu.click() - }) - }) - test('Modeling.View.Named views.Create named view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Named views.Create named view' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Create named view' - expect(actual).toBe(expected) - }) - test('Modeling.View.Named views.Load named view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Named views.Load named view' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Load named view' - expect(actual).toBe(expected) - }) - test('Modeling.View.Named views.Delete named view', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Named views.Delete named view' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Delete named view' - expect(actual).toBe(expected) - }) - test('Modeling.View.Panes.Feature tree', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Panes.Feature tree' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - - const button = page.getByTestId('feature-tree-pane-button') - const isPressed = await button.getAttribute('aria-pressed') - expect(isPressed).toBe('true') - }) - test('Modeling.View.Panes.KCL code', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Panes.KCL code' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - - const button = page.getByTestId('code-pane-button') - const isPressed = await button.getAttribute('aria-pressed') - expect(isPressed).toBe('true') - }) - test('Modeling.View.Panes.Project files', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Panes.Project files' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - - const button = page.getByTestId('files-pane-button') - const isPressed = await button.getAttribute('aria-pressed') - expect(isPressed).toBe('true') - }) - test('Modeling.View.Panes.Variables', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'View.Panes.Variables' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - - const button = page.getByTestId('variables-pane-button') - const isPressed = await button.getAttribute('aria-pressed') - expect(isPressed).toBe('true') - }) - test('Modeling.View.Panes.Logs', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById('View.Panes.Logs') - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - - const button = page.getByTestId('logs-pane-button') - const isPressed = await button.getAttribute('aria-pressed') - expect(isPressed).toBe('true') - }) - }) - test.describe('Design role', () => { - // TODO Start sketch - test('Modeling.Design.Create an offset plane', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create an offset plane' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Offset plane' - expect(actual).toBe(expected) - }) - test('Modeling.Design.Create a helix', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create a helix' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Helix' - expect(actual).toBe(expected) - }) - test('Modeling.Design.Create a parameter', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create a parameter' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Create parameter' - expect(actual).toBe(expected) - }) - - test('Modeling.Design.Create an additive feature.Extrude', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create an additive feature.Extrude' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Extrude' - expect(actual).toBe(expected) - }) - test('Modeling.Design.Create an additive feature.Revolve', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create an additive feature.Revolve' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Revolve' - expect(actual).toBe(expected) - }) - test('Modeling.Design.Create an additive feature.Sweep', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create an additive feature.Sweep' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Sweep' - expect(actual).toBe(expected) - }) - test('Modeling.Design.Create an additive feature.Loft', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create an additive feature.Loft' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Loft' - expect(actual).toBe(expected) - }) - test('Modeling.Design.Apply modification feature.Fillet', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Apply modification feature.Fillet' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Fillet' - expect(actual).toBe(expected) - }) - test('Modeling.Design.Apply modification feature.Chamfer', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Apply modification feature.Chamfer' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Chamfer' - expect(actual).toBe(expected) - }) - - test('Modeling.Design.Apply modification feature.Shell', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Apply modification feature.Shell' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Shell' - expect(actual).toBe(expected) - }) - - test('Modeling.Design.Insert from project file', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.settled(cmdBar) - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const openProject = app.applicationMenu.getMenuItemById( - 'Design.Insert from project file' - ) - if (!openProject) { - throw new Error('Design.Insert from project file') - } - openProject.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Insert' - expect(actual).toBe(expected) - }) - - test('Modeling.Design.Create with Zoo Text-To-CAD', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Create with Zoo Text-To-CAD' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Text-to-CAD' - expect(actual).toBe(expected) - }) - - test('Modeling.Design.Modify with Zoo Text-To-CAD', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) { - throw new Error('app or app.applicationMenu is missing') - } - const menu = app.applicationMenu.getMenuItemById( - 'Design.Modify with Zoo Text-To-CAD' - ) - if (!menu) { - throw new Error('menu missing') - } - menu.click() - }) - // Check that the command bar is opened - await expect(cmdBar.cmdBarElement).toBeVisible() - // Check the placeholder project name exists - const actual = await cmdBar.cmdBarElement - .getByTestId('command-name') - .textContent() - const expected = 'Prompt-to-edit' - expect(actual).toBe(expected) - }) - }) - test.describe('Help role', () => { - test('Modeling.Help.Show all commands', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) fail() - const menu = app.applicationMenu.getMenuItemById( - 'Help.Show all commands' - ) - if (!menu) fail() - menu.click() - }) - // Check the placeholder project name exists - const actual = cmdBar.cmdBarElement.getByTestId('cmd-bar-search') - await expect(actual).toBeVisible() - }) - test('Modeling.Help.KCL code samples', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) fail() - const menu = app.applicationMenu.getMenuItemById( - 'Help.KCL code samples' - ) - if (!menu) fail() - }) - }) - test('Modeling.Help.Report a bug', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - toolbar, - }) => { - // TODO: this test has been dead dead on the idle stream branch - test.fixme(orRunWhenFullSuiteEnabled()) - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await expect - .poll( - async () => - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) return false - const menu = - app.applicationMenu.getMenuItemById('Help.Report a bug') - if (!menu) return false - menu.click() - return true - }) - ) - .toBe(true) - // Core dump and refresh magic number timeout - await scene.connectionEstablished() - await expect(toolbar.startSketchBtn).toBeVisible() - }) - test('Modeling.Help.Reset onboarding', async ({ - tronApp, - cmdBar, - page, - homePage, - scene, - }) => { - if (!tronApp) { - throwTronAppMissing() - return - } - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Run electron snippet to find the Menu! - await page.waitForTimeout(100) // wait for createModelingPageMenu() to run - await tronApp.electron.evaluate(async ({ app }) => { - if (!app || !app.applicationMenu) fail() - const menu = app.applicationMenu.getMenuItemById( - 'Help.Reset onboarding' - ) - if (!menu) fail() - menu.click() - }) - - const actual = page.getByText( - `This is a hardware design tool that lets you edit visually, with code, or both. It's powered by the KittyCAD Design API, the first API created for anyone to build hardware design tools.` - ) - await expect(actual).toBeVisible() - }) - }) - }) -}) + } +) diff --git a/e2e/playwright/point-click-assemblies.spec.ts b/e2e/playwright/point-click-assemblies.spec.ts index 437836040..cf7e32b80 100644 --- a/e2e/playwright/point-click-assemblies.spec.ts +++ b/e2e/playwright/point-click-assemblies.spec.ts @@ -6,6 +6,7 @@ import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture' import { executorInputPath, getUtils, + kclSamplesPath, testsInputPath, } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' @@ -472,4 +473,94 @@ test.describe('Point-and-click assemblies tests', () => { }) } ) + + test( + 'Assembly gets reexecuted when imported models are updated externally', + { tag: ['@electron'] }, + async ({ context, page, homePage, scene, toolbar, cmdBar, tronApp }) => { + if (!tronApp) { + fail() + } + + const midPoint = { x: 500, y: 250 } + const washerPoint = { x: 645, y: 250 } + const partColor: [number, number, number] = [120, 120, 120] + const redPartColor: [number, number, number] = [200, 0, 0] + const bgColor: [number, number, number] = [30, 30, 30] + const tolerance = 50 + const projectName = 'assembly' + + await test.step('Setup parts and expect imported model', async () => { + await context.folderSetupFn(async (dir) => { + const projectDir = path.join(dir, projectName) + await fsp.mkdir(projectDir, { recursive: true }) + await Promise.all([ + fsp.copyFile( + executorInputPath('cube.kcl'), + path.join(projectDir, 'cube.kcl') + ), + fsp.copyFile( + kclSamplesPath( + path.join( + 'pipe-flange-assembly', + 'mcmaster-parts', + '98017a257-washer.step' + ) + ), + path.join(projectDir, 'foreign.step') + ), + fsp.writeFile( + path.join(projectDir, 'main.kcl'), + ` +import "cube.kcl" as cube +import "foreign.step" as foreign +cube +foreign + |> translate(x = 40, z = 10)` + ), + ]) + }) + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.openProject(projectName) + await scene.settled(cmdBar) + await toolbar.closePane('code') + await scene.expectPixelColor(partColor, midPoint, tolerance) + }) + + await test.step('Change imported kcl file and expect change', async () => { + await context.folderSetupFn(async (dir) => { + // Append appearance to the cube.kcl file + await fsp.appendFile( + path.join(dir, projectName, 'cube.kcl'), + `\n |> appearance(color = "#ff0000")` + ) + }) + await scene.settled(cmdBar) + await toolbar.closePane('code') + await scene.expectPixelColor(redPartColor, midPoint, tolerance) + await scene.expectPixelColor(partColor, washerPoint, tolerance) + }) + + await test.step('Change imported step file and expect change', async () => { + await context.folderSetupFn(async (dir) => { + // Replace the washer with a pipe + await fsp.copyFile( + kclSamplesPath( + path.join( + 'pipe-flange-assembly', + 'mcmaster-parts', + '1120t74-pipe.step' + ) + ), + path.join(dir, projectName, 'foreign.step') + ) + }) + await scene.settled(cmdBar) + await toolbar.closePane('code') + // Expect pipe to take over the red cube but leave some space where the washer was + await scene.expectPixelColor(partColor, midPoint, tolerance) + await scene.expectPixelColor(bgColor, washerPoint, tolerance) + }) + } + ) }) diff --git a/e2e/playwright/projects.spec.ts b/e2e/playwright/projects.spec.ts index ab64e548d..4dc31b4b1 100644 --- a/e2e/playwright/projects.spec.ts +++ b/e2e/playwright/projects.spec.ts @@ -18,7 +18,7 @@ import { expect, test } from '@e2e/playwright/zoo-test' test( 'projects reload if a new one is created, deleted, or renamed externally', - { tag: '@electron' }, + { tag: ['@electron', '@macos', '@windows'] }, async ({ context, page }, testInfo) => { let externalCreatedProjectName = 'external-created-project' diff --git a/e2e/playwright/prompt-to-edit.spec.ts b/e2e/playwright/prompt-to-edit.spec.ts index accd52b28..b42a021d1 100644 --- a/e2e/playwright/prompt-to-edit.spec.ts +++ b/e2e/playwright/prompt-to-edit.spec.ts @@ -29,7 +29,7 @@ sketch003 = startSketchOn(XY) extrude003 = extrude(sketch003, length = 20) ` -test.describe('Prompt-to-edit tests', { tag: '@skipWin' }, () => { +test.describe('Prompt-to-edit tests', () => { test.describe('Check the happy path, for basic changing color', () => { const cases = [ { diff --git a/e2e/playwright/regression-tests.spec.ts b/e2e/playwright/regression-tests.spec.ts index 53932437d..149068d03 100644 --- a/e2e/playwright/regression-tests.spec.ts +++ b/e2e/playwright/regression-tests.spec.ts @@ -14,7 +14,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Regression tests', { tag: ['@skipWin'] }, () => { +test.describe('Regression tests', () => { // bugs we found that don't fit neatly into other categories test('bad model has inline error #3251', async ({ context, @@ -239,17 +239,18 @@ extrude001 = extrude(sketch001, length = 50) await expect(zooLogo).not.toHaveAttribute('href') }) - test( - 'Position _ Is Out Of Range... regression test', - { tag: ['@skipWin'] }, - async ({ context, page, homePage }) => { - const u = await getUtils(page) - // const PUR = 400 / 37.5 //pixeltoUnitRatio - await page.setBodyDimensions({ width: 1200, height: 500 }) - await context.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `exampleSketch = startSketchOn("XZ") + test('Position _ Is Out Of Range... regression test', async ({ + context, + page, + homePage, + }) => { + const u = await getUtils(page) + // const PUR = 400 / 37.5 //pixeltoUnitRatio + await page.setBodyDimensions({ width: 1200, height: 500 }) + await context.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `exampleSketch = startSketchOn("XZ") |> startProfileAt([0, 0], %) |> angledLine(angle = 50, length = 45 ) |> yLine(endAbsolute = 0) @@ -258,55 +259,55 @@ extrude001 = extrude(sketch001, length = 50) example = extrude(exampleSketch, length = 5) shell(exampleSketch, faces = ['end'], thickness = 0.25)` - ) + ) + }) + + await expect(async () => { + await homePage.goToModelingScene() + await u.waitForPageLoad() + + // error in guter + await expect(page.locator('.cm-lint-marker-error')).toBeVisible({ + timeout: 1_000, }) + await page.waitForTimeout(200) + // expect it still to be there (sometimes it just clears for a bit?) + await expect(page.locator('.cm-lint-marker-error')).toBeVisible({ + timeout: 1_000, + }) + }).toPass({ timeout: 40_000, intervals: [1_000] }) - await expect(async () => { - await homePage.goToModelingScene() - await u.waitForPageLoad() + // error text on hover + await page.hover('.cm-lint-marker-error') + await expect(page.getByText('Unexpected token: |').first()).toBeVisible() - // error in guter - await expect(page.locator('.cm-lint-marker-error')).toBeVisible({ - timeout: 1_000, - }) - await page.waitForTimeout(200) - // expect it still to be there (sometimes it just clears for a bit?) - await expect(page.locator('.cm-lint-marker-error')).toBeVisible({ - timeout: 1_000, - }) - }).toPass({ timeout: 40_000, intervals: [1_000] }) + // Okay execution finished, let's start editing text below the error. + await u.codeLocator.click() + // Go to the end of the editor + // This bug happens when there is a diagnostic in the editor and you try to + // edit text below it. + // Or delete a huge chunk of text and then try to edit below it. + await page.keyboard.press('End') + await page.keyboard.down('Shift') + await page.keyboard.press('ArrowUp') + await page.keyboard.press('ArrowUp') + await page.keyboard.press('ArrowUp') + await page.keyboard.press('ArrowUp') + await page.keyboard.press('ArrowUp') + await page.keyboard.press('End') + await page.keyboard.up('Shift') + await page.keyboard.press('Backspace') + await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() - // error text on hover - await page.hover('.cm-lint-marker-error') - await expect(page.getByText('Unexpected token: |').first()).toBeVisible() + await page.keyboard.press('Enter') + await page.keyboard.press('Enter') + await page.keyboard.type('thing: "blah"', { delay: 100 }) + await page.keyboard.press('Enter') + await page.keyboard.press('ArrowLeft') - // Okay execution finished, let's start editing text below the error. - await u.codeLocator.click() - // Go to the end of the editor - // This bug happens when there is a diagnostic in the editor and you try to - // edit text below it. - // Or delete a huge chunk of text and then try to edit below it. - await page.keyboard.press('End') - await page.keyboard.down('Shift') - await page.keyboard.press('ArrowUp') - await page.keyboard.press('ArrowUp') - await page.keyboard.press('ArrowUp') - await page.keyboard.press('ArrowUp') - await page.keyboard.press('ArrowUp') - await page.keyboard.press('End') - await page.keyboard.up('Shift') - await page.keyboard.press('Backspace') - await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() - - await page.keyboard.press('Enter') - await page.keyboard.press('Enter') - await page.keyboard.type('thing: "blah"', { delay: 100 }) - await page.keyboard.press('Enter') - await page.keyboard.press('ArrowLeft') - - await expect( - page.locator('.cm-content') - ).toContainText(`exampleSketch = startSketchOn("XZ") + await expect( + page.locator('.cm-content') + ).toContainText(`exampleSketch = startSketchOn("XZ") |> startProfileAt([0, 0], %) |> angledLine(angle = 50, length = 45 ) |> yLine(endAbsolute = 0) @@ -314,9 +315,8 @@ extrude001 = extrude(sketch001, length = 50) thing: "blah"`) - await expect(page.locator('.cm-lint-marker-error')).toBeVisible() - } - ) + await expect(page.locator('.cm-lint-marker-error')).toBeVisible() + }) test( 'window resize updates should reconfigure the stream', @@ -486,82 +486,81 @@ extrude002 = extrude(profile002, length = 150) } ) // We updated this test such that you can have multiple exports going at once. - test( - 'ensure you CAN export while an export is already going', - { tag: ['@skipLinux', '@skipWin'] }, - async ({ page, homePage }) => { - const u = await getUtils(page) - await test.step('Set up the code and durations', async () => { - await page.addInitScript( - async ({ code }) => { - localStorage.setItem('persistCode', code) - ;(window as any).playwrightSkipFilePicker = true - }, - { - code: bracket, - } - ) + test('ensure you CAN export while an export is already going', async ({ + page, + homePage, + }) => { + const u = await getUtils(page) + await test.step('Set up the code and durations', async () => { + await page.addInitScript( + async ({ code }) => { + localStorage.setItem('persistCode', code) + ;(window as any).playwrightSkipFilePicker = true + }, + { + code: bracket, + } + ) - await page.setBodyDimensions({ width: 1000, height: 500 }) + await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await u.waitForPageLoad() + await homePage.goToModelingScene() + await u.waitForPageLoad() - // wait for execution done - await u.openDebugPanel() - await u.expectCmdLog('[data-message-type="execution-done"]') - await u.closeDebugPanel() + // wait for execution done + await u.openDebugPanel() + await u.expectCmdLog('[data-message-type="execution-done"]') + await u.closeDebugPanel() - // expect zero errors in guter - await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() - }) + // expect zero errors in guter + await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() + }) - const errorToastMessage = page.getByText(`Error while exporting`) - const exportingToastMessage = page.getByText(`Exporting...`) - const engineErrorToastMessage = page.getByText(`Nothing to export`) - const alreadyExportingToastMessage = page.getByText(`Already exporting`) - const successToastMessage = page.getByText(`Exported successfully`) + const errorToastMessage = page.getByText(`Error while exporting`) + const exportingToastMessage = page.getByText(`Exporting...`) + const engineErrorToastMessage = page.getByText(`Nothing to export`) + const alreadyExportingToastMessage = page.getByText(`Already exporting`) + const successToastMessage = page.getByText(`Exported successfully`) - await test.step('second export', async () => { - await clickExportButton(page) + await test.step('second export', async () => { + await clickExportButton(page) - await expect(exportingToastMessage).toBeVisible() + await expect(exportingToastMessage).toBeVisible() - await clickExportButton(page) + await clickExportButton(page) - await test.step('The first export still succeeds', async () => { - await Promise.all([ - expect(exportingToastMessage).not.toBeVisible({ timeout: 15_000 }), - expect(errorToastMessage).not.toBeVisible(), - expect(engineErrorToastMessage).not.toBeVisible(), - expect(successToastMessage).toBeVisible({ timeout: 15_000 }), - expect(alreadyExportingToastMessage).not.toBeVisible({ - timeout: 15_000, - }), - ]) - }) - }) - - await test.step('Successful, unblocked export', async () => { - // Try exporting again. - await clickExportButton(page) - - // Find the toast. - // Look out for the toast message - await expect(exportingToastMessage).toBeVisible() - - // Expect it to succeed. + await test.step('The first export still succeeds', async () => { await Promise.all([ - expect(exportingToastMessage).not.toBeVisible(), + expect(exportingToastMessage).not.toBeVisible({ timeout: 15_000 }), expect(errorToastMessage).not.toBeVisible(), expect(engineErrorToastMessage).not.toBeVisible(), - expect(alreadyExportingToastMessage).not.toBeVisible(), + expect(successToastMessage).toBeVisible({ timeout: 15_000 }), + expect(alreadyExportingToastMessage).not.toBeVisible({ + timeout: 15_000, + }), ]) - - await expect(successToastMessage).toHaveCount(2) }) - } - ) + }) + + await test.step('Successful, unblocked export', async () => { + // Try exporting again. + await clickExportButton(page) + + // Find the toast. + // Look out for the toast message + await expect(exportingToastMessage).toBeVisible() + + // Expect it to succeed. + await Promise.all([ + expect(exportingToastMessage).not.toBeVisible(), + expect(errorToastMessage).not.toBeVisible(), + expect(engineErrorToastMessage).not.toBeVisible(), + expect(alreadyExportingToastMessage).not.toBeVisible(), + ]) + + await expect(successToastMessage).toHaveCount(2) + }) + }) test( `Network health indicator only appears in modeling view`, diff --git a/e2e/playwright/sketch-tests.spec.ts b/e2e/playwright/sketch-tests.spec.ts index d25eca3a6..be1106f2c 100644 --- a/e2e/playwright/sketch-tests.spec.ts +++ b/e2e/playwright/sketch-tests.spec.ts @@ -16,7 +16,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Sketch tests', { tag: ['@skipWin'] }, () => { +test.describe('Sketch tests', () => { test('multi-sketch file shows multiple Edit Sketch buttons', async ({ page, context, @@ -393,54 +393,58 @@ sketch001 = startProfileAt([12.34, -12.34], sketch002) |> close() `) } - test( - 'code pane open at start-handles', - { tag: ['@skipWin'] }, - async ({ page, homePage, scene, toolbar, cmdBar }) => { - // Load the app with the code panes - await page.addInitScript(async () => { - localStorage.setItem( - 'store', - JSON.stringify({ - state: { - openPanes: ['code'], - }, - version: 0, - }) - ) - }) - await doEditSegmentsByDraggingHandle( - page, - homePage, - ['code'], - scene, - toolbar, - cmdBar + test('code pane open at start-handles', async ({ + page, + homePage, + scene, + toolbar, + cmdBar, + }) => { + // Load the app with the code panes + await page.addInitScript(async () => { + localStorage.setItem( + 'store', + JSON.stringify({ + state: { + openPanes: ['code'], + }, + version: 0, + }) ) - } - ) + }) + await doEditSegmentsByDraggingHandle( + page, + homePage, + ['code'], + scene, + toolbar, + cmdBar + ) + }) - test( - 'code pane closed at start-handles', - { tag: ['@skipWin'] }, - async ({ page, homePage, scene, toolbar, cmdBar }) => { - // Load the app with the code panes - await page.addInitScript(async (persistModelingContext) => { - localStorage.setItem( - persistModelingContext, - JSON.stringify({ openPanes: [] }) - ) - }, PERSIST_MODELING_CONTEXT) - await doEditSegmentsByDraggingHandle( - page, - homePage, - [], - scene, - toolbar, - cmdBar + test('code pane closed at start-handles', async ({ + page, + homePage, + scene, + toolbar, + cmdBar, + }) => { + // Load the app with the code panes + await page.addInitScript(async (persistModelingContext) => { + localStorage.setItem( + persistModelingContext, + JSON.stringify({ openPanes: [] }) ) - } - ) + }, PERSIST_MODELING_CONTEXT) + await doEditSegmentsByDraggingHandle( + page, + homePage, + [], + scene, + toolbar, + cmdBar + ) + }) }) test('Can edit a circle center and radius by dragging its handles', async ({ @@ -1389,16 +1393,21 @@ offsetPlane001 = offsetPlane(XY, offset = 10)` }) test.describe('multi-profile sketching', () => { - test( - `test it removes half-finished expressions when changing tools in sketch mode`, - { tag: ['@skipWin'] }, - async ({ context, page, scene, toolbar, editor, homePage, cmdBar }) => { - test.fixme(orRunWhenFullSuiteEnabled()) - // We seed the scene with a single offset plane - await context.addInitScript(() => { - localStorage.setItem( - 'persistCode', - `yo = 5 + test(`test it removes half-finished expressions when changing tools in sketch mode`, async ({ + context, + page, + scene, + toolbar, + editor, + homePage, + cmdBar, + }) => { + test.fixme(orRunWhenFullSuiteEnabled()) + // We seed the scene with a single offset plane + await context.addInitScript(() => { + localStorage.setItem( + 'persistCode', + `yo = 5 sketch001 = startSketchOn(XZ) profile001 = startProfileAt([121.52, 168.25], sketch001) |> line(end = [115.04, 113.61]) @@ -1410,127 +1419,129 @@ profile002 = startProfileAt([117.2, 56.08], sketch001) |> yLine(length = -107.86) ` - ) - }) + ) + }) - const [continueProfile2Clk] = scene.makeMouseHelpers(954, 282) + const [continueProfile2Clk] = scene.makeMouseHelpers(954, 282) - await homePage.goToModelingScene() - await scene.settled(cmdBar) - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await homePage.goToModelingScene() + await scene.settled(cmdBar) + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - await (await toolbar.getFeatureTreeOperation('Sketch', 0)).dblclick() - await page.waitForTimeout(600) + await (await toolbar.getFeatureTreeOperation('Sketch', 0)).dblclick() + await page.waitForTimeout(600) - const [circlePoint1] = scene.makeMouseHelpers(700, 200) + const [circlePoint1] = scene.makeMouseHelpers(700, 200) - await test.step('equip circle tool and click first point', async () => { - // await page.waitForTimeout(100) - await expect - .poll(async () => { - await toolbar.circleBtn.click() - return toolbar.circleBtn.getAttribute('aria-pressed') - }) - .toBe('true') - await page.waitForTimeout(100) - await circlePoint1() - await editor.expectEditor.toContain( - 'profile003 = circle(sketch001, center = [' - ) - }) + await test.step('equip circle tool and click first point', async () => { + // await page.waitForTimeout(100) + await expect + .poll(async () => { + await toolbar.circleBtn.click() + return toolbar.circleBtn.getAttribute('aria-pressed') + }) + .toBe('true') + await page.waitForTimeout(100) + await circlePoint1() + await editor.expectEditor.toContain( + 'profile003 = circle(sketch001, center = [' + ) + }) - await test.step('equip line tool and verify circle code is removed', async () => { - await toolbar.lineBtn.click() - await editor.expectEditor.not.toContain('profile003 = circle(') - }) + await test.step('equip line tool and verify circle code is removed', async () => { + await toolbar.lineBtn.click() + await editor.expectEditor.not.toContain('profile003 = circle(') + }) - const [circle3Point1] = scene.makeMouseHelpers(650, 200) - const [circle3Point2] = scene.makeMouseHelpers(750, 200) - // const [circle3Point3] = scene.makeMouseHelpers(700, 150) + const [circle3Point1] = scene.makeMouseHelpers(650, 200) + const [circle3Point2] = scene.makeMouseHelpers(750, 200) + // const [circle3Point3] = scene.makeMouseHelpers(700, 150) - await test.step('equip three point circle tool and click first two points', async () => { - await toolbar.selectCircleThreePoint() - await page.waitForTimeout(100) - await circle3Point1() - await page.waitForTimeout(100) - await circle3Point2() - await editor.expectEditor.toContain('profile003 = circleThreePoint(') - }) + await test.step('equip three point circle tool and click first two points', async () => { + await toolbar.selectCircleThreePoint() + await page.waitForTimeout(100) + await circle3Point1() + await page.waitForTimeout(100) + await circle3Point2() + await editor.expectEditor.toContain('profile003 = circleThreePoint(') + }) - await test.step('equip line tool and verify three-point circle code is removed', async () => { - await toolbar.lineBtn.click() - await editor.expectEditor.not.toContain( - 'profile003 = circleThreePoint(' - ) - }) + await test.step('equip line tool and verify three-point circle code is removed', async () => { + await toolbar.lineBtn.click() + await editor.expectEditor.not.toContain('profile003 = circleThreePoint(') + }) - await test.step('equip three-point-arc tool and click first two points', async () => { - await page.waitForTimeout(200) - await toolbar.selectThreePointArc() - await page.waitForTimeout(200) - await circle3Point1() - await page.waitForTimeout(200) - await circle3Point2() - await editor.expectEditor.toContain('arc(') - }) + await test.step('equip three-point-arc tool and click first two points', async () => { + await page.waitForTimeout(200) + await toolbar.selectThreePointArc() + await page.waitForTimeout(200) + await circle3Point1() + await page.waitForTimeout(200) + await circle3Point2() + await editor.expectEditor.toContain('arc(') + }) - await test.step('equip line tool and verify three-point-arc code is removed after second click', async () => { - await toolbar.lineBtn.click() - await editor.expectEditor.not.toContain('arc(') - }) + await test.step('equip line tool and verify three-point-arc code is removed after second click', async () => { + await toolbar.lineBtn.click() + await editor.expectEditor.not.toContain('arc(') + }) - const [cornerRectPoint1] = scene.makeMouseHelpers(600, 300) + const [cornerRectPoint1] = scene.makeMouseHelpers(600, 300) - await test.step('equip corner rectangle tool and click first point', async () => { - await toolbar.rectangleBtn.click() - await page.waitForTimeout(100) - await cornerRectPoint1() - await editor.expectEditor.toContain('profile004 = startProfileAt(') - }) + await test.step('equip corner rectangle tool and click first point', async () => { + await toolbar.rectangleBtn.click() + await page.waitForTimeout(100) + await cornerRectPoint1() + await editor.expectEditor.toContain('profile004 = startProfileAt(') + }) - await test.step('equip line tool and verify corner rectangle code is removed', async () => { - await toolbar.lineBtn.click() - await editor.expectEditor.not.toContain('profile004 = startProfileAt(') - }) + await test.step('equip line tool and verify corner rectangle code is removed', async () => { + await toolbar.lineBtn.click() + await editor.expectEditor.not.toContain('profile004 = startProfileAt(') + }) - const [centerRectPoint1] = scene.makeMouseHelpers(700, 300) + const [centerRectPoint1] = scene.makeMouseHelpers(700, 300) - await test.step('equip center rectangle tool and click first point', async () => { - await toolbar.selectCenterRectangle() - await page.waitForTimeout(100) - await centerRectPoint1() - await editor.expectEditor.toContain('profile004 = startProfileAt(') - }) + await test.step('equip center rectangle tool and click first point', async () => { + await toolbar.selectCenterRectangle() + await page.waitForTimeout(100) + await centerRectPoint1() + await editor.expectEditor.toContain('profile004 = startProfileAt(') + }) - await test.step('equip line tool and verify center rectangle code is removed', async () => { - await toolbar.lineBtn.click() - await editor.expectEditor.not.toContain('profile004 = startProfileAt(') - }) + await test.step('equip line tool and verify center rectangle code is removed', async () => { + await toolbar.lineBtn.click() + await editor.expectEditor.not.toContain('profile004 = startProfileAt(') + }) - await test.step('continue profile002 with the three point arc tool, and then switch back to the line tool to verify it only removes the last expression in the pipe', async () => { - await toolbar.selectThreePointArc() - await page.waitForTimeout(200) - await continueProfile2Clk() - await page.waitForTimeout(200) - await circle3Point1() - await editor.expectEditor.toContain('arc(') - await toolbar.lineBtn.click() - await editor.expectEditor.not.toContain('arc(') - await editor.expectEditor.toContain('profile002') - }) - } - ) - test( - `snapToProfile start only works for current profile`, - { tag: ['@skipWin'] }, - async ({ context, page, scene, toolbar, editor, homePage, cmdBar }) => { - // We seed the scene with a single offset plane - await context.addInitScript(() => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + await test.step('continue profile002 with the three point arc tool, and then switch back to the line tool to verify it only removes the last expression in the pipe', async () => { + await toolbar.selectThreePointArc() + await page.waitForTimeout(200) + await continueProfile2Clk() + await page.waitForTimeout(200) + await circle3Point1() + await editor.expectEditor.toContain('arc(') + await toolbar.lineBtn.click() + await editor.expectEditor.not.toContain('arc(') + await editor.expectEditor.toContain('profile002') + }) + }) + test(`snapToProfile start only works for current profile`, async ({ + context, + page, + scene, + toolbar, + editor, + homePage, + cmdBar, + }) => { + // We seed the scene with a single offset plane + await context.addInitScript(() => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) profile002 = startProfileAt([40.68, 87.67], sketch001) @@ -1538,74 +1549,73 @@ profile002 = startProfileAt([40.68, 87.67], sketch001) profile003 = startProfileAt([206.63, -56.73], sketch001) |> xLine(length = -156.32) ` - ) - }) - - await homePage.goToModelingScene() - await scene.settled(cmdBar) - - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() - - const [onSegmentClick] = scene.makeMouseHelpers(604, 349) - const [endOfLowerSegClick, endOfLowerSegMove] = scene.makeMouseHelpers( - 697, - 360 ) - const [profileStartOfHigherSegClick, profileStartOfHigherSegMove] = - scene.makeMouseHelpers(677, 78) - const tanArcLocation = { x: 624, y: 340 } as const + }) - await test.step('enter sketch mode', async () => { - await onSegmentClick({ shouldDbClick: true }) - await page.waitForTimeout(600) - }) + await homePage.goToModelingScene() + await scene.settled(cmdBar) - const codeFromTangentialArc = ` |> tangentialArc(endAbsolute = [39.49, 88.22])` - await test.step('check that tangential tool does not snap to other profile starts', async () => { - await toolbar.tangentialArcBtn.click() - await page.waitForTimeout(1000) - await endOfLowerSegMove() - await page.waitForTimeout(1000) - await endOfLowerSegClick() - await page.waitForTimeout(1000) - await profileStartOfHigherSegClick() - await page.waitForTimeout(1000) - await editor.expectEditor.toContain(codeFromTangentialArc) - await editor.expectEditor.not.toContain( - `[profileStartX(%), profileStartY(%)]` - ) - }) + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - await test.step('remove tangential arc code to reset', async () => { - await scene.expectPixelColor(TEST_COLORS.WHITE, tanArcLocation, 15) - await editor.replaceCode(codeFromTangentialArc, '') - // check pixel is now gray at tanArcLocation to verify code has executed - await scene.expectPixelColor([26, 26, 26], tanArcLocation, 15) - await editor.expectEditor.not.toContain( - `tangentialArc(endAbsolute = [39.49, 88.22])` - ) - }) + const [onSegmentClick] = scene.makeMouseHelpers(604, 349) + const [endOfLowerSegClick, endOfLowerSegMove] = scene.makeMouseHelpers( + 697, + 360 + ) + const [profileStartOfHigherSegClick, profileStartOfHigherSegMove] = + scene.makeMouseHelpers(677, 78) + const tanArcLocation = { x: 624, y: 340 } as const - await test.step('check that tangential tool does snap to current profile start', async () => { - await expect - .poll(async () => { - await toolbar.lineBtn.click() - return toolbar.lineBtn.getAttribute('aria-pressed') - }) - .toBe('true') - await profileStartOfHigherSegMove() - await endOfLowerSegMove() - await endOfLowerSegClick() - await profileStartOfHigherSegClick() - await editor.expectEditor.toContain('line(end = [-10.82, 144.95])') - await editor.expectEditor.not.toContain( - `[profileStartX(%), profileStartY(%)]` - ) - }) - } - ) + await test.step('enter sketch mode', async () => { + await onSegmentClick({ shouldDbClick: true }) + await page.waitForTimeout(600) + }) + + const codeFromTangentialArc = ` |> tangentialArc(endAbsolute = [39.49, 88.22])` + await test.step('check that tangential tool does not snap to other profile starts', async () => { + await toolbar.selectTangentialArc() + await page.waitForTimeout(1000) + await endOfLowerSegMove() + await page.waitForTimeout(1000) + await endOfLowerSegClick() + await page.waitForTimeout(1000) + await profileStartOfHigherSegClick() + await page.waitForTimeout(1000) + await editor.expectEditor.toContain(codeFromTangentialArc) + await editor.expectEditor.not.toContain( + `[profileStartX(%), profileStartY(%)]` + ) + }) + + await test.step('remove tangential arc code to reset', async () => { + await scene.expectPixelColor(TEST_COLORS.WHITE, tanArcLocation, 15) + await editor.replaceCode(codeFromTangentialArc, '') + // check pixel is now gray at tanArcLocation to verify code has executed + await scene.expectPixelColor([26, 26, 26], tanArcLocation, 15) + await editor.expectEditor.not.toContain( + `tangentialArc(endAbsolute = [39.49, 88.22])` + ) + }) + + await test.step('check that tangential tool does snap to current profile start', async () => { + await expect + .poll(async () => { + await toolbar.lineBtn.click() + return toolbar.lineBtn.getAttribute('aria-pressed') + }) + .toBe('true') + await profileStartOfHigherSegMove() + await endOfLowerSegMove() + await endOfLowerSegClick() + await profileStartOfHigherSegClick() + await editor.expectEditor.toContain('line(end = [-10.82, 144.95])') + await editor.expectEditor.not.toContain( + `[profileStartX(%), profileStartY(%)]` + ) + }) + }) test('can enter sketch mode for sketch with no profiles', async ({ scene, toolbar, @@ -1777,7 +1787,7 @@ profile003 = startProfileAt([206.63, -56.73], sketch001) await endLineStartTanArc() await editor.expectEditor.toContain(`|> line(end = [9.02, -0.55])`) - await toolbar.tangentialArcBtn.click() + await toolbar.selectTangentialArc() await page.waitForTimeout(300) await page.mouse.click(745, 359) await page.waitForTimeout(300) @@ -2071,14 +2081,17 @@ profile003 = startProfileAt([206.63, -56.73], sketch001) }) }) - test( - 'Can edit a sketch with multiple profiles, dragging segments to edit them, and adding one new profile', - { tag: ['@skipWin'] }, - async ({ homePage, scene, toolbar, editor, page }) => { - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + test('Can edit a sketch with multiple profiles, dragging segments to edit them, and adding one new profile', async ({ + homePage, + scene, + toolbar, + editor, + page, + }) => { + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) profile001 = startProfileAt([6.24, 4.54], sketch001) |> line(end = [-0.41, 6.99]) @@ -2093,127 +2106,130 @@ profile002 = startProfileAt([11.19, 5.02], sketch001) profile003 = circle(sketch001, center = [6.92, -4.2], radius = 3.16) profile004 = circleThreePoint(sketch001, p1 = [13.44, -6.8], p2 = [13.39, -2.07], p3 = [18.75, -4.41]) ` - ) - }) + ) + }) - await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.goToModelingScene() - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - // The text to prompt popover gets in the way of pointOnSegment click otherwise - const moveToClearToolBarPopover = scene.makeMouseHelpers(590, 500)[1] + // The text to prompt popover gets in the way of pointOnSegment click otherwise + const moveToClearToolBarPopover = scene.makeMouseHelpers(590, 500)[1] - const [pointOnSegment] = scene.makeMouseHelpers(590, 141) - const [profileEnd] = scene.makeMouseHelpers(970, 105) - const profileEndMv = scene.makeMouseHelpers(951, 101)[1] - const [newProfileEnd] = scene.makeMouseHelpers(764, 104) - const dragSegmentTo = scene.makeMouseHelpers(850, 104)[1] + const [pointOnSegment] = scene.makeMouseHelpers(590, 141) + const [profileEnd] = scene.makeMouseHelpers(970, 105) + const profileEndMv = scene.makeMouseHelpers(951, 101)[1] + const [newProfileEnd] = scene.makeMouseHelpers(764, 104) + const dragSegmentTo = scene.makeMouseHelpers(850, 104)[1] - const rectHandle = scene.makeMouseHelpers(901, 150)[1] - const rectDragTo = scene.makeMouseHelpers(901, 180)[1] + const rectHandle = scene.makeMouseHelpers(901, 150)[1] + const rectDragTo = scene.makeMouseHelpers(901, 180)[1] - const circleEdge = scene.makeMouseHelpers(691, 331)[1] - const dragCircleTo = scene.makeMouseHelpers(720, 331)[1] + const circleEdge = scene.makeMouseHelpers(691, 331)[1] + const dragCircleTo = scene.makeMouseHelpers(720, 331)[1] - const [rectStart] = scene.makeMouseHelpers(794, 322) - const [rectEnd] = scene.makeMouseHelpers(757, 395) + const [rectStart] = scene.makeMouseHelpers(794, 322) + const [rectEnd] = scene.makeMouseHelpers(757, 395) - const [circ3PStart] = scene.makeMouseHelpers(854, 332) - const [circ3PEnd] = scene.makeMouseHelpers(870, 275) + const [circ3PStart] = scene.makeMouseHelpers(854, 332) + const [circ3PEnd] = scene.makeMouseHelpers(870, 275) - await test.step('enter sketch and setup', async () => { - await moveToClearToolBarPopover() - await page.waitForTimeout(1000) - await pointOnSegment({ shouldDbClick: true }) - await page.waitForTimeout(2000) + await test.step('enter sketch and setup', async () => { + await moveToClearToolBarPopover() + await page.waitForTimeout(1000) + await pointOnSegment({ shouldDbClick: true }) + await page.waitForTimeout(2000) - await toolbar.lineBtn.click() - await page.waitForTimeout(100) - }) + await toolbar.lineBtn.click() + await page.waitForTimeout(100) + }) - await test.step('extend existing profile', async () => { - await profileEnd() - await page.waitForTimeout(100) - await newProfileEnd() - await editor.expectEditor.toContain(`|> line(end = [-11.35, 0.73])`) - await toolbar.lineBtn.click() - await page.waitForTimeout(100) - }) + await test.step('extend existing profile', async () => { + await profileEnd() + await page.waitForTimeout(100) + await newProfileEnd() + await editor.expectEditor.toContain(`|> line(end = [-11.35, 0.73])`) + await toolbar.lineBtn.click() + await page.waitForTimeout(100) + }) - await test.step('edit existing profile', async () => { - await profileEndMv() - await page.mouse.down() - await dragSegmentTo() - await page.mouse.up() - await editor.expectEditor.toContain(`line(end = [4.22, -4.49])`) - }) + await test.step('edit existing profile', async () => { + await profileEndMv() + await page.mouse.down() + await dragSegmentTo() + await page.mouse.up() + await editor.expectEditor.toContain(`line(end = [4.22, -4.49])`) + }) - await test.step('edit existing rect', async () => { - await rectHandle() - await page.mouse.down() - await rectDragTo() - await page.mouse.up() - await editor.expectEditor.toContain( - `angledLine(angle = -7, length = 10.27, tag = $rectangleSegmentA001)` - ) - }) + await test.step('edit existing rect', async () => { + await rectHandle() + await page.mouse.down() + await rectDragTo() + await page.mouse.up() + await editor.expectEditor.toContain( + `angledLine(angle = -7, length = 10.27, tag = $rectangleSegmentA001)` + ) + }) - await test.step('edit existing circl', async () => { - await circleEdge() - await page.mouse.down() - await dragCircleTo() - await page.mouse.up() - await editor.expectEditor.toContain( - `profile003 = circle(sketch001, center = [6.92, -4.2], radius = 4.81)` - ) - }) + await test.step('edit existing circl', async () => { + await circleEdge() + await page.mouse.down() + await dragCircleTo() + await page.mouse.up() + await editor.expectEditor.toContain( + `profile003 = circle(sketch001, center = [6.92, -4.2], radius = 4.81)` + ) + }) - await test.step('edit existing circle three point', async () => { - await circ3PStart() - await page.mouse.down() - await circ3PEnd() - await page.mouse.up() - await editor.expectEditor.toContain( - `profile004 = circleThreePoint( + await test.step('edit existing circle three point', async () => { + await circ3PStart() + await page.mouse.down() + await circ3PEnd() + await page.mouse.up() + await editor.expectEditor.toContain( + `profile004 = circleThreePoint( sketch001, p1 = [13.44, -6.8], p2 = [13.39, -2.07], p3 = [19.73, -1.33], )`, - { shouldNormalise: true } - ) - }) + { shouldNormalise: true } + ) + }) - await test.step('add new profile', async () => { - await toolbar.rectangleBtn.click() - await page.waitForTimeout(100) - await rectStart() - await editor.expectEditor.toContain( - `profile005 = startProfileAt([15.68, -3.84], sketch001)` - ) - await page.waitForTimeout(100) - await rectEnd() - await editor.expectEditor.toContain( - `|> angledLine(angle = 180, length = 1.97, tag = $rectangleSegmentA002) + await test.step('add new profile', async () => { + await toolbar.rectangleBtn.click() + await page.waitForTimeout(100) + await rectStart() + await editor.expectEditor.toContain( + `profile005 = startProfileAt([15.68, -3.84], sketch001)` + ) + await page.waitForTimeout(100) + await rectEnd() + await editor.expectEditor.toContain( + `|> angledLine(angle = 180, length = 1.97, tag = $rectangleSegmentA002) |> angledLine(angle = segAng(rectangleSegmentA002) + 90, length = 3.89) |> angledLine(angle = segAng(rectangleSegmentA002), length = -segLen(rectangleSegmentA002)) |> line(endAbsolute = [profileStartX(%), profileStartY(%)]) |> close()`.replaceAll('\n', '') - ) - }) - } - ) - test( - 'Can delete a profile in the editor while is sketch mode, and sketch mode does not break, can ctrl+z to undo after constraint with variable was added', - { tag: ['@skipWin', '@skipLinux'] }, - async ({ scene, toolbar, editor, cmdBar, page, homePage }) => { - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + ) + }) + }) + test('Can delete a profile in the editor while is sketch mode, and sketch mode does not break, can ctrl+z to undo after constraint with variable was added', async ({ + scene, + toolbar, + editor, + cmdBar, + page, + homePage, + }) => { + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) profile001 = startProfileAt([6.24, 4.54], sketch001) |> line(end = [-0.41, 6.99]) @@ -2227,93 +2243,95 @@ profile002 = startProfileAt([11.19, 5.02], sketch001) |> close() profile003 = circle(sketch001, center = [6.92, -4.2], radius = 3.16) ` - ) - }) + ) + }) - await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await scene.settled(cmdBar) - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.goToModelingScene() + await scene.settled(cmdBar) + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - // The text to prompt popover gets in the way of pointOnSegment click otherwise - const moveToClearToolBarPopover = scene.makeMouseHelpers(590, 500)[1] + // The text to prompt popover gets in the way of pointOnSegment click otherwise + const moveToClearToolBarPopover = scene.makeMouseHelpers(590, 500)[1] - const [pointOnSegment] = scene.makeMouseHelpers(590, 141) - const [segment1Click] = scene.makeMouseHelpers(616, 131) - const sketchIsDrawnProperly = async () => { - await test.step('check the sketch is still drawn properly', async () => { - await page.waitForTimeout(200) - await scene.expectPixelColor([255, 255, 255], { x: 617, y: 163 }, 15) - await scene.expectPixelColor([255, 255, 255], { x: 629, y: 331 }, 15) - }) - } - - await test.step('enter sketch and setup', async () => { - await moveToClearToolBarPopover() - await pointOnSegment({ shouldDbClick: true }) - await page.waitForTimeout(600) - }) - - await test.step('select and delete code for a profile', async () => {}) - await page.getByText('close()').click() - await page.keyboard.down('Shift') - for (let i = 0; i < 11; i++) { - await page.keyboard.press('ArrowUp') - } - await page.keyboard.press('Home') - await page.keyboard.up('Shift') - await page.keyboard.press('Backspace') - - await sketchIsDrawnProperly() - - await test.step('add random new var between profiles', async () => { - await page.keyboard.type('myVar = 5') - await page.keyboard.press('Enter') - // If this timeout isn't long enough, the test breaks. - // TODO: fix https://github.com/KittyCAD/modeling-app/issues/5437 - await page.waitForTimeout(3_000) - }) - - await sketchIsDrawnProperly() - - await test.step('Adding a constraint with a variable, and than ctrl-z-ing which will remove the variable again does not break sketch mode', async () => { - await expect(async () => { - await segment1Click() - await editor.expectState({ - diagnostics: [], - activeLines: ['|>line(end = [-0.41,6.99])'], - highlightedCode: 'line(end = [-0.41,6.99])', - }) - }).toPass({ timeout: 30_000, intervals: [1500] }) - - await toolbar.lengthConstraintBtn.click() - await cmdBar.progressCmdBar() - await editor.expectEditor.toContain('length001 = 7') - - // wait for execute defer - await page.waitForTimeout(600) - await sketchIsDrawnProperly() - - await page.keyboard.down('Meta') - await page.keyboard.press('KeyZ') - await page.keyboard.up('Meta') - - await editor.expectEditor.not.toContain('length001 = 7') - await sketchIsDrawnProperly() + const [pointOnSegment] = scene.makeMouseHelpers(590, 141) + const [segment1Click] = scene.makeMouseHelpers(616, 131) + const sketchIsDrawnProperly = async () => { + await test.step('check the sketch is still drawn properly', async () => { + await page.waitForTimeout(200) + await scene.expectPixelColor([255, 255, 255], { x: 617, y: 163 }, 15) + await scene.expectPixelColor([255, 255, 255], { x: 629, y: 331 }, 15) }) } - ) - test( - 'can enter sketch when there is an extrude', - { tag: ['@skipWin'] }, - async ({ homePage, scene, toolbar, page, cmdBar }) => { - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + await test.step('enter sketch and setup', async () => { + await moveToClearToolBarPopover() + await pointOnSegment({ shouldDbClick: true }) + await page.waitForTimeout(600) + }) + + await test.step('select and delete code for a profile', async () => {}) + await page.getByText('close()').click() + await page.keyboard.down('Shift') + for (let i = 0; i < 11; i++) { + await page.keyboard.press('ArrowUp') + } + await page.keyboard.press('Home') + await page.keyboard.up('Shift') + await page.keyboard.press('Backspace') + + await sketchIsDrawnProperly() + + await test.step('add random new var between profiles', async () => { + await page.keyboard.type('myVar = 5') + await page.keyboard.press('Enter') + // If this timeout isn't long enough, the test breaks. + // TODO: fix https://github.com/KittyCAD/modeling-app/issues/5437 + await page.waitForTimeout(3_000) + }) + + await sketchIsDrawnProperly() + + await test.step('Adding a constraint with a variable, and than ctrl-z-ing which will remove the variable again does not break sketch mode', async () => { + await expect(async () => { + await segment1Click() + await editor.expectState({ + diagnostics: [], + activeLines: ['|>line(end = [-0.41,6.99])'], + highlightedCode: 'line(end = [-0.41,6.99])', + }) + }).toPass({ timeout: 30_000, intervals: [1500] }) + + await toolbar.lengthConstraintBtn.click() + await cmdBar.progressCmdBar() + await editor.expectEditor.toContain('length001 = 7') + + // wait for execute defer + await page.waitForTimeout(600) + await sketchIsDrawnProperly() + + await page.keyboard.down('Meta') + await page.keyboard.press('KeyZ') + await page.keyboard.up('Meta') + + await editor.expectEditor.not.toContain('length001 = 7') + await sketchIsDrawnProperly() + }) + }) + + test('can enter sketch when there is an extrude', async ({ + homePage, + scene, + toolbar, + page, + cmdBar, + }) => { + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) profile001 = startProfileAt([-63.43, 193.08], sketch001) |> line(end = [168.52, 149.87]) @@ -2336,33 +2354,32 @@ profile004 = circle( extrude002 = extrude(profile001, length = 50) extrude001 = extrude(profile003, length = 5) ` - ) - }) + ) + }) - await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await scene.connectionEstablished() - await scene.settled(cmdBar) - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.goToModelingScene() + await scene.connectionEstablished() + await scene.settled(cmdBar) + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - const [pointOnSegment] = scene.makeMouseHelpers(574, 207) + const [pointOnSegment] = scene.makeMouseHelpers(574, 207) - await pointOnSegment() - await toolbar.editSketch() - // wait for engine animation - await page.waitForTimeout(600) + await pointOnSegment() + await toolbar.editSketch() + // wait for engine animation + await page.waitForTimeout(600) - await test.step('check the sketch is still drawn properly', async () => { - await Promise.all([ - scene.expectPixelColor(TEST_COLORS.WHITE, { x: 596, y: 165 }, 15), - scene.expectPixelColor(TEST_COLORS.WHITE, { x: 641, y: 220 }, 15), - scene.expectPixelColor(TEST_COLORS.WHITE, { x: 763, y: 214 }, 15), - ]) - }) - } - ) + await test.step('check the sketch is still drawn properly', async () => { + await Promise.all([ + scene.expectPixelColor(TEST_COLORS.WHITE, { x: 596, y: 165 }, 15), + scene.expectPixelColor(TEST_COLORS.WHITE, { x: 641, y: 220 }, 15), + scene.expectPixelColor(TEST_COLORS.WHITE, { x: 763, y: 214 }, 15), + ]) + }) + }) test('exit new sketch without drawing anything should not be a problem', async ({ homePage, scene, @@ -2416,14 +2433,17 @@ extrude001 = extrude(profile003, length = 5) await scene.expectPixelColor([255, 255, 255], { x: 633, y: 211 }, 15) }) }) - test( - 'A sketch with only "startProfileAt" and no segments should still be able to be continued', - { tag: ['@skipWin'] }, - async ({ homePage, scene, toolbar, editor, page }) => { - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + test('A sketch with only "startProfileAt" and no segments should still be able to be continued', async ({ + homePage, + scene, + toolbar, + editor, + page, + }) => { + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) profile001 = startProfileAt([85.19, 338.59], sketch001) |> line(end = [213.3, -94.52]) @@ -2434,39 +2454,41 @@ sketch002 = startSketchOn(XY) profile002 = startProfileAt([85.81, 52.55], sketch002) ` - ) - }) + ) + }) - await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.goToModelingScene() + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - const [startProfileAt] = scene.makeMouseHelpers(606, 184) - const [nextPoint] = scene.makeMouseHelpers(763, 130) - await page.getByText('startProfileAt([85.81, 52.55], sketch002)').click() - await toolbar.editSketch(1) - // timeout wait for engine animation is unavoidable - await page.waitForTimeout(600) + const [startProfileAt] = scene.makeMouseHelpers(606, 184) + const [nextPoint] = scene.makeMouseHelpers(763, 130) + await page.getByText('startProfileAt([85.81, 52.55], sketch002)').click() + await toolbar.editSketch(1) + // timeout wait for engine animation is unavoidable + await page.waitForTimeout(600) - // equip line tool - await toolbar.lineBtn.click() - await page.waitForTimeout(100) - await startProfileAt() - await page.waitForTimeout(100) - await nextPoint() - await editor.expectEditor.toContain(`|> line(end = [126.05, 44.12])`) - } - ) - test( - 'old style sketch all in one pipe (with extrude) will break up to allow users to add a new profile to the same sketch', - { tag: ['@skipWin'] }, - async ({ homePage, scene, toolbar, editor, page }) => { - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + // equip line tool + await toolbar.lineBtn.click() + await page.waitForTimeout(100) + await startProfileAt() + await page.waitForTimeout(100) + await nextPoint() + await editor.expectEditor.toContain(`|> line(end = [126.05, 44.12])`) + }) + test('old style sketch all in one pipe (with extrude) will break up to allow users to add a new profile to the same sketch', async ({ + homePage, + scene, + toolbar, + editor, + page, + }) => { + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) thePart = startSketchOn(XZ) |> startProfileAt([7.53, 10.51], %) |> line(end = [12.54, 1.83]) @@ -2476,59 +2498,62 @@ thePart = startSketchOn(XZ) |> close() extrude001 = extrude(thePart, length = 75) ` - ) - }) + ) + }) - await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.goToModelingScene() + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() - const [objClick] = scene.makeMouseHelpers(565, 343) - const [profilePoint1] = scene.makeMouseHelpers(609, 289) - const [profilePoint2] = scene.makeMouseHelpers(714, 389) + const [objClick] = scene.makeMouseHelpers(565, 343) + const [profilePoint1] = scene.makeMouseHelpers(609, 289) + const [profilePoint2] = scene.makeMouseHelpers(714, 389) - await test.step('enter sketch and setup', async () => { - await objClick() - await toolbar.editSketch() - // timeout wait for engine animation is unavoidable - await page.waitForTimeout(600) - }) + await test.step('enter sketch and setup', async () => { + await objClick() + await toolbar.editSketch() + // timeout wait for engine animation is unavoidable + await page.waitForTimeout(600) + }) - await test.step('expect code to match initial conditions still', async () => { - await editor.expectEditor.toContain( - `thePart = startSketchOn(XZ) |> startProfileAt([7.53, 10.51], %)` - ) - }) + await test.step('expect code to match initial conditions still', async () => { + await editor.expectEditor.toContain( + `thePart = startSketchOn(XZ) |> startProfileAt([7.53, 10.51], %)` + ) + }) - await test.step('equiping the line tool should break up the pipe expression', async () => { - await toolbar.lineBtn.click() - await editor.expectEditor.toContain( - `sketch001 = startSketchOn(XZ)thePart = startProfileAt([7.53, 10.51], sketch001)` - ) - }) + await test.step('equiping the line tool should break up the pipe expression', async () => { + await toolbar.lineBtn.click() + await editor.expectEditor.toContain( + `sketch001 = startSketchOn(XZ)thePart = startProfileAt([7.53, 10.51], sketch001)` + ) + }) - await test.step('can continue on to add a new profile to this sketch', async () => { - await profilePoint1() - await editor.expectEditor.toContain( - `profile001 = startProfileAt([19.69, -7.05], sketch001)` - ) - await profilePoint2() - await editor.expectEditor.toContain(`|> line(end = [18.97, -18.06])`) - }) - } - ) - test( - 'Can enter sketch on sketch of wall and cap for segment, solid2d, extrude-wall, extrude-cap selections', - { tag: ['@skipWin'] }, - async ({ homePage, scene, toolbar, editor, page, cmdBar }) => { - // TODO this test should include a test for selecting revolve walls and caps + await test.step('can continue on to add a new profile to this sketch', async () => { + await profilePoint1() + await editor.expectEditor.toContain( + `profile001 = startProfileAt([19.69, -7.05], sketch001)` + ) + await profilePoint2() + await editor.expectEditor.toContain(`|> line(end = [18.97, -18.06])`) + }) + }) + test('Can enter sketch on sketch of wall and cap for segment, solid2d, extrude-wall, extrude-cap selections', async ({ + homePage, + scene, + toolbar, + editor, + page, + cmdBar, + }) => { + // TODO this test should include a test for selecting revolve walls and caps - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) profile001 = startProfileAt([6.71, -3.66], sketch001) |> line(end = [2.65, 9.02], tag = $seg02) @@ -2592,80 +2617,80 @@ extrude003 = extrude(profile011, length = 2.5) // TODO this breaks the test, // revolve002 = revolve(profile008, angle = 45, axis = seg02) ` - ) + ) + }) + + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.goToModelingScene() + await scene.connectionEstablished() + await scene.settled(cmdBar) + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() + + const camPositionForSelectingSketchOnWallProfiles = () => + scene.moveCameraTo( + { x: 834, y: -680, z: 534 }, + { x: -54, y: -476, z: 148 } + ) + const wallSelectionOptions = [ + { + title: 'select wall segment', + selectClick: scene.makeMouseHelpers(598, 211)[0], + }, + { + title: 'select wall solid 2d', + selectClick: scene.makeMouseHelpers(677, 236)[0], + }, + { + title: 'select wall circle', + selectClick: scene.makeMouseHelpers(811, 247)[0], + }, + { + title: 'select wall extrude wall', + selectClick: scene.makeMouseHelpers(793, 136)[0], + }, + { + title: 'select wall extrude cap', + selectClick: scene.makeMouseHelpers(836, 103)[0], + }, + ] as const + + const verifyWallProfilesAreDrawn = async () => + test.step('verify wall profiles are drawn', async () => { + await Promise.all([ + // open polygon + scene.expectPixelColor(TEST_COLORS.WHITE, { x: 599, y: 168 }, 15), + // closed polygon + scene.expectPixelColor(TEST_COLORS.WHITE, { x: 656, y: 171 }, 15), + // revolved profile + scene.expectPixelColor(TEST_COLORS.WHITE, { x: 655, y: 264 }, 15), + // extruded profile + scene.expectPixelColor(TEST_COLORS.WHITE, { x: 808, y: 396 }, 15), + // circle (When entering via the circle, it's selected and therefore blue) + scene.expectPixelColor( + [TEST_COLORS.WHITE, TEST_COLORS.BLUE], + { x: 742, y: 386 }, + 15 + ), + ]) }) - await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await scene.connectionEstablished() - await scene.settled(cmdBar) - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() - - const camPositionForSelectingSketchOnWallProfiles = () => - scene.moveCameraTo( - { x: 834, y: -680, z: 534 }, - { x: -54, y: -476, z: 148 } - ) - const wallSelectionOptions = [ - { - title: 'select wall segment', - selectClick: scene.makeMouseHelpers(598, 211)[0], - }, - { - title: 'select wall solid 2d', - selectClick: scene.makeMouseHelpers(677, 236)[0], - }, - { - title: 'select wall circle', - selectClick: scene.makeMouseHelpers(811, 247)[0], - }, - { - title: 'select wall extrude wall', - selectClick: scene.makeMouseHelpers(793, 136)[0], - }, - { - title: 'select wall extrude cap', - selectClick: scene.makeMouseHelpers(836, 103)[0], - }, - ] as const - - const verifyWallProfilesAreDrawn = async () => - test.step('verify wall profiles are drawn', async () => { - await Promise.all([ - // open polygon - scene.expectPixelColor(TEST_COLORS.WHITE, { x: 599, y: 168 }, 15), - // closed polygon - scene.expectPixelColor(TEST_COLORS.WHITE, { x: 656, y: 171 }, 15), - // revolved profile - scene.expectPixelColor(TEST_COLORS.WHITE, { x: 655, y: 264 }, 15), - // extruded profile - scene.expectPixelColor(TEST_COLORS.WHITE, { x: 808, y: 396 }, 15), - // circle (When entering via the circle, it's selected and therefore blue) - scene.expectPixelColor( - [TEST_COLORS.WHITE, TEST_COLORS.BLUE], - { x: 742, y: 386 }, - 15 - ), - ]) + await test.step('select wall profiles', async () => { + for (const { title, selectClick } of wallSelectionOptions) { + await test.step(title, async () => { + await camPositionForSelectingSketchOnWallProfiles() + await selectClick() + await toolbar.editSketch(1) + await page.waitForTimeout(600) + await verifyWallProfilesAreDrawn() + await toolbar.exitSketchBtn.click() + await page.waitForTimeout(100) }) + } + }) - await test.step('select wall profiles', async () => { - for (const { title, selectClick } of wallSelectionOptions) { - await test.step(title, async () => { - await camPositionForSelectingSketchOnWallProfiles() - await selectClick() - await toolbar.editSketch(1) - await page.waitForTimeout(600) - await verifyWallProfilesAreDrawn() - await toolbar.exitSketchBtn.click() - await page.waitForTimeout(100) - }) - } - }) - - /* FIXME: the cap part of this test is insanely flaky, and I'm not sure + /* FIXME: the cap part of this test is insanely flaky, and I'm not sure * why. * await test.step('select cap profiles', async () => { for (const { title, selectClick } of capSelectionOptions) { @@ -2681,16 +2706,18 @@ extrude003 = extrude(profile011, length = 2.5) }) } }) */ - } - ) - test( - 'Can enter sketch loft edges, base and continue sketch', - { tag: ['@skipWin'] }, - async ({ homePage, scene, toolbar, editor, page }) => { - await page.addInitScript(async () => { - localStorage.setItem( - 'persistCode', - `@settings(defaultLengthUnit = in) + }) + test('Can enter sketch loft edges, base and continue sketch', async ({ + homePage, + scene, + toolbar, + editor, + page, + }) => { + await page.addInitScript(async () => { + localStorage.setItem( + 'persistCode', + `@settings(defaultLengthUnit = in) sketch001 = startSketchOn(XZ) profile001 = startProfileAt([34, 42.66], sketch001) |> line(end = [102.65, 151.99]) @@ -2707,37 +2734,36 @@ profile002 = startProfileAt([39.43, 172.21], sketch002) loft([profile001, profile002]) ` - ) - }) - - await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled() - - const [baseProfileEdgeClick] = scene.makeMouseHelpers(621, 292) - - const [rect1Crn1] = scene.makeMouseHelpers(592, 283) - const [rect1Crn2] = scene.makeMouseHelpers(797, 268) - - await baseProfileEdgeClick() - await toolbar.editSketch() - await page.waitForTimeout(600) - await scene.expectPixelColor(TEST_COLORS.WHITE, { x: 562, y: 172 }, 15) - - await toolbar.rectangleBtn.click() - await page.waitForTimeout(100) - await rect1Crn1() - await editor.expectEditor.toContain( - `profile003 = startProfileAt([50.72, -18.19], sketch001)` ) - await rect1Crn2() - await editor.expectEditor.toContain( - `angledLine(angle = 0, length = 113.01, tag = $rectangleSegmentA001)` - ) - } - ) + }) + + await page.setBodyDimensions({ width: 1000, height: 500 }) + await homePage.goToModelingScene() + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled() + + const [baseProfileEdgeClick] = scene.makeMouseHelpers(621, 292) + + const [rect1Crn1] = scene.makeMouseHelpers(592, 283) + const [rect1Crn2] = scene.makeMouseHelpers(797, 268) + + await baseProfileEdgeClick() + await toolbar.editSketch() + await page.waitForTimeout(600) + await scene.expectPixelColor(TEST_COLORS.WHITE, { x: 562, y: 172 }, 15) + + await toolbar.rectangleBtn.click() + await page.waitForTimeout(100) + await rect1Crn1() + await editor.expectEditor.toContain( + `profile003 = startProfileAt([50.72, -18.19], sketch001)` + ) + await rect1Crn2() + await editor.expectEditor.toContain( + `angledLine(angle = 0, length = 113.01, tag = $rectangleSegmentA001)` + ) + }) test('Can enter sketch loft edges offsetPlane and continue sketch', async ({ scene, toolbar, @@ -2970,7 +2996,7 @@ test.describe('Redirecting to home page and back to the original file should cle await click00r(200, -200) // Draw arc - await toolbar.tangentialArcBtn.click() + await toolbar.selectTangentialArc() await click00r(0, 0) await click00r(100, 100) @@ -3245,96 +3271,100 @@ profile003 = startProfileAt([-201.08, 254.17], sketch002) ).toBeVisible() }) }) - test( - 'adding a syntax error, recovers after fixing', - { tag: ['@skipWin'] }, - async ({ page, homePage, context, scene, editor, toolbar, cmdBar }) => { - const file = await fs.readFile( - path.resolve( - __dirname, - '../../', - './rust/kcl-lib/e2e/executor/inputs/e2e-can-sketch-on-chamfer.kcl' - ), - 'utf-8' + test('adding a syntax error, recovers after fixing', async ({ + page, + homePage, + context, + scene, + editor, + toolbar, + cmdBar, + }) => { + const file = await fs.readFile( + path.resolve( + __dirname, + '../../', + './rust/kcl-lib/e2e/executor/inputs/e2e-can-sketch-on-chamfer.kcl' + ), + 'utf-8' + ) + await context.addInitScript((file) => { + localStorage.setItem('persistCode', file) + }, file) + await homePage.goToModelingScene() + + const [objClick] = scene.makeMouseHelpers(600, 250) + const arrowHeadLocation = { x: 706, y: 129 } as const + const arrowHeadWhite = TEST_COLORS.WHITE + const backgroundGray: [number, number, number] = [28, 28, 28] + const verifyArrowHeadColor = async (c: [number, number, number]) => + scene.expectPixelColor(c, arrowHeadLocation, 15) + + // wait for scene to load + await scene.settled(cmdBar) + + await test.step('check chamfer selection changes cursor positon', async () => { + await expect(async () => { + // sometimes initial click doesn't register + await objClick() + await editor.expectActiveLinesToBe([ + '|> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag]', + ]) + }).toPass({ timeout: 15_000, intervals: [500] }) + }) + + await test.step('enter sketch and sanity check segments have been drawn', async () => { + await toolbar.editSketch() + // this checks sketch segments have been drawn + await verifyArrowHeadColor(arrowHeadWhite) + }) + + await test.step('Make typo and check the segments have Disappeared and there is a syntax error', async () => { + await editor.replaceCode( + 'line(endAbsolute = [pro', + 'badBadBadFn(endAbsolute = [pro' ) - await context.addInitScript((file) => { - localStorage.setItem('persistCode', file) - }, file) - await homePage.goToModelingScene() - - const [objClick] = scene.makeMouseHelpers(600, 250) - const arrowHeadLocation = { x: 706, y: 129 } as const - const arrowHeadWhite = TEST_COLORS.WHITE - const backgroundGray: [number, number, number] = [28, 28, 28] - const verifyArrowHeadColor = async (c: [number, number, number]) => - scene.expectPixelColor(c, arrowHeadLocation, 15) - - // wait for scene to load - await scene.settled(cmdBar) - - await test.step('check chamfer selection changes cursor positon', async () => { - await expect(async () => { - // sometimes initial click doesn't register - await objClick() - await editor.expectActiveLinesToBe([ - '|> startProfileAt([75.8, 317.2], %) // [$startCapTag, $EndCapTag]', - ]) - }).toPass({ timeout: 15_000, intervals: [500] }) + await editor.expectState({ + activeLines: [], + diagnostics: ['`badBadBadFn`isnotdefined'], + highlightedCode: '', }) - - await test.step('enter sketch and sanity check segments have been drawn', async () => { - await toolbar.editSketch() - // this checks sketch segments have been drawn - await verifyArrowHeadColor(arrowHeadWhite) - }) - - await test.step('Make typo and check the segments have Disappeared and there is a syntax error', async () => { - await editor.replaceCode( - 'line(endAbsolute = [pro', - 'badBadBadFn(endAbsolute = [pro' + await expect( + page.getByText( + "Error in kcl script, sketch cannot be drawn until it's fixed" ) - await editor.expectState({ - activeLines: [], - diagnostics: ['`badBadBadFn`isnotdefined'], - highlightedCode: '', - }) - await expect( - page.getByText( - "Error in kcl script, sketch cannot be drawn until it's fixed" - ) - ).toBeVisible() - // this checks sketch segments have failed to be drawn - await verifyArrowHeadColor(backgroundGray) - }) + ).toBeVisible() + // this checks sketch segments have failed to be drawn + await verifyArrowHeadColor(backgroundGray) + }) - await test.step('', async () => { - await editor.replaceCode( - 'badBadBadFn(endAbsolute = [pro', - 'line(endAbsolute = [pro' - ) - await editor.expectState({ - activeLines: [], - diagnostics: [], - highlightedCode: '', - }) - // this checks sketch segments have been drawn - await verifyArrowHeadColor(arrowHeadWhite) + await test.step('', async () => { + await editor.replaceCode( + 'badBadBadFn(endAbsolute = [pro', + 'line(endAbsolute = [pro' + ) + await editor.expectState({ + activeLines: [], + diagnostics: [], + highlightedCode: '', }) + // this checks sketch segments have been drawn + await verifyArrowHeadColor(arrowHeadWhite) + }) - await test.step('make a change to the code and expect pixel color to change', async () => { - // defends against a regression where sketch would duplicate in the scene - // https://github.com/KittyCAD/modeling-app/issues/6345 - await editor.replaceCode( - 'startProfileAt([75.8, 317.2', - 'startProfileAt([75.8, 217.2' - ) - // expect not white anymore - await scene.expectPixelColorNotToBe( - TEST_COLORS.WHITE, - arrowHeadLocation, - 15 - ) - }) - } - ) + await test.step('make a change to the code and expect pixel color to change', async () => { + // defends against a regression where sketch would duplicate in the scene + // https://github.com/KittyCAD/modeling-app/issues/6345 + await editor.replaceCode( + 'startProfileAt([75.8, 317.2', + 'startProfileAt([75.8, 217.2' + ) + // expect not white anymore + await scene.expectPixelColorNotToBe( + TEST_COLORS.WHITE, + arrowHeadLocation, + 15 + ) + }) + }) }) diff --git a/e2e/playwright/snapshot-tests.spec.ts b/e2e/playwright/snapshot-tests.spec.ts index 908b649dd..48c8efc81 100644 --- a/e2e/playwright/snapshot-tests.spec.ts +++ b/e2e/playwright/snapshot-tests.spec.ts @@ -47,7 +47,7 @@ test.setTimeout(60_000) // up with another PR if we want this back. test( 'exports of each format should work', - { tag: ['@snapshot', '@skipWin', '@skipMacos'] }, + { tag: ['@snapshot'] }, async ({ page, context, scene, cmdBar, tronApp }) => { if (!tronApp) { fail() @@ -464,9 +464,7 @@ test( |> xLine(length = 184.3)` await expect(page.locator('.cm-content')).toHaveText(code) - await page - .getByRole('button', { name: 'arc Tangential Arc', exact: true }) - .click() + await toolbar.selectTangentialArc() // click on the end of the profile to continue it await page.waitForTimeout(500) @@ -621,7 +619,7 @@ test.describe( 'Client side scene scale should match engine scale', { tag: '@snapshot' }, () => { - test('Inch scale', async ({ page, cmdBar, scene }) => { + test('Inch scale', async ({ page, cmdBar, scene, toolbar }) => { const u = await getUtils(page) await page.setViewportSize({ width: 1200, height: 500 }) const PUR = 400 / 37.5 //pixeltoUnitRatio @@ -655,9 +653,7 @@ test.describe( |> xLine(length = 184.3)` await expect(u.codeLocator).toHaveText(code) - await page - .getByRole('button', { name: 'arc Tangential Arc', exact: true }) - .click() + await toolbar.selectTangentialArc() await page.waitForTimeout(100) // click to continue profile @@ -671,9 +667,8 @@ test.describe( await expect(u.codeLocator).toHaveText(code) // click tangential arc tool again to unequip it - await page - .getByRole('button', { name: 'arc Tangential Arc', exact: true }) - .click() + // it will be available directly in the toolbar since it was last equipped + await toolbar.tangentialArcBtn.click() await page.waitForTimeout(100) // screen shot should show the sketch @@ -696,7 +691,13 @@ test.describe( }) }) - test('Millimeter scale', async ({ page, context, cmdBar, scene }) => { + test('Millimeter scale', async ({ + page, + context, + cmdBar, + scene, + toolbar, + }) => { await context.addInitScript( async ({ settingsKey, settings }) => { localStorage.setItem(settingsKey, settings) @@ -749,9 +750,7 @@ test.describe( |> xLine(length = 184.3)` await expect(u.codeLocator).toHaveText(code) - await page - .getByRole('button', { name: 'arc Tangential Arc', exact: true }) - .click() + await toolbar.selectTangentialArc() await page.waitForTimeout(100) // click to continue profile @@ -764,9 +763,7 @@ test.describe( |> tangentialArc(endAbsolute = [551.2, -62.01])` await expect(u.codeLocator).toHaveText(code) - await page - .getByRole('button', { name: 'arc Tangential Arc', exact: true }) - .click() + await toolbar.tangentialArcBtn.click() await page.waitForTimeout(100) // screen shot should show the sketch diff --git a/e2e/playwright/test-network-and-connection-issues.spec.ts b/e2e/playwright/test-network-and-connection-issues.spec.ts index fe0b5dab6..3ae1f6f23 100644 --- a/e2e/playwright/test-network-and-connection-issues.spec.ts +++ b/e2e/playwright/test-network-and-connection-issues.spec.ts @@ -8,228 +8,235 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Test network and connection issues', () => { - test( - 'simulate network down and network little widget', - { tag: '@skipLocalEngine' }, - async ({ page, homePage }) => { - test.fixme(orRunWhenFullSuiteEnabled()) - const u = await getUtils(page) - await page.setBodyDimensions({ width: 1200, height: 500 }) +test.describe( + 'Test network and connection issues', + { + tag: ['@macos', '@windows'], + }, + () => { + test( + 'simulate network down and network little widget', + { tag: '@skipLocalEngine' }, + async ({ page, homePage }) => { + test.fixme(orRunWhenFullSuiteEnabled()) + const u = await getUtils(page) + await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() + await homePage.goToModelingScene() - const networkToggle = page.getByTestId('network-toggle') + const networkToggle = page.getByTestId('network-toggle') - // This is how we wait until the stream is online - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled({ timeout: 15000 }) + // This is how we wait until the stream is online + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled({ timeout: 15000 }) - const networkWidget = page.locator('[data-testid="network-toggle"]') - await expect(networkWidget).toBeVisible() - await networkWidget.hover() + const networkWidget = page.locator('[data-testid="network-toggle"]') + await expect(networkWidget).toBeVisible() + await networkWidget.hover() - const networkPopover = page.locator('[data-testid="network-popover"]') - await expect(networkPopover).not.toBeVisible() + const networkPopover = page.locator('[data-testid="network-popover"]') + await expect(networkPopover).not.toBeVisible() - // (First check) Expect the network to be up - await expect(networkToggle).toContainText('Connected') + // (First check) Expect the network to be up + await expect(networkToggle).toContainText('Connected') - // Click the network widget - await networkWidget.click() + // Click the network widget + await networkWidget.click() - // Check the modal opened. - await expect(networkPopover).toBeVisible() + // Check the modal opened. + await expect(networkPopover).toBeVisible() - // Click off the modal. - await page.mouse.click(100, 100) - await expect(networkPopover).not.toBeVisible() + // Click off the modal. + await page.mouse.click(100, 100) + await expect(networkPopover).not.toBeVisible() - // Turn off the network - await u.emulateNetworkConditions({ - offline: true, - // values of 0 remove any active throttling. crbug.com/456324#c9 - latency: 0, - downloadThroughput: -1, - uploadThroughput: -1, - }) + // Turn off the network + await u.emulateNetworkConditions({ + offline: true, + // values of 0 remove any active throttling. crbug.com/456324#c9 + latency: 0, + downloadThroughput: -1, + uploadThroughput: -1, + }) - // Expect the network to be down - await expect(networkToggle).toContainText('Problem') + // Expect the network to be down + await expect(networkToggle).toContainText('Problem') - // Click the network widget - await networkWidget.click() + // Click the network widget + await networkWidget.click() - // Check the modal opened. - await expect(networkPopover).toBeVisible() + // Check the modal opened. + await expect(networkPopover).toBeVisible() - // Click off the modal. - await page.mouse.click(0, 0) - await expect(networkPopover).not.toBeVisible() + // Click off the modal. + await page.mouse.click(0, 0) + await expect(networkPopover).not.toBeVisible() - // Turn back on the network - await u.emulateNetworkConditions({ - offline: false, - // values of 0 remove any active throttling. crbug.com/456324#c9 - latency: 0, - downloadThroughput: -1, - uploadThroughput: -1, - }) + // Turn back on the network + await u.emulateNetworkConditions({ + offline: false, + // values of 0 remove any active throttling. crbug.com/456324#c9 + latency: 0, + downloadThroughput: -1, + uploadThroughput: -1, + }) - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled({ timeout: 15000 }) + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled({ timeout: 15000 }) - // (Second check) expect the network to be up - await expect(networkToggle).toContainText('Connected') - } - ) + // (Second check) expect the network to be up + await expect(networkToggle).toContainText('Connected') + } + ) - test( - 'Engine disconnect & reconnect in sketch mode', - { tag: '@skipLocalEngine' }, - async ({ page, homePage, toolbar, scene, cmdBar }) => { - test.fixme(orRunWhenFullSuiteEnabled()) - const networkToggle = page.getByTestId('network-toggle') + test( + 'Engine disconnect & reconnect in sketch mode', + { tag: '@skipLocalEngine' }, + async ({ page, homePage, toolbar, scene, cmdBar }) => { + test.fixme(orRunWhenFullSuiteEnabled()) + const networkToggle = page.getByTestId('network-toggle') - const u = await getUtils(page) - await page.setBodyDimensions({ width: 1200, height: 500 }) - const PUR = 400 / 37.5 //pixeltoUnitRatio + const u = await getUtils(page) + await page.setBodyDimensions({ width: 1200, height: 500 }) + const PUR = 400 / 37.5 //pixeltoUnitRatio - await homePage.goToModelingScene() - await u.waitForPageLoad() + await homePage.goToModelingScene() + await u.waitForPageLoad() - await u.openDebugPanel() - // click on "Start Sketch" button - await u.clearCommandLogs() - await page.getByRole('button', { name: 'Start Sketch' }).click() - await page.waitForTimeout(100) + await u.openDebugPanel() + // click on "Start Sketch" button + await u.clearCommandLogs() + await page.getByRole('button', { name: 'Start Sketch' }).click() + await page.waitForTimeout(100) - // select a plane - await page.mouse.click(700, 200) + // select a plane + await page.mouse.click(700, 200) - await expect(page.locator('.cm-content')).toHaveText( - `sketch001 = startSketchOn(XZ)` - ) - await u.closeDebugPanel() + await expect(page.locator('.cm-content')).toHaveText( + `sketch001 = startSketchOn(XZ)` + ) + await u.closeDebugPanel() - await page.waitForTimeout(500) // TODO detect animation ending, or disable animation + await page.waitForTimeout(500) // TODO detect animation ending, or disable animation - const startXPx = 600 - await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) - await expect(page.locator('.cm-content')).toHaveText( - `sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001)` - ) - await page.waitForTimeout(100) + const startXPx = 600 + await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 10) + await expect(page.locator('.cm-content')).toHaveText( + `sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001)` + ) + await page.waitForTimeout(100) - await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) - await page.waitForTimeout(100) + await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 10) + await page.waitForTimeout(100) - await expect( - page.locator('.cm-content') - ).toHaveText(`sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001) + await expect( + page.locator('.cm-content') + ).toHaveText(`sketch001 = startSketchOn(XZ)profile001 = startProfileAt(${commonPoints.startAt}, sketch001) |> xLine(length = ${commonPoints.num1})`) - // Expect the network to be up - await expect(networkToggle).toContainText('Connected') + // Expect the network to be up + await expect(networkToggle).toContainText('Connected') - // simulate network down - await u.emulateNetworkConditions({ - offline: true, - // values of 0 remove any active throttling. crbug.com/456324#c9 - latency: 0, - downloadThroughput: -1, - uploadThroughput: -1, - }) + // simulate network down + await u.emulateNetworkConditions({ + offline: true, + // values of 0 remove any active throttling. crbug.com/456324#c9 + latency: 0, + downloadThroughput: -1, + uploadThroughput: -1, + }) - // Expect the network to be down - await expect(networkToggle).toContainText('Problem') + // Expect the network to be down + await expect(networkToggle).toContainText('Problem') - // Ensure we are not in sketch mode - await expect( - page.getByRole('button', { name: 'Exit Sketch' }) - ).not.toBeVisible() - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).toBeVisible() + // Ensure we are not in sketch mode + await expect( + page.getByRole('button', { name: 'Exit Sketch' }) + ).not.toBeVisible() + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).toBeVisible() - // simulate network up - await u.emulateNetworkConditions({ - offline: false, - // values of 0 remove any active throttling. crbug.com/456324#c9 - latency: 0, - downloadThroughput: -1, - uploadThroughput: -1, - }) + // simulate network up + await u.emulateNetworkConditions({ + offline: false, + // values of 0 remove any active throttling. crbug.com/456324#c9 + latency: 0, + downloadThroughput: -1, + uploadThroughput: -1, + }) - // Wait for the app to be ready for use - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).not.toBeDisabled({ timeout: 15000 }) + // Wait for the app to be ready for use + await expect( + page.getByRole('button', { name: 'Start Sketch' }) + ).not.toBeDisabled({ timeout: 15000 }) - // Expect the network to be up - await expect(networkToggle).toContainText('Connected') - await scene.settled(cmdBar) + // Expect the network to be up + await expect(networkToggle).toContainText('Connected') + await scene.settled(cmdBar) - // Click off the code pane. - await page.mouse.click(100, 100) + // Click off the code pane. + await page.mouse.click(100, 100) - // select a line - await page - .getByText(`startProfileAt(${commonPoints.startAt}, sketch001)`) - .click() + // select a line + await page + .getByText(`startProfileAt(${commonPoints.startAt}, sketch001)`) + .click() - // enter sketch again - await toolbar.editSketch() + // enter sketch again + await toolbar.editSketch() - // Click the line tool - await page.getByRole('button', { name: 'line Line', exact: true }).click() + // Click the line tool + await page + .getByRole('button', { name: 'line Line', exact: true }) + .click() - await page.waitForTimeout(150) + await page.waitForTimeout(150) - const camCommand: EngineCommand = { - type: 'modeling_cmd_req', - cmd_id: uuidv4(), - cmd: { - type: 'default_camera_look_at', - center: { x: 109, y: 0, z: -152 }, - vantage: { x: 115, y: -505, z: -152 }, - up: { x: 0, y: 0, z: 1 }, - }, - } - const updateCamCommand: EngineCommand = { - type: 'modeling_cmd_req', - cmd_id: uuidv4(), - cmd: { - type: 'default_camera_get_settings', - }, - } - await toolbar.openPane('debug') - await u.sendCustomCmd(camCommand) - await page.waitForTimeout(100) - await u.sendCustomCmd(updateCamCommand) - await page.waitForTimeout(100) + const camCommand: EngineCommand = { + type: 'modeling_cmd_req', + cmd_id: uuidv4(), + cmd: { + type: 'default_camera_look_at', + center: { x: 109, y: 0, z: -152 }, + vantage: { x: 115, y: -505, z: -152 }, + up: { x: 0, y: 0, z: 1 }, + }, + } + const updateCamCommand: EngineCommand = { + type: 'modeling_cmd_req', + cmd_id: uuidv4(), + cmd: { + type: 'default_camera_get_settings', + }, + } + await toolbar.openPane('debug') + await u.sendCustomCmd(camCommand) + await page.waitForTimeout(100) + await u.sendCustomCmd(updateCamCommand) + await page.waitForTimeout(100) - // click to continue profile - await page.mouse.click(1007, 400) - await page.waitForTimeout(100) - // Ensure we can continue sketching - await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 20) - await expect - .poll(u.normalisedEditorCode) - .toBe(`sketch001 = startSketchOn(XZ) + // click to continue profile + await page.mouse.click(1007, 400) + await page.waitForTimeout(100) + // Ensure we can continue sketching + await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 20) + await expect + .poll(u.normalisedEditorCode) + .toBe(`sketch001 = startSketchOn(XZ) profile001 = startProfileAt([12.34, -12.34], sketch001) |> xLine(length = 12.34) |> line(end = [-12.34, 12.34]) `) - await page.waitForTimeout(100) - await page.mouse.click(startXPx, 500 - PUR * 20) + await page.waitForTimeout(100) + await page.mouse.click(startXPx, 500 - PUR * 20) - await expect - .poll(u.normalisedEditorCode) - .toBe(`sketch001 = startSketchOn(XZ) + await expect + .poll(u.normalisedEditorCode) + .toBe(`sketch001 = startSketchOn(XZ) profile001 = startProfileAt([12.34, -12.34], sketch001) |> xLine(length = 12.34) |> line(end = [-12.34, 12.34]) @@ -237,21 +244,22 @@ profile001 = startProfileAt([12.34, -12.34], sketch001) `) - // Unequip line tool - await page.keyboard.press('Escape') - // Make sure we didn't pop out of sketch mode. - await expect( - page.getByRole('button', { name: 'Exit Sketch' }) - ).toBeVisible() - await expect( - page.getByRole('button', { name: 'line Line', exact: true }) - ).not.toHaveAttribute('aria-pressed', 'true') + // Unequip line tool + await page.keyboard.press('Escape') + // Make sure we didn't pop out of sketch mode. + await expect( + page.getByRole('button', { name: 'Exit Sketch' }) + ).toBeVisible() + await expect( + page.getByRole('button', { name: 'line Line', exact: true }) + ).not.toHaveAttribute('aria-pressed', 'true') - // Exit sketch - await page.keyboard.press('Escape') - await expect( - page.getByRole('button', { name: 'Exit Sketch' }) - ).not.toBeVisible() - } - ) -}) + // Exit sketch + await page.keyboard.press('Escape') + await expect( + page.getByRole('button', { name: 'Exit Sketch' }) + ).not.toBeVisible() + } + ) + } +) diff --git a/e2e/playwright/test-utils.ts b/e2e/playwright/test-utils.ts index 44813baa9..ebf3eba2a 100644 --- a/e2e/playwright/test-utils.ts +++ b/e2e/playwright/test-utils.ts @@ -1024,6 +1024,10 @@ export function testsInputPath(fileName: string): string { return path.join('rust', 'kcl-lib', 'tests', 'inputs', fileName) } +export function kclSamplesPath(fileName: string): string { + return path.join('public', 'kcl-samples', fileName) +} + export async function doAndWaitForImageDiff( page: Page, fn: () => Promise, diff --git a/e2e/playwright/testing-camera-movement.spec.ts b/e2e/playwright/testing-camera-movement.spec.ts index dd63f5605..816d6ce51 100644 --- a/e2e/playwright/testing-camera-movement.spec.ts +++ b/e2e/playwright/testing-camera-movement.spec.ts @@ -4,7 +4,7 @@ import { uuidv4 } from '@src/lib/utils' import { getUtils, orRunWhenFullSuiteEnabled } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Testing Camera Movement', { tag: ['@skipWin'] }, () => { +test.describe('Testing Camera Movement', () => { test('Can move camera reliably', async ({ page, context, diff --git a/e2e/playwright/testing-constraints.spec.ts b/e2e/playwright/testing-constraints.spec.ts index b4fc908f6..1d33df616 100644 --- a/e2e/playwright/testing-constraints.spec.ts +++ b/e2e/playwright/testing-constraints.spec.ts @@ -10,7 +10,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Testing constraints', { tag: ['@skipWin'] }, () => { +test.describe('Testing constraints', () => { test('Can constrain line length', async ({ page, homePage }) => { await page.addInitScript(async () => { localStorage.setItem( diff --git a/e2e/playwright/testing-gizmo.spec.ts b/e2e/playwright/testing-gizmo.spec.ts index a46467160..5f220cd56 100644 --- a/e2e/playwright/testing-gizmo.spec.ts +++ b/e2e/playwright/testing-gizmo.spec.ts @@ -4,7 +4,7 @@ import { TEST_CODE_GIZMO } from '@e2e/playwright/storageStates' import { getUtils } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Testing Gizmo', { tag: ['@skipWin'] }, () => { +test.describe('Testing Gizmo', () => { const cases = [ { testDescription: 'top view', diff --git a/e2e/playwright/testing-segment-overlays.spec.ts b/e2e/playwright/testing-segment-overlays.spec.ts index 0e5d184bf..adaa148aa 100644 --- a/e2e/playwright/testing-segment-overlays.spec.ts +++ b/e2e/playwright/testing-segment-overlays.spec.ts @@ -11,7 +11,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Testing segment overlays', { tag: ['@skipWin'] }, () => { +test.describe('Testing segment overlays', () => { test('Hover over a segment should show its overlay, hovering over the input overlays should show its popover, clicking the input overlay should constrain/unconstrain it:\nfor the following segments', () => { // TODO: fix this test on mac after the electron migration test.fixme(orRunWhenFullSuiteEnabled()) diff --git a/e2e/playwright/testing-selections.spec.ts b/e2e/playwright/testing-selections.spec.ts index 13eb29d49..8f4193952 100644 --- a/e2e/playwright/testing-selections.spec.ts +++ b/e2e/playwright/testing-selections.spec.ts @@ -9,7 +9,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Testing selections', { tag: ['@skipWin'] }, () => { +test.describe('Testing selections', () => { test.setTimeout(90_000) test('Selections work on fresh and edited sketch', async ({ page, @@ -39,12 +39,12 @@ test.describe('Testing selections', { tag: ['@skipWin'] }, () => { }) const emptySpaceHover = () => test.step('Hover over empty space', async () => { - await page.mouse.move(700, 143, { steps: 5 }) + await page.mouse.move(1000, 143, { steps: 5 }) await expect(page.locator('.hover-highlight')).not.toBeVisible() }) const emptySpaceClick = () => test.step(`Click in empty space`, async () => { - await page.mouse.click(700, 143) + await page.mouse.click(1000, 143) await expect(page.locator('.cm-line').last()).toHaveClass( /cm-activeLine/ ) diff --git a/e2e/playwright/testing-settings.spec.ts b/e2e/playwright/testing-settings.spec.ts index dfd80d2b6..933435b24 100644 --- a/e2e/playwright/testing-settings.spec.ts +++ b/e2e/playwright/testing-settings.spec.ts @@ -26,725 +26,752 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Testing settings', () => { - test('Stored settings are validated and fall back to defaults', async ({ - page, - homePage, - tronApp, - }) => { - if (!tronApp) { - fail() - } - // Override beforeEach test setup - // with corrupted settings - await tronApp.cleanProjectDir( - TEST_SETTINGS_CORRUPTED as DeepPartial - ) - - await page.setBodyDimensions({ width: 1200, height: 500 }) - - // Check the settings were reset - const storedSettings = tomlToSettings( - await page.evaluate( - ({ settingsKey }) => localStorage.getItem(settingsKey) || '', - { settingsKey: TEST_SETTINGS_KEY } +test.describe( + 'Testing settings', + { + tag: ['@macos', '@windows'], + }, + () => { + test('Stored settings are validated and fall back to defaults', async ({ + page, + homePage, + tronApp, + }) => { + if (!tronApp) { + fail() + } + // Override beforeEach test setup + // with corrupted settings + await tronApp.cleanProjectDir( + TEST_SETTINGS_CORRUPTED as DeepPartial ) - ) - expect(storedSettings.settings?.app?.appearance?.theme).toBe('dark') - - // Check that the invalid settings were changed to good defaults - expect(storedSettings.settings?.modeling?.base_unit).toBe('in') - expect(storedSettings.settings?.modeling?.mouse_controls).toBe('zoo') - // Commenting this out because tests need this to be set to work properly. - // expect(storedSettings.settings?.app?.project_directory).toBe('') - expect(storedSettings.settings?.project?.default_project_name).toBe( - 'untitled' - ) - }) - - // The behavior is actually broken. Parent always takes precedence - test('Project settings can be set and override user settings', async ({ - page, - homePage, - }) => { - test.fixme(orRunWhenFullSuiteEnabled()) - const u = await getUtils(page) - await test.step(`Setup`, async () => { await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() + + // Check the settings were reset + const storedSettings = tomlToSettings( + await page.evaluate( + ({ settingsKey }) => localStorage.getItem(settingsKey) || '', + { settingsKey: TEST_SETTINGS_KEY } + ) + ) + + expect(storedSettings.settings?.app?.appearance?.theme).toBe('dark') + + // Check that the invalid settings were changed to good defaults + expect(storedSettings.settings?.modeling?.base_unit).toBe('in') + expect(storedSettings.settings?.modeling?.mouse_controls).toBe('zoo') + // Commenting this out because tests need this to be set to work properly. + // expect(storedSettings.settings?.app?.project_directory).toBe('') + expect(storedSettings.settings?.project?.default_project_name).toBe( + 'untitled' + ) + }) + + // The behavior is actually broken. Parent always takes precedence + test('Project settings can be set and override user settings', async ({ + page, + homePage, + }) => { + test.fixme(orRunWhenFullSuiteEnabled()) + const u = await getUtils(page) + await test.step(`Setup`, async () => { + await page.setBodyDimensions({ width: 1200, height: 500 }) + await homePage.goToModelingScene() + await page + .getByRole('button', { name: 'Start Sketch' }) + .waitFor({ state: 'visible' }) + }) + + // Selectors and constants + const paneButtonLocator = page.getByTestId('debug-pane-button') + const headingLocator = page.getByRole('heading', { + name: 'Settings', + exact: true, + }) + const inputLocator = page.locator('input[name="app-showDebugPanel"]') + + await test.step('Open settings dialog and set "Show debug panel" to on', async () => { + await page.keyboard.press('ControlOrMeta+,') + await expect(headingLocator).toBeVisible() + + /** Test to close https://github.com/KittyCAD/modeling-app/issues/2713 */ + await test.step(`Confirm that this dialog has a solid background`, async () => { + await expect + .poll( + () => u.getGreatestPixDiff({ x: 600, y: 250 }, [28, 28, 28]), + { + timeout: 1000, + message: + 'Checking for solid background, should not see default plane colors', + } + ) + .toBeLessThan(15) + }) + + await page.locator('#showDebugPanel').getByText('OffOn').click() + }) + + // Close it and open again with keyboard shortcut, while KCL editor is focused + // Put the cursor in the editor + await test.step('Open settings with keyboard shortcut', async () => { + await page.getByTestId('settings-close-button').click() + await page.locator('.cm-content').click() + await page.keyboard.press('ControlOrMeta+,') + await expect(headingLocator).toBeVisible() + }) + + // Verify the toast appeared + await expect( + page.getByText(`Set show debug panel to "false" for this project`) + ).toBeVisible() + await expect( + page.getByText(`Set show debug panel to "false" for this project`) + ).not.toBeVisible() + + // Check that the debug panel button is gone + await expect(paneButtonLocator).not.toBeVisible() + + // Check that the user setting was not changed + await page.getByRole('radio', { name: 'User' }).click() + await expect(inputLocator).toBeChecked() + + // Roll back to default of "off" + await await page + .getByText( + 'show debug panelRoll back show debug panelRoll back to match' + ) + .hover() await page - .getByRole('button', { name: 'Start Sketch' }) - .waitFor({ state: 'visible' }) + .getByRole('button', { + name: 'Roll back show debug panel', + }) + .click() + await expect(inputLocator).not.toBeChecked() + + // Check that the project setting did not change + await page.getByRole('radio', { name: 'Project' }).click() + await expect( + page.locator('input[name="app-showDebugPanel"]') + ).not.toBeChecked() }) - // Selectors and constants - const paneButtonLocator = page.getByTestId('debug-pane-button') - const headingLocator = page.getByRole('heading', { - name: 'Settings', - exact: true, - }) - const inputLocator = page.locator('input[name="app-showDebugPanel"]') - - await test.step('Open settings dialog and set "Show debug panel" to on', async () => { - await page.keyboard.press('ControlOrMeta+,') - await expect(headingLocator).toBeVisible() - - /** Test to close https://github.com/KittyCAD/modeling-app/issues/2713 */ - await test.step(`Confirm that this dialog has a solid background`, async () => { - await expect - .poll(() => u.getGreatestPixDiff({ x: 600, y: 250 }, [28, 28, 28]), { - timeout: 1000, - message: - 'Checking for solid background, should not see default plane colors', - }) - .toBeLessThan(15) - }) - - await page.locator('#showDebugPanel').getByText('OffOn').click() - }) - - // Close it and open again with keyboard shortcut, while KCL editor is focused - // Put the cursor in the editor - await test.step('Open settings with keyboard shortcut', async () => { - await page.getByTestId('settings-close-button').click() - await page.locator('.cm-content').click() - await page.keyboard.press('ControlOrMeta+,') - await expect(headingLocator).toBeVisible() - }) - - // Verify the toast appeared - await expect( - page.getByText(`Set show debug panel to "false" for this project`) - ).toBeVisible() - await expect( - page.getByText(`Set show debug panel to "false" for this project`) - ).not.toBeVisible() - - // Check that the debug panel button is gone - await expect(paneButtonLocator).not.toBeVisible() - - // Check that the user setting was not changed - await page.getByRole('radio', { name: 'User' }).click() - await expect(inputLocator).toBeChecked() - - // Roll back to default of "off" - await await page - .getByText('show debug panelRoll back show debug panelRoll back to match') - .hover() - await page - .getByRole('button', { - name: 'Roll back show debug panel', - }) - .click() - await expect(inputLocator).not.toBeChecked() - - // Check that the project setting did not change - await page.getByRole('radio', { name: 'Project' }).click() - await expect( - page.locator('input[name="app-showDebugPanel"]') - ).not.toBeChecked() - }) - - test('Keybindings display the correct hotkey for Command Palette', async ({ - page, - homePage, - }) => { - // TODO: fix this test on windows after the electron migration - test.skip(process.platform === 'win32', 'Skip on windows') - const u = await getUtils(page) - await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() - await u.waitForPageLoad() - - await test.step('Open keybindings settings', async () => { - // Open the settings modal with the keyboard shortcut - await page.keyboard.press('ControlOrMeta+,') - - // Go to Keybindings tab. - const keybindingsTab = page.getByRole('radio', { name: 'Keybindings' }) - await keybindingsTab.click() - }) - - // Go to the hotkey for Command Palette. - const commandPalette = page.getByText('Toggle Command Palette') - await commandPalette.scrollIntoViewIfNeeded() - - // The heading is above it and should be in view now. - const commandPaletteHeading = page.getByRole('heading', { - name: 'Command Palette', - }) - // The hotkey is in a kbd element next to the heading. - const hotkey = commandPaletteHeading.locator('+ div kbd') - const text = process.platform === 'darwin' ? 'Command+K' : 'Control+K' - await expect(hotkey).toHaveText(text) - }) - - test('Project and user settings can be reset', async ({ page, homePage }) => { - const u = await getUtils(page) - await test.step(`Setup`, async () => { + test('Keybindings display the correct hotkey for Command Palette', async ({ + page, + homePage, + }) => { + // TODO: fix this test on windows after the electron migration + test.skip(process.platform === 'win32', 'Skip on windows') + const u = await getUtils(page) await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() await u.waitForPageLoad() - await page.waitForTimeout(1000) - }) - // Selectors and constants - const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) - const userSettingsTab = page.getByRole('radio', { name: 'User' }) - const resetButton = (level: SettingsLevel) => - page.getByRole('button', { - name: `Reset ${level}-level settings`, - }) - const themeColorSetting = page.locator('#themeColor').getByRole('slider') - const settingValues = { - default: '259', - user: '120', - project: '50', - } - const resetToast = (level: SettingsLevel) => - page.getByText(`${level}-level settings were reset`) + await test.step('Open keybindings settings', async () => { + // Open the settings modal with the keyboard shortcut + await page.keyboard.press('ControlOrMeta+,') - await test.step(`Open the settings modal`, async () => { - await page.getByRole('link', { name: 'Settings' }).last().click() - await expect( - page.getByRole('heading', { name: 'Settings', exact: true }) - ).toBeVisible() - }) - - await test.step('Set up theme color', async () => { - // Verify we're looking at the project-level settings, - // and it's set to default value - await expect(projectSettingsTab).toBeChecked() - await expect(themeColorSetting).toHaveValue(settingValues.default) - - // Set project-level value to 50 - await themeColorSetting.fill(settingValues.project) - - // Set user-level value to 120 - await userSettingsTab.click() - await themeColorSetting.fill(settingValues.user) - await projectSettingsTab.click() - }) - - await test.step('Reset project settings', async () => { - // Click the reset settings button. - await resetButton('project').click() - - await expect(resetToast('project')).toBeVisible() - await expect(resetToast('project')).not.toBeVisible() - - // Verify it is now set to the inherited user value - await expect(themeColorSetting).toHaveValue(settingValues.user) - - await test.step(`Check that the user settings did not change`, async () => { - await userSettingsTab.click() - await expect(themeColorSetting).toHaveValue(settingValues.user) + // Go to Keybindings tab. + const keybindingsTab = page.getByRole('radio', { name: 'Keybindings' }) + await keybindingsTab.click() }) - await test.step(`Set project-level again to test the user-level reset`, async () => { - await projectSettingsTab.click() - await themeColorSetting.fill(settingValues.project) - await userSettingsTab.click() + // Go to the hotkey for Command Palette. + const commandPalette = page.getByText('Toggle Command Palette') + await commandPalette.scrollIntoViewIfNeeded() + + // The heading is above it and should be in view now. + const commandPaletteHeading = page.getByRole('heading', { + name: 'Command Palette', }) + // The hotkey is in a kbd element next to the heading. + const hotkey = commandPaletteHeading.locator('+ div kbd') + const text = process.platform === 'darwin' ? 'Command+K' : 'Control+K' + await expect(hotkey).toHaveText(text) }) - await test.step('Reset user settings', async () => { - // Click the reset settings button. - await resetButton('user').click() - - await expect(resetToast('user')).toBeVisible() - await expect(resetToast('user')).not.toBeVisible() - - // Verify it is now set to the default value - await expect(themeColorSetting).toHaveValue(settingValues.default) - - await test.step(`Check that the project settings did not change`, async () => { - await projectSettingsTab.click() - await expect(themeColorSetting).toHaveValue(settingValues.project) + test('Project and user settings can be reset', async ({ + page, + homePage, + }) => { + const u = await getUtils(page) + await test.step(`Setup`, async () => { + await page.setBodyDimensions({ width: 1200, height: 500 }) + await homePage.goToModelingScene() + await u.waitForPageLoad() + await page.waitForTimeout(1000) }) - }) - }) - - test( - `Project settings override user settings on desktop`, - { tag: ['@electron', '@skipWin'] }, - async ({ context, page }, testInfo) => { - test.fixme(orRunWhenFullSuiteEnabled()) - const projectName = 'bracket' - const { dir: projectDirName } = await context.folderSetupFn( - async (dir) => { - const bracketDir = join(dir, projectName) - await fsp.mkdir(bracketDir, { recursive: true }) - await fsp.copyFile( - executorInputPath('cylinder-inches.kcl'), - join(bracketDir, 'main.kcl') - ) - } - ) - - await page.setBodyDimensions({ width: 1200, height: 500 }) // Selectors and constants - const tempProjectSettingsFilePath = join( - projectDirName, - projectName, - PROJECT_SETTINGS_FILE_NAME - ) - const tempUserSettingsFilePath = join(projectDirName, SETTINGS_FILE_NAME) - const userThemeColor = '120' - const projectThemeColor = '50' - const settingsOpenButton = page.getByRole('link', { - name: 'settings Settings', - }) - const themeColorSetting = page.locator('#themeColor').getByRole('slider') const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) const userSettingsTab = page.getByRole('radio', { name: 'User' }) - const settingsCloseButton = page.getByTestId('settings-close-button') - const projectLink = page.getByText('bracket') - const logoLink = page.getByTestId('app-logo') - - await test.step('Set user theme color on home', async () => { - await expect(settingsOpenButton).toBeVisible() - await settingsOpenButton.click() - // The user tab should be selected by default on home - await expect(userSettingsTab).toBeChecked() - await themeColorSetting.fill(userThemeColor) - await expect(logoLink).toHaveCSS('--primary-hue', userThemeColor) - await settingsCloseButton.click() - await expect - .poll(async () => fsp.readFile(tempUserSettingsFilePath, 'utf-8'), { - message: 'Setting should now be written to the file', - timeout: 5_000, - }) - .toContain(`themeColor = "${userThemeColor}"`) - }) - - await test.step('Set project theme color', async () => { - // Open the project - await projectLink.click() - await settingsOpenButton.click() - // The project tab should be selected by default within a project - await expect(projectSettingsTab).toBeChecked() - await themeColorSetting.fill(projectThemeColor) - await expect(logoLink).toHaveCSS('--primary-hue', projectThemeColor) - await settingsCloseButton.click() - // Make sure that the project settings file has been written to before continuing - await expect - .poll( - async () => fsp.readFile(tempProjectSettingsFilePath, 'utf-8'), - { - message: 'Setting should now be written to the file', - timeout: 5_000, - } - ) - .toContain(`themeColor = "${projectThemeColor}"`) - }) - - await test.step('Refresh the application and see project setting applied', async () => { - // Make sure we're done navigating before we reload - await expect(settingsCloseButton).not.toBeVisible() - - await page.reload({ waitUntil: 'domcontentloaded' }) - await expect(logoLink).toHaveCSS('--primary-hue', projectThemeColor) - }) - - await test.step(`Navigate back to the home view and see user setting applied`, async () => { - await logoLink.click() - await expect(logoLink).toHaveCSS('--primary-hue', userThemeColor) - }) - } - ) - - test( - `Load desktop app with no settings file`, - { - tag: '@electron', - }, - async ({ page }, testInfo) => { - await page.setBodyDimensions({ width: 1200, height: 500 }) - - // Selectors and constants - const errorHeading = page.getByRole('heading', { - name: 'An unexpected error occurred', - }) - const projectDirLink = page.getByText('Loaded from') - - // If the app loads without exploding we're in the clear - await expect(errorHeading).not.toBeVisible() - await expect(projectDirLink).toBeVisible() - } - ) - - test( - `Load desktop app with a settings file, but no project directory setting`, - { - tag: '@electron', - }, - async ({ context, page, tronApp }, testInfo) => { - if (!tronApp) { - fail() + const resetButton = (level: SettingsLevel) => + page.getByRole('button', { + name: `Reset ${level}-level settings`, + }) + const themeColorSetting = page.locator('#themeColor').getByRole('slider') + const settingValues = { + default: '259', + user: '120', + project: '50', } - await tronApp.cleanProjectDir({ - app: { - appearance: { - color: 259, - }, - }, - }) + const resetToast = (level: SettingsLevel) => + page.getByText(`${level}-level settings were reset`) - await page.setBodyDimensions({ width: 1200, height: 500 }) - - // Selectors and constants - const errorHeading = page.getByRole('heading', { - name: 'An unexpected error occurred', - }) - const projectDirLink = page.getByText('Loaded from') - - // If the app loads without exploding we're in the clear - await expect(errorHeading).not.toBeVisible() - await expect(projectDirLink).toBeVisible() - } - ) - - // It was much easier to test the logo color than the background stream color. - test( - 'user settings reload on external change, on project and modeling view', - { - tag: '@electron', - }, - async ({ context, page, tronApp }, testInfo) => { - test.fixme(orRunWhenFullSuiteEnabled()) - if (!tronApp) { - fail() - } - - await tronApp.cleanProjectDir({ - app: { - appearance: { - // Doesn't matter what you set it to. It will - // default to 264.5 - - color: 0, - }, - }, - }) - - const { dir: projectDirName } = await context.folderSetupFn( - async () => {} - ) - - await page.setBodyDimensions({ width: 1200, height: 500 }) - - const logoLink = page.getByTestId('app-logo') - const projectDirLink = page.getByText('Loaded from') - - await test.step('Wait for project view', async () => { - await expect(projectDirLink).toBeVisible() - await expect(logoLink).toHaveCSS('--primary-hue', '264.5') - }) - - const changeColor = async (color: string) => { - const tempSettingsFilePath = join(projectDirName, SETTINGS_FILE_NAME) - let tomlStr = await fsp.readFile(tempSettingsFilePath, 'utf-8') - tomlStr = tomlStr.replace(/(themeColor = ")[0-9]+(")/, `$1${color}$2`) - await fsp.writeFile(tempSettingsFilePath, tomlStr) - } - - await test.step('Check color of logo changed', async () => { - await changeColor('99') - await expect(logoLink).toHaveCSS('--primary-hue', '99') - }) - - await test.step('Check color of logo changed when in modeling view', async () => { - await createProject({ name: 'project-000', page }) - await changeColor('58') - await expect(logoLink).toHaveCSS('--primary-hue', '58') - }) - - await test.step('Check going back to projects view still changes the color', async () => { - await logoLink.click() - await expect(projectDirLink).toBeVisible() - await changeColor('21') - await expect(logoLink).toHaveCSS('--primary-hue', '21') - }) - } - ) - - test( - 'project settings reload on external change', - { tag: '@electron' }, - async ({ context, page }, testInfo) => { - test.fixme(orRunWhenFullSuiteEnabled()) - const { dir: projectDirName } = await context.folderSetupFn( - async () => {} - ) - - await page.setBodyDimensions({ width: 1200, height: 500 }) - - const logoLink = page.getByTestId('app-logo') - const projectDirLink = page.getByText('Loaded from') - - await test.step('Wait for project view', async () => { - await expect(projectDirLink).toBeVisible() - }) - - await createProject({ name: 'project-000', page }) - - const changeColorFs = async (color: string) => { - const tempSettingsFilePath = join( - projectDirName, - 'project-000', - PROJECT_SETTINGS_FILE_NAME - ) - await fsp.writeFile( - tempSettingsFilePath, - `[settings.app]\nthemeColor = "${color}"` - ) - } - - await test.step('Check the color is first starting as we expect', async () => { - await expect(logoLink).toHaveCSS('--primary-hue', '264.5') - }) - - await test.step('Check color of logo changed', async () => { - await changeColorFs('99') - await expect(logoLink).toHaveCSS('--primary-hue', '99') - }) - } - ) - - test( - `Closing settings modal should go back to the original file being viewed`, - { tag: '@electron' }, - async ({ context, page }, testInfo) => { - // TODO: fix this test on windows after the electron migration - test.skip(process.platform === 'win32', 'Skip on windows') - await context.folderSetupFn(async (dir) => { - const bracketDir = join(dir, 'project-000') - await fsp.mkdir(bracketDir, { recursive: true }) - await fsp.copyFile( - executorInputPath('cube.kcl'), - join(bracketDir, 'main.kcl') - ) - await fsp.copyFile( - executorInputPath('cylinder.kcl'), - join(bracketDir, '2.kcl') - ) - }) - const kclCube = await fsp.readFile(executorInputPath('cube.kcl'), 'utf-8') - const kclCylinder = await fsp.readFile( - executorInputPath('cylinder.kcl'), - 'utf8' - ) - - const { - openKclCodePanel, - openFilePanel, - waitForPageLoad, - selectFile, - editorTextMatches, - } = await getUtils(page, test) - - await page.setBodyDimensions({ width: 1200, height: 500 }) - page.on('console', console.log) - - await test.step('Precondition: Open to second project file', async () => { - await expect(page.getByTestId('home-section')).toBeVisible() - await page.getByText('project-000').click() - await waitForPageLoad() - await openKclCodePanel() - await openFilePanel() - await editorTextMatches(kclCube) - - await selectFile('2.kcl') - await editorTextMatches(kclCylinder) - }) - - const settingsOpenButton = page.getByRole('link', { - name: 'settings Settings', - }) - const settingsCloseButton = page.getByTestId('settings-close-button') - - await test.step('Open and close settings', async () => { - await settingsOpenButton.click() + await test.step(`Open the settings modal`, async () => { + await page.getByRole('link', { name: 'Settings' }).last().click() await expect( page.getByRole('heading', { name: 'Settings', exact: true }) ).toBeVisible() - await settingsCloseButton.click() }) - await test.step('Postcondition: Same file content is in editor as before settings opened', async () => { - await editorTextMatches(kclCylinder) - }) - } - ) + await test.step('Set up theme color', async () => { + // Verify we're looking at the project-level settings, + // and it's set to default value + await expect(projectSettingsTab).toBeChecked() + await expect(themeColorSetting).toHaveValue(settingValues.default) - test('Changing modeling default unit', async ({ page, homePage }) => { - await test.step(`Test setup`, async () => { - // TODO: fix this test on windows after the electron migration - test.skip(process.platform === 'win32', 'Skip on windows') - await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() - const toastMessage = page.getByText(`Successfully created "testDefault"`) - await expect(toastMessage).not.toBeVisible() - await page - .getByRole('button', { name: 'Start Sketch' }) - .waitFor({ state: 'visible' }) + // Set project-level value to 50 + await themeColorSetting.fill(settingValues.project) + + // Set user-level value to 120 + await userSettingsTab.click() + await themeColorSetting.fill(settingValues.user) + await projectSettingsTab.click() + }) + + await test.step('Reset project settings', async () => { + // Click the reset settings button. + await resetButton('project').click() + + await expect(resetToast('project')).toBeVisible() + await expect(resetToast('project')).not.toBeVisible() + + // Verify it is now set to the inherited user value + await expect(themeColorSetting).toHaveValue(settingValues.user) + + await test.step(`Check that the user settings did not change`, async () => { + await userSettingsTab.click() + await expect(themeColorSetting).toHaveValue(settingValues.user) + }) + + await test.step(`Set project-level again to test the user-level reset`, async () => { + await projectSettingsTab.click() + await themeColorSetting.fill(settingValues.project) + await userSettingsTab.click() + }) + }) + + await test.step('Reset user settings', async () => { + // Click the reset settings button. + await resetButton('user').click() + + await expect(resetToast('user')).toBeVisible() + await expect(resetToast('user')).not.toBeVisible() + + // Verify it is now set to the default value + await expect(themeColorSetting).toHaveValue(settingValues.default) + + await test.step(`Check that the project settings did not change`, async () => { + await projectSettingsTab.click() + await expect(themeColorSetting).toHaveValue(settingValues.project) + }) + }) }) - // Selectors and constants - const userSettingsTab = page.getByRole('radio', { name: 'User' }) - const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) - const defaultUnitSection = page.getByText( - 'default unitRoll back default unitRoll back to match' + test( + `Project settings override user settings on desktop`, + { tag: ['@electron'] }, + async ({ context, page }, testInfo) => { + test.fixme(orRunWhenFullSuiteEnabled()) + const projectName = 'bracket' + const { dir: projectDirName } = await context.folderSetupFn( + async (dir) => { + const bracketDir = join(dir, projectName) + await fsp.mkdir(bracketDir, { recursive: true }) + await fsp.copyFile( + executorInputPath('cylinder-inches.kcl'), + join(bracketDir, 'main.kcl') + ) + } + ) + + await page.setBodyDimensions({ width: 1200, height: 500 }) + + // Selectors and constants + const tempProjectSettingsFilePath = join( + projectDirName, + projectName, + PROJECT_SETTINGS_FILE_NAME + ) + const tempUserSettingsFilePath = join( + projectDirName, + SETTINGS_FILE_NAME + ) + const userThemeColor = '120' + const projectThemeColor = '50' + const settingsOpenButton = page.getByRole('link', { + name: 'settings Settings', + }) + const themeColorSetting = page + .locator('#themeColor') + .getByRole('slider') + const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) + const userSettingsTab = page.getByRole('radio', { name: 'User' }) + const settingsCloseButton = page.getByTestId('settings-close-button') + const projectLink = page.getByText('bracket') + const logoLink = page.getByTestId('app-logo') + + await test.step('Set user theme color on home', async () => { + await expect(settingsOpenButton).toBeVisible() + await settingsOpenButton.click() + // The user tab should be selected by default on home + await expect(userSettingsTab).toBeChecked() + await themeColorSetting.fill(userThemeColor) + await expect(logoLink).toHaveCSS('--primary-hue', userThemeColor) + await settingsCloseButton.click() + await expect + .poll(async () => fsp.readFile(tempUserSettingsFilePath, 'utf-8'), { + message: 'Setting should now be written to the file', + timeout: 5_000, + }) + .toContain(`themeColor = "${userThemeColor}"`) + }) + + await test.step('Set project theme color', async () => { + // Open the project + await projectLink.click() + await settingsOpenButton.click() + // The project tab should be selected by default within a project + await expect(projectSettingsTab).toBeChecked() + await themeColorSetting.fill(projectThemeColor) + await expect(logoLink).toHaveCSS('--primary-hue', projectThemeColor) + await settingsCloseButton.click() + // Make sure that the project settings file has been written to before continuing + await expect + .poll( + async () => fsp.readFile(tempProjectSettingsFilePath, 'utf-8'), + { + message: 'Setting should now be written to the file', + timeout: 5_000, + } + ) + .toContain(`themeColor = "${projectThemeColor}"`) + }) + + await test.step('Refresh the application and see project setting applied', async () => { + // Make sure we're done navigating before we reload + await expect(settingsCloseButton).not.toBeVisible() + + await page.reload({ waitUntil: 'domcontentloaded' }) + await expect(logoLink).toHaveCSS('--primary-hue', projectThemeColor) + }) + + await test.step(`Navigate back to the home view and see user setting applied`, async () => { + await logoLink.click() + await expect(logoLink).toHaveCSS('--primary-hue', userThemeColor) + }) + } ) - const defaultUnitRollbackButton = page.getByRole('button', { - name: 'Roll back default unit', - }) - await test.step(`Open the settings modal`, async () => { - await page.getByRole('link', { name: 'Settings' }).last().click() - await expect( - page.getByRole('heading', { name: 'Settings', exact: true }) - ).toBeVisible() - }) + test( + `Load desktop app with no settings file`, + { + tag: '@electron', + }, + async ({ page }, testInfo) => { + await page.setBodyDimensions({ width: 1200, height: 500 }) - await test.step(`Reset unit setting`, async () => { + // Selectors and constants + const errorHeading = page.getByRole('heading', { + name: 'An unexpected error occurred', + }) + const projectDirLink = page.getByText('Loaded from') + + // If the app loads without exploding we're in the clear + await expect(errorHeading).not.toBeVisible() + await expect(projectDirLink).toBeVisible() + } + ) + + test( + `Load desktop app with a settings file, but no project directory setting`, + { + tag: '@electron', + }, + async ({ context, page, tronApp }, testInfo) => { + if (!tronApp) { + fail() + } + await tronApp.cleanProjectDir({ + app: { + appearance: { + color: 259, + }, + }, + }) + + await page.setBodyDimensions({ width: 1200, height: 500 }) + + // Selectors and constants + const errorHeading = page.getByRole('heading', { + name: 'An unexpected error occurred', + }) + const projectDirLink = page.getByText('Loaded from') + + // If the app loads without exploding we're in the clear + await expect(errorHeading).not.toBeVisible() + await expect(projectDirLink).toBeVisible() + } + ) + + // It was much easier to test the logo color than the background stream color. + test( + 'user settings reload on external change, on project and modeling view', + { + tag: '@electron', + }, + async ({ context, page, tronApp }, testInfo) => { + test.fixme(orRunWhenFullSuiteEnabled()) + if (!tronApp) { + fail() + } + + await tronApp.cleanProjectDir({ + app: { + appearance: { + // Doesn't matter what you set it to. It will + // default to 264.5 + + color: 0, + }, + }, + }) + + const { dir: projectDirName } = await context.folderSetupFn( + async () => {} + ) + + await page.setBodyDimensions({ width: 1200, height: 500 }) + + const logoLink = page.getByTestId('app-logo') + const projectDirLink = page.getByText('Loaded from') + + await test.step('Wait for project view', async () => { + await expect(projectDirLink).toBeVisible() + await expect(logoLink).toHaveCSS('--primary-hue', '264.5') + }) + + const changeColor = async (color: string) => { + const tempSettingsFilePath = join(projectDirName, SETTINGS_FILE_NAME) + let tomlStr = await fsp.readFile(tempSettingsFilePath, 'utf-8') + tomlStr = tomlStr.replace(/(themeColor = ")[0-9]+(")/, `$1${color}$2`) + await fsp.writeFile(tempSettingsFilePath, tomlStr) + } + + await test.step('Check color of logo changed', async () => { + await changeColor('99') + await expect(logoLink).toHaveCSS('--primary-hue', '99') + }) + + await test.step('Check color of logo changed when in modeling view', async () => { + await createProject({ name: 'project-000', page }) + await changeColor('58') + await expect(logoLink).toHaveCSS('--primary-hue', '58') + }) + + await test.step('Check going back to projects view still changes the color', async () => { + await logoLink.click() + await expect(projectDirLink).toBeVisible() + await changeColor('21') + await expect(logoLink).toHaveCSS('--primary-hue', '21') + }) + } + ) + + test( + 'project settings reload on external change', + { tag: '@electron' }, + async ({ context, page }, testInfo) => { + test.fixme(orRunWhenFullSuiteEnabled()) + const { dir: projectDirName } = await context.folderSetupFn( + async () => {} + ) + + await page.setBodyDimensions({ width: 1200, height: 500 }) + + const logoLink = page.getByTestId('app-logo') + const projectDirLink = page.getByText('Loaded from') + + await test.step('Wait for project view', async () => { + await expect(projectDirLink).toBeVisible() + }) + + await createProject({ name: 'project-000', page }) + + const changeColorFs = async (color: string) => { + const tempSettingsFilePath = join( + projectDirName, + 'project-000', + PROJECT_SETTINGS_FILE_NAME + ) + await fsp.writeFile( + tempSettingsFilePath, + `[settings.app]\nthemeColor = "${color}"` + ) + } + + await test.step('Check the color is first starting as we expect', async () => { + await expect(logoLink).toHaveCSS('--primary-hue', '264.5') + }) + + await test.step('Check color of logo changed', async () => { + await changeColorFs('99') + await expect(logoLink).toHaveCSS('--primary-hue', '99') + }) + } + ) + + test( + `Closing settings modal should go back to the original file being viewed`, + { tag: '@electron' }, + async ({ context, page }, testInfo) => { + // TODO: fix this test on windows after the electron migration + test.skip(process.platform === 'win32', 'Skip on windows') + await context.folderSetupFn(async (dir) => { + const bracketDir = join(dir, 'project-000') + await fsp.mkdir(bracketDir, { recursive: true }) + await fsp.copyFile( + executorInputPath('cube.kcl'), + join(bracketDir, 'main.kcl') + ) + await fsp.copyFile( + executorInputPath('cylinder.kcl'), + join(bracketDir, '2.kcl') + ) + }) + const kclCube = await fsp.readFile( + executorInputPath('cube.kcl'), + 'utf-8' + ) + const kclCylinder = await fsp.readFile( + executorInputPath('cylinder.kcl'), + 'utf8' + ) + + const { + openKclCodePanel, + openFilePanel, + waitForPageLoad, + selectFile, + editorTextMatches, + } = await getUtils(page, test) + + await page.setBodyDimensions({ width: 1200, height: 500 }) + page.on('console', console.log) + + await test.step('Precondition: Open to second project file', async () => { + await expect(page.getByTestId('home-section')).toBeVisible() + await page.getByText('project-000').click() + await waitForPageLoad() + await openKclCodePanel() + await openFilePanel() + await editorTextMatches(kclCube) + + await selectFile('2.kcl') + await editorTextMatches(kclCylinder) + }) + + const settingsOpenButton = page.getByRole('link', { + name: 'settings Settings', + }) + const settingsCloseButton = page.getByTestId('settings-close-button') + + await test.step('Open and close settings', async () => { + await settingsOpenButton.click() + await expect( + page.getByRole('heading', { name: 'Settings', exact: true }) + ).toBeVisible() + await settingsCloseButton.click() + }) + + await test.step('Postcondition: Same file content is in editor as before settings opened', async () => { + await editorTextMatches(kclCylinder) + }) + } + ) + + test('Changing modeling default unit', async ({ page, homePage }) => { + await test.step(`Test setup`, async () => { + // TODO: fix this test on windows after the electron migration + test.skip(process.platform === 'win32', 'Skip on windows') + await page.setBodyDimensions({ width: 1200, height: 500 }) + await homePage.goToModelingScene() + const toastMessage = page.getByText( + `Successfully created "testDefault"` + ) + await expect(toastMessage).not.toBeVisible() + await page + .getByRole('button', { name: 'Start Sketch' }) + .waitFor({ state: 'visible' }) + }) + + // Selectors and constants + const userSettingsTab = page.getByRole('radio', { name: 'User' }) + const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) + const defaultUnitSection = page.getByText( + 'default unitRoll back default unitRoll back to match' + ) + const defaultUnitRollbackButton = page.getByRole('button', { + name: 'Roll back default unit', + }) + + await test.step(`Open the settings modal`, async () => { + await page.getByRole('link', { name: 'Settings' }).last().click() + await expect( + page.getByRole('heading', { name: 'Settings', exact: true }) + ).toBeVisible() + }) + + await test.step(`Reset unit setting`, async () => { + await userSettingsTab.click() + await defaultUnitSection.hover() + await defaultUnitRollbackButton.click() + await projectSettingsTab.hover() + await projectSettingsTab.click() + await page.waitForTimeout(1000) + }) + + await test.step('Change modeling default unit within project tab', async () => { + const changeUnitOfMeasureInProjectTab = async ( + unitOfMeasure: string + ) => { + await test.step(`Set modeling default unit to ${unitOfMeasure}`, async () => { + await page + .getByTestId('modeling-defaultUnit') + .selectOption(`${unitOfMeasure}`) + const toastMessage = page.getByText( + `Set default unit to "${unitOfMeasure}" for this project` + ) + + // Assert visibility and disapperance + await expect(toastMessage).toBeVisible() + await expect(toastMessage).not.toBeVisible() + }) + } + await changeUnitOfMeasureInProjectTab('in') + await changeUnitOfMeasureInProjectTab('ft') + await changeUnitOfMeasureInProjectTab('yd') + await changeUnitOfMeasureInProjectTab('mm') + await changeUnitOfMeasureInProjectTab('cm') + await changeUnitOfMeasureInProjectTab('m') + }) + + // Go to the user tab + await userSettingsTab.hover() await userSettingsTab.click() - await defaultUnitSection.hover() - await defaultUnitRollbackButton.click() - await projectSettingsTab.hover() - await projectSettingsTab.click() await page.waitForTimeout(1000) - }) - await test.step('Change modeling default unit within project tab', async () => { - const changeUnitOfMeasureInProjectTab = async (unitOfMeasure: string) => { - await test.step(`Set modeling default unit to ${unitOfMeasure}`, async () => { - await page - .getByTestId('modeling-defaultUnit') - .selectOption(`${unitOfMeasure}`) + await test.step('Change modeling default unit within user tab', async () => { + const changeUnitOfMeasureInUserTab = async (unitOfMeasure: string) => { + await test.step(`Set modeling default unit to ${unitOfMeasure}`, async () => { + await page + .getByTestId('modeling-defaultUnit') + .selectOption(`${unitOfMeasure}`) + const toastMessage = page.getByText( + `Set default unit to "${unitOfMeasure}" as a user default` + ) + await expect(toastMessage).toBeVisible() + await expect(toastMessage).not.toBeVisible() + }) + } + await changeUnitOfMeasureInUserTab('in') + await changeUnitOfMeasureInUserTab('ft') + await changeUnitOfMeasureInUserTab('yd') + await changeUnitOfMeasureInUserTab('mm') + await changeUnitOfMeasureInUserTab('cm') + await changeUnitOfMeasureInUserTab('m') + }) + + // Close settings + const settingsCloseButton = page.getByTestId('settings-close-button') + await settingsCloseButton.click() + + await test.step('Change modeling default unit within command bar', async () => { + const commands = page.getByRole('button', { name: 'Commands' }) + const changeUnitOfMeasureInCommandBar = async ( + unitOfMeasure: string + ) => { + // Open command bar + await commands.click() + const settingsModelingDefaultUnitCommand = page.getByText( + 'Settings · modeling · default unit' + ) + await settingsModelingDefaultUnitCommand.click() + + const commandOption = page.getByRole('option', { + name: unitOfMeasure, + exact: true, + }) + await commandOption.click() + const toastMessage = page.getByText( `Set default unit to "${unitOfMeasure}" for this project` ) - - // Assert visibility and disapperance await expect(toastMessage).toBeVisible() - await expect(toastMessage).not.toBeVisible() - }) - } - await changeUnitOfMeasureInProjectTab('in') - await changeUnitOfMeasureInProjectTab('ft') - await changeUnitOfMeasureInProjectTab('yd') - await changeUnitOfMeasureInProjectTab('mm') - await changeUnitOfMeasureInProjectTab('cm') - await changeUnitOfMeasureInProjectTab('m') - }) + } + await changeUnitOfMeasureInCommandBar('in') + await changeUnitOfMeasureInCommandBar('ft') + await changeUnitOfMeasureInCommandBar('yd') + await changeUnitOfMeasureInCommandBar('mm') + await changeUnitOfMeasureInCommandBar('cm') + await changeUnitOfMeasureInCommandBar('m') + }) - // Go to the user tab - await userSettingsTab.hover() - await userSettingsTab.click() - await page.waitForTimeout(1000) - - await test.step('Change modeling default unit within user tab', async () => { - const changeUnitOfMeasureInUserTab = async (unitOfMeasure: string) => { - await test.step(`Set modeling default unit to ${unitOfMeasure}`, async () => { - await page - .getByTestId('modeling-defaultUnit') - .selectOption(`${unitOfMeasure}`) + await test.step('Change modeling default unit within gizmo', async () => { + const changeUnitOfMeasureInGizmo = async ( + unitOfMeasure: string, + copy: string + ) => { + const gizmo = page.getByRole('button', { + name: 'Current units are: ', + }) + await gizmo.click() + const button = page.locator('ul').getByRole('button', { + name: copy, + exact: true, + }) + await button.click() const toastMessage = page.getByText( - `Set default unit to "${unitOfMeasure}" as a user default` + `Updated per-file units to ${unitOfMeasure}` ) await expect(toastMessage).toBeVisible() - await expect(toastMessage).not.toBeVisible() - }) - } - await changeUnitOfMeasureInUserTab('in') - await changeUnitOfMeasureInUserTab('ft') - await changeUnitOfMeasureInUserTab('yd') - await changeUnitOfMeasureInUserTab('mm') - await changeUnitOfMeasureInUserTab('cm') - await changeUnitOfMeasureInUserTab('m') + } + + await changeUnitOfMeasureInGizmo('ft', 'Feet') + await changeUnitOfMeasureInGizmo('in', 'Inches') + await changeUnitOfMeasureInGizmo('yd', 'Yards') + await changeUnitOfMeasureInGizmo('mm', 'Millimeters') + await changeUnitOfMeasureInGizmo('cm', 'Centimeters') + await changeUnitOfMeasureInGizmo('m', 'Meters') + }) }) - // Close settings - const settingsCloseButton = page.getByTestId('settings-close-button') - await settingsCloseButton.click() - - await test.step('Change modeling default unit within command bar', async () => { - const commands = page.getByRole('button', { name: 'Commands' }) - const changeUnitOfMeasureInCommandBar = async (unitOfMeasure: string) => { - // Open command bar - await commands.click() - const settingsModelingDefaultUnitCommand = page.getByText( - 'Settings · modeling · default unit' - ) - await settingsModelingDefaultUnitCommand.click() - - const commandOption = page.getByRole('option', { - name: unitOfMeasure, - exact: true, - }) - await commandOption.click() - - const toastMessage = page.getByText( - `Set default unit to "${unitOfMeasure}" for this project` - ) - await expect(toastMessage).toBeVisible() - } - await changeUnitOfMeasureInCommandBar('in') - await changeUnitOfMeasureInCommandBar('ft') - await changeUnitOfMeasureInCommandBar('yd') - await changeUnitOfMeasureInCommandBar('mm') - await changeUnitOfMeasureInCommandBar('cm') - await changeUnitOfMeasureInCommandBar('m') - }) - - await test.step('Change modeling default unit within gizmo', async () => { - const changeUnitOfMeasureInGizmo = async ( - unitOfMeasure: string, - copy: string - ) => { - const gizmo = page.getByRole('button', { - name: 'Current units are: ', - }) - await gizmo.click() - const button = page.locator('ul').getByRole('button', { - name: copy, - exact: true, - }) - await button.click() - const toastMessage = page.getByText( - `Updated per-file units to ${unitOfMeasure}` - ) - await expect(toastMessage).toBeVisible() - } - - await changeUnitOfMeasureInGizmo('ft', 'Feet') - await changeUnitOfMeasureInGizmo('in', 'Inches') - await changeUnitOfMeasureInGizmo('yd', 'Yards') - await changeUnitOfMeasureInGizmo('mm', 'Millimeters') - await changeUnitOfMeasureInGizmo('cm', 'Centimeters') - await changeUnitOfMeasureInGizmo('m', 'Meters') - }) - }) - - test('Changing theme in sketch mode', async ({ - context, - page, - homePage, - toolbar, - scene, - cmdBar, - }) => { - // TODO: fix this test on windows after the electron migration - test.skip(process.platform === 'win32', 'Skip on windows') - const u = await getUtils(page) - await context.addInitScript(() => { - localStorage.setItem( - 'persistCode', - `sketch001 = startSketchOn(XZ) + test('Changing theme in sketch mode', async ({ + context, + page, + homePage, + toolbar, + scene, + cmdBar, + }) => { + // TODO: fix this test on windows after the electron migration + test.skip(process.platform === 'win32', 'Skip on windows') + const u = await getUtils(page) + await context.addInitScript(() => { + localStorage.setItem( + 'persistCode', + `sketch001 = startSketchOn(XZ) |> startProfileAt([0, 0], %) |> line(end = [5, 0]) |> line(end = [0, 5]) @@ -753,308 +780,312 @@ test.describe('Testing settings', () => { |> close() extrude001 = extrude(sketch001, length = 5) ` - ) - }) - await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() - await expect(toolbar.startSketchBtn).toBeEnabled({ timeout: 15_000 }) - await scene.settled(cmdBar) - await page.waitForTimeout(1000) - - // Selectors and constants - const lineToolButton = page.getByTestId('line') - const segmentOverlays = page.getByTestId('segment-overlay') - const sketchOriginLocation = { x: 600, y: 250 } - const darkThemeSegmentColor: [number, number, number] = [215, 215, 215] - const lightThemeSegmentColor: [number, number, number] = [90, 90, 90] - - await test.step(`Get into sketch mode`, async () => { - await page.mouse.click(700, 200) - await toolbar.editSketch() - - // We use the line tool as a proxy for sketch mode - await expect(lineToolButton).toBeVisible() - await expect(segmentOverlays).toHaveCount(4) - // but we allow more time to pass for animating to the sketch - await page.waitForTimeout(1000) - }) - - await test.step(`Check the sketch line color before`, async () => { - await expect - .poll(() => - u.getGreatestPixDiff(sketchOriginLocation, darkThemeSegmentColor) ) - .toBeLessThan(15) - }) - - await test.step(`Change theme to light using command palette`, async () => { - await page.keyboard.press('ControlOrMeta+K') - await page.getByRole('option', { name: 'theme' }).click() - await page.getByRole('option', { name: 'light' }).click() - await expect(page.getByText('theme to "light"')).toBeVisible() - - // Make sure we haven't left sketch mode - await expect(lineToolButton).toBeVisible() - }) - - await test.step(`Check the sketch line color after`, async () => { - await expect - .poll(() => - u.getGreatestPixDiff(sketchOriginLocation, lightThemeSegmentColor) - ) - .toBeLessThan(15) - }) - }) - - test(`Changing system theme preferences (via media query) should update UI and stream`, async ({ - page, - homePage, - tronApp, - }) => { - if (!tronApp) { - fail() - } - - await tronApp.cleanProjectDir({ - // Override the settings so that the theme is set to `system` - ...TEST_SETTINGS_DEFAULT_THEME, - }) - - const u = await getUtils(page) - - // Selectors and constants - const darkBackgroundCss = 'oklch(0.3012 0 264.5)' - const lightBackgroundCss = 'oklch(0.9911 0 264.5)' - const darkBackgroundColor = TEST_COLORS.DARK_MODE_BKGD - const lightBackgroundColor: [number, number, number] = [245, 245, 245] - const streamBackgroundPixelIsColor = async ( - color: [number, number, number] - ) => { - return u.getGreatestPixDiff({ x: 1000, y: 200 }, color) - } - const toolbar = page.locator('menu').filter({ hasText: 'Start Sketch' }) - - await test.step(`Test setup`, async () => { + }) await page.setBodyDimensions({ width: 1200, height: 500 }) await homePage.goToModelingScene() - await u.waitForPageLoad() + await expect(toolbar.startSketchBtn).toBeEnabled({ timeout: 15_000 }) + await scene.settled(cmdBar) await page.waitForTimeout(1000) - await expect(toolbar).toBeVisible() + + // Selectors and constants + const lineToolButton = page.getByTestId('line') + const segmentOverlays = page.getByTestId('segment-overlay') + const sketchOriginLocation = { x: 600, y: 250 } + const darkThemeSegmentColor: [number, number, number] = [215, 215, 215] + const lightThemeSegmentColor: [number, number, number] = [90, 90, 90] + + await test.step(`Get into sketch mode`, async () => { + await page.mouse.click(700, 200) + await toolbar.editSketch() + + // We use the line tool as a proxy for sketch mode + await expect(lineToolButton).toBeVisible() + await expect(segmentOverlays).toHaveCount(4) + // but we allow more time to pass for animating to the sketch + await page.waitForTimeout(1000) + }) + + await test.step(`Check the sketch line color before`, async () => { + await expect + .poll(() => + u.getGreatestPixDiff(sketchOriginLocation, darkThemeSegmentColor) + ) + .toBeLessThan(15) + }) + + await test.step(`Change theme to light using command palette`, async () => { + await page.keyboard.press('ControlOrMeta+K') + await page.getByRole('option', { name: 'theme' }).click() + await page.getByRole('option', { name: 'light' }).click() + await expect(page.getByText('theme to "light"')).toBeVisible() + + // Make sure we haven't left sketch mode + await expect(lineToolButton).toBeVisible() + }) + + await test.step(`Check the sketch line color after`, async () => { + await expect + .poll(() => + u.getGreatestPixDiff(sketchOriginLocation, lightThemeSegmentColor) + ) + .toBeLessThan(15) + }) }) - await test.step(`Check the background color is light before`, async () => { - await expect(toolbar).toHaveCSS('background-color', lightBackgroundCss) - await expect - .poll(() => streamBackgroundPixelIsColor(lightBackgroundColor)) - .toBeLessThan(15) + test(`Changing system theme preferences (via media query) should update UI and stream`, async ({ + page, + homePage, + tronApp, + }) => { + if (!tronApp) { + fail() + } + + await tronApp.cleanProjectDir({ + // Override the settings so that the theme is set to `system` + ...TEST_SETTINGS_DEFAULT_THEME, + }) + + const u = await getUtils(page) + + // Selectors and constants + const darkBackgroundCss = 'oklch(0.3012 0 264.5)' + const lightBackgroundCss = 'oklch(0.9911 0 264.5)' + const darkBackgroundColor = TEST_COLORS.DARK_MODE_BKGD + const lightBackgroundColor: [number, number, number] = [245, 245, 245] + const streamBackgroundPixelIsColor = async ( + color: [number, number, number] + ) => { + return u.getGreatestPixDiff({ x: 1000, y: 200 }, color) + } + const toolbar = page.locator('menu').filter({ hasText: 'Start Sketch' }) + + await test.step(`Test setup`, async () => { + await page.setBodyDimensions({ width: 1200, height: 500 }) + await homePage.goToModelingScene() + await u.waitForPageLoad() + await page.waitForTimeout(1000) + await expect(toolbar).toBeVisible() + }) + + await test.step(`Check the background color is light before`, async () => { + await expect(toolbar).toHaveCSS('background-color', lightBackgroundCss) + await expect + .poll(() => streamBackgroundPixelIsColor(lightBackgroundColor)) + .toBeLessThan(15) + }) + + await test.step(`Change media query preference to dark, emulating dusk with system theme`, async () => { + await page.emulateMedia({ colorScheme: 'dark' }) + }) + + await test.step(`Check the background color is dark after`, async () => { + await expect(toolbar).toHaveCSS('background-color', darkBackgroundCss) + await expect + .poll(() => streamBackgroundPixelIsColor(darkBackgroundColor)) + .toBeLessThan(15) + }) }) - await test.step(`Change media query preference to dark, emulating dusk with system theme`, async () => { - await page.emulateMedia({ colorScheme: 'dark' }) - }) + test(`Turning off "Show debug panel" with debug panel open leaves no phantom panel`, async ({ + context, + page, + homePage, + tronApp, + scene, + cmdBar, + }) => { + if (!tronApp) { + fail() + } - await test.step(`Check the background color is dark after`, async () => { - await expect(toolbar).toHaveCSS('background-color', darkBackgroundCss) - await expect - .poll(() => streamBackgroundPixelIsColor(darkBackgroundColor)) - .toBeLessThan(15) - }) - }) + await tronApp.cleanProjectDir({ + // Override beforeEach test setup + // with debug panel open + // but "show debug panel" set to false + ...TEST_SETTINGS, + app: { ...TEST_SETTINGS.app, show_debug_panel: false }, + modeling: { ...TEST_SETTINGS.modeling }, + }) - test(`Turning off "Show debug panel" with debug panel open leaves no phantom panel`, async ({ - context, - page, - homePage, - tronApp, - scene, - cmdBar, - }) => { - if (!tronApp) { - fail() - } + const u = await getUtils(page) - await tronApp.cleanProjectDir({ - // Override beforeEach test setup - // with debug panel open - // but "show debug panel" set to false - ...TEST_SETTINGS, - app: { ...TEST_SETTINGS.app, show_debug_panel: false }, - modeling: { ...TEST_SETTINGS.modeling }, - }) - - const u = await getUtils(page) - - await context.addInitScript(async () => { - localStorage.setItem('persistModelingContext', '{"openPanes":["debug"]}') - }) - await page.setBodyDimensions({ width: 1200, height: 500 }) - await homePage.goToModelingScene() - await scene.connectionEstablished() - - // Constants and locators - const resizeHandle = page.locator('.sidebar-resize-handles > div.block') - const debugPaneButton = page.getByTestId('debug-pane-button') - const commandsButton = page.getByRole('button', { name: 'Commands' }) - const debugPaneOption = page.getByRole('option', { - name: 'Settings · app · show debug panel', - }) - - async function setShowDebugPanelTo(value: 'On' | 'Off') { - await commandsButton.click() - await debugPaneOption.scrollIntoViewIfNeeded() - await debugPaneOption.click() - await page.getByRole('option', { name: value }).click() - await expect( - page.getByText( - `Set show debug panel to "${value === 'On'}" for this project` + await context.addInitScript(async () => { + localStorage.setItem( + 'persistModelingContext', + '{"openPanes":["debug"]}' ) - ).toBeVisible() - } + }) + await page.setBodyDimensions({ width: 1200, height: 500 }) + await homePage.goToModelingScene() + await scene.connectionEstablished() - await test.step(`Initial load with corrupted settings`, async () => { - // Check that the debug panel is not visible - await expect(debugPaneButton).not.toBeVisible() - // Check the pane resize handle wrapper is not visible - await expect(resizeHandle).not.toBeVisible() + // Constants and locators + const resizeHandle = page.locator('.sidebar-resize-handles > div.block') + const debugPaneButton = page.getByTestId('debug-pane-button') + const commandsButton = page.getByRole('button', { name: 'Commands' }) + const debugPaneOption = page.getByRole('option', { + name: 'Settings · app · show debug panel', + }) + + async function setShowDebugPanelTo(value: 'On' | 'Off') { + await commandsButton.click() + await debugPaneOption.scrollIntoViewIfNeeded() + await debugPaneOption.click() + await page.getByRole('option', { name: value }).click() + await expect( + page.getByText( + `Set show debug panel to "${value === 'On'}" for this project` + ) + ).toBeVisible() + } + + await test.step(`Initial load with corrupted settings`, async () => { + // Check that the debug panel is not visible + await expect(debugPaneButton).not.toBeVisible() + // Check the pane resize handle wrapper is not visible + await expect(resizeHandle).not.toBeVisible() + }) + + await test.step(`Open code pane to verify we see the resize handles`, async () => { + await u.openKclCodePanel() + await expect(resizeHandle).toBeVisible() + await u.closeKclCodePanel() + }) + + await test.step(`Turn on debug panel, open it`, async () => { + await setShowDebugPanelTo('On') + await expect(debugPaneButton).toBeVisible() + // We want the logic to clear the phantom panel, so we shouldn't see + // the real panel (and therefore the resize handle) yet + await expect(resizeHandle).not.toBeVisible() + await u.openDebugPanel() + await expect(resizeHandle).toBeVisible() + }) + + await test.step(`Turn off debug panel setting with it open`, async () => { + await setShowDebugPanelTo('Off') + await expect(debugPaneButton).not.toBeVisible() + await expect(resizeHandle).not.toBeVisible() + }) }) - await test.step(`Open code pane to verify we see the resize handles`, async () => { - await u.openKclCodePanel() - await expect(resizeHandle).toBeVisible() - await u.closeKclCodePanel() - }) + test(`Change inline units setting`, async ({ + page, + homePage, + context, + editor, + }) => { + const initialInlineUnits = 'yd' + const editedInlineUnits = { short: 'mm', long: 'Millimeters' } + const inlineSettingsString = (s: string) => + `@settings(defaultLengthUnit = ${s})` + const unitsIndicator = page.getByRole('button', { + name: 'Current units are:', + }) + const unitsChangeButton = (name: string) => + page.getByRole('button', { name, exact: true }) - await test.step(`Turn on debug panel, open it`, async () => { - await setShowDebugPanelTo('On') - await expect(debugPaneButton).toBeVisible() - // We want the logic to clear the phantom panel, so we shouldn't see - // the real panel (and therefore the resize handle) yet - await expect(resizeHandle).not.toBeVisible() - await u.openDebugPanel() - await expect(resizeHandle).toBeVisible() - }) + await context.folderSetupFn(async (dir) => { + const bracketDir = join(dir, 'project-000') + await fsp.mkdir(bracketDir, { recursive: true }) + await fsp.copyFile( + executorInputPath('cube.kcl'), + join(bracketDir, 'main.kcl') + ) + }) - await test.step(`Turn off debug panel setting with it open`, async () => { - await setShowDebugPanelTo('Off') - await expect(debugPaneButton).not.toBeVisible() - await expect(resizeHandle).not.toBeVisible() - }) - }) + await test.step(`Initial units from settings are ignored`, async () => { + await homePage.openProject('project-000') + await expect(unitsIndicator).toHaveText('Current units are: mm') + }) - test(`Change inline units setting`, async ({ - page, - homePage, - context, - editor, - }) => { - const initialInlineUnits = 'yd' - const editedInlineUnits = { short: 'mm', long: 'Millimeters' } - const inlineSettingsString = (s: string) => - `@settings(defaultLengthUnit = ${s})` - const unitsIndicator = page.getByRole('button', { - name: 'Current units are:', - }) - const unitsChangeButton = (name: string) => - page.getByRole('button', { name, exact: true }) - - await context.folderSetupFn(async (dir) => { - const bracketDir = join(dir, 'project-000') - await fsp.mkdir(bracketDir, { recursive: true }) - await fsp.copyFile( - executorInputPath('cube.kcl'), - join(bracketDir, 'main.kcl') - ) - }) - - await test.step(`Initial units from settings are ignored`, async () => { - await homePage.openProject('project-000') - await expect(unitsIndicator).toHaveText('Current units are: mm') - }) - - await test.step(`Manually write inline settings`, async () => { - await editor.openPane() - await editor.replaceCode( - `fn cube`, - `${inlineSettingsString(initialInlineUnits)} + await test.step(`Manually write inline settings`, async () => { + await editor.openPane() + await editor.replaceCode( + `fn cube`, + `${inlineSettingsString(initialInlineUnits)} fn cube` - ) - await expect(unitsIndicator).toContainText(initialInlineUnits) + ) + await expect(unitsIndicator).toContainText(initialInlineUnits) + }) + + await test.step(`Change units setting via lower-right control`, async () => { + await unitsIndicator.click() + await unitsChangeButton(editedInlineUnits.long).click() + await expect( + page.getByText(`Updated per-file units to ${editedInlineUnits.short}`) + ).toBeVisible() + }) }) - await test.step(`Change units setting via lower-right control`, async () => { - await unitsIndicator.click() - await unitsChangeButton(editedInlineUnits.long).click() - await expect( - page.getByText(`Updated per-file units to ${editedInlineUnits.short}`) - ).toBeVisible() - }) - }) + /** + * This test assumes that the default value of the "highlight edges" setting is "on". + */ + test(`Toggle stream settings multiple times`, async ({ + page, + scene, + homePage, + context, + toolbar, + cmdBar, + }, testInfo) => { + test.fixme(orRunWhenFullSuiteEnabled()) + await context.folderSetupFn(async (dir) => { + const projectDir = join(dir, 'project-000') + await fsp.mkdir(projectDir, { recursive: true }) + await fsp.copyFile( + executorInputPath('cube.kcl'), + join(projectDir, 'main.kcl') + ) + }) - /** - * This test assumes that the default value of the "highlight edges" setting is "on". - */ - test(`Toggle stream settings multiple times`, async ({ - page, - scene, - homePage, - context, - toolbar, - cmdBar, - }, testInfo) => { - test.fixme(orRunWhenFullSuiteEnabled()) - await context.folderSetupFn(async (dir) => { - const projectDir = join(dir, 'project-000') - await fsp.mkdir(projectDir, { recursive: true }) - await fsp.copyFile( - executorInputPath('cube.kcl'), - join(projectDir, 'main.kcl') - ) - }) + await test.step(`First snapshot`, async () => { + await homePage.openProject('project-000') + await toolbar.closePane('code') + await expect(toolbar.startSketchBtn).toBeEnabled({ timeout: 20_000 }) + await scene.clickNoWhere() + }) - await test.step(`First snapshot`, async () => { - await homePage.openProject('project-000') - await toolbar.closePane('code') - await expect(toolbar.startSketchBtn).toBeEnabled({ timeout: 20_000 }) - await scene.clickNoWhere() - }) + const toast = (value: boolean) => + page.getByText( + `Set highlight edges to "${String(value)}" as a user default` + ) - const toast = (value: boolean) => - page.getByText( - `Set highlight edges to "${String(value)}" as a user default` + await test.step(`Toggle highlightEdges off`, async () => { + await cmdBar.openCmdBar() + await cmdBar.chooseCommand('Settings · modeling · highlight edges') + await cmdBar.selectOption({ name: 'off' }).click() + const falseToast = toast(false) + await expect(falseToast).toBeVisible() + await falseToast.waitFor({ state: 'detached' }) + }) + + await expect(scene.streamWrapper).not.toHaveScreenshot( + 'toggle-settings-initial.png', + { + maxDiffPixels: 15, + mask: networkingMasks(page), + } ) - await test.step(`Toggle highlightEdges off`, async () => { - await cmdBar.openCmdBar() - await cmdBar.chooseCommand('Settings · modeling · highlight edges') - await cmdBar.selectOption({ name: 'off' }).click() - const falseToast = toast(false) - await expect(falseToast).toBeVisible() - await falseToast.waitFor({ state: 'detached' }) + await test.step(`Toggle highlightEdges on`, async () => { + await cmdBar.openCmdBar() + await cmdBar.chooseCommand('Settings · modeling · highlight edges') + await cmdBar.selectOption({ name: 'on' }).click() + const trueToast = toast(true) + await expect(trueToast).toBeVisible() + await trueToast.waitFor({ state: 'detached' }) + }) + + await expect(scene.streamWrapper).toHaveScreenshot( + 'toggle-settings-initial.png', + { + maxDiffPixels: 15, + mask: networkingMasks(page), + } + ) }) - - await expect(scene.streamWrapper).not.toHaveScreenshot( - 'toggle-settings-initial.png', - { - maxDiffPixels: 15, - mask: networkingMasks(page), - } - ) - - await test.step(`Toggle highlightEdges on`, async () => { - await cmdBar.openCmdBar() - await cmdBar.chooseCommand('Settings · modeling · highlight edges') - await cmdBar.selectOption({ name: 'on' }).click() - const trueToast = toast(true) - await expect(trueToast).toBeVisible() - await trueToast.waitFor({ state: 'detached' }) - }) - - await expect(scene.streamWrapper).toHaveScreenshot( - 'toggle-settings-initial.png', - { - maxDiffPixels: 15, - mask: networkingMasks(page), - } - ) - }) -}) + } +) diff --git a/e2e/playwright/text-to-cad-tests.spec.ts b/e2e/playwright/text-to-cad-tests.spec.ts index 7c3a8e0ad..6e0e4c7aa 100644 --- a/e2e/playwright/text-to-cad-tests.spec.ts +++ b/e2e/playwright/text-to-cad-tests.spec.ts @@ -9,7 +9,7 @@ import { } from '@e2e/playwright/test-utils' import { expect, test } from '@e2e/playwright/zoo-test' -test.describe('Text-to-CAD tests', { tag: ['@skipWin'] }, () => { +test.describe('Text-to-CAD tests', () => { test('basic lego happy case', async ({ page, homePage }) => { const u = await getUtils(page) @@ -436,93 +436,92 @@ test.describe('Text-to-CAD tests', { tag: ['@skipWin'] }, () => { }) // This will be fine once greg makes prompt at top of file deterministic - test( - 'can do many at once and get many prompts back, and interact with many', - { tag: ['@skipWin'] }, - async ({ page, homePage }) => { - test.fixme(orRunWhenFullSuiteEnabled()) - // Let this test run longer since we've seen it timeout. - test.setTimeout(180_000) + test('can do many at once and get many prompts back, and interact with many', async ({ + page, + homePage, + }) => { + test.fixme(orRunWhenFullSuiteEnabled()) + // Let this test run longer since we've seen it timeout. + test.setTimeout(180_000) - const u = await getUtils(page) + const u = await getUtils(page) - await page.setBodyDimensions({ width: 1000, height: 500 }) + await page.setBodyDimensions({ width: 1000, height: 500 }) - await homePage.goToModelingScene() - await u.waitForPageLoad() + await homePage.goToModelingScene() + await u.waitForPageLoad() - await sendPromptFromCommandBar(page, 'a 2x4 lego') + await sendPromptFromCommandBar(page, 'a 2x4 lego') - await sendPromptFromCommandBar(page, 'a 2x8 lego') + await sendPromptFromCommandBar(page, 'a 2x8 lego') - await sendPromptFromCommandBar(page, 'a 2x10 lego') + await sendPromptFromCommandBar(page, 'a 2x10 lego') - // Find the toast. - // Look out for the toast message - const submittingToastMessage = page.getByText( - `Submitting to Text-to-CAD API...` - ) - await expect(submittingToastMessage.first()).toBeVisible() + // Find the toast. + // Look out for the toast message + const submittingToastMessage = page.getByText( + `Submitting to Text-to-CAD API...` + ) + await expect(submittingToastMessage.first()).toBeVisible() - const generatingToastMessage = page.getByText( - `Generating parametric model...` - ) - await expect(generatingToastMessage.first()).toBeVisible({ - timeout: 10_000, - }) + const generatingToastMessage = page.getByText( + `Generating parametric model...` + ) + await expect(generatingToastMessage.first()).toBeVisible({ + timeout: 10_000, + }) - const successToastMessage = page.getByText(`Text-to-CAD successful`) - // We should have three success toasts. - await expect(successToastMessage).toHaveCount(3, { timeout: 25_000 }) + const successToastMessage = page.getByText(`Text-to-CAD successful`) + // We should have three success toasts. + await expect(successToastMessage).toHaveCount(3, { timeout: 25_000 }) - await expect(page.getByText(`a 2x4 lego`)).toBeVisible() - await expect(page.getByText(`a 2x8 lego`)).toBeVisible() - await expect(page.getByText(`a 2x10 lego`)).toBeVisible() + await expect(page.getByText(`a 2x4 lego`)).toBeVisible() + await expect(page.getByText(`a 2x8 lego`)).toBeVisible() + await expect(page.getByText(`a 2x10 lego`)).toBeVisible() - // Ensure if you reject one, the others stay. - const rejectButton = page.getByRole('button', { name: 'Reject' }) - await expect(rejectButton.first()).toBeVisible() - // Click the reject button on the first toast. - await rejectButton.first().click() + // Ensure if you reject one, the others stay. + const rejectButton = page.getByRole('button', { name: 'Reject' }) + await expect(rejectButton.first()).toBeVisible() + // Click the reject button on the first toast. + await rejectButton.first().click() - // The first toast should disappear, but not the others. - await expect(page.getByText(`a 2x10 lego`)).not.toBeVisible() - await expect(page.getByText(`a 2x8 lego`)).toBeVisible() - await expect(page.getByText(`a 2x4 lego`)).toBeVisible() + // The first toast should disappear, but not the others. + await expect(page.getByText(`a 2x10 lego`)).not.toBeVisible() + await expect(page.getByText(`a 2x8 lego`)).toBeVisible() + await expect(page.getByText(`a 2x4 lego`)).toBeVisible() - // Ensure you can copy the code for one of the models remaining. - const copyToClipboardButton = page.getByRole('button', { - name: 'Accept', - }) - await expect(copyToClipboardButton.first()).toBeVisible() - // Click the button. - await copyToClipboardButton.first().click() + // Ensure you can copy the code for one of the models remaining. + const copyToClipboardButton = page.getByRole('button', { + name: 'Accept', + }) + await expect(copyToClipboardButton.first()).toBeVisible() + // Click the button. + await copyToClipboardButton.first().click() - // Do NOT do AI tests like this: "Expect the code to be pasted." - // Reason: AI tests are NONDETERMINISTIC. Thus we need to be as most - // general as we can for the assertion. - // We can use Kolmogorov complexity as a measurement of the - // "probably most minimal version of this program" to have a lower - // bound to work with. It is completely by feel because there are - // no proofs that any program is its smallest self. - const code2x8 = await page.locator('.cm-content').innerText() - await expect(code2x8.length).toBeGreaterThan(249) + // Do NOT do AI tests like this: "Expect the code to be pasted." + // Reason: AI tests are NONDETERMINISTIC. Thus we need to be as most + // general as we can for the assertion. + // We can use Kolmogorov complexity as a measurement of the + // "probably most minimal version of this program" to have a lower + // bound to work with. It is completely by feel because there are + // no proofs that any program is its smallest self. + const code2x8 = await page.locator('.cm-content').innerText() + await expect(code2x8.length).toBeGreaterThan(249) - // Ensure the final toast remains. - await expect(page.getByText(`a 2x10 lego`)).not.toBeVisible() - await expect(page.getByText(`Prompt: "a 2x8 lego`)).not.toBeVisible() - await expect(page.getByText(`a 2x4 lego`)).toBeVisible() + // Ensure the final toast remains. + await expect(page.getByText(`a 2x10 lego`)).not.toBeVisible() + await expect(page.getByText(`Prompt: "a 2x8 lego`)).not.toBeVisible() + await expect(page.getByText(`a 2x4 lego`)).toBeVisible() - // Ensure you can copy the code for the final model. - await expect(copyToClipboardButton).toBeVisible() - // Click the button. - await copyToClipboardButton.click() + // Ensure you can copy the code for the final model. + await expect(copyToClipboardButton).toBeVisible() + // Click the button. + await copyToClipboardButton.click() - // Expect the code to be pasted. - const code2x4 = await page.locator('.cm-content').innerText() - await expect(code2x4.length).toBeGreaterThan(249) - } - ) + // Expect the code to be pasted. + const code2x4 = await page.locator('.cm-content').innerText() + await expect(code2x4.length).toBeGreaterThan(249) + }) test('can do many at once with errors, clicking dismiss error does not dismiss all', async ({ page, diff --git a/e2e/playwright/various.spec.ts b/e2e/playwright/various.spec.ts index 28c849391..9382e72f7 100644 --- a/e2e/playwright/various.spec.ts +++ b/e2e/playwright/various.spec.ts @@ -178,6 +178,7 @@ test('Keyboard shortcuts can be viewed through the help menu', async ({ test('First escape in tool pops you out of tool, second exits sketch mode', async ({ page, homePage, + toolbar, }) => { // Wait for the app to be ready for use const u = await getUtils(page) @@ -188,15 +189,6 @@ test('First escape in tool pops you out of tool, second exits sketch mode', asyn await u.expectCmdLog('[data-message-type="execution-done"]') await u.closeDebugPanel() - const lineButton = page.getByRole('button', { - name: 'line Line', - exact: true, - }) - const arcButton = page.getByRole('button', { - name: 'arc Tangential Arc', - exact: true, - }) - // Test these hotkeys perform actions when // focus is on the canvas await page.mouse.move(600, 250) @@ -207,8 +199,8 @@ test('First escape in tool pops you out of tool, second exits sketch mode', asyn await page.mouse.move(800, 300) await page.mouse.click(800, 300) await page.waitForTimeout(1000) - await expect(lineButton).toBeVisible() - await expect(lineButton).toHaveAttribute('aria-pressed', 'true') + await expect(toolbar.lineBtn).toBeVisible() + await expect(toolbar.lineBtn).toHaveAttribute('aria-pressed', 'true') // Draw a line await page.mouse.move(700, 200, { steps: 5 }) @@ -224,10 +216,9 @@ test('First escape in tool pops you out of tool, second exits sketch mode', asyn await page.keyboard.press('Escape') // Make sure we didn't pop out of sketch mode. await expect(page.getByRole('button', { name: 'Exit Sketch' })).toBeVisible() - await expect(lineButton).not.toHaveAttribute('aria-pressed', 'true') + await expect(toolbar.lineBtn).not.toHaveAttribute('aria-pressed', 'true') // Equip arc tool - await page.keyboard.press('a') - await expect(arcButton).toHaveAttribute('aria-pressed', 'true') + await toolbar.selectTangentialArc() // click in the same position again to continue the profile await page.mouse.move(secondMousePosition.x, secondMousePosition.y, { @@ -238,11 +229,14 @@ test('First escape in tool pops you out of tool, second exits sketch mode', asyn await page.mouse.move(1000, 100, { steps: 5 }) await page.mouse.click(1000, 100) await page.keyboard.press('Escape') - await expect(arcButton).toHaveAttribute('aria-pressed', 'false') + await expect(toolbar.tangentialArcBtn).toHaveAttribute( + 'aria-pressed', + 'false' + ) await expect .poll(async () => { await page.keyboard.press('l') - return lineButton.getAttribute('aria-pressed') + return toolbar.lineBtn.getAttribute('aria-pressed') }) .toBe('true') @@ -251,8 +245,11 @@ test('First escape in tool pops you out of tool, second exits sketch mode', asyn // Unequip line tool await page.keyboard.press('Escape') - await expect(lineButton).toHaveAttribute('aria-pressed', 'false') - await expect(arcButton).toHaveAttribute('aria-pressed', 'false') + await expect(toolbar.lineBtn).toHaveAttribute('aria-pressed', 'false') + await expect(toolbar.tangentialArcBtn).toHaveAttribute( + 'aria-pressed', + 'false' + ) // Make sure we didn't pop out of sketch mode. await expect(page.getByRole('button', { name: 'Exit Sketch' })).toBeVisible() // Exit sketch diff --git a/interface.d.ts b/interface.d.ts index 96caab368..66f4cba1f 100644 --- a/interface.d.ts +++ b/interface.d.ts @@ -20,6 +20,7 @@ export interface IElectronAPI { open: typeof dialog.showOpenDialog save: typeof dialog.showSaveDialog openExternal: typeof shell.openExternal + openInNewWindow: (name: string) => void takeElectronWindowScreenshot: ({ width, height, diff --git a/known-circular.txt b/known-circular.txt index a98bc438b..a5d6d3595 100644 --- a/known-circular.txt +++ b/known-circular.txt @@ -3,14 +3,13 @@ > dpdm --no-warning --no-tree -T --skip-dynamic-imports=circular src/index.tsx • Circular Dependencies - 01) src/lang/std/sketch.ts -> src/lang/modifyAst.ts -> src/lang/modifyAst/addEdgeTreatment.ts - 02) src/lang/std/sketch.ts -> src/lang/modifyAst.ts - 03) src/lang/std/sketch.ts -> src/lang/modifyAst.ts -> src/lang/std/sketchcombos.ts - 04) src/lib/singletons.ts -> src/editor/manager.ts -> src/lib/selections.ts - 05) src/lib/singletons.ts -> src/editor/manager.ts -> src/lib/selections.ts -> src/machines/appMachine.ts -> src/machines/engineStreamMachine.ts - 06) src/lib/singletons.ts -> src/editor/manager.ts -> src/lib/selections.ts -> src/machines/appMachine.ts -> src/machines/settingsMachine.ts - 07) src/machines/appMachine.ts -> src/machines/settingsMachine.ts -> src/machines/commandBarMachine.ts -> src/lib/commandBarConfigs/authCommandConfig.ts - 08) src/lib/singletons.ts -> src/lang/codeManager.ts - 09) src/lib/singletons.ts -> src/clientSideScene/sceneEntities.ts -> src/clientSideScene/segments.ts -> src/components/Toolbar/angleLengthInfo.ts - 10) src/hooks/useModelingContext.ts -> src/components/ModelingMachineProvider.tsx -> src/components/Toolbar/Intersect.tsx -> src/components/SetHorVertDistanceModal.tsx -> src/lib/useCalculateKclExpression.ts - 11) src/routes/Onboarding/index.tsx -> src/routes/Onboarding/Camera.tsx -> src/routes/Onboarding/utils.tsx + 1) src/lang/std/sketch.ts -> src/lang/modifyAst.ts -> src/lang/modifyAst/addEdgeTreatment.ts + 2) src/lang/std/sketch.ts -> src/lang/modifyAst.ts + 3) src/lang/std/sketch.ts -> src/lang/modifyAst.ts -> src/lang/std/sketchcombos.ts + 4) src/lib/singletons.ts -> src/editor/manager.ts -> src/lib/selections.ts + 5) src/lib/singletons.ts -> src/editor/manager.ts -> src/lib/selections.ts + 6) src/lib/singletons.ts -> src/lang/codeManager.ts + 7) src/lib/singletons.ts -> src/clientSideScene/sceneEntities.ts -> src/clientSideScene/segments.ts -> src/components/Toolbar/angleLengthInfo.ts + 8) src/lib/singletons.ts -> src/clientSideScene/sceneEntities.ts -> src/clientSideScene/segments.ts + 9) src/hooks/useModelingContext.ts -> src/components/ModelingMachineProvider.tsx -> src/components/Toolbar/Intersect.tsx -> src/components/SetHorVertDistanceModal.tsx -> src/lib/useCalculateKclExpression.ts + 10) src/routes/Onboarding/index.tsx -> src/routes/Onboarding/Camera.tsx -> src/routes/Onboarding/utils.tsx diff --git a/package.json b/package.json index fa6a2d4fb..cb3e6598d 100644 --- a/package.json +++ b/package.json @@ -76,6 +76,7 @@ "yargs": "^17.7.2" }, "scripts": { + "prepare": "husky", "install:rust": "curl --proto '=https' --tlsv1.2 -sSf https://sh.rustup.rs | sh -s -- -y --default-toolchain none && source \"$HOME/.cargo/env\" && (cd rust && (rustup show active-toolchain || rustup toolchain install))", "install:rust:windows": "winget install Microsoft.VisualStudio.2022.Community --silent --override \"--wait --quiet --add ProductLang En-us --add Microsoft.VisualStudio.Workload.NativeDesktop --includeRecommended\" && winget install Rustlang.Rustup", "install:wasm-pack:sh": ". $HOME/.cargo/env && curl https://rustwasm.github.io/wasm-pack/installer/init.sh -sSf | sh -s -- -f", @@ -114,6 +115,7 @@ "circular-deps": "dpdm --no-warning --no-tree -T --skip-dynamic-imports=circular src/index.tsx", "circular-deps:overwrite": "npm run circular-deps | sed '$d' | grep -v '^npm run' > known-circular.txt", "circular-deps:diff": "./scripts/diff-circular-deps.sh", + "circular-deps:diff:nodejs": "npm run circular-deps:diff || node ./scripts/diff.js", "files:set-version": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json", "files:set-notes": "./scripts/set-files-notes.sh", "files:flip-to-nightly": "./scripts/flip-files-to-nightly.sh", @@ -125,7 +127,7 @@ "generate:machine-api": "npx openapi-typescript ./openapi/machine-api.json -o src/lib/machine-api.d.ts", "generate:samples-manifest": "cd public/kcl-samples && node generate-manifest.js", "tron:start": "electron-forge start", - "chrome:test": "PLATFORM=web NODE_ENV=development playwright test --config=playwright.config.ts --project='Google Chrome' --grep-invert='@snapshot'", + "chrome:test": "PLATFORM=web NODE_ENV=development playwright test --config=playwright.config.ts --project='Google Chrome' --grep-invert=@snapshot", "tronb:vite:dev": "vite build -c vite.main.config.ts -m development && vite build -c vite.preload.config.ts -m development && vite build -c vite.renderer.config.ts -m development", "tronb:vite:prod": "vite build -c vite.main.config.ts && vite build -c vite.preload.config.ts && vite build -c vite.renderer.config.ts", "tronb:package:dev": "npm run tronb:vite:dev && electron-builder --config electron-builder.yml", @@ -135,15 +137,15 @@ "test:snapshots": "PLATFORM=web NODE_ENV=development playwright test --config=playwright.config.ts --grep=@snapshot --trace=on --shard=1/1", "test:unit": "vitest run --mode development --exclude **/kclSamples.test.ts", "test:unit:kcl-samples": "vitest run --mode development ./src/lang/kclSamples.test.ts", - "test:playwright:electron": "playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", - "test:playwright:electron:windows": "playwright test --config=playwright.electron.config.ts --grep-invert=\"@skipWin|@snapshot\" --quiet", - "test:playwright:electron:macos": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot' --quiet", - "test:playwright:electron:ubuntu": "playwright test --config=playwright.electron.config.ts --grep-invert='@skipLinux|@snapshot' --quiet", - "test:playwright:electron:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@snapshot'", - "test:playwright:electron:windows:local": "npm run tronb:vite:dev && set NODE_ENV='development' && playwright test --config=playwright.electron.config.ts --grep-invert=\"@skipWin|@snapshot\"", - "test:playwright:electron:macos:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipMacos|@snapshot'", - "test:playwright:electron:ubuntu:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipLinux|@snapshot'", - "test:playwright:electron:ubuntu:engine:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert='@skipLinux|@snapshot|@skipLocalEngine'", + "test:playwright:electron": "playwright test --config=playwright.electron.config.ts --grep-invert=@snapshot", + "test:playwright:electron:windows": "playwright test --config=playwright.electron.config.ts --grep=@windows --quiet", + "test:playwright:electron:macos": "playwright test --config=playwright.electron.config.ts --grep=@macos --quiet", + "test:playwright:electron:ubuntu": "playwright test --config=playwright.electron.config.ts --grep-invert=@snapshot --quiet", + "test:playwright:electron:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert=@snapshot", + "test:playwright:electron:windows:local": "npm run tronb:vite:dev && set NODE_ENV='development' && playwright test --config=playwright.electron.config.ts --grep-invert=@snapshot", + "test:playwright:electron:macos:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert=@snapshot", + "test:playwright:electron:ubuntu:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert=@snapshot", + "test:playwright:electron:ubuntu:engine:local": "npm run tronb:vite:dev && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep-invert=@snapshot|@skipLocalEngine", "test:unit:local": "npm run simpleserver:bg && npm run test:unit; kill-port 3000", "test:unit:kcl-samples:local": "npm run simpleserver:bg && npm run test:unit:kcl-samples; kill-port 3000" }, diff --git a/public/kcl-samples/screenshots/80-20-rail.png b/public/kcl-samples/screenshots/80-20-rail.png index 4a8b011a6..e005eff3c 100644 Binary files a/public/kcl-samples/screenshots/80-20-rail.png and b/public/kcl-samples/screenshots/80-20-rail.png differ diff --git a/public/kcl-samples/screenshots/color-cube.png b/public/kcl-samples/screenshots/color-cube.png index bfa4877fd..38f5ecc3d 100644 Binary files a/public/kcl-samples/screenshots/color-cube.png and b/public/kcl-samples/screenshots/color-cube.png differ diff --git a/public/kcl-samples/screenshots/dodecahedron.png b/public/kcl-samples/screenshots/dodecahedron.png index 3b3dbea4a..46fff2ff0 100644 Binary files a/public/kcl-samples/screenshots/dodecahedron.png and b/public/kcl-samples/screenshots/dodecahedron.png differ diff --git a/public/kcl-samples/screenshots/i-beam.png b/public/kcl-samples/screenshots/i-beam.png index 7e97b598c..19411c20e 100644 Binary files a/public/kcl-samples/screenshots/i-beam.png and b/public/kcl-samples/screenshots/i-beam.png differ diff --git a/rust/kcl-lib/src/docs/kcl_doc.rs b/rust/kcl-lib/src/docs/kcl_doc.rs index 3a79e486f..d2d180506 100644 --- a/rust/kcl-lib/src/docs/kcl_doc.rs +++ b/rust/kcl-lib/src/docs/kcl_doc.rs @@ -1076,7 +1076,7 @@ mod test { #[for_each_std_mod] #[tokio::test(flavor = "multi_thread")] - async fn test_examples() { + async fn kcl_test_examples() { let std = walk_prelude(); let mut errs = Vec::new(); for d in std { diff --git a/rust/kcl-lib/src/execution/exec_ast.rs b/rust/kcl-lib/src/execution/exec_ast.rs index b8fc39a38..e60c14869 100644 --- a/rust/kcl-lib/src/execution/exec_ast.rs +++ b/rust/kcl-lib/src/execution/exec_ast.rs @@ -459,7 +459,7 @@ impl ExecutorContext { exec_state.add_path_to_source_id(resolved_path.clone(), id); let format = super::import::format_from_annotations(attrs, path, source_range)?; let geom = super::import::import_foreign(path, format, exec_state, self, source_range).await?; - exec_state.add_module(id, resolved_path, ModuleRepr::Foreign(geom)); + exec_state.add_module(id, resolved_path, ModuleRepr::Foreign(geom, None)); Ok(id) } ImportPath::Std { .. } => { @@ -501,7 +501,7 @@ impl ExecutorContext { *cache = Some((val, er, items.clone())); (er, items) }), - ModuleRepr::Foreign(geom) => Err(KclError::Semantic(KclErrorDetails { + ModuleRepr::Foreign(geom, _) => Err(KclError::Semantic(KclErrorDetails { message: "Cannot import items from foreign modules".to_owned(), source_ranges: vec![geom.source_range], })), @@ -546,9 +546,20 @@ impl ExecutorContext { Err(e) => Err(e), } } - ModuleRepr::Foreign(geom) => super::import::send_to_engine(geom.clone(), self) - .await - .map(|geom| Some(KclValue::ImportedGeometry(geom))), + ModuleRepr::Foreign(_, Some(imported)) => Ok(Some(imported.clone())), + ModuleRepr::Foreign(geom, cached) => { + let result = super::import::send_to_engine(geom.clone(), self) + .await + .map(|geom| Some(KclValue::ImportedGeometry(geom))); + + match result { + Ok(val) => { + *cached = val.clone(); + Ok(val) + } + Err(e) => Err(e), + } + } ModuleRepr::Dummy => unreachable!(), }; diff --git a/rust/kcl-lib/src/execution/mod.rs b/rust/kcl-lib/src/execution/mod.rs index 2a86db786..b46ccccb9 100644 --- a/rust/kcl-lib/src/execution/mod.rs +++ b/rust/kcl-lib/src/execution/mod.rs @@ -752,26 +752,31 @@ impl ExecutorContext { let mut universe = std::collections::HashMap::new(); let default_planes = self.engine.get_default_planes().read().await.clone(); - crate::walk::import_universe(self, &program.ast, &mut universe, exec_state) - .await - .map_err(|err| { - let module_id_to_module_path: IndexMap = exec_state - .global - .path_to_source_id - .iter() - .map(|(k, v)| ((*v), k.clone())) - .collect(); + crate::walk::import_universe( + self, + &ModuleRepr::Kcl(program.ast.clone(), None), + &mut universe, + exec_state, + ) + .await + .map_err(|err| { + let module_id_to_module_path: IndexMap = exec_state + .global + .path_to_source_id + .iter() + .map(|(k, v)| ((*v), k.clone())) + .collect(); - KclErrorWithOutputs::new( - err, - exec_state.global.operations.clone(), - exec_state.global.artifact_commands.clone(), - exec_state.global.artifact_graph.clone(), - module_id_to_module_path, - exec_state.global.id_to_source.clone(), - default_planes.clone(), - ) - })?; + KclErrorWithOutputs::new( + err, + exec_state.global.operations.clone(), + exec_state.global.artifact_commands.clone(), + exec_state.global.artifact_graph.clone(), + module_id_to_module_path, + exec_state.global.id_to_source.clone(), + default_planes.clone(), + ) + })?; for modules in crate::walk::import_graph(&universe, self) .map_err(|err| { @@ -799,16 +804,12 @@ impl ExecutorContext { #[allow(clippy::type_complexity)] let (results_tx, mut results_rx): ( - tokio::sync::mpsc::Sender<( - ModuleId, - ModulePath, - Result<(Option, EnvironmentRef, Vec), KclError>, - )>, + tokio::sync::mpsc::Sender<(ModuleId, ModulePath, Result)>, tokio::sync::mpsc::Receiver<_>, ) = tokio::sync::mpsc::channel(1); for module in modules { - let Some((import_stmt, module_id, module_path, program)) = universe.get(&module) else { + let Some((import_stmt, module_id, module_path, repr)) = universe.get(&module) else { return Err(KclErrorWithOutputs::no_outputs(KclError::Internal(KclErrorDetails { message: format!("Module {module} not found in universe"), source_ranges: Default::default(), @@ -816,12 +817,41 @@ impl ExecutorContext { }; let module_id = *module_id; let module_path = module_path.clone(); - let program = program.clone(); + let repr = repr.clone(); let exec_state = exec_state.clone(); let exec_ctxt = self.clone(); let results_tx = results_tx.clone(); let source_range = SourceRange::from(import_stmt); + let exec_module = async |exec_ctxt: &ExecutorContext, + repr: &ModuleRepr, + module_id: ModuleId, + module_path: &ModulePath, + exec_state: &mut ExecState, + source_range: SourceRange| + -> Result { + match repr { + ModuleRepr::Kcl(program, _) => { + let result = exec_ctxt + .exec_module_from_ast(program, module_id, module_path, exec_state, source_range, false) + .await; + + result.map(|val| ModuleRepr::Kcl(program.clone(), Some(val))) + } + ModuleRepr::Foreign(geom, _) => { + let result = crate::execution::import::send_to_engine(geom.clone(), exec_ctxt) + .await + .map(|geom| Some(KclValue::ImportedGeometry(geom))); + + result.map(|val| ModuleRepr::Foreign(geom.clone(), val)) + } + ModuleRepr::Dummy | ModuleRepr::Root => Err(KclError::Internal(KclErrorDetails { + message: format!("Module {module_path} not found in universe"), + source_ranges: vec![source_range], + })), + } + }; + #[cfg(target_arch = "wasm32")] { wasm_bindgen_futures::spawn_local(async move { @@ -829,16 +859,15 @@ impl ExecutorContext { let mut exec_state = exec_state; let exec_ctxt = exec_ctxt; - let result = exec_ctxt - .exec_module_from_ast( - &program, - module_id, - &module_path, - &mut exec_state, - source_range, - false, - ) - .await; + let result = exec_module( + &exec_ctxt, + &repr, + module_id, + &module_path, + &mut exec_state, + source_range, + ) + .await; results_tx .send((module_id, module_path, result)) @@ -852,16 +881,15 @@ impl ExecutorContext { let mut exec_state = exec_state; let exec_ctxt = exec_ctxt; - let result = exec_ctxt - .exec_module_from_ast( - &program, - module_id, - &module_path, - &mut exec_state, - source_range, - false, - ) - .await; + let result = exec_module( + &exec_ctxt, + &repr, + module_id, + &module_path, + &mut exec_state, + source_range, + ) + .await; results_tx .send((module_id, module_path, result)) @@ -875,13 +903,24 @@ impl ExecutorContext { while let Some((module_id, _, result)) = results_rx.recv().await { match result { - Ok((val, session_data, variables)) => { + Ok(new_repr) => { let mut repr = exec_state.global.module_infos[&module_id].take_repr(); - let ModuleRepr::Kcl(_, cache) = &mut repr else { - continue; - }; - *cache = Some((val, session_data, variables)); + match &mut repr { + ModuleRepr::Kcl(_, cache) => { + let ModuleRepr::Kcl(_, session_data) = new_repr else { + unreachable!(); + }; + *cache = session_data; + } + ModuleRepr::Foreign(_, cache) => { + let ModuleRepr::Foreign(_, session_data) = new_repr else { + unreachable!(); + }; + *cache = session_data; + } + ModuleRepr::Dummy | ModuleRepr::Root => unreachable!(), + } exec_state.global.module_infos[&module_id].restore_repr(repr); } diff --git a/rust/kcl-lib/src/execution/types.rs b/rust/kcl-lib/src/execution/types.rs index 8b6c76b80..c00585793 100644 --- a/rust/kcl-lib/src/execution/types.rs +++ b/rust/kcl-lib/src/execution/types.rs @@ -28,6 +28,10 @@ pub enum RuntimeType { } impl RuntimeType { + pub fn edge() -> Self { + RuntimeType::Primitive(PrimitiveType::Edge) + } + pub fn sketch() -> Self { RuntimeType::Primitive(PrimitiveType::Sketch) } @@ -2115,4 +2119,73 @@ d = cos(30) assert_value_and_type("c", &result, 1.0, NumericType::count()); assert_value_and_type("d", &result, 0.0, NumericType::count()); } + + #[tokio::test(flavor = "multi_thread")] + async fn coerce_nested_array() { + let mut exec_state = ExecState::new(&crate::ExecutorContext::new_mock().await); + + let mixed1 = KclValue::MixedArray { + value: vec![ + KclValue::Number { + value: 0.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + KclValue::Number { + value: 1.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + KclValue::HomArray { + value: vec![ + KclValue::Number { + value: 2.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + KclValue::Number { + value: 3.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + ], + ty: RuntimeType::Primitive(PrimitiveType::Number(NumericType::count())), + }, + ], + meta: Vec::new(), + }; + + // Principal types + let tym1 = RuntimeType::Array( + Box::new(RuntimeType::Primitive(PrimitiveType::Number(NumericType::count()))), + ArrayLen::NonEmpty, + ); + + let result = KclValue::HomArray { + value: vec![ + KclValue::Number { + value: 0.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + KclValue::Number { + value: 1.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + KclValue::Number { + value: 2.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + KclValue::Number { + value: 3.0, + ty: NumericType::count(), + meta: Vec::new(), + }, + ], + ty: RuntimeType::Primitive(PrimitiveType::Number(NumericType::count())), + }; + assert_coerce_results(&mixed1, &tym1, &result, &mut exec_state); + } } diff --git a/rust/kcl-lib/src/modules.rs b/rust/kcl-lib/src/modules.rs index bd6a869f4..dde9636a3 100644 --- a/rust/kcl-lib/src/modules.rs +++ b/rust/kcl-lib/src/modules.rs @@ -124,7 +124,7 @@ pub enum ModuleRepr { Root, // AST, memory, exported names Kcl(Node, Option<(Option, EnvironmentRef, Vec)>), - Foreign(PreImportedGeometry), + Foreign(PreImportedGeometry, Option), Dummy, } diff --git a/rust/kcl-lib/src/simulation_tests.rs b/rust/kcl-lib/src/simulation_tests.rs index 541d0f5fe..4d3731f49 100644 --- a/rust/kcl-lib/src/simulation_tests.rs +++ b/rust/kcl-lib/src/simulation_tests.rs @@ -2579,6 +2579,28 @@ mod loop_tag { super::execute(TEST_NAME, true).await } } +mod multiple_foreign_imports_all_render { + const TEST_NAME: &str = "multiple-foreign-imports-all-render"; + + /// Test parsing KCL. + #[test] + fn parse() { + super::parse(TEST_NAME) + } + + /// Test that parsing and unparsing KCL produces the original KCL input. + #[tokio::test(flavor = "multi_thread")] + async fn unparse() { + super::unparse(TEST_NAME).await + } + + /// Test that KCL is executed correctly. + #[tokio::test(flavor = "multi_thread")] + async fn kcl_test_execute() { + super::execute(TEST_NAME, true).await + } +} + mod involute_fail { const TEST_NAME: &str = "involute_fail"; diff --git a/rust/kcl-lib/src/std/args.rs b/rust/kcl-lib/src/std/args.rs index d4dda688b..dec978fae 100644 --- a/rust/kcl-lib/src/std/args.rs +++ b/rust/kcl-lib/src/std/args.rs @@ -659,13 +659,6 @@ impl Args { Ok((sketches, sketch)) } - pub(crate) fn get_data<'a, T>(&'a self) -> Result - where - T: FromArgs<'a>, - { - FromArgs::from_args(self, 0) - } - pub(crate) fn get_data_and_sketch_surface(&self) -> Result<([TyF64; 2], SketchSurface, Option), KclError> { FromArgs::from_args(self, 0) } diff --git a/rust/kcl-lib/src/std/clone.rs b/rust/kcl-lib/src/std/clone.rs index e391cebbe..ba52ec386 100644 --- a/rust/kcl-lib/src/std/clone.rs +++ b/rust/kcl-lib/src/std/clone.rs @@ -559,6 +559,8 @@ clonedCube = clone(cube) assert_eq!(cube.tags.len(), 0); assert_eq!(cloned_cube.tags.len(), 0); + + ctx.close().await; } // Ensure the clone function returns a solid with different ids for all the internal paths and @@ -615,6 +617,8 @@ clonedCube = clone(cube) assert_eq!(cube.edge_cuts.len(), 0); assert_eq!(cloned_cube.edge_cuts.len(), 0); + + ctx.close().await; } // Ensure the clone function returns a sketch with different ids for all the internal paths and @@ -668,6 +672,8 @@ clonedCube = clone(cube) assert_eq!(tag_info.surface, None); assert_eq!(cloned_tag_info.surface, None); } + + ctx.close().await; } // Ensure the clone function returns a solid with different ids for all the internal paths and @@ -734,6 +740,8 @@ clonedCube = clone(cube) assert_eq!(cube.edge_cuts.len(), 0); assert_eq!(cloned_cube.edge_cuts.len(), 0); + + ctx.close().await; } // Ensure we can get all paths even on a sketch where we closed it and it was already closed. @@ -807,6 +815,8 @@ clonedCube = clone(cube) assert_ne!(edge_cut.edge_id(), cloned_edge_cut.edge_id()); assert_eq!(edge_cut.tag(), cloned_edge_cut.tag()); } + + ctx.close().await; } // Ensure the clone function returns a solid with different ids for all the internal paths and @@ -905,5 +915,7 @@ clonedCube = clone(cube) assert_ne!(edge_cut.edge_id(), cloned_edge_cut.edge_id()); assert_eq!(edge_cut.tag(), cloned_edge_cut.tag()); } + + ctx.close().await; } } diff --git a/rust/kcl-lib/src/std/edge.rs b/rust/kcl-lib/src/std/edge.rs index 41ce68b67..39b5e2093 100644 --- a/rust/kcl-lib/src/std/edge.rs +++ b/rust/kcl-lib/src/std/edge.rs @@ -8,15 +8,15 @@ use uuid::Uuid; use crate::{ errors::{KclError, KclErrorDetails}, - execution::{ExecState, ExtrudeSurface, KclValue, TagIdentifier}, + execution::{types::RuntimeType, ExecState, ExtrudeSurface, KclValue, TagIdentifier}, std::Args, }; /// Get the opposite edge to the edge given. pub async fn get_opposite_edge(exec_state: &mut ExecState, args: Args) -> Result { - let tag: TagIdentifier = args.get_data()?; + let input_edge = args.get_unlabeled_kw_arg_typed("edge", &RuntimeType::edge(), exec_state)?; - let edge = inner_get_opposite_edge(tag, exec_state, args.clone()).await?; + let edge = inner_get_opposite_edge(input_edge, exec_state, args.clone()).await?; Ok(KclValue::Uuid { value: edge, meta: vec![args.source_range.into()], @@ -53,15 +53,24 @@ pub async fn get_opposite_edge(exec_state: &mut ExecState, args: Args) -> Result /// ``` #[stdlib { name = "getOppositeEdge", + keywords = true, + unlabeled_first = true, + args = { + edge = { docs = "The tag of the edge you want to find the opposite edge of." }, + } }] -async fn inner_get_opposite_edge(tag: TagIdentifier, exec_state: &mut ExecState, args: Args) -> Result { +async fn inner_get_opposite_edge( + edge: TagIdentifier, + exec_state: &mut ExecState, + args: Args, +) -> Result { if args.ctx.no_engine_commands().await { return Ok(exec_state.next_uuid()); } - let face_id = args.get_adjacent_face_to_tag(exec_state, &tag, false).await?; + let face_id = args.get_adjacent_face_to_tag(exec_state, &edge, false).await?; let id = exec_state.next_uuid(); - let tagged_path = args.get_tag_engine_info(exec_state, &tag)?; + let tagged_path = args.get_tag_engine_info(exec_state, &edge)?; let resp = args .send_modeling_cmd( @@ -88,9 +97,9 @@ async fn inner_get_opposite_edge(tag: TagIdentifier, exec_state: &mut ExecState, /// Get the next adjacent edge to the edge given. pub async fn get_next_adjacent_edge(exec_state: &mut ExecState, args: Args) -> Result { - let tag: TagIdentifier = args.get_data()?; + let input_edge = args.get_unlabeled_kw_arg_typed("edge", &RuntimeType::edge(), exec_state)?; - let edge = inner_get_next_adjacent_edge(tag, exec_state, args.clone()).await?; + let edge = inner_get_next_adjacent_edge(input_edge, exec_state, args.clone()).await?; Ok(KclValue::Uuid { value: edge, meta: vec![args.source_range.into()], @@ -127,19 +136,24 @@ pub async fn get_next_adjacent_edge(exec_state: &mut ExecState, args: Args) -> R /// ``` #[stdlib { name = "getNextAdjacentEdge", + keywords = true, + unlabeled_first = true, + args = { + edge = { docs = "The tag of the edge you want to find the next adjacent edge of." }, + } }] async fn inner_get_next_adjacent_edge( - tag: TagIdentifier, + edge: TagIdentifier, exec_state: &mut ExecState, args: Args, ) -> Result { if args.ctx.no_engine_commands().await { return Ok(exec_state.next_uuid()); } - let face_id = args.get_adjacent_face_to_tag(exec_state, &tag, false).await?; + let face_id = args.get_adjacent_face_to_tag(exec_state, &edge, false).await?; let id = exec_state.next_uuid(); - let tagged_path = args.get_tag_engine_info(exec_state, &tag)?; + let tagged_path = args.get_tag_engine_info(exec_state, &edge)?; let resp = args .send_modeling_cmd( @@ -167,7 +181,7 @@ async fn inner_get_next_adjacent_edge( adjacent_edge.edge.ok_or_else(|| { KclError::Type(KclErrorDetails { - message: format!("No edge found next adjacent to tag: `{}`", tag.value), + message: format!("No edge found next adjacent to tag: `{}`", edge.value), source_ranges: vec![args.source_range], }) }) @@ -175,9 +189,9 @@ async fn inner_get_next_adjacent_edge( /// Get the previous adjacent edge to the edge given. pub async fn get_previous_adjacent_edge(exec_state: &mut ExecState, args: Args) -> Result { - let tag: TagIdentifier = args.get_data()?; + let input_edge = args.get_unlabeled_kw_arg_typed("edge", &RuntimeType::edge(), exec_state)?; - let edge = inner_get_previous_adjacent_edge(tag, exec_state, args.clone()).await?; + let edge = inner_get_previous_adjacent_edge(input_edge, exec_state, args.clone()).await?; Ok(KclValue::Uuid { value: edge, meta: vec![args.source_range.into()], @@ -214,19 +228,24 @@ pub async fn get_previous_adjacent_edge(exec_state: &mut ExecState, args: Args) /// ``` #[stdlib { name = "getPreviousAdjacentEdge", + keywords = true, + unlabeled_first = true, + args = { + edge = { docs = "The tag of the edge you want to find the previous adjacent edge of." }, + } }] async fn inner_get_previous_adjacent_edge( - tag: TagIdentifier, + edge: TagIdentifier, exec_state: &mut ExecState, args: Args, ) -> Result { if args.ctx.no_engine_commands().await { return Ok(exec_state.next_uuid()); } - let face_id = args.get_adjacent_face_to_tag(exec_state, &tag, false).await?; + let face_id = args.get_adjacent_face_to_tag(exec_state, &edge, false).await?; let id = exec_state.next_uuid(); - let tagged_path = args.get_tag_engine_info(exec_state, &tag)?; + let tagged_path = args.get_tag_engine_info(exec_state, &edge)?; let resp = args .send_modeling_cmd( @@ -253,7 +272,7 @@ async fn inner_get_previous_adjacent_edge( adjacent_edge.edge.ok_or_else(|| { KclError::Type(KclErrorDetails { - message: format!("No edge found previous adjacent to tag: `{}`", tag.value), + message: format!("No edge found previous adjacent to tag: `{}`", edge.value), source_ranges: vec![args.source_range], }) }) diff --git a/rust/kcl-lib/src/walk/import_graph.rs b/rust/kcl-lib/src/walk/import_graph.rs index 46c9dc795..b57b68b63 100644 --- a/rust/kcl-lib/src/walk/import_graph.rs +++ b/rust/kcl-lib/src/walk/import_graph.rs @@ -8,7 +8,7 @@ use anyhow::Result; use crate::{ errors::KclErrorDetails, modules::{ModulePath, ModuleRepr}, - parsing::ast::types::{ImportPath, ImportStatement, Node as AstNode, NodeRef, Program}, + parsing::ast::types::{ImportPath, ImportStatement, Node as AstNode}, walk::{Node, Visitable}, ExecState, ExecutorContext, KclError, ModuleId, SourceRange, }; @@ -20,7 +20,7 @@ type Dependency = (String, String); type Graph = Vec; -type DependencyInfo = (AstNode, ModuleId, ModulePath, AstNode); +type DependencyInfo = (AstNode, ModuleId, ModulePath, ModuleRepr); type Universe = HashMap; /// Process a number of programs, returning the graph of dependencies. @@ -32,9 +32,9 @@ type Universe = HashMap; pub fn import_graph(progs: &Universe, ctx: &ExecutorContext) -> Result>, KclError> { let mut graph = Graph::new(); - for (name, (_, _, _, program)) in progs.iter() { + for (name, (_, _, _, repr)) in progs.iter() { graph.extend( - import_dependencies(program, ctx)? + import_dependencies(repr, ctx)? .into_iter() .map(|(dependency, _, _)| (name.clone(), dependency)) .collect::>(), @@ -118,28 +118,42 @@ fn topsort(all_modules: &[&str], graph: Graph) -> Result>, KclEr type ImportDependencies = Vec<(String, AstNode, ModulePath)>; -pub(crate) fn import_dependencies( - prog: NodeRef, - ctx: &ExecutorContext, -) -> Result { - let ret = Arc::new(Mutex::new(vec![])); +pub(crate) fn import_dependencies(repr: &ModuleRepr, ctx: &ExecutorContext) -> Result { + let ModuleRepr::Kcl(prog, _) = repr else { + // It has no dependencies, so return an empty list. + return Ok(vec![]); + }; + let ret = Arc::new(Mutex::new(vec![])); fn walk(ret: Arc>, node: Node<'_>, ctx: &ExecutorContext) -> Result<(), KclError> { if let Node::ImportStatement(is) = node { - // We only care about Kcl imports for now. - if let ImportPath::Kcl { filename } = &is.path { - let resolved_path = ModulePath::from_import_path(&is.path, &ctx.settings.project_directory); - - // We need to lock the mutex to push the dependency. - // This is a bit of a hack, but it works for now. - ret.lock() - .map_err(|err| { - KclError::Internal(KclErrorDetails { - message: format!("Failed to lock mutex: {}", err), - source_ranges: Default::default(), - }) - })? - .push((filename.to_string(), is.clone(), resolved_path)); + // We only care about Kcl and Foreign imports for now. + let resolved_path = ModulePath::from_import_path(&is.path, &ctx.settings.project_directory); + match &is.path { + ImportPath::Kcl { filename } => { + // We need to lock the mutex to push the dependency. + // This is a bit of a hack, but it works for now. + ret.lock() + .map_err(|err| { + KclError::Internal(KclErrorDetails { + message: format!("Failed to lock mutex: {}", err), + source_ranges: Default::default(), + }) + })? + .push((filename.to_string(), is.clone(), resolved_path)); + } + ImportPath::Foreign { path } => { + ret.lock() + .map_err(|err| { + KclError::Internal(KclErrorDetails { + message: format!("Failed to lock mutex: {}", err), + source_ranges: Default::default(), + }) + })? + .push((path.to_string(), is.clone(), resolved_path)); + } + ImportPath::Std { .. } => { // do nothing + } } } @@ -164,11 +178,11 @@ pub(crate) fn import_dependencies( pub(crate) async fn import_universe( ctx: &ExecutorContext, - prog: NodeRef<'_, Program>, + repr: &ModuleRepr, out: &mut Universe, exec_state: &mut ExecState, ) -> Result<(), KclError> { - let modules = import_dependencies(prog, ctx)?; + let modules = import_dependencies(repr, ctx)?; for (filename, import_stmt, module_path) in modules { if out.contains_key(&filename) { continue; @@ -178,26 +192,21 @@ pub(crate) async fn import_universe( .open_module(&import_stmt.path, &[], exec_state, Default::default()) .await?; - let program = { + let repr = { let Some(module_info) = exec_state.get_module(module_id) else { return Err(KclError::Internal(KclErrorDetails { message: format!("Module {} not found", module_id), source_ranges: vec![import_stmt.into()], })); }; - let ModuleRepr::Kcl(program, _) = &module_info.repr else { - // if it's not a KCL module we can skip it since it has no - // dependencies. - continue; - }; - program.clone() + module_info.repr.clone() }; out.insert( filename.clone(), - (import_stmt.clone(), module_id, module_path.clone(), program.clone()), + (import_stmt.clone(), module_id, module_path.clone(), repr.clone()), ); - Box::pin(import_universe(ctx, &program, out, exec_state)).await?; + Box::pin(import_universe(ctx, &repr, out, exec_state)).await?; } Ok(()) @@ -206,7 +215,7 @@ pub(crate) async fn import_universe( #[cfg(test)] mod tests { use super::*; - use crate::parsing::ast::types::ImportSelector; + use crate::parsing::ast::types::{ImportSelector, Program}; macro_rules! kcl { ( $kcl:expr ) => {{ @@ -224,7 +233,7 @@ mod tests { }), ModuleId::default(), ModulePath::Local { value: "".into() }, - program, + ModuleRepr::Kcl(program, None), ) } diff --git a/rust/kcl-lib/tests/kcl_samples/80-20-rail/rendered_model.png b/rust/kcl-lib/tests/kcl_samples/80-20-rail/rendered_model.png index 4a8b011a6..e005eff3c 100644 Binary files a/rust/kcl-lib/tests/kcl_samples/80-20-rail/rendered_model.png and b/rust/kcl-lib/tests/kcl_samples/80-20-rail/rendered_model.png differ diff --git a/rust/kcl-lib/tests/kcl_samples/color-cube/rendered_model.png b/rust/kcl-lib/tests/kcl_samples/color-cube/rendered_model.png index bfa4877fd..38f5ecc3d 100644 Binary files a/rust/kcl-lib/tests/kcl_samples/color-cube/rendered_model.png and b/rust/kcl-lib/tests/kcl_samples/color-cube/rendered_model.png differ diff --git a/rust/kcl-lib/tests/kcl_samples/i-beam/rendered_model.png b/rust/kcl-lib/tests/kcl_samples/i-beam/rendered_model.png index 7e97b598c..19411c20e 100644 Binary files a/rust/kcl-lib/tests/kcl_samples/i-beam/rendered_model.png and b/rust/kcl-lib/tests/kcl_samples/i-beam/rendered_model.png differ diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/anothercube.kcl b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/anothercube.kcl new file mode 100644 index 000000000..59e5f233d --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/anothercube.kcl @@ -0,0 +1,3 @@ +import "../inputs/cube.step" as cube + +clone(cube) diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_commands.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_commands.snap new file mode 100644 index 000000000..75ec9a78f --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_commands.snap @@ -0,0 +1,16040 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Artifact commands multiple-foreign-imports-all-render.kcl +--- +[ + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "edge_lines_visible", + "hidden": false + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "object_visible", + "object_id": "[uuid]", + "hidden": true + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "object_visible", + "object_id": "[uuid]", + "hidden": true + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "import_files", + "files": [ + { + "path": "cube.step", + "data": [ + 73, + 83, + 79, + 45, + 49, + 48, + 51, + 48, + 51, + 45, + 50, + 49, + 59, + 13, + 10, + 72, + 69, + 65, + 68, + 69, + 82, + 59, + 13, + 10, + 70, + 73, + 76, + 69, + 95, + 68, + 69, + 83, + 67, + 82, + 73, + 80, + 84, + 73, + 79, + 78, + 32, + 40, + 40, + 32, + 39, + 83, + 84, + 69, + 80, + 32, + 65, + 80, + 50, + 48, + 51, + 39, + 32, + 41, + 44, + 13, + 10, + 32, + 32, + 32, + 32, + 39, + 49, + 39, + 32, + 41, + 59, + 13, + 10, + 70, + 73, + 76, + 69, + 95, + 78, + 65, + 77, + 69, + 32, + 40, + 39, + 67, + 117, + 98, + 101, + 95, + 49, + 109, + 46, + 83, + 84, + 69, + 80, + 39, + 44, + 13, + 10, + 32, + 32, + 32, + 32, + 39, + 50, + 48, + 50, + 51, + 45, + 48, + 55, + 45, + 51, + 49, + 84, + 49, + 53, + 58, + 49, + 48, + 58, + 49, + 57, + 39, + 44, + 13, + 10, + 32, + 32, + 32, + 32, + 40, + 32, + 39, + 39, + 32, + 41, + 44, + 13, + 10, + 32, + 32, + 32, + 32, + 40, + 32, + 39, + 39, + 32, + 41, + 44, + 13, + 10, + 32, + 32, + 32, + 32, + 39, + 83, + 119, + 83, + 84, + 69, + 80, + 32, + 50, + 46, + 48, + 39, + 44, + 13, + 10, + 32, + 32, + 32, + 32, + 39, + 83, + 111, + 108, + 105, + 100, + 87, + 111, + 114, + 107, + 115, + 32, + 50, + 48, + 50, + 51, + 39, + 44, + 13, + 10, + 32, + 32, + 32, + 32, + 39, + 39, + 32, + 41, + 59, + 13, + 10, + 70, + 73, + 76, + 69, + 95, + 83, + 67, + 72, + 69, + 77, + 65, + 32, + 40, + 40, + 32, + 39, + 67, + 79, + 78, + 70, + 73, + 71, + 95, + 67, + 79, + 78, + 84, + 82, + 79, + 76, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 39, + 32, + 41, + 41, + 59, + 13, + 10, + 69, + 78, + 68, + 83, + 69, + 67, + 59, + 13, + 10, + 13, + 10, + 68, + 65, + 84, + 65, + 59, + 13, + 10, + 35, + 49, + 32, + 61, + 32, + 65, + 88, + 73, + 83, + 50, + 95, + 80, + 76, + 65, + 67, + 69, + 77, + 69, + 78, + 84, + 95, + 51, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 51, + 51, + 44, + 32, + 35, + 49, + 49, + 54, + 44, + 32, + 35, + 55, + 50, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 32, + 61, + 40, + 32, + 76, + 69, + 78, + 71, + 84, + 72, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 41, + 32, + 78, + 65, + 77, + 69, + 68, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 42, + 32, + 41, + 32, + 83, + 73, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 46, + 77, + 73, + 76, + 76, + 73, + 46, + 44, + 32, + 46, + 77, + 69, + 84, + 82, + 69, + 46, + 32, + 41, + 32, + 41, + 59, + 13, + 10, + 35, + 51, + 32, + 61, + 32, + 70, + 65, + 67, + 69, + 95, + 79, + 85, + 84, + 69, + 82, + 95, + 66, + 79, + 85, + 78, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 57, + 57, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 51, + 51, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 53, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 55, + 53, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 48, + 53, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 32, + 61, + 32, + 76, + 79, + 67, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 49, + 49, + 44, + 32, + 49, + 48, + 44, + 32, + 49, + 57, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 35, + 56, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 32, + 61, + 32, + 80, + 76, + 65, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 35, + 50, + 50, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 52, + 48, + 44, + 32, + 35, + 53, + 56, + 44, + 32, + 35, + 49, + 51, + 57, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 32, + 61, + 32, + 67, + 65, + 76, + 69, + 78, + 68, + 65, + 82, + 95, + 68, + 65, + 84, + 69, + 32, + 40, + 32, + 50, + 48, + 50, + 51, + 44, + 32, + 51, + 49, + 44, + 32, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 32, + 61, + 32, + 65, + 88, + 73, + 83, + 50, + 95, + 80, + 76, + 65, + 67, + 69, + 77, + 69, + 78, + 84, + 95, + 51, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 55, + 53, + 44, + 32, + 35, + 55, + 51, + 44, + 32, + 35, + 49, + 50, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 32, + 61, + 32, + 80, + 76, + 65, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 35, + 49, + 52, + 53, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 32, + 61, + 40, + 32, + 71, + 69, + 79, + 77, + 69, + 84, + 82, + 73, + 67, + 95, + 82, + 69, + 80, + 82, + 69, + 83, + 69, + 78, + 84, + 65, + 84, + 73, + 79, + 78, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 51, + 32, + 41, + 32, + 71, + 76, + 79, + 66, + 65, + 76, + 95, + 85, + 78, + 67, + 69, + 82, + 84, + 65, + 73, + 78, + 84, + 89, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 69, + 68, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 40, + 32, + 35, + 49, + 53, + 50, + 32, + 41, + 32, + 41, + 32, + 71, + 76, + 79, + 66, + 65, + 76, + 95, + 85, + 78, + 73, + 84, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 69, + 68, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 40, + 32, + 35, + 50, + 44, + 32, + 35, + 49, + 54, + 56, + 44, + 32, + 35, + 49, + 55, + 52, + 32, + 41, + 32, + 41, + 32, + 82, + 69, + 80, + 82, + 69, + 83, + 69, + 78, + 84, + 65, + 84, + 73, + 79, + 78, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 39, + 87, + 79, + 82, + 75, + 65, + 83, + 80, + 65, + 67, + 69, + 39, + 32, + 41, + 32, + 41, + 59, + 13, + 10, + 35, + 50, + 50, + 32, + 61, + 32, + 70, + 65, + 67, + 69, + 95, + 79, + 85, + 84, + 69, + 82, + 95, + 66, + 79, + 85, + 78, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 51, + 55, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 32, + 61, + 32, + 80, + 82, + 79, + 68, + 85, + 67, + 84, + 95, + 68, + 69, + 70, + 73, + 78, + 73, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 85, + 78, + 75, + 78, + 79, + 87, + 78, + 39, + 44, + 32, + 39, + 39, + 44, + 32, + 35, + 53, + 49, + 44, + 32, + 35, + 56, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 52, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 53, + 32, + 61, + 32, + 65, + 80, + 80, + 76, + 73, + 67, + 65, + 84, + 73, + 79, + 78, + 95, + 80, + 82, + 79, + 84, + 79, + 67, + 79, + 76, + 95, + 68, + 69, + 70, + 73, + 78, + 73, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 105, + 110, + 116, + 101, + 114, + 110, + 97, + 116, + 105, + 111, + 110, + 97, + 108, + 32, + 115, + 116, + 97, + 110, + 100, + 97, + 114, + 100, + 39, + 44, + 32, + 39, + 99, + 111, + 110, + 102, + 105, + 103, + 95, + 99, + 111, + 110, + 116, + 114, + 111, + 108, + 95, + 100, + 101, + 115, + 105, + 103, + 110, + 39, + 44, + 32, + 49, + 57, + 57, + 52, + 44, + 32, + 35, + 53, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 54, + 32, + 61, + 32, + 80, + 76, + 65, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 35, + 49, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 55, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 76, + 79, + 79, + 80, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 52, + 49, + 44, + 32, + 35, + 56, + 44, + 32, + 35, + 49, + 50, + 49, + 44, + 32, + 35, + 49, + 50, + 51, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 56, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 57, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 48, + 32, + 61, + 32, + 70, + 65, + 67, + 69, + 95, + 79, + 85, + 84, + 69, + 82, + 95, + 66, + 79, + 85, + 78, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 55, + 56, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 49, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 56, + 51, + 44, + 32, + 35, + 49, + 55, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 50, + 32, + 61, + 32, + 80, + 76, + 65, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 35, + 49, + 55, + 51, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 51, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 52, + 52, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 52, + 32, + 61, + 32, + 68, + 65, + 84, + 69, + 95, + 65, + 78, + 68, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 49, + 54, + 44, + 32, + 35, + 50, + 50, + 52, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 53, + 32, + 61, + 32, + 65, + 68, + 86, + 65, + 78, + 67, + 69, + 68, + 95, + 70, + 65, + 67, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 57, + 56, + 32, + 41, + 44, + 32, + 35, + 50, + 54, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 54, + 32, + 61, + 32, + 67, + 65, + 76, + 69, + 78, + 68, + 65, + 82, + 95, + 68, + 65, + 84, + 69, + 32, + 40, + 32, + 50, + 48, + 50, + 51, + 44, + 32, + 51, + 49, + 44, + 32, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 55, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 54, + 49, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 56, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 32, + 40, + 32, + 35, + 50, + 49, + 48, + 44, + 32, + 40, + 32, + 35, + 53, + 49, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 51, + 57, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 48, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 57, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 49, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 49, + 57, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 50, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 54, + 57, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 51, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 77, + 69, + 78, + 84, + 32, + 40, + 32, + 35, + 49, + 54, + 55, + 44, + 32, + 35, + 49, + 56, + 56, + 44, + 32, + 40, + 32, + 35, + 49, + 56, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 52, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 53, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 54, + 32, + 61, + 32, + 65, + 80, + 80, + 76, + 73, + 67, + 65, + 84, + 73, + 79, + 78, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 39, + 99, + 111, + 110, + 102, + 105, + 103, + 117, + 114, + 97, + 116, + 105, + 111, + 110, + 32, + 99, + 111, + 110, + 116, + 114, + 111, + 108, + 108, + 101, + 100, + 32, + 51, + 100, + 32, + 100, + 101, + 115, + 105, + 103, + 110, + 115, + 32, + 111, + 102, + 32, + 109, + 101, + 99, + 104, + 97, + 110, + 105, + 99, + 97, + 108, + 32, + 112, + 97, + 114, + 116, + 115, + 32, + 97, + 110, + 100, + 32, + 97, + 115, + 115, + 101, + 109, + 98, + 108, + 105, + 101, + 115, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 55, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 48, + 51, + 44, + 32, + 35, + 49, + 48, + 49, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 56, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 52, + 52, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 52, + 57, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 48, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 49, + 32, + 61, + 32, + 80, + 82, + 79, + 68, + 85, + 67, + 84, + 95, + 68, + 69, + 70, + 73, + 78, + 73, + 84, + 73, + 79, + 78, + 95, + 70, + 79, + 82, + 77, + 65, + 84, + 73, + 79, + 78, + 95, + 87, + 73, + 84, + 72, + 95, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 95, + 83, + 79, + 85, + 82, + 67, + 69, + 32, + 40, + 32, + 39, + 65, + 78, + 89, + 39, + 44, + 32, + 39, + 39, + 44, + 32, + 35, + 54, + 56, + 44, + 32, + 46, + 78, + 79, + 84, + 95, + 75, + 78, + 79, + 87, + 78, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 50, + 32, + 61, + 32, + 68, + 65, + 84, + 69, + 95, + 65, + 78, + 68, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 49, + 49, + 48, + 44, + 32, + 35, + 49, + 56, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 51, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 77, + 69, + 78, + 84, + 32, + 40, + 32, + 35, + 49, + 57, + 50, + 44, + 32, + 35, + 57, + 50, + 44, + 32, + 40, + 32, + 35, + 54, + 56, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 52, + 32, + 61, + 32, + 68, + 65, + 84, + 69, + 95, + 65, + 78, + 68, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 49, + 49, + 55, + 44, + 32, + 35, + 57, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 53, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 48, + 57, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 54, + 32, + 61, + 32, + 80, + 76, + 65, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 35, + 49, + 50, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 55, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 50, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 56, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 52, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 53, + 57, + 32, + 61, + 32, + 65, + 80, + 80, + 76, + 73, + 67, + 65, + 84, + 73, + 79, + 78, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 39, + 99, + 111, + 110, + 102, + 105, + 103, + 117, + 114, + 97, + 116, + 105, + 111, + 110, + 32, + 99, + 111, + 110, + 116, + 114, + 111, + 108, + 108, + 101, + 100, + 32, + 51, + 100, + 32, + 100, + 101, + 115, + 105, + 103, + 110, + 115, + 32, + 111, + 102, + 32, + 109, + 101, + 99, + 104, + 97, + 110, + 105, + 99, + 97, + 108, + 32, + 112, + 97, + 114, + 116, + 115, + 32, + 97, + 110, + 100, + 32, + 97, + 115, + 115, + 101, + 109, + 98, + 108, + 105, + 101, + 115, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 48, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 53, + 55, + 44, + 32, + 35, + 56, + 48, + 44, + 32, + 35, + 49, + 56, + 53, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 49, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 50, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 50, + 49, + 49, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 51, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 51, + 50, + 44, + 32, + 35, + 50, + 51, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 52, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 57, + 54, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 53, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 54, + 32, + 61, + 32, + 80, + 76, + 65, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 35, + 55, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 55, + 32, + 61, + 32, + 67, + 76, + 79, + 83, + 69, + 68, + 95, + 83, + 72, + 69, + 76, + 76, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 50, + 50, + 54, + 44, + 32, + 35, + 49, + 54, + 52, + 44, + 32, + 35, + 49, + 54, + 57, + 44, + 32, + 35, + 50, + 51, + 52, + 44, + 32, + 35, + 51, + 53, + 44, + 32, + 35, + 50, + 49, + 57, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 56, + 32, + 61, + 32, + 80, + 82, + 79, + 68, + 85, + 67, + 84, + 32, + 40, + 32, + 39, + 67, + 117, + 98, + 101, + 95, + 49, + 109, + 39, + 44, + 32, + 39, + 67, + 117, + 98, + 101, + 95, + 49, + 109, + 39, + 44, + 32, + 39, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 53, + 53, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 54, + 57, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 56, + 48, + 44, + 32, + 35, + 49, + 50, + 48, + 44, + 32, + 35, + 51, + 49, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 48, + 32, + 61, + 32, + 65, + 88, + 73, + 83, + 50, + 95, + 80, + 76, + 65, + 67, + 69, + 77, + 69, + 78, + 84, + 95, + 51, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 51, + 56, + 44, + 32, + 35, + 49, + 48, + 53, + 44, + 32, + 35, + 50, + 49, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 49, + 32, + 61, + 32, + 68, + 65, + 84, + 69, + 95, + 65, + 78, + 68, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 49, + 55, + 50, + 44, + 32, + 35, + 49, + 57, + 52, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 50, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 51, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 52, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 50, + 52, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 53, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 56, + 48, + 44, + 32, + 35, + 49, + 51, + 53, + 44, + 32, + 35, + 56, + 49, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 54, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 53, + 48, + 44, + 32, + 35, + 49, + 55, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 55, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 32, + 40, + 32, + 35, + 49, + 54, + 54, + 44, + 32, + 40, + 32, + 35, + 50, + 51, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 56, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 76, + 79, + 79, + 80, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 57, + 55, + 44, + 32, + 35, + 49, + 57, + 57, + 44, + 32, + 35, + 50, + 51, + 48, + 44, + 32, + 35, + 52, + 50, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 55, + 57, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 48, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 49, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 48, + 51, + 44, + 32, + 35, + 56, + 53, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 50, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 51, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 77, + 69, + 78, + 84, + 32, + 40, + 32, + 35, + 57, + 51, + 44, + 32, + 35, + 49, + 49, + 53, + 44, + 32, + 40, + 32, + 35, + 50, + 51, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 52, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 68, + 65, + 84, + 69, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 49, + 49, + 50, + 44, + 32, + 35, + 50, + 49, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 53, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 56, + 50, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 54, + 32, + 61, + 32, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 39, + 100, + 101, + 116, + 97, + 105, + 108, + 101, + 100, + 32, + 100, + 101, + 115, + 105, + 103, + 110, + 39, + 44, + 32, + 35, + 53, + 57, + 44, + 32, + 39, + 100, + 101, + 115, + 105, + 103, + 110, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 55, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 83, + 84, + 65, + 84, + 85, + 83, + 32, + 40, + 32, + 39, + 110, + 111, + 116, + 95, + 121, + 101, + 116, + 95, + 97, + 112, + 112, + 114, + 111, + 118, + 101, + 100, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 56, + 32, + 61, + 32, + 67, + 79, + 79, + 82, + 68, + 73, + 78, + 65, + 84, + 69, + 68, + 95, + 85, + 78, + 73, + 86, + 69, + 82, + 83, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 95, + 79, + 70, + 70, + 83, + 69, + 84, + 32, + 40, + 32, + 53, + 44, + 32, + 48, + 44, + 32, + 46, + 66, + 69, + 72, + 73, + 78, + 68, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 56, + 57, + 32, + 61, + 32, + 67, + 79, + 79, + 82, + 68, + 73, + 78, + 65, + 84, + 69, + 68, + 95, + 85, + 78, + 73, + 86, + 69, + 82, + 83, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 95, + 79, + 70, + 70, + 83, + 69, + 84, + 32, + 40, + 32, + 53, + 44, + 32, + 48, + 44, + 32, + 46, + 66, + 69, + 72, + 73, + 78, + 68, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 48, + 32, + 61, + 32, + 76, + 79, + 67, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 49, + 49, + 44, + 32, + 49, + 48, + 44, + 32, + 49, + 57, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 35, + 49, + 53, + 52, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 49, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 50, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 100, + 101, + 115, + 105, + 103, + 110, + 95, + 111, + 119, + 110, + 101, + 114, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 51, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 52, + 32, + 61, + 32, + 65, + 80, + 80, + 76, + 73, + 67, + 65, + 84, + 73, + 79, + 78, + 95, + 80, + 82, + 79, + 84, + 79, + 67, + 79, + 76, + 95, + 68, + 69, + 70, + 73, + 78, + 73, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 105, + 110, + 116, + 101, + 114, + 110, + 97, + 116, + 105, + 111, + 110, + 97, + 108, + 32, + 115, + 116, + 97, + 110, + 100, + 97, + 114, + 100, + 39, + 44, + 32, + 39, + 99, + 111, + 110, + 102, + 105, + 103, + 95, + 99, + 111, + 110, + 116, + 114, + 111, + 108, + 95, + 100, + 101, + 115, + 105, + 103, + 110, + 39, + 44, + 32, + 49, + 57, + 57, + 52, + 44, + 32, + 35, + 52, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 53, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 32, + 40, + 32, + 35, + 56, + 55, + 44, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 54, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 55, + 52, + 44, + 32, + 35, + 53, + 55, + 44, + 32, + 35, + 49, + 54, + 50, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 55, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 52, + 52, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 56, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 57, + 57, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 76, + 79, + 79, + 80, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 54, + 44, + 32, + 35, + 49, + 50, + 53, + 44, + 32, + 35, + 54, + 52, + 44, + 32, + 35, + 49, + 49, + 56, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 48, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 49, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 48, + 52, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 50, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 76, + 79, + 79, + 80, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 55, + 56, + 44, + 32, + 35, + 50, + 48, + 50, + 44, + 32, + 35, + 52, + 49, + 44, + 32, + 35, + 52, + 56, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 51, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 52, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 53, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 54, + 32, + 61, + 32, + 80, + 82, + 79, + 68, + 85, + 67, + 84, + 95, + 82, + 69, + 76, + 65, + 84, + 69, + 68, + 95, + 80, + 82, + 79, + 68, + 85, + 67, + 84, + 95, + 67, + 65, + 84, + 69, + 71, + 79, + 82, + 89, + 32, + 40, + 32, + 39, + 100, + 101, + 116, + 97, + 105, + 108, + 39, + 44, + 32, + 39, + 39, + 44, + 32, + 40, + 32, + 35, + 54, + 56, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 55, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 51, + 53, + 44, + 32, + 35, + 50, + 49, + 53, + 44, + 32, + 35, + 49, + 52, + 48, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 56, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 32, + 40, + 32, + 35, + 57, + 53, + 44, + 32, + 40, + 32, + 35, + 49, + 56, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 48, + 57, + 32, + 61, + 32, + 83, + 72, + 65, + 80, + 69, + 95, + 68, + 69, + 70, + 73, + 78, + 73, + 84, + 73, + 79, + 78, + 95, + 82, + 69, + 80, + 82, + 69, + 83, + 69, + 78, + 84, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 52, + 54, + 44, + 32, + 35, + 50, + 50, + 50, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 48, + 32, + 61, + 32, + 67, + 65, + 76, + 69, + 78, + 68, + 65, + 82, + 95, + 68, + 65, + 84, + 69, + 32, + 40, + 32, + 50, + 48, + 50, + 51, + 44, + 32, + 51, + 49, + 44, + 32, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 49, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 50, + 32, + 61, + 32, + 68, + 65, + 84, + 69, + 95, + 65, + 78, + 68, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 51, + 54, + 44, + 32, + 35, + 49, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 51, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 56, + 44, + 32, + 35, + 57, + 53, + 44, + 32, + 35, + 49, + 51, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 52, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 49, + 49, + 44, + 32, + 35, + 50, + 49, + 48, + 44, + 32, + 35, + 49, + 51, + 50, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 53, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 99, + 114, + 101, + 97, + 116, + 111, + 114, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 54, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 55, + 32, + 61, + 32, + 67, + 65, + 76, + 69, + 78, + 68, + 65, + 82, + 95, + 68, + 65, + 84, + 69, + 32, + 40, + 32, + 50, + 48, + 50, + 51, + 44, + 32, + 51, + 49, + 44, + 32, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 56, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 50, + 56, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 49, + 57, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 49, + 53, + 44, + 32, + 35, + 52, + 48, + 44, + 32, + 35, + 54, + 51, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 48, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 53, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 49, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 50, + 49, + 49, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 50, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 51, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 57, + 54, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 52, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 53, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 56, + 57, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 54, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 55, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 56, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 55, + 52, + 44, + 32, + 35, + 52, + 48, + 44, + 32, + 35, + 49, + 52, + 50, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 50, + 57, + 32, + 61, + 32, + 65, + 88, + 73, + 83, + 50, + 95, + 80, + 76, + 65, + 67, + 69, + 77, + 69, + 78, + 84, + 95, + 51, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 52, + 44, + 32, + 35, + 49, + 57, + 53, + 44, + 32, + 35, + 49, + 55, + 49, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 48, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 49, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 68, + 65, + 84, + 69, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 55, + 49, + 44, + 32, + 35, + 49, + 54, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 50, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 51, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 52, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 32, + 40, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 44, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 44, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 44, + 32, + 40, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 41, + 44, + 32, + 40, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 41, + 44, + 32, + 40, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 53, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 49, + 52, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 54, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 57, + 49, + 44, + 32, + 35, + 49, + 54, + 54, + 44, + 32, + 35, + 49, + 53, + 49, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 55, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 76, + 79, + 79, + 80, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 53, + 56, + 44, + 32, + 35, + 49, + 55, + 54, + 44, + 32, + 35, + 49, + 53, + 55, + 44, + 32, + 35, + 50, + 49, + 50, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 56, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 51, + 57, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 52, + 53, + 44, + 32, + 35, + 53, + 53, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 48, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 55, + 55, + 44, + 32, + 35, + 51, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 49, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 54, + 48, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 50, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 51, + 49, + 44, + 32, + 35, + 49, + 57, + 51, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 51, + 32, + 61, + 32, + 70, + 65, + 67, + 69, + 95, + 79, + 85, + 84, + 69, + 82, + 95, + 66, + 79, + 85, + 78, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 50, + 55, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 52, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 50, + 48, + 44, + 32, + 35, + 50, + 49, + 53, + 44, + 32, + 35, + 52, + 55, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 53, + 32, + 61, + 32, + 65, + 88, + 73, + 83, + 50, + 95, + 80, + 76, + 65, + 67, + 69, + 77, + 69, + 78, + 84, + 95, + 51, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 51, + 57, + 44, + 32, + 35, + 50, + 52, + 44, + 32, + 35, + 50, + 48, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 54, + 32, + 61, + 32, + 80, + 82, + 79, + 68, + 85, + 67, + 84, + 95, + 68, + 69, + 70, + 73, + 78, + 73, + 84, + 73, + 79, + 78, + 95, + 83, + 72, + 65, + 80, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 35, + 50, + 51, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 55, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 99, + 114, + 101, + 97, + 116, + 111, + 114, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 56, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 77, + 69, + 78, + 84, + 32, + 40, + 32, + 35, + 49, + 54, + 53, + 44, + 32, + 35, + 49, + 52, + 55, + 44, + 32, + 40, + 32, + 35, + 53, + 49, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 52, + 57, + 32, + 61, + 32, + 68, + 65, + 84, + 69, + 95, + 84, + 73, + 77, + 69, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 99, + 108, + 97, + 115, + 115, + 105, + 102, + 105, + 99, + 97, + 116, + 105, + 111, + 110, + 95, + 100, + 97, + 116, + 101, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 48, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 53, + 56, + 44, + 32, + 35, + 49, + 50, + 48, + 44, + 32, + 35, + 49, + 54, + 51, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 49, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 50, + 32, + 61, + 32, + 85, + 78, + 67, + 69, + 82, + 84, + 65, + 73, + 78, + 84, + 89, + 95, + 77, + 69, + 65, + 83, + 85, + 82, + 69, + 95, + 87, + 73, + 84, + 72, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 76, + 69, + 78, + 71, + 84, + 72, + 95, + 77, + 69, + 65, + 83, + 85, + 82, + 69, + 40, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 56, + 50, + 69, + 45, + 48, + 53, + 32, + 41, + 44, + 32, + 35, + 50, + 44, + 32, + 39, + 100, + 105, + 115, + 116, + 97, + 110, + 99, + 101, + 95, + 97, + 99, + 99, + 117, + 114, + 97, + 99, + 121, + 95, + 118, + 97, + 108, + 117, + 101, + 39, + 44, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 41, + 59, + 13, + 10, + 35, + 49, + 53, + 51, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 83, + 69, + 67, + 85, + 82, + 73, + 84, + 89, + 95, + 67, + 76, + 65, + 83, + 83, + 73, + 70, + 73, + 67, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 56, + 48, + 44, + 32, + 40, + 32, + 35, + 53, + 49, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 52, + 32, + 61, + 32, + 67, + 79, + 79, + 82, + 68, + 73, + 78, + 65, + 84, + 69, + 68, + 95, + 85, + 78, + 73, + 86, + 69, + 82, + 83, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 95, + 79, + 70, + 70, + 83, + 69, + 84, + 32, + 40, + 32, + 53, + 44, + 32, + 48, + 44, + 32, + 46, + 66, + 69, + 72, + 73, + 78, + 68, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 53, + 32, + 61, + 32, + 77, + 69, + 67, + 72, + 65, + 78, + 73, + 67, + 65, + 76, + 95, + 67, + 79, + 78, + 84, + 69, + 88, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 52, + 54, + 44, + 32, + 39, + 109, + 101, + 99, + 104, + 97, + 110, + 105, + 99, + 97, + 108, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 54, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 50, + 57, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 55, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 54, + 48, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 56, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 53, + 48, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 53, + 57, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 48, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 49, + 57, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 49, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 50, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 50, + 44, + 32, + 35, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 51, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 54, + 53, + 44, + 32, + 35, + 53, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 52, + 32, + 61, + 32, + 65, + 68, + 86, + 65, + 78, + 67, + 69, + 68, + 95, + 70, + 65, + 67, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 51, + 48, + 32, + 41, + 44, + 32, + 35, + 51, + 50, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 53, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 54, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 32, + 40, + 32, + 35, + 49, + 57, + 54, + 44, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 55, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 54, + 56, + 32, + 61, + 40, + 32, + 78, + 65, + 77, + 69, + 68, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 42, + 32, + 41, + 32, + 80, + 76, + 65, + 78, + 69, + 95, + 65, + 78, + 71, + 76, + 69, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 41, + 32, + 83, + 73, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 36, + 44, + 32, + 46, + 82, + 65, + 68, + 73, + 65, + 78, + 46, + 32, + 41, + 32, + 41, + 59, + 13, + 10, + 35, + 49, + 54, + 57, + 32, + 61, + 32, + 65, + 68, + 86, + 65, + 78, + 67, + 69, + 68, + 95, + 70, + 65, + 67, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 52, + 51, + 32, + 41, + 44, + 32, + 35, + 54, + 54, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 48, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 55, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 49, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 50, + 32, + 61, + 32, + 67, + 65, + 76, + 69, + 78, + 68, + 65, + 82, + 95, + 68, + 65, + 84, + 69, + 32, + 40, + 32, + 50, + 48, + 50, + 51, + 44, + 32, + 51, + 49, + 44, + 32, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 51, + 32, + 61, + 32, + 65, + 88, + 73, + 83, + 50, + 95, + 80, + 76, + 65, + 67, + 69, + 77, + 69, + 78, + 84, + 95, + 51, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 48, + 52, + 44, + 32, + 35, + 49, + 56, + 52, + 44, + 32, + 35, + 49, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 52, + 32, + 61, + 40, + 32, + 78, + 65, + 77, + 69, + 68, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 42, + 32, + 41, + 32, + 83, + 73, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 36, + 44, + 32, + 46, + 83, + 84, + 69, + 82, + 65, + 68, + 73, + 65, + 78, + 46, + 32, + 41, + 32, + 83, + 79, + 76, + 73, + 68, + 95, + 65, + 78, + 71, + 76, + 69, + 95, + 85, + 78, + 73, + 84, + 32, + 40, + 32, + 41, + 32, + 41, + 59, + 13, + 10, + 35, + 49, + 55, + 53, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 54, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 54, + 57, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 55, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 56, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 53, + 48, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 55, + 57, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 51, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 48, + 32, + 61, + 32, + 83, + 69, + 67, + 85, + 82, + 73, + 84, + 89, + 95, + 67, + 76, + 65, + 83, + 83, + 73, + 70, + 73, + 67, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 39, + 44, + 32, + 39, + 39, + 44, + 32, + 35, + 50, + 50, + 51, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 49, + 32, + 61, + 32, + 70, + 65, + 67, + 69, + 95, + 79, + 85, + 84, + 69, + 82, + 95, + 66, + 79, + 85, + 78, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 48, + 50, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 50, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 48, + 55, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 51, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 52, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 53, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 52, + 44, + 32, + 35, + 49, + 53, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 54, + 32, + 61, + 32, + 76, + 79, + 67, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 49, + 49, + 44, + 32, + 49, + 48, + 44, + 32, + 49, + 57, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 35, + 50, + 50, + 49, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 55, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 68, + 65, + 84, + 69, + 95, + 65, + 78, + 68, + 95, + 84, + 73, + 77, + 69, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 77, + 69, + 78, + 84, + 32, + 40, + 32, + 35, + 53, + 52, + 44, + 32, + 35, + 49, + 57, + 55, + 44, + 32, + 40, + 32, + 35, + 50, + 51, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 56, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 99, + 108, + 97, + 115, + 115, + 105, + 102, + 105, + 99, + 97, + 116, + 105, + 111, + 110, + 95, + 111, + 102, + 102, + 105, + 99, + 101, + 114, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 56, + 57, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 53, + 55, + 44, + 32, + 35, + 53, + 56, + 44, + 32, + 35, + 55, + 54, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 48, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 83, + 84, + 65, + 84, + 85, + 83, + 32, + 40, + 32, + 39, + 110, + 111, + 116, + 95, + 121, + 101, + 116, + 95, + 97, + 112, + 112, + 114, + 111, + 118, + 101, + 100, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 49, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 100, + 101, + 115, + 105, + 103, + 110, + 95, + 115, + 117, + 112, + 112, + 108, + 105, + 101, + 114, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 50, + 32, + 61, + 32, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 35, + 49, + 51, + 52, + 44, + 32, + 35, + 50, + 50, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 51, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 49, + 54, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 52, + 32, + 61, + 32, + 76, + 79, + 67, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 49, + 49, + 44, + 32, + 49, + 48, + 44, + 32, + 49, + 57, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 35, + 50, + 48, + 56, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 53, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 54, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 83, + 84, + 65, + 84, + 85, + 83, + 32, + 40, + 32, + 39, + 110, + 111, + 116, + 95, + 121, + 101, + 116, + 95, + 97, + 112, + 112, + 114, + 111, + 118, + 101, + 100, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 55, + 32, + 61, + 32, + 68, + 65, + 84, + 69, + 95, + 84, + 73, + 77, + 69, + 95, + 82, + 79, + 76, + 69, + 32, + 40, + 32, + 39, + 99, + 114, + 101, + 97, + 116, + 105, + 111, + 110, + 95, + 100, + 97, + 116, + 101, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 56, + 32, + 61, + 32, + 70, + 65, + 67, + 69, + 95, + 79, + 85, + 84, + 69, + 82, + 95, + 66, + 79, + 85, + 78, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 55, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 49, + 57, + 57, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 48, + 55, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 48, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 50, + 56, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 49, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 50, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 53, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 51, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 52, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 53, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 54, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 55, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 56, + 32, + 61, + 32, + 67, + 79, + 79, + 82, + 68, + 73, + 78, + 65, + 84, + 69, + 68, + 95, + 85, + 78, + 73, + 86, + 69, + 82, + 83, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 95, + 79, + 70, + 70, + 83, + 69, + 84, + 32, + 40, + 32, + 53, + 44, + 32, + 48, + 44, + 32, + 46, + 66, + 69, + 72, + 73, + 78, + 68, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 48, + 57, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 48, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 32, + 40, + 32, + 35, + 49, + 57, + 48, + 44, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 49, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 67, + 85, + 82, + 86, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 51, + 53, + 44, + 32, + 35, + 55, + 52, + 44, + 32, + 35, + 50, + 49, + 51, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 50, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 49, + 56, + 57, + 44, + 32, + 46, + 84, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 51, + 32, + 61, + 32, + 76, + 73, + 78, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 48, + 49, + 44, + 32, + 35, + 51, + 51, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 52, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 53, + 32, + 61, + 32, + 86, + 69, + 82, + 84, + 69, + 88, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 50, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 54, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 55, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 56, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 68, + 65, + 84, + 69, + 95, + 65, + 78, + 68, + 95, + 84, + 73, + 77, + 69, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 77, + 69, + 78, + 84, + 32, + 40, + 32, + 35, + 53, + 50, + 44, + 32, + 35, + 49, + 52, + 57, + 44, + 32, + 40, + 32, + 35, + 49, + 56, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 49, + 57, + 32, + 61, + 32, + 65, + 68, + 86, + 65, + 78, + 67, + 69, + 68, + 95, + 70, + 65, + 67, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 56, + 49, + 32, + 41, + 44, + 32, + 35, + 49, + 49, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 48, + 32, + 61, + 32, + 65, + 88, + 73, + 83, + 50, + 95, + 80, + 76, + 65, + 67, + 69, + 77, + 69, + 78, + 84, + 95, + 51, + 68, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 55, + 57, + 44, + 32, + 35, + 49, + 54, + 49, + 44, + 32, + 35, + 50, + 48, + 54, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 49, + 32, + 61, + 32, + 67, + 79, + 79, + 82, + 68, + 73, + 78, + 65, + 84, + 69, + 68, + 95, + 85, + 78, + 73, + 86, + 69, + 82, + 83, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 95, + 79, + 70, + 70, + 83, + 69, + 84, + 32, + 40, + 32, + 53, + 44, + 32, + 48, + 44, + 32, + 46, + 66, + 69, + 72, + 73, + 78, + 68, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 50, + 32, + 61, + 32, + 65, + 68, + 86, + 65, + 78, + 67, + 69, + 68, + 95, + 66, + 82, + 69, + 80, + 95, + 83, + 72, + 65, + 80, + 69, + 95, + 82, + 69, + 80, + 82, + 69, + 83, + 69, + 78, + 84, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 67, + 117, + 98, + 101, + 95, + 49, + 109, + 39, + 44, + 32, + 40, + 32, + 35, + 50, + 51, + 53, + 44, + 32, + 35, + 49, + 32, + 41, + 44, + 32, + 35, + 50, + 49, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 51, + 32, + 61, + 32, + 83, + 69, + 67, + 85, + 82, + 73, + 84, + 89, + 95, + 67, + 76, + 65, + 83, + 83, + 73, + 70, + 73, + 67, + 65, + 84, + 73, + 79, + 78, + 95, + 76, + 69, + 86, + 69, + 76, + 32, + 40, + 32, + 39, + 117, + 110, + 99, + 108, + 97, + 115, + 115, + 105, + 102, + 105, + 101, + 100, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 52, + 32, + 61, + 32, + 76, + 79, + 67, + 65, + 76, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 49, + 49, + 44, + 32, + 49, + 48, + 44, + 32, + 49, + 57, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 35, + 56, + 57, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 53, + 32, + 61, + 32, + 67, + 67, + 95, + 68, + 69, + 83, + 73, + 71, + 78, + 95, + 80, + 69, + 82, + 83, + 79, + 78, + 95, + 65, + 78, + 68, + 95, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 95, + 65, + 83, + 83, + 73, + 71, + 78, + 77, + 69, + 78, + 84, + 32, + 40, + 32, + 35, + 50, + 48, + 44, + 32, + 35, + 49, + 57, + 49, + 44, + 32, + 40, + 32, + 35, + 53, + 49, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 54, + 32, + 61, + 32, + 65, + 68, + 86, + 65, + 78, + 67, + 69, + 68, + 95, + 70, + 65, + 67, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 50, + 50, + 32, + 41, + 44, + 32, + 35, + 53, + 54, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 55, + 32, + 61, + 32, + 69, + 68, + 71, + 69, + 95, + 76, + 79, + 79, + 80, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 49, + 54, + 48, + 44, + 32, + 35, + 50, + 48, + 48, + 44, + 32, + 35, + 54, + 50, + 44, + 32, + 35, + 49, + 56, + 50, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 56, + 32, + 61, + 32, + 79, + 82, + 71, + 65, + 78, + 73, + 90, + 65, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 44, + 32, + 39, + 85, + 78, + 83, + 80, + 69, + 67, + 73, + 70, + 73, + 69, + 68, + 39, + 44, + 32, + 39, + 39, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 50, + 57, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 48, + 32, + 61, + 32, + 79, + 82, + 73, + 69, + 78, + 84, + 69, + 68, + 95, + 69, + 68, + 71, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 42, + 44, + 32, + 42, + 44, + 32, + 35, + 55, + 53, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 49, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 50, + 32, + 61, + 32, + 67, + 65, + 82, + 84, + 69, + 83, + 73, + 65, + 78, + 95, + 80, + 79, + 73, + 78, + 84, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 53, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 51, + 32, + 61, + 32, + 68, + 73, + 82, + 69, + 67, + 84, + 73, + 79, + 78, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 32, + 40, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 44, + 32, + 45, + 49, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 52, + 32, + 61, + 32, + 65, + 68, + 86, + 65, + 78, + 67, + 69, + 68, + 95, + 70, + 65, + 67, + 69, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 40, + 32, + 35, + 51, + 32, + 41, + 44, + 32, + 35, + 49, + 57, + 44, + 32, + 46, + 70, + 46, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 53, + 32, + 61, + 32, + 77, + 65, + 78, + 73, + 70, + 79, + 76, + 68, + 95, + 83, + 79, + 76, + 73, + 68, + 95, + 66, + 82, + 69, + 80, + 32, + 40, + 32, + 39, + 66, + 111, + 115, + 115, + 45, + 69, + 120, + 116, + 114, + 117, + 100, + 101, + 49, + 39, + 44, + 32, + 35, + 54, + 55, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 54, + 32, + 61, + 32, + 86, + 69, + 67, + 84, + 79, + 82, + 32, + 40, + 32, + 39, + 78, + 79, + 78, + 69, + 39, + 44, + 32, + 35, + 49, + 50, + 50, + 44, + 32, + 49, + 48, + 48, + 48, + 46, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 48, + 32, + 41, + 32, + 59, + 13, + 10, + 35, + 50, + 51, + 55, + 32, + 61, + 32, + 65, + 80, + 80, + 82, + 79, + 86, + 65, + 76, + 95, + 68, + 65, + 84, + 69, + 95, + 84, + 73, + 77, + 69, + 32, + 40, + 32, + 35, + 51, + 52, + 44, + 32, + 35, + 57, + 53, + 32, + 41, + 32, + 59, + 13, + 10, + 69, + 78, + 68, + 83, + 69, + 67, + 59, + 13, + 10, + 69, + 78, + 68, + 45, + 73, + 83, + 79, + 45, + 49, + 48, + 51, + 48, + 51, + 45, + 50, + 49, + 59, + 13, + 10 + ] + } + ], + "format": { + "type": "step", + "split_closed_faces": false + } + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "set_object_transform", + "object_id": "[uuid]", + "transforms": [ + { + "translate": { + "property": { + "x": 1020.0, + "y": 0.0, + "z": 0.0 + }, + "set": false, + "is_local": true + }, + "rotate_rpy": null, + "rotate_angle_axis": null, + "scale": null + } + ] + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "object_set_material_params_pbr", + "object_id": "[uuid]", + "color": { + "r": 1.0, + "g": 0.0, + "b": 0.12156863, + "a": 100.0 + }, + "metalness": 0.5, + "roughness": 0.5, + "ambient_occlusion": 0.0 + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "set_object_transform", + "object_id": "[uuid]", + "transforms": [ + { + "translate": { + "property": { + "x": -1020.0, + "y": 0.0, + "z": 0.0 + }, + "set": false, + "is_local": true + }, + "rotate_rpy": null, + "rotate_angle_axis": null, + "scale": null + } + ] + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "object_set_material_params_pbr", + "object_id": "[uuid]", + "color": { + "r": 1.0, + "g": 0.0, + "b": 0.0, + "a": 100.0 + }, + "metalness": 0.5, + "roughness": 0.5, + "ambient_occlusion": 0.0 + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "entity_clone", + "entity_id": "[uuid]" + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "entity_get_all_child_uuids", + "entity_id": "[uuid]" + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "entity_get_all_child_uuids", + "entity_id": "[uuid]" + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "entity_clone", + "entity_id": "[uuid]" + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "entity_get_all_child_uuids", + "entity_id": "[uuid]" + } + }, + { + "cmdId": "[uuid]", + "range": [], + "command": { + "type": "entity_get_all_child_uuids", + "entity_id": "[uuid]" + } + } +] diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_graph_flowchart.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_graph_flowchart.snap new file mode 100644 index 000000000..a9d7d8847 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_graph_flowchart.snap @@ -0,0 +1,6 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Artifact graph flowchart multiple-foreign-imports-all-render.kcl +extension: md +snapshot_kind: binary +--- diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_graph_flowchart.snap.md b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_graph_flowchart.snap.md new file mode 100644 index 000000000..13e533509 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/artifact_graph_flowchart.snap.md @@ -0,0 +1,3 @@ +```mermaid +flowchart LR +``` diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/ast.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/ast.snap new file mode 100644 index 000000000..bf8aa47a3 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/ast.snap @@ -0,0 +1,504 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Result of parsing multiple-foreign-imports-all-render.kcl +--- +{ + "Ok": { + "body": [ + { + "commentStart": 0, + "end": 0, + "path": { + "type": "Foreign", + "path": "../inputs/cube.step" + }, + "selector": { + "type": "None", + "alias": { + "commentStart": 0, + "end": 0, + "name": "cube", + "start": 0, + "type": "Identifier" + } + }, + "start": 0, + "type": "ImportStatement", + "type": "ImportStatement" + }, + { + "commentStart": 0, + "end": 0, + "path": { + "type": "Kcl", + "filename": "othercube.kcl" + }, + "selector": { + "type": "None", + "alias": { + "commentStart": 0, + "end": 0, + "name": "othercube", + "start": 0, + "type": "Identifier" + } + }, + "start": 0, + "type": "ImportStatement", + "type": "ImportStatement" + }, + { + "commentStart": 0, + "end": 0, + "path": { + "type": "Kcl", + "filename": "anothercube.kcl" + }, + "selector": { + "type": "None", + "alias": { + "commentStart": 0, + "end": 0, + "name": "anothercube", + "start": 0, + "type": "Identifier" + } + }, + "start": 0, + "type": "ImportStatement", + "type": "ImportStatement" + }, + { + "commentStart": 0, + "declaration": { + "commentStart": 0, + "end": 0, + "id": { + "commentStart": 0, + "end": 0, + "name": "model", + "start": 0, + "type": "Identifier" + }, + "init": { + "abs_path": false, + "commentStart": 0, + "end": 0, + "name": { + "commentStart": 0, + "end": 0, + "name": "cube", + "start": 0, + "type": "Identifier" + }, + "path": [], + "start": 0, + "type": "Name", + "type": "Name" + }, + "start": 0, + "type": "VariableDeclarator" + }, + "end": 0, + "kind": "const", + "start": 0, + "type": "VariableDeclaration", + "type": "VariableDeclaration" + }, + { + "commentStart": 0, + "end": 0, + "expression": { + "body": [ + { + "abs_path": false, + "commentStart": 0, + "end": 0, + "name": { + "commentStart": 0, + "end": 0, + "name": "othercube", + "start": 0, + "type": "Identifier" + }, + "path": [], + "start": 0, + "type": "Name", + "type": "Name" + }, + { + "arguments": [ + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "x", + "start": 0, + "type": "Identifier" + }, + "arg": { + "commentStart": 0, + "end": 0, + "raw": "1020", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": { + "value": 1020.0, + "suffix": "None" + } + } + } + ], + "callee": { + "abs_path": false, + "commentStart": 0, + "end": 0, + "name": { + "commentStart": 0, + "end": 0, + "name": "translate", + "start": 0, + "type": "Identifier" + }, + "path": [], + "start": 0, + "type": "Name" + }, + "commentStart": 0, + "end": 0, + "start": 0, + "type": "CallExpressionKw", + "type": "CallExpressionKw", + "unlabeled": null + }, + { + "arguments": [ + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "color", + "start": 0, + "type": "Identifier" + }, + "arg": { + "commentStart": 0, + "end": 0, + "raw": "\"#ff001f\"", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": "#ff001f" + } + }, + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "metalness", + "start": 0, + "type": "Identifier" + }, + "arg": { + "commentStart": 0, + "end": 0, + "raw": "50", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": { + "value": 50.0, + "suffix": "None" + } + } + }, + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "roughness", + "start": 0, + "type": "Identifier" + }, + "arg": { + "commentStart": 0, + "end": 0, + "raw": "50", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": { + "value": 50.0, + "suffix": "None" + } + } + } + ], + "callee": { + "abs_path": false, + "commentStart": 0, + "end": 0, + "name": { + "commentStart": 0, + "end": 0, + "name": "appearance", + "start": 0, + "type": "Identifier" + }, + "path": [], + "start": 0, + "type": "Name" + }, + "commentStart": 0, + "end": 0, + "start": 0, + "type": "CallExpressionKw", + "type": "CallExpressionKw", + "unlabeled": null + } + ], + "commentStart": 0, + "end": 0, + "start": 0, + "type": "PipeExpression", + "type": "PipeExpression" + }, + "start": 0, + "type": "ExpressionStatement", + "type": "ExpressionStatement" + }, + { + "commentStart": 0, + "end": 0, + "expression": { + "body": [ + { + "abs_path": false, + "commentStart": 0, + "end": 0, + "name": { + "commentStart": 0, + "end": 0, + "name": "anothercube", + "start": 0, + "type": "Identifier" + }, + "path": [], + "start": 0, + "type": "Name", + "type": "Name" + }, + { + "arguments": [ + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "x", + "start": 0, + "type": "Identifier" + }, + "arg": { + "argument": { + "commentStart": 0, + "end": 0, + "raw": "1020", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": { + "value": 1020.0, + "suffix": "None" + } + }, + "commentStart": 0, + "end": 0, + "operator": "-", + "start": 0, + "type": "UnaryExpression", + "type": "UnaryExpression" + } + } + ], + "callee": { + "abs_path": false, + "commentStart": 0, + "end": 0, + "name": { + "commentStart": 0, + "end": 0, + "name": "translate", + "start": 0, + "type": "Identifier" + }, + "path": [], + "start": 0, + "type": "Name" + }, + "commentStart": 0, + "end": 0, + "start": 0, + "type": "CallExpressionKw", + "type": "CallExpressionKw", + "unlabeled": null + }, + { + "arguments": [ + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "color", + "start": 0, + "type": "Identifier" + }, + "arg": { + "commentStart": 0, + "end": 0, + "raw": "\"#ff0000\"", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": "#ff0000" + } + }, + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "metalness", + "start": 0, + "type": "Identifier" + }, + "arg": { + "commentStart": 0, + "end": 0, + "raw": "50", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": { + "value": 50.0, + "suffix": "None" + } + } + }, + { + "type": "LabeledArg", + "label": { + "commentStart": 0, + "end": 0, + "name": "roughness", + "start": 0, + "type": "Identifier" + }, + "arg": { + "commentStart": 0, + "end": 0, + "raw": "50", + "start": 0, + "type": "Literal", + "type": "Literal", + "value": { + "value": 50.0, + "suffix": "None" + } + } + } + ], + "callee": { + "abs_path": false, + "commentStart": 0, + "end": 0, + "name": { + "commentStart": 0, + "end": 0, + "name": "appearance", + "start": 0, + "type": "Identifier" + }, + "path": [], + "start": 0, + "type": "Name" + }, + "commentStart": 0, + "end": 0, + "start": 0, + "type": "CallExpressionKw", + "type": "CallExpressionKw", + "unlabeled": null + } + ], + "commentStart": 0, + "end": 0, + "start": 0, + "type": "PipeExpression", + "type": "PipeExpression" + }, + "start": 0, + "type": "ExpressionStatement", + "type": "ExpressionStatement" + } + ], + "commentStart": 0, + "end": 0, + "nonCodeMeta": { + "nonCodeNodes": { + "2": [ + { + "commentStart": 0, + "end": 0, + "start": 0, + "type": "NonCodeNode", + "value": { + "type": "newLine" + } + } + ], + "3": [ + { + "commentStart": 0, + "end": 0, + "start": 0, + "type": "NonCodeNode", + "value": { + "type": "newLine" + } + } + ], + "4": [ + { + "commentStart": 0, + "end": 0, + "start": 0, + "type": "NonCodeNode", + "value": { + "type": "newLine" + } + } + ], + "5": [ + { + "commentStart": 0, + "end": 0, + "start": 0, + "type": "NonCodeNode", + "value": { + "type": "newLine" + } + } + ] + }, + "startNodes": [] + }, + "start": 0 + } +} diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/input.kcl b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/input.kcl new file mode 100644 index 000000000..bab3abe34 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/input.kcl @@ -0,0 +1,22 @@ +import "../inputs/cube.step" as cube +import "othercube.kcl" as othercube +import "anothercube.kcl" as anothercube + +model = cube + +othercube + |> translate(x=1020) + |> appearance( + color = "#ff001f", + metalness = 50, + roughness = 50 + ) + +anothercube + |> translate(x=-1020) + |> appearance( + color = "#ff0000", + metalness = 50, + roughness = 50 + ) + diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/ops.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/ops.snap new file mode 100644 index 000000000..08715ecd8 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/ops.snap @@ -0,0 +1,96 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Operations executed multiple-foreign-imports-all-render.kcl +--- +[ + { + "type": "GroupBegin", + "group": { + "type": "ModuleInstance", + "name": "cube", + "moduleId": 6 + }, + "sourceRange": [] + }, + { + "type": "GroupEnd" + }, + { + "type": "GroupBegin", + "group": { + "type": "ModuleInstance", + "name": "othercube", + "moduleId": 7 + }, + "sourceRange": [] + }, + { + "type": "GroupBegin", + "group": { + "type": "ModuleInstance", + "name": "cube", + "moduleId": 6 + }, + "sourceRange": [] + }, + { + "type": "GroupEnd" + }, + { + "labeledArgs": { + "geometry": { + "value": { + "type": "ImportedGeometry", + "artifact_id": "[uuid]" + }, + "sourceRange": [] + } + }, + "name": "clone", + "sourceRange": [], + "type": "StdLibCall", + "unlabeledArg": null + }, + { + "type": "GroupEnd" + }, + { + "type": "GroupBegin", + "group": { + "type": "ModuleInstance", + "name": "anothercube", + "moduleId": 8 + }, + "sourceRange": [] + }, + { + "type": "GroupBegin", + "group": { + "type": "ModuleInstance", + "name": "cube", + "moduleId": 6 + }, + "sourceRange": [] + }, + { + "type": "GroupEnd" + }, + { + "labeledArgs": { + "geometry": { + "value": { + "type": "ImportedGeometry", + "artifact_id": "[uuid]" + }, + "sourceRange": [] + } + }, + "name": "clone", + "sourceRange": [], + "type": "StdLibCall", + "unlabeledArg": null + }, + { + "type": "GroupEnd" + } +] diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/othercube.kcl b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/othercube.kcl new file mode 100644 index 000000000..59e5f233d --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/othercube.kcl @@ -0,0 +1,3 @@ +import "../inputs/cube.step" as cube + +clone(cube) diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/program_memory.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/program_memory.snap new file mode 100644 index 000000000..ea2a86513 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/program_memory.snap @@ -0,0 +1,25 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Variables in memory after executing multiple-foreign-imports-all-render.kcl +--- +{ + "anothercube": { + "type": "Module", + "value": 8 + }, + "cube": { + "type": "Module", + "value": 6 + }, + "model": { + "type": "ImportedGeometry", + "id": "[uuid]", + "value": [ + "cube.step" + ] + }, + "othercube": { + "type": "Module", + "value": 7 + } +} diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/rendered_model.png b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/rendered_model.png new file mode 100644 index 000000000..434ba67e3 Binary files /dev/null and b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/rendered_model.png differ diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed.snap new file mode 100644 index 000000000..22a29c083 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed.snap @@ -0,0 +1,17 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Result of unparsing multiple-foreign-imports-all-render.kcl +--- +import "../inputs/cube.step" as cube +import "othercube.kcl" as othercube +import "anothercube.kcl" as anothercube + +model = cube + +othercube + |> translate(x = 1020) + |> appearance(color = "#ff001f", metalness = 50, roughness = 50) + +anothercube + |> translate(x = -1020) + |> appearance(color = "#ff0000", metalness = 50, roughness = 50) diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed@anothercube.kcl.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed@anothercube.kcl.snap new file mode 100644 index 000000000..3aab90f30 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed@anothercube.kcl.snap @@ -0,0 +1,7 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Result of unparsing tests/multiple-foreign-imports-all-render/anothercube.kcl +--- +import "../inputs/cube.step" as cube + +clone(cube) diff --git a/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed@othercube.kcl.snap b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed@othercube.kcl.snap new file mode 100644 index 000000000..ac56b1f84 --- /dev/null +++ b/rust/kcl-lib/tests/multiple-foreign-imports-all-render/unparsed@othercube.kcl.snap @@ -0,0 +1,7 @@ +--- +source: kcl-lib/src/simulation_tests.rs +description: Result of unparsing tests/multiple-foreign-imports-all-render/othercube.kcl +--- +import "../inputs/cube.step" as cube + +clone(cube) diff --git a/rust/kcl-lib/tests/outputs/serial_test_example_hole1.png b/rust/kcl-lib/tests/outputs/serial_test_example_hole1.png index ef3bdb543..399c5e7c4 100644 Binary files a/rust/kcl-lib/tests/outputs/serial_test_example_hole1.png and b/rust/kcl-lib/tests/outputs/serial_test_example_hole1.png differ diff --git a/rust/kcl-lib/tests/outputs/serial_test_example_hollow1.png b/rust/kcl-lib/tests/outputs/serial_test_example_hollow1.png index f78db1301..0eb238b5d 100644 Binary files a/rust/kcl-lib/tests/outputs/serial_test_example_hollow1.png and b/rust/kcl-lib/tests/outputs/serial_test_example_hollow1.png differ diff --git a/rust/kcl-lib/tests/outputs/serial_test_example_hollow2.png b/rust/kcl-lib/tests/outputs/serial_test_example_hollow2.png index 79c7e6a7d..3e549e7d8 100644 Binary files a/rust/kcl-lib/tests/outputs/serial_test_example_hollow2.png and b/rust/kcl-lib/tests/outputs/serial_test_example_hollow2.png differ diff --git a/rust/kcl-lib/tests/outputs/serial_test_example_shell1.png b/rust/kcl-lib/tests/outputs/serial_test_example_shell1.png index 4247c54a5..2c4add6eb 100644 Binary files a/rust/kcl-lib/tests/outputs/serial_test_example_shell1.png and b/rust/kcl-lib/tests/outputs/serial_test_example_shell1.png differ diff --git a/rust/kcl-lib/tests/outputs/serial_test_example_shell4.png b/rust/kcl-lib/tests/outputs/serial_test_example_shell4.png index af2779855..b5ec6d68c 100644 Binary files a/rust/kcl-lib/tests/outputs/serial_test_example_shell4.png and b/rust/kcl-lib/tests/outputs/serial_test_example_shell4.png differ diff --git a/scripts/diff.js b/scripts/diff.js new file mode 100644 index 000000000..807f68d1d --- /dev/null +++ b/scripts/diff.js @@ -0,0 +1,59 @@ +const fs = require('fs') +const latestRun = fs.readFileSync('/tmp/circular-deps.txt','utf-8') +const knownCircular = fs.readFileSync('./known-circular.txt','utf-8') + +function parseLine (line) { + let num = null + let depPath = null + const res = line.split(")",2) + if (res.length === 2) { + // should be a dep line + num = parseInt(res[0]) + depPath = res[1] + } + return { + num, + depPath + } +} + +function makeDependencyHash (file) { + const deps = {} + file.split("\n").forEach((line)=>{ + const {num, depPath} = parseLine(line) + if (depPath && !isNaN(num)) { + deps[depPath] = 1 + } + }) + return deps +} + +const latestRunDepHash = makeDependencyHash(latestRun) +const knownDepHash = makeDependencyHash(knownCircular) + +const dup1 = JSON.parse(JSON.stringify(latestRunDepHash)) +const dup2 = JSON.parse(JSON.stringify(knownDepHash)) +Object.keys(knownDepHash).forEach((key)=>{ + delete dup1[key] +}) + +Object.keys(latestRunDepHash).forEach((key)=>{ + delete dup2[key] +}) + +console.log(" ") +console.log("diff.js - line item diff") +console.log(" ") +console.log("Added(+)") +Object.keys(dup1).forEach((dep, index)=>{ + console.log(`${index+1}) ${dep}`) +}) + +console.log(" ") +console.log("Removed(-)") +if (Object.keys(dup2).length === 0) { + console.log("None") +} +Object.keys(dup2).forEach((dep, index)=>{ + console.log(`${index+1}) ${dep}`) +}) diff --git a/src/App.tsx b/src/App.tsx index 931798ade..4e501a20e 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -28,13 +28,9 @@ import { PATHS } from '@src/lib/paths' import { takeScreenshotOfVideoStreamCanvas } from '@src/lib/screenshot' import { sceneInfra } from '@src/lib/singletons' import { maybeWriteToDisk } from '@src/lib/telemetry' -import type { IndexLoaderData } from '@src/lib/types' -import { - engineStreamActor, - useSettings, - useToken, -} from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { type IndexLoaderData } from '@src/lib/types' +import { engineStreamActor, useSettings, useToken } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' import { EngineStreamTransition } from '@src/machines/engineStreamMachine' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { CommandBarOpenButton } from '@src/components/CommandBarOpenButton' diff --git a/src/Auth.tsx b/src/Auth.tsx index 5ccc6e801..e4df0e622 100644 --- a/src/Auth.tsx +++ b/src/Auth.tsx @@ -1,5 +1,5 @@ import Loading from '@src/components/Loading' -import { useAuthState } from '@src/machines/appMachine' +import { useAuthState } from '@src/lib/singletons' // Wrapper around protected routes, used in src/Router.tsx export const Auth = ({ children }: React.PropsWithChildren) => { diff --git a/src/Root.tsx b/src/Root.tsx new file mode 100644 index 000000000..9986bc772 --- /dev/null +++ b/src/Root.tsx @@ -0,0 +1,35 @@ +import { AppStateProvider } from '@src/AppState' +import LspProvider from '@src/components/LspProvider' +import { MachineManagerProvider } from '@src/components/MachineManagerProvider' +import { OpenInDesktopAppHandler } from '@src/components/OpenInDesktopAppHandler' +import { SystemIOMachineLogicListenerDesktop } from '@src/components/Providers/SystemIOProviderDesktop' +import { SystemIOMachineLogicListenerWeb } from '@src/components/Providers/SystemIOProviderWeb' +import { RouteProvider } from '@src/components/RouteProvider' +import { KclContextProvider } from '@src/lang/KclProvider' +import { Outlet } from 'react-router-dom' +import { isDesktop } from '@src/lib/isDesktop' +// Root component will live for the entire applications runtime +function RootLayout() { + return ( + + + + + + + {isDesktop() ? ( + + ) : ( + + )} + + + + + + + + ) +} + +export default RootLayout diff --git a/src/Router.tsx b/src/Router.tsx index 16e0a720f..f592783de 100644 --- a/src/Router.tsx +++ b/src/Router.tsx @@ -9,22 +9,15 @@ import { } from 'react-router-dom' import { App } from '@src/App' -import { AppStateProvider } from '@src/AppState' import { Auth } from '@src/Auth' import { CommandBar } from '@src/components/CommandBar/CommandBar' import DownloadAppBanner from '@src/components/DownloadAppBanner' import { ErrorPage } from '@src/components/ErrorPage' import FileMachineProvider from '@src/components/FileMachineProvider' -import LspProvider from '@src/components/LspProvider' -import { MachineManagerProvider } from '@src/components/MachineManagerProvider' import ModelingMachineProvider from '@src/components/ModelingMachineProvider' -import { OpenInDesktopAppHandler } from '@src/components/OpenInDesktopAppHandler' -import { ProjectsContextProvider } from '@src/components/ProjectsContextProvider' -import { RouteProvider } from '@src/components/RouteProvider' import { WasmErrBanner } from '@src/components/WasmErrBanner' import { NetworkContext } from '@src/hooks/useNetworkContext' import { useNetworkStatus } from '@src/hooks/useNetworkStatus' -import { KclContextProvider } from '@src/lang/KclProvider' import { coreDump } from '@src/lang/wasm' import { ASK_TO_OPEN_QUERY_PARAM, @@ -42,7 +35,8 @@ import { rustContext, } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' -import { useToken } from '@src/machines/appMachine' +import { useToken } from '@src/lib/singletons' +import RootLayout from '@src/Root' import Home from '@src/routes/Home' import Onboarding, { onboardingRoutes } from '@src/routes/Onboarding' import { Settings } from '@src/routes/Settings' @@ -54,27 +48,13 @@ const createRouter = isDesktop() ? createHashRouter : createBrowserRouter const router = createRouter([ { id: PATHS.INDEX, - element: ( - - - - - - - - - - - - - - - - ), - errorElement: , + element: , + // Gotcha: declaring errorElement on the root will unmount the element causing our forever React components to unmount. + // Leave errorElement on the child components, this allows for the entire react context on error pages as well. children: [ { path: PATHS.INDEX, + errorElement: , loader: async ({ request }) => { const onDesktop = isDesktop() const url = new URL(request.url) @@ -95,6 +75,7 @@ const router = createRouter([ loader: fileLoader, id: PATHS.FILE, path: PATHS.FILE + '/:id', + errorElement: , element: ( @@ -141,6 +122,7 @@ const router = createRouter([ }, { path: PATHS.HOME, + errorElement: , element: ( @@ -169,6 +151,7 @@ const router = createRouter([ }, { path: PATHS.SIGN_IN, + errorElement: , element: , }, ], diff --git a/src/Toolbar.tsx b/src/Toolbar.tsx index fc9aa65d2..872fc9d96 100644 --- a/src/Toolbar.tsx +++ b/src/Toolbar.tsx @@ -26,7 +26,7 @@ import type { } from '@src/lib/toolbar' import { isToolbarItemResolvedDropdown, toolbarConfig } from '@src/lib/toolbar' import { isArray } from '@src/lib/utils' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' export function Toolbar({ className = '', diff --git a/src/clientSideScene/ClientSideSceneComp.tsx b/src/clientSideScene/ClientSideSceneComp.tsx index 58a36be52..566e1e01a 100644 --- a/src/clientSideScene/ClientSideSceneComp.tsx +++ b/src/clientSideScene/ClientSideSceneComp.tsx @@ -40,8 +40,8 @@ import { } from '@src/lib/singletons' import { err, reportRejection, trap } from '@src/lib/trap' import { throttle, toSync } from '@src/lib/utils' -import type { useSettings } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import type { useSettings } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' import type { SegmentOverlay } from '@src/machines/modelingMachine' function useShouldHideScene(): { hideClient: boolean; hideServer: boolean } { diff --git a/src/clientSideScene/segments.ts b/src/clientSideScene/segments.ts index d5b1ce334..a70177918 100644 --- a/src/clientSideScene/segments.ts +++ b/src/clientSideScene/segments.ts @@ -84,7 +84,7 @@ import { getThemeColorForThreeJs } from '@src/lib/theme' import { err } from '@src/lib/trap' import { isClockwise, normaliseAngle, roundOff } from '@src/lib/utils' import { getTangentPointFromPreviousArc } from '@src/lib/utils2d' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import type { SegmentOverlay, SegmentOverlayPayload, diff --git a/src/components/AppHeader.tsx b/src/components/AppHeader.tsx index 552c3068b..d6b3eb61c 100644 --- a/src/components/AppHeader.tsx +++ b/src/components/AppHeader.tsx @@ -4,7 +4,7 @@ import ProjectSidebarMenu from '@src/components/ProjectSidebarMenu' import UserSidebarMenu from '@src/components/UserSidebarMenu' import { isDesktop } from '@src/lib/isDesktop' import { type IndexLoaderData } from '@src/lib/types' -import { useUser } from '@src/machines/appMachine' +import { useUser } from '@src/lib/singletons' import styles from './AppHeader.module.css' diff --git a/src/components/CameraProjectionToggle.tsx b/src/components/CameraProjectionToggle.tsx index 314fc56f5..aa0c3f7b3 100644 --- a/src/components/CameraProjectionToggle.tsx +++ b/src/components/CameraProjectionToggle.tsx @@ -1,7 +1,7 @@ import { Switch } from '@headlessui/react' import { useEffect, useState } from 'react' -import { settingsActor, useSettings } from '@src/machines/appMachine' +import { settingsActor, useSettings } from '@src/lib/singletons' export function CameraProjectionToggle() { const settings = useSettings() diff --git a/src/components/CommandBar/CommandArgOptionInput.tsx b/src/components/CommandBar/CommandArgOptionInput.tsx index ac4195f0f..a7b34ab1f 100644 --- a/src/components/CommandBar/CommandArgOptionInput.tsx +++ b/src/components/CommandBar/CommandArgOptionInput.tsx @@ -8,10 +8,7 @@ import type { CommandArgument, CommandArgumentOption, } from '@src/lib/commandTypes' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' const contextSelector = (snapshot: StateFrom | undefined) => snapshot?.context diff --git a/src/components/CommandBar/CommandBar.tsx b/src/components/CommandBar/CommandBar.tsx index 09d157699..5bc3efc6f 100644 --- a/src/components/CommandBar/CommandBar.tsx +++ b/src/components/CommandBar/CommandBar.tsx @@ -11,10 +11,7 @@ import { useNetworkContext } from '@src/hooks/useNetworkContext' import { EngineConnectionStateType } from '@src/lang/std/engineConnection' import useHotkeyWrapper from '@src/lib/hotkeyWrapper' import { engineCommandManager } from '@src/lib/singletons' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' import toast from 'react-hot-toast' export const COMMAND_PALETTE_HOTKEY = 'mod+k' diff --git a/src/components/CommandBar/CommandBarArgument.tsx b/src/components/CommandBar/CommandBarArgument.tsx index 6f15e9630..64d6c1f22 100644 --- a/src/components/CommandBar/CommandBarArgument.tsx +++ b/src/components/CommandBar/CommandBarArgument.tsx @@ -7,10 +7,7 @@ import CommandBarSelectionInput from '@src/components/CommandBar/CommandBarSelec import CommandBarSelectionMixedInput from '@src/components/CommandBar/CommandBarSelectionMixedInput' import CommandBarTextareaInput from '@src/components/CommandBar/CommandBarTextareaInput' import type { CommandArgument } from '@src/lib/commandTypes' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' function CommandBarArgument({ stepBack }: { stepBack: () => void }) { const commandBarState = useCommandBarState() diff --git a/src/components/CommandBar/CommandBarBasicInput.tsx b/src/components/CommandBar/CommandBarBasicInput.tsx index 137653bc1..fa315a1bb 100644 --- a/src/components/CommandBar/CommandBarBasicInput.tsx +++ b/src/components/CommandBar/CommandBarBasicInput.tsx @@ -3,10 +3,7 @@ import { useEffect, useMemo, useRef } from 'react' import { useHotkeys } from 'react-hotkeys-hook' import type { CommandArgument } from '@src/lib/commandTypes' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' import type { AnyStateMachine, SnapshotFrom } from 'xstate' // TODO: remove the need for this selector once we decouple all actors from React diff --git a/src/components/CommandBar/CommandBarHeader.tsx b/src/components/CommandBar/CommandBarHeader.tsx index 7d72e31dd..5b95bdd7f 100644 --- a/src/components/CommandBar/CommandBarHeader.tsx +++ b/src/components/CommandBar/CommandBarHeader.tsx @@ -11,10 +11,7 @@ import type { import type { Selections } from '@src/lib/selections' import { getSelectionTypeDisplayText } from '@src/lib/selections' import { roundOff } from '@src/lib/utils' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' function CommandBarHeader({ children }: React.PropsWithChildren) { const commandBarState = useCommandBarState() diff --git a/src/components/CommandBar/CommandBarKclInput.tsx b/src/components/CommandBar/CommandBarKclInput.tsx index 68204a2a7..283ae46b9 100644 --- a/src/components/CommandBar/CommandBarKclInput.tsx +++ b/src/components/CommandBar/CommandBarKclInput.tsx @@ -29,11 +29,8 @@ import { err } from '@src/lib/trap' import { useCalculateKclExpression } from '@src/lib/useCalculateKclExpression' import { roundOff } from '@src/lib/utils' import { varMentions } from '@src/lib/varCompletionExtension' -import { useSettings } from '@src/machines/appMachine' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { useSettings } from '@src/lib/singletons' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' import styles from './CommandBarKclInput.module.css' diff --git a/src/components/CommandBar/CommandBarPathInput.tsx b/src/components/CommandBar/CommandBarPathInput.tsx index 3f57c4acb..a90f02c9a 100644 --- a/src/components/CommandBar/CommandBarPathInput.tsx +++ b/src/components/CommandBar/CommandBarPathInput.tsx @@ -5,10 +5,7 @@ import { ActionButton } from '@src/components/ActionButton' import type { CommandArgument } from '@src/lib/commandTypes' import { reportRejection } from '@src/lib/trap' import { isArray, toSync } from '@src/lib/utils' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' import { useSelector } from '@xstate/react' import type { AnyStateMachine, SnapshotFrom } from 'xstate' diff --git a/src/components/CommandBar/CommandBarReview.tsx b/src/components/CommandBar/CommandBarReview.tsx index 8eb52e9ad..6eb37309e 100644 --- a/src/components/CommandBar/CommandBarReview.tsx +++ b/src/components/CommandBar/CommandBarReview.tsx @@ -1,10 +1,7 @@ import { useHotkeys } from 'react-hotkeys-hook' import CommandBarHeader from '@src/components/CommandBar/CommandBarHeader' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' function CommandBarReview({ stepBack }: { stepBack: () => void }) { const commandBarState = useCommandBarState() diff --git a/src/components/CommandBar/CommandBarSelectionInput.tsx b/src/components/CommandBar/CommandBarSelectionInput.tsx index db06d6473..9b97a2f8b 100644 --- a/src/components/CommandBar/CommandBarSelectionInput.tsx +++ b/src/components/CommandBar/CommandBarSelectionInput.tsx @@ -12,10 +12,7 @@ import { import { engineCommandManager, kclManager } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' import { toSync } from '@src/lib/utils' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' import type { modelingMachine } from '@src/machines/modelingMachine' const semanticEntityNames: { diff --git a/src/components/CommandBar/CommandBarSelectionMixedInput.tsx b/src/components/CommandBar/CommandBarSelectionMixedInput.tsx index f1291c874..a68c556ac 100644 --- a/src/components/CommandBar/CommandBarSelectionMixedInput.tsx +++ b/src/components/CommandBar/CommandBarSelectionMixedInput.tsx @@ -8,10 +8,7 @@ import { getSelectionCountByType, } from '@src/lib/selections' import { kclManager } from '@src/lib/singletons' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' const selectionSelector = (snapshot: any) => snapshot?.context.selectionRanges diff --git a/src/components/CommandBar/CommandBarTextareaInput.tsx b/src/components/CommandBar/CommandBarTextareaInput.tsx index 78c9cb885..ab20a9917 100644 --- a/src/components/CommandBar/CommandBarTextareaInput.tsx +++ b/src/components/CommandBar/CommandBarTextareaInput.tsx @@ -3,10 +3,7 @@ import { useEffect, useRef } from 'react' import { useHotkeys } from 'react-hotkeys-hook' import type { CommandArgument } from '@src/lib/commandTypes' -import { - commandBarActor, - useCommandBarState, -} from '@src/machines/commandBarMachine' +import { commandBarActor, useCommandBarState } from '@src/lib/singletons' function CommandBarTextareaInput({ arg, diff --git a/src/components/CommandBarOpenButton.tsx b/src/components/CommandBarOpenButton.tsx index e90dff26a..4e8d0b844 100644 --- a/src/components/CommandBarOpenButton.tsx +++ b/src/components/CommandBarOpenButton.tsx @@ -1,7 +1,7 @@ import { COMMAND_PALETTE_HOTKEY } from '@src/components/CommandBar/CommandBar' import usePlatform from '@src/hooks/usePlatform' import { hotkeyDisplay } from '@src/lib/hotkeyWrapper' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import { CustomIcon } from '@src/components/CustomIcon' export function CommandBarOpenButton() { diff --git a/src/components/CommandComboBox.tsx b/src/components/CommandComboBox.tsx index 983dfea61..7c3905ea6 100644 --- a/src/components/CommandComboBox.tsx +++ b/src/components/CommandComboBox.tsx @@ -6,7 +6,7 @@ import { CustomIcon } from '@src/components/CustomIcon' import type { Command } from '@src/lib/commandTypes' import { sortCommands } from '@src/lib/commandUtils' import { getActorNextEvents } from '@src/lib/utils' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' function CommandComboBox({ options, diff --git a/src/components/DownloadAppBanner.tsx b/src/components/DownloadAppBanner.tsx index ee955a4fa..0081e1b75 100644 --- a/src/components/DownloadAppBanner.tsx +++ b/src/components/DownloadAppBanner.tsx @@ -3,7 +3,7 @@ import { useState } from 'react' import { ActionButton } from '@src/components/ActionButton' import { CREATE_FILE_URL_PARAM } from '@src/lib/constants' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' import { useSearchParams } from 'react-router-dom' const DownloadAppBanner = () => { diff --git a/src/components/EngineStream.tsx b/src/components/EngineStream.tsx index 842b8b7bb..761a8e066 100644 --- a/src/components/EngineStream.tsx +++ b/src/components/EngineStream.tsx @@ -18,8 +18,8 @@ import { REASONABLE_TIME_TO_REFRESH_STREAM_SIZE } from '@src/lib/timings' import { err, reportRejection, trap } from '@src/lib/trap' import type { IndexLoaderData } from '@src/lib/types' import { uuidv4 } from '@src/lib/utils' -import { engineStreamActor, useSettings } from '@src/machines/appMachine' -import { useCommandBarState } from '@src/machines/commandBarMachine' +import { engineStreamActor, useSettings } from '@src/lib/singletons' +import { useCommandBarState } from '@src/lib/singletons' import { EngineStreamState, EngineStreamTransition, diff --git a/src/components/FileMachineProvider.tsx b/src/components/FileMachineProvider.tsx index 4f596a59f..2d9ffb3c8 100644 --- a/src/components/FileMachineProvider.tsx +++ b/src/components/FileMachineProvider.tsx @@ -33,8 +33,8 @@ import { markOnce } from '@src/lib/performance' import { codeManager, kclManager } from '@src/lib/singletons' import { err, reportRejection } from '@src/lib/trap' import { type IndexLoaderData } from '@src/lib/types' -import { useSettings, useToken } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { useSettings, useToken } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' import { fileMachine } from '@src/machines/fileMachine' import { modelingMenuCallbackMostActions } from '@src/menu/register' @@ -194,6 +194,14 @@ export const FileMachineProvider = ({ navigate(`..${PATHS.FILE}/${encodeURIComponent(event.output.path)}`) } }, + openFileInNewWindow: ({ event }) => { + if (event.type !== 'Open file in new window') { + return + } + + commandBarActor.send({ type: 'Close' }) + window.electron.openInNewWindow(event.data.name) + }, }, actors: { readFiles: fromPromise(async ({ input }) => { diff --git a/src/components/FileTree.tsx b/src/components/FileTree.tsx index da5bc0ec8..342a92bc3 100644 --- a/src/components/FileTree.tsx +++ b/src/components/FileTree.tsx @@ -24,14 +24,15 @@ import { sortFilesAndDirectories } from '@src/lib/desktopFS' import useHotkeyWrapper from '@src/lib/hotkeyWrapper' import { PATHS } from '@src/lib/paths' import type { FileEntry } from '@src/lib/project' -import { codeManager, kclManager } from '@src/lib/singletons' +import { codeManager, kclManager, rustContext } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' import type { IndexLoaderData } from '@src/lib/types' import { ToastInsert } from '@src/components/ToastInsert' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import toast from 'react-hot-toast' import styles from './FileTree.module.css' +import { jsAppSettings } from '@src/lib/settings/settingsUtils' function getIndentationCSS(level: number) { return `calc(1rem * ${level + 1})` @@ -163,6 +164,7 @@ const FileTreeItem = ({ onCreateFile, onCreateFolder, onCloneFileOrFolder, + onOpenInNewWindow, newTreeEntry, level = 0, treeSelection, @@ -183,6 +185,7 @@ const FileTreeItem = ({ onCreateFile: (name: string) => void onCreateFolder: (name: string) => void onCloneFileOrFolder: (path: string) => void + onOpenInNewWindow: (path: string) => void newTreeEntry: TreeEntry level?: number treeSelection: FileEntry | undefined @@ -212,10 +215,25 @@ const FileTreeItem = ({ return } + // TODO: make this not just name based but sub path instead + const isImportedInCurrentFile = kclManager.ast.body.some( + (n) => + n.type === 'ImportStatement' && + ((n.path.type === 'Kcl' && + n.path.filename.includes(fileOrDir.name)) || + (n.path.type === 'Foreign' && n.path.path.includes(fileOrDir.name))) + ) + if (isCurrentFile && eventType === 'change') { let code = await window.electron.readFile(path, { encoding: 'utf-8' }) code = normalizeLineEndings(code) codeManager.updateCodeStateEditor(code) + } else if (isImportedInCurrentFile && eventType === 'change') { + await rustContext.clearSceneAndBustCache( + { settings: await jsAppSettings() }, + codeManager?.currentFilePath || undefined + ) + await kclManager.executeAst() } fileSend({ type: 'Refresh' }) }, @@ -439,6 +457,7 @@ const FileTreeItem = ({ onCreateFile={onCreateFile} onCreateFolder={onCreateFolder} onCloneFileOrFolder={onCloneFileOrFolder} + onOpenInNewWindow={onOpenInNewWindow} newTreeEntry={newTreeEntry} lastDirectoryClicked={lastDirectoryClicked} onClickDirectory={onClickDirectory} @@ -479,6 +498,7 @@ const FileTreeItem = ({ onRename={addCurrentItemToRenaming} onDelete={() => setIsConfirmingDelete(true)} onClone={() => onCloneFileOrFolder(fileOrDir.path)} + onOpenInNewWindow={() => onOpenInNewWindow(fileOrDir.path)} /> ) @@ -489,6 +509,7 @@ interface FileTreeContextMenuProps { onRename: () => void onDelete: () => void onClone: () => void + onOpenInNewWindow: () => void } function FileTreeContextMenu({ @@ -496,6 +517,7 @@ function FileTreeContextMenu({ onRename, onDelete, onClone, + onOpenInNewWindow, }: FileTreeContextMenuProps) { const platform = usePlatform() const metaKey = platform === 'macos' ? '⌘' : 'Ctrl' @@ -525,6 +547,12 @@ function FileTreeContextMenu({ > Clone , + + Open in new window + , ]} /> ) @@ -636,11 +664,21 @@ export const useFileTreeOperations = () => { }) } + function openInNewWindow(args: { path: string }) { + send({ + type: 'Open file in new window', + data: { + name: args.path, + }, + }) + } + return { createFile, createFolder, cloneFileOrDir, newTreeEntry, + openInNewWindow, } } @@ -648,8 +686,13 @@ export const FileTree = ({ className = '', onNavigateToFile: closePanel, }: FileTreeProps) => { - const { createFile, createFolder, cloneFileOrDir, newTreeEntry } = - useFileTreeOperations() + const { + createFile, + createFolder, + cloneFileOrDir, + openInNewWindow, + newTreeEntry, + } = useFileTreeOperations() return (
@@ -666,6 +709,7 @@ export const FileTree = ({ onCreateFile={(name: string) => createFile({ dryRun: false, name })} onCreateFolder={(name: string) => createFolder({ dryRun: false, name })} onCloneFileOrFolder={(path: string) => cloneFileOrDir({ path })} + onOpenInNewWindow={(path: string) => openInNewWindow({ path })} />
) @@ -676,11 +720,13 @@ export const FileTreeInner = ({ onCreateFile, onCreateFolder, onCloneFileOrFolder, + onOpenInNewWindow, newTreeEntry, }: { onCreateFile: (name: string) => void onCreateFolder: (name: string) => void onCloneFileOrFolder: (path: string) => void + onOpenInNewWindow: (path: string) => void newTreeEntry: TreeEntry onNavigateToFile?: () => void }) => { @@ -792,6 +838,7 @@ export const FileTreeInner = ({ onCreateFile={onCreateFile} onCreateFolder={onCreateFolder} onCloneFileOrFolder={onCloneFileOrFolder} + onOpenInNewWindow={onOpenInNewWindow} newTreeEntry={newTreeEntry} onClickDirectory={onClickDirectory} onNavigateToFile={onNavigateToFile_} diff --git a/src/components/Gizmo.tsx b/src/components/Gizmo.tsx index b05dddd88..fadd0d14e 100644 --- a/src/components/Gizmo.tsx +++ b/src/components/Gizmo.tsx @@ -27,7 +27,7 @@ import { useModelingContext } from '@src/hooks/useModelingContext' import { AxisNames } from '@src/lib/constants' import { sceneInfra } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' const CANVAS_SIZE = 80 const FRUSTUM_SIZE = 0.5 diff --git a/src/components/HelpMenu.tsx b/src/components/HelpMenu.tsx index 64c83063d..7274658a4 100644 --- a/src/components/HelpMenu.tsx +++ b/src/components/HelpMenu.tsx @@ -10,7 +10,7 @@ import { createAndOpenNewTutorialProject } from '@src/lib/desktopFS' import { openExternalBrowserIfDesktop } from '@src/lib/openWindow' import { PATHS } from '@src/lib/paths' import { reportRejection } from '@src/lib/trap' -import { settingsActor } from '@src/machines/appMachine' +import { settingsActor } from '@src/lib/singletons' import type { WebContentSendPayload } from '@src/menu/channels' const HelpMenuDivider = () => ( diff --git a/src/components/LspProvider.tsx b/src/components/LspProvider.tsx index 8fd358065..5e9155bec 100644 --- a/src/components/LspProvider.tsx +++ b/src/components/LspProvider.tsx @@ -27,7 +27,7 @@ import { PATHS } from '@src/lib/paths' import type { FileEntry } from '@src/lib/project' import { codeManager } from '@src/lib/singletons' import { err } from '@src/lib/trap' -import { useToken } from '@src/machines/appMachine' +import { useToken } from '@src/lib/singletons' function getWorkspaceFolders(): LSP.WorkspaceFolder[] { return [] diff --git a/src/components/MachineManagerProvider.tsx b/src/components/MachineManagerProvider.tsx index 8a4ca2e3c..6bb2c214f 100644 --- a/src/components/MachineManagerProvider.tsx +++ b/src/components/MachineManagerProvider.tsx @@ -5,7 +5,7 @@ import type { components } from '@src/lib/machine-api' import { engineCommandManager } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' import { toSync } from '@src/lib/utils' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' export type MachinesListing = Array< components['schemas']['MachineInfoResponse'] diff --git a/src/components/ModelingMachineProvider.tsx b/src/components/ModelingMachineProvider.tsx index 8840e6e1a..5624ff039 100644 --- a/src/components/ModelingMachineProvider.tsx +++ b/src/components/ModelingMachineProvider.tsx @@ -111,8 +111,8 @@ import { submitAndAwaitTextToKcl } from '@src/lib/textToCad' import { err, reject, reportRejection, trap } from '@src/lib/trap' import type { IndexLoaderData } from '@src/lib/types' import { platform, uuidv4 } from '@src/lib/utils' -import { useSettings, useToken } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { useSettings, useToken } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' import { kclEditorActor } from '@src/machines/kclEditorMachine' import { getPersistedContext, diff --git a/src/components/ModelingSidebar/ModelingPane.tsx b/src/components/ModelingSidebar/ModelingPane.tsx index 734b8dbd6..b919b8ff0 100644 --- a/src/components/ModelingSidebar/ModelingPane.tsx +++ b/src/components/ModelingSidebar/ModelingPane.tsx @@ -5,7 +5,7 @@ import { ActionButton } from '@src/components/ActionButton' import { ActionIcon } from '@src/components/ActionIcon' import type { CustomIconName } from '@src/components/CustomIcon' import Tooltip from '@src/components/Tooltip' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' import { onboardingPaths } from '@src/routes/Onboarding/paths' import styles from './ModelingPane.module.css' diff --git a/src/components/ModelingSidebar/ModelingPanes/FeatureTreeMenu.tsx b/src/components/ModelingSidebar/ModelingPanes/FeatureTreeMenu.tsx index 964c950f3..bfd99f176 100644 --- a/src/components/ModelingSidebar/ModelingPanes/FeatureTreeMenu.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/FeatureTreeMenu.tsx @@ -2,7 +2,7 @@ import { Menu } from '@headlessui/react' import type { PropsWithChildren } from 'react' import { ActionIcon } from '@src/components/ActionIcon' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import styles from './KclEditorMenu.module.css' diff --git a/src/components/ModelingSidebar/ModelingPanes/KclEditorMenu.tsx b/src/components/ModelingSidebar/ModelingPanes/KclEditorMenu.tsx index 21f38f75c..c21e4db13 100644 --- a/src/components/ModelingSidebar/ModelingPanes/KclEditorMenu.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/KclEditorMenu.tsx @@ -9,7 +9,7 @@ import { useConvertToVariable } from '@src/hooks/useToolbarGuards' import { openExternalBrowserIfDesktop } from '@src/lib/openWindow' import { kclManager } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import styles from './KclEditorMenu.module.css' diff --git a/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx b/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx index 79daaef53..66d847d7a 100644 --- a/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx @@ -47,7 +47,7 @@ import { codeManagerHistoryCompartment } from '@src/lang/codeManager' import { codeManager, editorManager, kclManager } from '@src/lib/singletons' import { Themes, getSystemTheme } from '@src/lib/theme' import { onMouseDragMakeANewNumber, onMouseDragRegex } from '@src/lib/utils' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' import { editorIsMountedSelector, kclEditorActor, diff --git a/src/components/ModelingSidebar/ModelingPanes/index.tsx b/src/components/ModelingSidebar/ModelingPanes/index.tsx index 00ef6224e..ebfcb9020 100644 --- a/src/components/ModelingSidebar/ModelingPanes/index.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/index.tsx @@ -132,8 +132,13 @@ export const sidebarPanes: SidebarPane[] = [ icon: 'folder', sidebarName: 'Project Files', Content: (props: { id: SidebarType; onClose: () => void }) => { - const { createFile, createFolder, cloneFileOrDir, newTreeEntry } = - useFileTreeOperations() + const { + createFile, + createFolder, + cloneFileOrDir, + openInNewWindow, + newTreeEntry, + } = useFileTreeOperations() return ( <> @@ -155,6 +160,7 @@ export const sidebarPanes: SidebarPane[] = [ createFolder({ dryRun: false, name }) } onCloneFileOrFolder={(path: string) => cloneFileOrDir({ path })} + onOpenInNewWindow={(path: string) => openInNewWindow({ path })} newTreeEntry={newTreeEntry} /> diff --git a/src/components/ModelingSidebar/ModelingSidebar.tsx b/src/components/ModelingSidebar/ModelingSidebar.tsx index c19ed466e..146e4c366 100644 --- a/src/components/ModelingSidebar/ModelingSidebar.tsx +++ b/src/components/ModelingSidebar/ModelingSidebar.tsx @@ -22,8 +22,8 @@ import { useKclContext } from '@src/lang/KclProvider' import { EngineConnectionStateType } from '@src/lang/std/engineConnection' import { SIDEBAR_BUTTON_SUFFIX } from '@src/lib/constants' import { isDesktop } from '@src/lib/isDesktop' -import { useSettings } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { useSettings } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { reportRejection } from '@src/lib/trap' import { refreshPage } from '@src/lib/utils' diff --git a/src/components/ProjectSidebarMenu.tsx b/src/components/ProjectSidebarMenu.tsx index 37af452bd..799a8b18f 100644 --- a/src/components/ProjectSidebarMenu.tsx +++ b/src/components/ProjectSidebarMenu.tsx @@ -23,8 +23,8 @@ import { kclManager, } from '@src/lib/singletons' import { type IndexLoaderData } from '@src/lib/types' -import { useToken } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { useToken } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' const ProjectSidebarMenu = ({ project, diff --git a/src/components/ProjectsContextProvider.tsx b/src/components/ProjectsContextProvider.tsx deleted file mode 100644 index 1766f54ff..000000000 --- a/src/components/ProjectsContextProvider.tsx +++ /dev/null @@ -1,492 +0,0 @@ -import { useMachine } from '@xstate/react' -import { createContext, useCallback, useEffect, useState } from 'react' -import toast from 'react-hot-toast' -import { useLocation, useNavigate, useSearchParams } from 'react-router-dom' -import type { Actor, AnyStateMachine, Prop, StateFrom } from 'xstate' -import { fromPromise } from 'xstate' - -import { useLspContext } from '@src/components/LspProvider' -import { useFileSystemWatcher } from '@src/hooks/useFileSystemWatcher' -import { useProjectsLoader } from '@src/hooks/useProjectsLoader' -import useStateMachineCommands from '@src/hooks/useStateMachineCommands' -import { newKclFile } from '@src/lang/project' -import { projectsCommandBarConfig } from '@src/lib/commandBarConfigs/projectsCommandConfig' -import { - CREATE_FILE_URL_PARAM, - FILE_EXT, - PROJECT_ENTRYPOINT, -} from '@src/lib/constants' -import { - createNewProjectDirectory, - listProjects, - renameProjectDirectory, -} from '@src/lib/desktop' -import { - doesProjectNameNeedInterpolated, - getNextFileName, - getNextProjectIndex, - getUniqueProjectName, - interpolateProjectNameWithIndex, -} from '@src/lib/desktopFS' -import { isDesktop } from '@src/lib/isDesktop' -import { PATHS } from '@src/lib/paths' -import type { Project } from '@src/lib/project' -import { codeManager, kclManager } from '@src/lib/singletons' -import { err } from '@src/lib/trap' -import { useSettings } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' -import { projectsMachine } from '@src/machines/projectsMachine' - -type MachineContext = { - state?: StateFrom - send: Prop, 'send'> -} - -export const ProjectsMachineContext = createContext( - {} as MachineContext -) - -/** - * Watches the project directory and provides project management-related commands, - * like "Create project", "Open project", "Delete project", etc. - * - * If in the future we implement full-fledge project management in the web version, - * we can unify these components but for now, we need this to be only for the desktop version. - */ -export const ProjectsContextProvider = ({ - children, -}: { - children: React.ReactNode -}) => { - return isDesktop() ? ( - {children} - ) : ( - {children} - ) -} - -/** - * We need some of the functionality of the ProjectsContextProvider in the web version - * but we can't perform file system operations in the browser, - * so most of the behavior of this machine is stubbed out. - */ -const ProjectsContextWeb = ({ children }: { children: React.ReactNode }) => { - const [searchParams, setSearchParams] = useSearchParams() - const clearImportSearchParams = useCallback(() => { - // Clear the search parameters related to the "Import file from URL" command - // or we'll never be able cancel or submit it. - searchParams.delete(CREATE_FILE_URL_PARAM) - searchParams.delete('code') - searchParams.delete('name') - searchParams.delete('units') - setSearchParams(searchParams) - }, [searchParams, setSearchParams]) - const settings = useSettings() - - const [state, send, actor] = useMachine( - projectsMachine.provide({ - actions: { - navigateToProject: () => {}, - navigateToProjectIfNeeded: () => {}, - navigateToFile: () => {}, - toastSuccess: ({ event }) => - toast.success( - ('data' in event && typeof event.data === 'string' && event.data) || - ('output' in event && - 'message' in event.output && - typeof event.output.message === 'string' && - event.output.message) || - '' - ), - toastError: ({ event }) => - toast.error( - ('data' in event && typeof event.data === 'string' && event.data) || - ('output' in event && - typeof event.output === 'string' && - event.output) || - '' - ), - }, - actors: { - readProjects: fromPromise(async () => [] as Project[]), - createProject: fromPromise(async () => ({ - message: 'not implemented on web', - })), - renameProject: fromPromise(async () => ({ - message: 'not implemented on web', - oldName: '', - newName: '', - })), - deleteProject: fromPromise(async () => ({ - message: 'not implemented on web', - name: '', - })), - createFile: fromPromise(async ({ input }) => { - // Browser version doesn't navigate, just overwrites the current file - clearImportSearchParams() - - const codeToWrite = newKclFile( - input.code, - settings.modeling.defaultUnit.current - ) - if (err(codeToWrite)) return Promise.reject(codeToWrite) - codeManager.updateCodeStateEditor(codeToWrite) - await codeManager.writeToFile() - await kclManager.executeCode() - - return { - message: 'File overwritten successfully', - fileName: input.name, - projectName: '', - } - }), - }, - }), - { - input: { - projects: [], - defaultProjectName: settings.projects.defaultProjectName.current, - defaultDirectory: settings.app.projectDirectory.current, - hasListedProjects: false, - }, - } - ) - - // register all project-related command palette commands - useStateMachineCommands({ - machineId: 'projects', - send, - state, - commandBarConfig: projectsCommandBarConfig, - actor, - onCancel: clearImportSearchParams, - }) - - return ( - - {children} - - ) -} - -const ProjectsContextDesktop = ({ - children, -}: { - children: React.ReactNode -}) => { - const navigate = useNavigate() - const location = useLocation() - const [searchParams, setSearchParams] = useSearchParams() - const clearImportSearchParams = useCallback(() => { - // Clear the search parameters related to the "Import file from URL" command - // or we'll never be able cancel or submit it. - searchParams.delete(CREATE_FILE_URL_PARAM) - searchParams.delete('code') - searchParams.delete('name') - searchParams.delete('units') - setSearchParams(searchParams) - }, [searchParams, setSearchParams]) - const { onProjectOpen } = useLspContext() - const settings = useSettings() - const [projectsLoaderTrigger, setProjectsLoaderTrigger] = useState(0) - const { projectPaths, projectsDir } = useProjectsLoader([ - projectsLoaderTrigger, - ]) - - const [state, send, actor] = useMachine( - projectsMachine.provide({ - actions: { - navigateToProject: ({ context, event }) => { - const nameFromEventData = - 'data' in event && - event.data && - 'name' in event.data && - event.data.name - const nameFromOutputData = - 'output' in event && - event.output && - 'name' in event.output && - event.output.name - - const name = nameFromEventData || nameFromOutputData - - if (name) { - let projectPath = - context.defaultDirectory + window.electron.path.sep + name - onProjectOpen( - { - name, - path: projectPath, - }, - null - ) - commandBarActor.send({ type: 'Close' }) - const newPathName = `${PATHS.FILE}/${encodeURIComponent( - projectPath - )}` - navigate(newPathName) - } - }, - navigateToProjectIfNeeded: ({ event }) => { - if ( - event.type.startsWith('xstate.done.actor.') && - 'output' in event - ) { - const isInAProject = location.pathname.startsWith(PATHS.FILE) - const isInDeletedProject = - event.type === 'xstate.done.actor.delete-project' && - isInAProject && - decodeURIComponent(location.pathname).includes(event.output.name) - if (isInDeletedProject) { - navigate(PATHS.HOME) - return - } - - const isInRenamedProject = - event.type === 'xstate.done.actor.rename-project' && - isInAProject && - decodeURIComponent(location.pathname).includes( - event.output.oldName - ) - - if (isInRenamedProject) { - // TODO: In future, we can navigate to the new project path - // directly, but we need to coordinate with - // @lf94's useFileSystemWatcher in SettingsAuthProvider.tsx:224 - // Because it's beating us to the punch and updating the route - // const newPathName = location.pathname.replace( - // encodeURIComponent(event.output.oldName), - // encodeURIComponent(event.output.newName) - // ) - // navigate(newPathName) - return - } - } - }, - navigateToFile: ({ context, event }) => { - if (event.type !== 'xstate.done.actor.create-file') return - // For now, the browser version of create-file doesn't need to navigate - // since it just overwrites the current file. - if (!isDesktop()) return - let projectPath = window.electron.join( - context.defaultDirectory, - event.output.projectName - ) - let filePath = window.electron.join( - projectPath, - event.output.fileName - ) - onProjectOpen( - { - name: event.output.projectName, - path: projectPath, - }, - null - ) - const pathToNavigateTo = `${PATHS.FILE}/${encodeURIComponent( - filePath - )}` - navigate(pathToNavigateTo) - }, - toastSuccess: ({ event }) => - toast.success( - ('data' in event && typeof event.data === 'string' && event.data) || - ('output' in event && - 'message' in event.output && - typeof event.output.message === 'string' && - event.output.message) || - '' - ), - toastError: ({ event }) => - toast.error( - ('data' in event && typeof event.data === 'string' && event.data) || - ('output' in event && - typeof event.output === 'string' && - event.output) || - ('error' in event && - event.error instanceof Error && - event.error.message) || - '' - ), - }, - actors: { - readProjects: fromPromise(() => { - return listProjects() - }), - createProject: fromPromise(async ({ input }) => { - let name = ( - input && 'name' in input && input.name - ? input.name - : settings.projects.defaultProjectName.current - ).trim() - - const uniqueName = getUniqueProjectName(name, input.projects) - await createNewProjectDirectory(uniqueName) - - return { - message: `Successfully created "${uniqueName}"`, - name: uniqueName, - } - }), - renameProject: fromPromise(async ({ input }) => { - const { - oldName, - newName, - defaultProjectName, - defaultDirectory, - projects, - } = input - let name = newName ? newName : defaultProjectName - if (doesProjectNameNeedInterpolated(name)) { - const nextIndex = getNextProjectIndex(name, projects) - name = interpolateProjectNameWithIndex(name, nextIndex) - } - - // Toast an error if the project name is taken - if (projects.find((p) => p.name === name)) { - return Promise.reject( - new Error(`Project with name "${name}" already exists`) - ) - } - - await renameProjectDirectory( - window.electron.path.join(defaultDirectory, oldName), - name - ) - return { - message: `Successfully renamed "${oldName}" to "${name}"`, - oldName: oldName, - newName: name, - } - }), - deleteProject: fromPromise(async ({ input }) => { - await window.electron.rm( - window.electron.path.join(input.defaultDirectory, input.name), - { - recursive: true, - } - ) - return { - message: `Successfully deleted "${input.name}"`, - name: input.name, - } - }), - createFile: fromPromise(async ({ input }) => { - let projectName = - (input.method === 'newProject' ? input.name : input.projectName) || - settings.projects.defaultProjectName.current - let fileName = - input.method === 'newProject' - ? PROJECT_ENTRYPOINT - : input.name.endsWith(FILE_EXT) - ? input.name - : input.name + FILE_EXT - let message = 'File created successfully' - - const needsInterpolated = doesProjectNameNeedInterpolated(projectName) - if (needsInterpolated) { - const nextIndex = getNextProjectIndex(projectName, input.projects) - projectName = interpolateProjectNameWithIndex( - projectName, - nextIndex - ) - } - - // Create the project around the file if newProject - let fileLoaded = false - if (input.method === 'newProject') { - await createNewProjectDirectory(projectName, input.code) - fileLoaded = true - message = `Project "${projectName}" created successfully with link contents` - } else { - message = `File "${fileName}" created successfully` - } - - // Create the file - let baseDir = window.electron.join( - settings.app.projectDirectory.current, - projectName - ) - const { name, path } = getNextFileName({ - entryName: fileName, - baseDir, - }) - - fileName = name - if (!fileLoaded) { - const codeToWrite = newKclFile( - input.code, - settings.modeling.defaultUnit.current - ) - if (err(codeToWrite)) return Promise.reject(codeToWrite) - await window.electron.writeFile(path, codeToWrite) - } - - // TODO: Return the project's file name if one was created. - return { - message, - fileName, - projectName, - } - }), - }, - }), - { - input: { - projects: projectPaths, - defaultProjectName: settings.projects.defaultProjectName.current, - defaultDirectory: settings.app.projectDirectory.current, - hasListedProjects: false, - }, - } - ) - - useFileSystemWatcher( - async () => { - // Gotcha: Chokidar is buggy. It will emit addDir or add on files that did not get created. - // This means while the application initialize and Chokidar initializes you cannot tell if - // a directory or file is actually created or they are buggy signals. This means you must - // ignore all signals during initialization because it is ambiguous. Once those signals settle - // you can actually start listening to real signals. - // If someone creates folders or files during initialization we ignore those events! - if (!actor.getSnapshot().context.hasListedProjects) { - return - } - return setProjectsLoaderTrigger(projectsLoaderTrigger + 1) - }, - projectsDir ? [projectsDir] : [] - ) - - // Gotcha: Triggers listProjects() on chokidar changes - // Gotcha: Load the projects when the projectDirectory changes. - const projectDirectory = settings.app.projectDirectory.current - useEffect(() => { - send({ type: 'Read projects', data: {} }) - }, [projectPaths, projectDirectory]) - - // register all project-related command palette commands - useStateMachineCommands({ - machineId: 'projects', - send, - state, - commandBarConfig: projectsCommandBarConfig, - actor, - onCancel: clearImportSearchParams, - }) - - return ( - - {children} - - ) -} diff --git a/src/components/Providers/SystemIOProviderDesktop.tsx b/src/components/Providers/SystemIOProviderDesktop.tsx new file mode 100644 index 000000000..f1a6028a7 --- /dev/null +++ b/src/components/Providers/SystemIOProviderDesktop.tsx @@ -0,0 +1,107 @@ +import { useFileSystemWatcher } from '@src/hooks/useFileSystemWatcher' +import { PATHS } from '@src/lib/paths' +import { systemIOActor, useSettings } from '@src/lib/singletons' +import { + useHasListedProjects, + useProjectDirectoryPath, + useRequestedFileName, + useRequestedProjectName, +} from '@src/machines/systemIO/hooks' +import { SystemIOMachineEvents } from '@src/machines/systemIO/utils' +import { useNavigate } from 'react-router-dom' +import { useEffect } from 'react' + +export function SystemIOMachineLogicListenerDesktop() { + const requestedProjectName = useRequestedProjectName() + const requestedFileName = useRequestedFileName() + const projectDirectoryPath = useProjectDirectoryPath() + const hasListedProjects = useHasListedProjects() + const navigate = useNavigate() + const settings = useSettings() + + const useGlobalProjectNavigation = () => { + useEffect(() => { + if (!requestedProjectName.name) { + return + } + let projectPathWithoutSpecificKCLFile = + projectDirectoryPath + + window.electron.path.sep + + requestedProjectName.name + + const requestedPath = `${PATHS.FILE}/${encodeURIComponent( + projectPathWithoutSpecificKCLFile + )}` + navigate(requestedPath) + }, [requestedProjectName]) + } + + const useGlobalFileNavigation = () => { + useEffect(() => { + if (!requestedFileName.file || !requestedFileName.project) { + return + } + const projectPath = window.electron.join( + projectDirectoryPath, + requestedFileName.project + ) + const filePath = window.electron.join(projectPath, requestedFileName.file) + const requestedPath = `${PATHS.FILE}/${encodeURIComponent(filePath)}` + navigate(requestedPath) + }, [requestedFileName]) + } + + const useApplicationProjectDirectory = () => { + useEffect(() => { + systemIOActor.send({ + type: SystemIOMachineEvents.setProjectDirectoryPath, + data: { + requestedProjectDirectoryPath: + settings.app.projectDirectory.current || '', + }, + }) + }, [settings.app.projectDirectory.current]) + } + + const useDefaultProjectName = () => { + useEffect(() => { + systemIOActor.send({ + type: SystemIOMachineEvents.setDefaultProjectFolderName, + data: { + requestedDefaultProjectFolderName: + settings.projects.defaultProjectName.current || '', + }, + }) + }, [settings.projects.defaultProjectName.current]) + } + + const useWatchingApplicationProjectDirectory = () => { + useFileSystemWatcher( + async () => { + // Gotcha: Chokidar is buggy. It will emit addDir or add on files that did not get created. + // This means while the application initialize and Chokidar initializes you cannot tell if + // a directory or file is actually created or they are buggy signals. This means you must + // ignore all signals during initialization because it is ambiguous. Once those signals settle + // you can actually start listening to real signals. + // If someone creates folders or files during initialization we ignore those events! + if (!hasListedProjects) { + return + } + systemIOActor.send({ + type: SystemIOMachineEvents.readFoldersFromProjectDirectory, + }) + }, + settings.app.projectDirectory.current + ? [settings.app.projectDirectory.current] + : [] + ) + } + + useGlobalProjectNavigation() + useGlobalFileNavigation() + useApplicationProjectDirectory() + useDefaultProjectName() + useWatchingApplicationProjectDirectory() + + return null +} diff --git a/src/components/Providers/SystemIOProviderWeb.tsx b/src/components/Providers/SystemIOProviderWeb.tsx new file mode 100644 index 000000000..6c342d7e4 --- /dev/null +++ b/src/components/Providers/SystemIOProviderWeb.tsx @@ -0,0 +1,29 @@ +import { useEffect, useCallback } from 'react' +import { useClearURLParams } from '@src/machines/systemIO/hooks' +import { useSearchParams } from 'react-router-dom' +import { CREATE_FILE_URL_PARAM } from '@src/lib/constants' + +export function SystemIOMachineLogicListenerWeb() { + const clearURLParams = useClearURLParams() + const [searchParams, setSearchParams] = useSearchParams() + const clearImportSearchParams = useCallback(() => { + // Clear the search parameters related to the "Import file from URL" command + // or we'll never be able cancel or submit it. + searchParams.delete(CREATE_FILE_URL_PARAM) + searchParams.delete('code') + searchParams.delete('name') + searchParams.delete('units') + setSearchParams(searchParams) + }, [searchParams, setSearchParams]) + + const useClearQueryParams = () => { + useEffect(() => { + if (clearURLParams.value) { + clearImportSearchParams() + } + }, [clearURLParams]) + } + + useClearQueryParams() + return null +} diff --git a/src/components/RouteProvider.tsx b/src/components/RouteProvider.tsx index c118d20f5..b18f55743 100644 --- a/src/components/RouteProvider.tsx +++ b/src/components/RouteProvider.tsx @@ -18,7 +18,7 @@ import { markOnce } from '@src/lib/performance' import { loadAndValidateSettings } from '@src/lib/settings/settingsUtils' import { trap } from '@src/lib/trap' import type { IndexLoaderData } from '@src/lib/types' -import { settingsActor, useSettings } from '@src/machines/appMachine' +import { settingsActor, useSettings } from '@src/lib/singletons' export const RouteProviderContext = createContext({}) diff --git a/src/components/Settings/AllSettingsFields.tsx b/src/components/Settings/AllSettingsFields.tsx index 7eda78850..9e50f2e8e 100644 --- a/src/components/Settings/AllSettingsFields.tsx +++ b/src/components/Settings/AllSettingsFields.tsx @@ -28,7 +28,7 @@ import { } from '@src/lib/settings/settingsUtils' import { reportRejection } from '@src/lib/trap' import { toSync } from '@src/lib/utils' -import { settingsActor, useSettings } from '@src/machines/appMachine' +import { settingsActor, useSettings } from '@src/lib/singletons' import { APP_VERSION, IS_NIGHTLY, getReleaseUrl } from '@src/routes/utils' import { waitFor } from 'xstate' diff --git a/src/components/Settings/SettingsFieldInput.tsx b/src/components/Settings/SettingsFieldInput.tsx index a5a45bfa3..545c0279c 100644 --- a/src/components/Settings/SettingsFieldInput.tsx +++ b/src/components/Settings/SettingsFieldInput.tsx @@ -9,7 +9,7 @@ import type { WildcardSetEvent, } from '@src/lib/settings/settingsTypes' import { getSettingInputType } from '@src/lib/settings/settingsUtils' -import { settingsActor, useSettings } from '@src/machines/appMachine' +import { settingsActor, useSettings } from '@src/lib/singletons' interface SettingsFieldInputProps { // We don't need the fancy types here, diff --git a/src/components/Settings/SettingsSearchBar.tsx b/src/components/Settings/SettingsSearchBar.tsx index b36429fb3..40e48190f 100644 --- a/src/components/Settings/SettingsSearchBar.tsx +++ b/src/components/Settings/SettingsSearchBar.tsx @@ -8,7 +8,7 @@ import { useNavigate } from 'react-router-dom' import { CustomIcon } from '@src/components/CustomIcon' import { interactionMap } from '@src/lib/settings/initialKeybindings' import type { SettingsLevel } from '@src/lib/settings/settingsTypes' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' type ExtendedSettingsLevel = SettingsLevel | 'keybindings' diff --git a/src/components/Settings/SettingsSectionsList.tsx b/src/components/Settings/SettingsSectionsList.tsx index f1fbfcfd5..e3dea57a5 100644 --- a/src/components/Settings/SettingsSectionsList.tsx +++ b/src/components/Settings/SettingsSectionsList.tsx @@ -3,7 +3,7 @@ import decamelize from 'decamelize' import type { Setting } from '@src/lib/settings/initialSettings' import type { SettingsLevel } from '@src/lib/settings/settingsTypes' import { shouldHideSetting } from '@src/lib/settings/settingsUtils' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' interface SettingsSectionsListProps { searchParamTab: SettingsLevel diff --git a/src/components/ShareButton.tsx b/src/components/ShareButton.tsx index 09a6e2c1e..92bfe08a6 100644 --- a/src/components/ShareButton.tsx +++ b/src/components/ShareButton.tsx @@ -2,7 +2,7 @@ import { CustomIcon } from '@src/components/CustomIcon' import Tooltip from '@src/components/Tooltip' import usePlatform from '@src/hooks/usePlatform' import { hotkeyDisplay } from '@src/lib/hotkeyWrapper' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import { useHotkeys } from 'react-hotkeys-hook' const shareHotkey = 'mod+alt+s' diff --git a/src/components/ToastTextToCad.tsx b/src/components/ToastTextToCad.tsx index c039acef3..bb4d8e3af 100644 --- a/src/components/ToastTextToCad.tsx +++ b/src/components/ToastTextToCad.tsx @@ -33,7 +33,7 @@ import { codeManager, kclManager } from '@src/lib/singletons' import { sendTelemetry } from '@src/lib/textToCadTelemetry' import type { Themes } from '@src/lib/theme' import { reportRejection } from '@src/lib/trap' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import type { fileMachine } from '@src/machines/fileMachine' const CANVAS_SIZE = 128 diff --git a/src/components/UserSidebarMenu.tsx b/src/components/UserSidebarMenu.tsx index 03b32c50b..6ba8085e1 100644 --- a/src/components/UserSidebarMenu.tsx +++ b/src/components/UserSidebarMenu.tsx @@ -11,7 +11,7 @@ import { useAbsoluteFilePath } from '@src/hooks/useAbsoluteFilePath' import usePlatform from '@src/hooks/usePlatform' import { isDesktop } from '@src/lib/isDesktop' import { PATHS } from '@src/lib/paths' -import { authActor } from '@src/machines/appMachine' +import { authActor } from '@src/lib/singletons' type User = Models['User_type'] diff --git a/src/components/ViewControlMenu.tsx b/src/components/ViewControlMenu.tsx index 123131beb..e8c3ce88b 100644 --- a/src/components/ViewControlMenu.tsx +++ b/src/components/ViewControlMenu.tsx @@ -12,7 +12,7 @@ import type { AxisNames } from '@src/lib/constants' import { VIEW_NAMES_SEMANTIC } from '@src/lib/constants' import { sceneInfra } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' export function useViewControlMenuItems() { const { state: modelingState, send: modelingSend } = useModelingContext() diff --git a/src/hooks/useAuthNavigation.tsx b/src/hooks/useAuthNavigation.tsx index 86414a2f3..6b337fc9c 100644 --- a/src/hooks/useAuthNavigation.tsx +++ b/src/hooks/useAuthNavigation.tsx @@ -2,7 +2,7 @@ import { useEffect } from 'react' import { useLocation, useNavigate } from 'react-router-dom' import { PATHS } from '@src/lib/paths' -import { useAuthState } from '@src/machines/appMachine' +import { useAuthState } from '@src/lib/singletons' /** * A simple hook that listens to the auth state of the app and navigates diff --git a/src/hooks/useCreateFileLinkQueryWatcher.ts b/src/hooks/useCreateFileLinkQueryWatcher.ts index 1a0f8cdb0..eff918925 100644 --- a/src/hooks/useCreateFileLinkQueryWatcher.ts +++ b/src/hooks/useCreateFileLinkQueryWatcher.ts @@ -9,7 +9,7 @@ import { CREATE_FILE_URL_PARAM, DEFAULT_FILE_NAME } from '@src/lib/constants' import { isDesktop } from '@src/lib/isDesktop' import type { FileLinkParams } from '@src/lib/links' import { PATHS } from '@src/lib/paths' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' // For initializing the command arguments, we actually want `method` to be undefined // so that we don't skip it in the command palette. diff --git a/src/hooks/useEngineConnectionSubscriptions.ts b/src/hooks/useEngineConnectionSubscriptions.ts index e0137c58a..860e5047e 100644 --- a/src/hooks/useEngineConnectionSubscriptions.ts +++ b/src/hooks/useEngineConnectionSubscriptions.ts @@ -27,7 +27,7 @@ import { } from '@src/lib/singletons' import { err, reportRejection } from '@src/lib/trap' import { getModuleId } from '@src/lib/utils' -import { engineStreamActor } from '@src/machines/appMachine' +import { engineStreamActor } from '@src/lib/singletons' import { EngineStreamState } from '@src/machines/engineStreamMachine' import type { EdgeCutInfo, diff --git a/src/hooks/useMenu.ts b/src/hooks/useMenu.ts index 2e7813d6a..e1d2c1124 100644 --- a/src/hooks/useMenu.ts +++ b/src/hooks/useMenu.ts @@ -2,7 +2,7 @@ import { NetworkHealthState } from '@src/hooks/useNetworkStatus' import { isDesktop } from '@src/lib/isDesktop' import type { ToolbarModeName } from '@src/lib/toolbar' import { reportRejection } from '@src/lib/trap' -import { useCommandBarState } from '@src/machines/commandBarMachine' +import { useCommandBarState } from '@src/lib/singletons' import type { MenuLabels, WebContentSendPayload } from '@src/menu/channels' import { useEffect } from 'react' diff --git a/src/hooks/useProjectsContext.ts b/src/hooks/useProjectsContext.ts deleted file mode 100644 index 2e68535a7..000000000 --- a/src/hooks/useProjectsContext.ts +++ /dev/null @@ -1,7 +0,0 @@ -import { useContext } from 'react' - -import { ProjectsMachineContext } from '@src/components/ProjectsContextProvider' - -export const useProjectsContext = () => { - return useContext(ProjectsMachineContext) -} diff --git a/src/hooks/useResolvedTheme.ts b/src/hooks/useResolvedTheme.ts index 4f13c94cc..8778071d0 100644 --- a/src/hooks/useResolvedTheme.ts +++ b/src/hooks/useResolvedTheme.ts @@ -1,5 +1,5 @@ import { Themes, getSystemTheme } from '@src/lib/theme' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' /** * Resolves the current theme based on the theme setting diff --git a/src/hooks/useStateMachineCommands.ts b/src/hooks/useStateMachineCommands.ts index 47c110092..56c29944a 100644 --- a/src/hooks/useStateMachineCommands.ts +++ b/src/hooks/useStateMachineCommands.ts @@ -12,9 +12,8 @@ import type { } from '@src/lib/commandTypes' import { createMachineCommand } from '@src/lib/createMachineCommand' import type { authMachine } from '@src/machines/authMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import type { modelingMachine } from '@src/machines/modelingMachine' -import type { projectsMachine } from '@src/machines/projectsMachine' import type { settingsMachine } from '@src/machines/settingsMachine' // This might not be necessary, AnyStateMachine from xstate is working @@ -22,7 +21,6 @@ export type AllMachines = | typeof modelingMachine | typeof settingsMachine | typeof authMachine - | typeof projectsMachine interface UseStateMachineCommandsArgs< T extends AllMachines, diff --git a/src/index.tsx b/src/index.tsx index 59d23eac4..1ae0620a2 100644 --- a/src/index.tsx +++ b/src/index.tsx @@ -13,7 +13,7 @@ import { initializeWindowExceptionHandler } from '@src/lib/exceptions' import { isDesktop } from '@src/lib/isDesktop' import { markOnce } from '@src/lib/performance' import { reportRejection } from '@src/lib/trap' -import { appActor } from '@src/machines/appMachine' +import { appActor } from '@src/lib/singletons' import reportWebVitals from '@src/reportWebVitals' markOnce('code/willAuth') diff --git a/src/lang/modifyAst/addEdgeTreatment.ts b/src/lang/modifyAst/addEdgeTreatment.ts index 0cdee28dc..bc840c50b 100644 --- a/src/lang/modifyAst/addEdgeTreatment.ts +++ b/src/lang/modifyAst/addEdgeTreatment.ts @@ -7,7 +7,6 @@ import type CodeManager from '@src/lang/codeManager' import { ARG_TAG } from '@src/lang/constants' import { createArrayExpression, - createCallExpressionStdLib, createCallExpressionStdLibKw, createLabeledArg, createLocalName, @@ -355,9 +354,9 @@ export function getEdgeTagCall( // Modify the tag based on selectionType if (artifact.type === 'sweepEdge' && artifact.subType === 'opposite') { - tagCall = createCallExpressionStdLib('getOppositeEdge', [tagCall]) + tagCall = createCallExpressionStdLibKw('getOppositeEdge', tagCall, []) } else if (artifact.type === 'sweepEdge' && artifact.subType === 'adjacent') { - tagCall = createCallExpressionStdLib('getNextAdjacentEdge', [tagCall]) + tagCall = createCallExpressionStdLibKw('getNextAdjacentEdge', tagCall, []) } return tagCall } diff --git a/src/lang/std/sketch.ts b/src/lang/std/sketch.ts index 6560db5a7..693dd567f 100644 --- a/src/lang/std/sketch.ts +++ b/src/lang/std/sketch.ts @@ -3468,23 +3468,36 @@ function addTagToChamfer( // e.g. chamfer(tags: [getOppositeEdge(tagOfInterest), tag2]) // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + // Note: Single unlabeled arg calls could be either CallExpression or + // CallExpressionKw. const tagMatchesOppositeTagType = edgeCutMeta?.subType === 'opposite' && - tag.type === 'CallExpression' && - tag.callee.name.name === 'getOppositeEdge' && - tag.arguments[0].type === 'Name' && - tag.arguments[0].name.name === edgeCutMeta.tagName + ((tag.type === 'CallExpression' && + tag.callee.name.name === 'getOppositeEdge' && + tag.arguments[0].type === 'Name' && + tag.arguments[0].name.name === edgeCutMeta.tagName) || + (tag.type === 'CallExpressionKw' && + tag.callee.name.name === 'getOppositeEdge' && + tag.unlabeled?.type === 'Name' && + tag.unlabeled.name.name === edgeCutMeta.tagName)) if (tagMatchesOppositeTagType) return true // e.g. chamfer(tags: [getNextAdjacentEdge(tagOfInterest), tag2]) // ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ + // Note: Single unlabeled arg calls could be either CallExpression or + // CallExpressionKw. const tagMatchesAdjacentTagType = edgeCutMeta?.subType === 'adjacent' && - tag.type === 'CallExpression' && - (tag.callee.name.name === 'getNextAdjacentEdge' || - tag.callee.name.name === 'getPrevAdjacentEdge') && - tag.arguments[0].type === 'Name' && - tag.arguments[0].name.name === edgeCutMeta.tagName + ((tag.type === 'CallExpression' && + (tag.callee.name.name === 'getNextAdjacentEdge' || + tag.callee.name.name === 'getPrevAdjacentEdge') && + tag.arguments[0].type === 'Name' && + tag.arguments[0].name.name === edgeCutMeta.tagName) || + (tag.type === 'CallExpressionKw' && + (tag.callee.name.name === 'getNextAdjacentEdge' || + tag.callee.name.name === 'getPrevAdjacentEdge') && + tag.unlabeled?.type === 'Name' && + tag.unlabeled.name.name === edgeCutMeta.tagName)) if (tagMatchesAdjacentTagType) return true return false }) diff --git a/src/lang/wasm.test.ts b/src/lang/wasm.test.ts index e8b094310..c47ec4591 100644 --- a/src/lang/wasm.test.ts +++ b/src/lang/wasm.test.ts @@ -2,7 +2,7 @@ import type { Node } from '@rust/kcl-lib/bindings/Node' import type { Program } from '@rust/kcl-lib/bindings/Program' import type { ParseResult } from '@src/lang/wasm' -import { formatNumber, parse } from '@src/lang/wasm' +import { formatNumber, parse, errFromErrWithOutputs } from '@src/lang/wasm' import { initPromise } from '@src/lang/wasmUtils' import { enginelessExecutor } from '@src/lib/testHelpers' import { err } from '@src/lib/trap' @@ -32,3 +32,15 @@ it('formats numbers with units', () => { expect(formatNumber(0.5, 'Mm')).toEqual('0.5mm') expect(formatNumber(-0.5, 'Mm')).toEqual('-0.5mm') }) + +describe('test errFromErrWithOutputs', () => { + it('converts KclErrorWithOutputs to KclError', () => { + const blob = + '{"error":{"kind":"internal","sourceRanges":[],"msg":"Cache busted"},"operations":[],"artifactCommands":[],"artifactGraph":{"map":{}},"filenames":{},"sourceFiles":{},"defaultPlanes":null}' + const error = errFromErrWithOutputs(blob) + const errorStr = JSON.stringify(error) + expect(errorStr).toEqual( + '{"kind":"internal","sourceRange":[0,0,0],"msg":"Cache busted","operations":[],"artifactCommands":[],"artifactGraph":{},"filenames":{},"defaultPlanes":null}' + ) + }) +}) diff --git a/src/lib/commandBarConfigs/authCommandConfig.ts b/src/lib/commandBarConfigs/authCommandConfig.ts index ad3f6af49..92040491e 100644 --- a/src/lib/commandBarConfigs/authCommandConfig.ts +++ b/src/lib/commandBarConfigs/authCommandConfig.ts @@ -1,26 +1,32 @@ import type { Command } from '@src/lib/commandTypes' -import { authActor } from '@src/machines/appMachine' import { ACTOR_IDS } from '@src/machines/machineConstants' import { refreshPage } from '@src/lib/utils' import { reportRejection } from '@src/lib/trap' +import type { ActorRefFrom } from 'xstate' +import type { authMachine } from '@src/machines/authMachine' -export const authCommands: Command[] = [ - { - groupId: ACTOR_IDS.AUTH, - name: 'log-out', - displayName: 'Log out', - icon: 'arrowLeft', - needsReview: false, - onSubmit: () => authActor.send({ type: 'Log out' }), - }, - { - groupId: ACTOR_IDS.AUTH, - name: 'refresh', - displayName: 'Refresh app', - icon: 'arrowRotateRight', - needsReview: false, - onSubmit: () => { - refreshPage('Command palette').catch(reportRejection) +export function createAuthCommands({ + authActor, +}: { authActor: ActorRefFrom }) { + const authCommands: Command[] = [ + { + groupId: ACTOR_IDS.AUTH, + name: 'log-out', + displayName: 'Log out', + icon: 'arrowLeft', + needsReview: false, + onSubmit: () => authActor.send({ type: 'Log out' }), }, - }, -] + { + groupId: ACTOR_IDS.AUTH, + name: 'refresh', + displayName: 'Refresh app', + icon: 'arrowRotateRight', + needsReview: false, + onSubmit: () => { + refreshPage('Command palette').catch(reportRejection) + }, + }, + ] + return authCommands +} diff --git a/src/lib/commandBarConfigs/modelingCommandConfig.ts b/src/lib/commandBarConfigs/modelingCommandConfig.ts index 35d38d049..6ef285da3 100644 --- a/src/lib/commandBarConfigs/modelingCommandConfig.ts +++ b/src/lib/commandBarConfigs/modelingCommandConfig.ts @@ -742,7 +742,6 @@ export const modelingMachineCommandConfig: StateMachineCommandSetConfig< Fillet: { description: 'Fillet edge', icon: 'fillet3d', - status: 'development', needsReview: true, args: { nodeToEdit: { @@ -772,7 +771,6 @@ export const modelingMachineCommandConfig: StateMachineCommandSetConfig< Chamfer: { description: 'Chamfer edge', icon: 'chamfer3d', - status: 'development', needsReview: true, args: { nodeToEdit: { diff --git a/src/lib/commandBarConfigs/namedViewsConfig.ts b/src/lib/commandBarConfigs/namedViewsConfig.ts index 997e3cd16..0c21af1c4 100644 --- a/src/lib/commandBarConfigs/namedViewsConfig.ts +++ b/src/lib/commandBarConfigs/namedViewsConfig.ts @@ -12,7 +12,7 @@ import type { Command, CommandArgumentOption } from '@src/lib/commandTypes' import { engineCommandManager } from '@src/lib/singletons' import { err, reportRejection } from '@src/lib/trap' import { uuidv4 } from '@src/lib/utils' -import { getSettings, settingsActor } from '@src/machines/appMachine' +import { getSettings, settingsActor } from '@src/lib/singletons' function isWorldCoordinateSystemType( x: string diff --git a/src/lib/commandBarConfigs/projectsCommandConfig.ts b/src/lib/commandBarConfigs/projectsCommandConfig.ts index bc6b4a7bb..5e8907b89 100644 --- a/src/lib/commandBarConfigs/projectsCommandConfig.ts +++ b/src/lib/commandBarConfigs/projectsCommandConfig.ts @@ -1,23 +1,10 @@ import { CommandBarOverwriteWarning } from '@src/components/CommandBarOverwriteWarning' -import type { StateMachineCommandSetConfig } from '@src/lib/commandTypes' import { isDesktop } from '@src/lib/isDesktop' -import type { projectsMachine } from '@src/machines/projectsMachine' - +import type { Command, CommandArgumentOption } from '@src/lib/commandTypes' +import { SystemIOMachineEvents } from '@src/machines/systemIO/utils' +import type { ActorRefFrom } from 'xstate' +import type { systemIOMachine } from '@src/machines/systemIO/systemIOMachine' export type ProjectsCommandSchema = { - 'Read projects': Record - 'Create project': { - name: string - } - 'Open project': { - name: string - } - 'Delete project': { - name: string - } - 'Rename project': { - oldName: string - newName: string - } 'Import file from URL': { name: string code?: string @@ -26,43 +13,101 @@ export type ProjectsCommandSchema = { } } -export const projectsCommandBarConfig: StateMachineCommandSetConfig< - typeof projectsMachine, - ProjectsCommandSchema -> = { - 'Open project': { +export function createProjectCommands({ + systemIOActor, +}: { + systemIOActor: ActorRefFrom +}) { + /** + * Helper functions instead of importing these due to circular deps. + * unable to resolve this in a cleaner way at the moment. + * This is safe in terms of logic but visually ugly. + * TODO: https://github.com/KittyCAD/modeling-app/issues/6032 + */ + const folderSnapshot = () => { + const { folders } = systemIOActor.getSnapshot().context + return folders + } + + const defaultProjectFolderNameSnapshot = () => { + const { defaultProjectFolderName } = systemIOActor.getSnapshot().context + return defaultProjectFolderName + } + + const openProjectCommand: Command = { icon: 'arrowRight', + name: 'Open project', + displayName: `Open project`, description: 'Open a project', - status: isDesktop() ? 'active' : 'inactive', + groupId: 'projects', + needsReview: false, + onSubmit: (record) => { + if (record) { + systemIOActor.send({ + type: SystemIOMachineEvents.navigateToProject, + data: { requestedProjectName: record.name }, + }) + } + }, args: { name: { + required: true, inputType: 'options', - required: true, - options: (_, context) => - context?.projects.map((p) => ({ - name: p.name, - value: p.name, - })) || [], + options: () => { + const folders = folderSnapshot() + const options: CommandArgumentOption[] = [] + folders.forEach((folder) => { + options.push({ + name: folder.name, + value: folder.name, + isCurrent: false, + }) + }) + return options + }, }, }, - }, - 'Create project': { - icon: 'folderPlus', + } + + const createProjectCommand: Command = { + icon: 'folder', + name: 'Create project', + displayName: `Create project`, description: 'Create a project', - status: isDesktop() ? 'active' : 'inactive', + groupId: 'projects', + needsReview: false, + onSubmit: (record) => { + if (record) { + systemIOActor.send({ + type: SystemIOMachineEvents.createProject, + data: { requestedProjectName: record.name }, + }) + } + }, args: { name: { - inputType: 'string', required: true, - defaultValueFromContext: (context) => context.defaultProjectName, + inputType: 'string', + defaultValue: defaultProjectFolderNameSnapshot, }, }, - }, - 'Delete project': { - icon: 'close', + } + + const deleteProjectCommand: Command = { + icon: 'folder', + name: 'Delete project', + displayName: `Delete project`, description: 'Delete a project', - status: isDesktop() ? 'active' : 'inactive', + groupId: 'projects', needsReview: true, + onSubmit: (record) => { + if (record) { + systemIOActor.send({ + type: SystemIOMachineEvents.deleteProject, + data: { requestedProjectName: record.name }, + }) + } + }, reviewMessage: ({ argumentsToSubmit }) => CommandBarOverwriteWarning({ heading: 'Are you sure you want to delete?', @@ -72,41 +117,83 @@ export const projectsCommandBarConfig: StateMachineCommandSetConfig< name: { inputType: 'options', required: true, - options: (_, context) => - context?.projects.map((p) => ({ - name: p.name, - value: p.name, - })) || [], + options: () => { + const folders = folderSnapshot() + const options: CommandArgumentOption[] = [] + folders.forEach((folder) => { + options.push({ + name: folder.name, + value: folder.name, + isCurrent: false, + }) + }) + return options + }, }, }, - }, - 'Rename project': { + } + + const renameProjectCommand: Command = { icon: 'folder', + name: 'Rename project', + displayName: `Rename project`, description: 'Rename a project', + groupId: 'projects', needsReview: true, - status: isDesktop() ? 'active' : 'inactive', + onSubmit: (record) => { + if (record) { + systemIOActor.send({ + type: SystemIOMachineEvents.renameProject, + data: { + requestedProjectName: record.newName, + projectName: record.oldName, + }, + }) + } + }, args: { oldName: { inputType: 'options', required: true, - options: (_, context) => - context?.projects.map((p) => ({ - name: p.name, - value: p.name, - })) || [], + options: () => { + const folders = folderSnapshot() + const options: CommandArgumentOption[] = [] + folders.forEach((folder) => { + options.push({ + name: folder.name, + value: folder.name, + isCurrent: false, + }) + }) + return options + }, }, newName: { inputType: 'string', required: true, - defaultValueFromContext: (context) => context.defaultProjectName, + defaultValue: defaultProjectFolderNameSnapshot, }, }, - }, - 'Import file from URL': { + } + + const importFileFromURL: Command = { + name: 'Import file from URL', + groupId: 'projects', icon: 'file', description: 'Create a file', needsReview: true, - status: 'active', + onSubmit: (record) => { + if (record) { + systemIOActor.send({ + type: SystemIOMachineEvents.importFileFromURL, + data: { + requestedProjectName: record.projectName, + requestedCode: record.code, + requestedFileName: record.name, + }, + }) + } + }, args: { method: { inputType: 'options', @@ -134,11 +221,18 @@ export const projectsCommandBarConfig: StateMachineCommandSetConfig< isDesktop() && commandsContext.argumentsToSubmit.method === 'existingProject', skip: true, - options: (_, context) => - context?.projects.map((p) => ({ - name: p.name, - value: p.name, - })) || [], + options: (_, context) => { + const folders = folderSnapshot() + const options: CommandArgumentOption[] = [] + folders.forEach((folder) => { + options.push({ + name: folder.name, + value: folder.name, + isCurrent: false, + }) + }) + return options + }, }, name: { inputType: 'string', @@ -168,5 +262,18 @@ export const projectsCommandBarConfig: StateMachineCommandSetConfig< }".` : `Will overwrite the contents of the current file with the contents from the URL.` }, - }, + } + + /** No disk-writing commands are available in the browser */ + const projectCommands = isDesktop() + ? [ + openProjectCommand, + createProjectCommand, + deleteProjectCommand, + renameProjectCommand, + importFileFromURL, + ] + : [importFileFromURL] + + return projectCommands } diff --git a/src/lib/desktop.ts b/src/lib/desktop.ts index 94de380fe..d4fd0bae2 100644 --- a/src/lib/desktop.ts +++ b/src/lib/desktop.ts @@ -85,10 +85,23 @@ export async function ensureProjectDirectoryExists( return projectDir } +export async function mkdirOrNOOP(directoryPath: string) { + try { + await window.electron.stat(directoryPath) + } catch (e) { + if (e === 'ENOENT') { + await window.electron.mkdir(directoryPath, { recursive: true }) + } + } + + return directoryPath +} + export async function createNewProjectDirectory( projectName: string, initialCode?: string, - configuration?: DeepPartial | Error + configuration?: DeepPartial | Error, + initialFileName?: string ): Promise { if (!configuration) { configuration = await readAppSettingsFile() @@ -114,7 +127,8 @@ export async function createNewProjectDirectory( } } - const projectFile = window.electron.path.join(projectDir, PROJECT_ENTRYPOINT) + const kclFileName = initialFileName || PROJECT_ENTRYPOINT + const projectFile = window.electron.path.join(projectDir, kclFileName) // When initialCode is present, we're loading existing code. If it's not // present, we're creating a new project, and we want to incorporate the // user's settings. diff --git a/src/lib/routeLoaders.ts b/src/lib/routeLoaders.ts index 8d547e366..de46b4c4a 100644 --- a/src/lib/routeLoaders.ts +++ b/src/lib/routeLoaders.ts @@ -20,7 +20,7 @@ import type { HomeLoaderData, IndexLoaderData, } from '@src/lib/types' -import { settingsActor } from '@src/machines/appMachine' +import { settingsActor } from '@src/lib/singletons' export const telemetryLoader: LoaderFunction = async ({ params, diff --git a/src/lib/selections.ts b/src/lib/selections.ts index 428df0f5e..2de46904d 100644 --- a/src/lib/selections.ts +++ b/src/lib/selections.ts @@ -43,7 +43,7 @@ import { isOverlap, uuidv4, } from '@src/lib/utils' -import { engineStreamActor } from '@src/machines/appMachine' +import { engineStreamActor } from '@src/lib/singletons' import type { ModelingMachineEvent } from '@src/machines/modelingMachine' import { showUnsupportedSelectionToast } from '@src/components/ToastUnsupportedSelection' diff --git a/src/lib/settings/settingsUtils.ts b/src/lib/settings/settingsUtils.ts index fe78ff1bb..93d340fc2 100644 --- a/src/lib/settings/settingsUtils.ts +++ b/src/lib/settings/settingsUtils.ts @@ -191,7 +191,7 @@ export function readLocalStorageAppSettingsFile(): } } -function readLocalStorageProjectSettingsFile(): +export function readLocalStorageProjectSettingsFile(): | DeepPartial | Error { // TODO: Remove backwards compatibility after a few releases. @@ -456,7 +456,8 @@ export function getSettingInputType(setting: Setting) { export const jsAppSettings = async () => { let jsAppSettings = default_app_settings() if (!TEST) { - const settings = await import('@src/machines/appMachine').then((module) => + // TODO: https://github.com/KittyCAD/modeling-app/issues/6445 + const settings = await import('@src/lib/singletons').then((module) => module.getSettings() ) if (settings) { diff --git a/src/lib/singletons.ts b/src/lib/singletons.ts index 7ff9134d0..c81c331a0 100644 --- a/src/lib/singletons.ts +++ b/src/lib/singletons.ts @@ -9,6 +9,26 @@ import { SceneEntities } from '@src/clientSideScene/sceneEntities' import { SceneInfra } from '@src/clientSideScene/sceneInfra' import type { BaseUnit } from '@src/lib/settings/settingsTypes' +import { useSelector } from '@xstate/react' +import type { SnapshotFrom } from 'xstate' +import { createActor, setup, assign } from 'xstate' + +import { isDesktop } from '@src/lib/isDesktop' +import { createSettings } from '@src/lib/settings/initialSettings' +import { authMachine } from '@src/machines/authMachine' +import { + engineStreamContextCreate, + engineStreamMachine, +} from '@src/machines/engineStreamMachine' +import { ACTOR_IDS } from '@src/machines/machineConstants' +import { settingsMachine } from '@src/machines/settingsMachine' +import { systemIOMachineDesktop } from '@src/machines/systemIO/systemIOMachineDesktop' +import { systemIOMachineWeb } from '@src/machines/systemIO/systemIOMachineWeb' +import type { AppMachineContext } from '@src/lib/types' +import { createAuthCommands } from '@src/lib/commandBarConfigs/authCommandConfig' +import { commandBarMachine } from '@src/machines/commandBarMachine' +import { createProjectCommands } from '@src/lib/commandBarConfigs/projectsCommandConfig' + export const codeManager = new CodeManager() export const engineCommandManager = new EngineCommandManager() export const rustContext = new RustContext(engineCommandManager) @@ -90,3 +110,122 @@ if (typeof window !== 'undefined') { }, }) } +const { AUTH, SETTINGS, SYSTEM_IO, ENGINE_STREAM, COMMAND_BAR } = ACTOR_IDS +const appMachineActors = { + [AUTH]: authMachine, + [SETTINGS]: settingsMachine, + [SYSTEM_IO]: isDesktop() ? systemIOMachineDesktop : systemIOMachineWeb, + [ENGINE_STREAM]: engineStreamMachine, + [COMMAND_BAR]: commandBarMachine, +} as const + +const appMachine = setup({ + types: {} as { + context: AppMachineContext + }, + actors: appMachineActors, +}).createMachine({ + id: 'modeling-app', + context: { + codeManager: codeManager, + kclManager: kclManager, + engineCommandManager: engineCommandManager, + sceneInfra: sceneInfra, + sceneEntitiesManager: sceneEntitiesManager, + }, + entry: [ + /** + * We originally wanted to use spawnChild but the inferred type blew up. The more children we + * created the type complexity went through the roof. This functionally should act the same. + * the system and parent internals are tracked properly. After reading the documentation + * it suggests either method but this method requires manual clean up as described in the gotcha + * comment block below. If this becomes an issue we can always move this spawn into createActor functions + * in javascript above and reference those directly but the system and parent internals within xstate + * will not work. + */ + assign({ + // Gotcha, if you use spawn, make sure you remove the ActorRef from context + // to prevent memory leaks when the spawned actor is no longer needed + authActor: ({ spawn }) => spawn(AUTH, { id: AUTH, systemId: AUTH }), + settingsActor: ({ spawn }) => + spawn(SETTINGS, { + id: SETTINGS, + systemId: SETTINGS, + input: createSettings(), + }), + systemIOActor: ({ spawn }) => + spawn(SYSTEM_IO, { id: SYSTEM_IO, systemId: SYSTEM_IO }), + engineStreamActor: ({ spawn }) => + spawn(ENGINE_STREAM, { + id: ENGINE_STREAM, + systemId: ENGINE_STREAM, + input: engineStreamContextCreate(), + }), + commandBarActor: ({ spawn }) => + spawn(COMMAND_BAR, { + id: COMMAND_BAR, + systemId: COMMAND_BAR, + input: { + commands: [], + }, + }), + }), + ], +}) + +export const appActor = createActor(appMachine, { + systemId: 'root', +}) + +/** + * GOTCHA: the type coercion of this actor works because it is spawned for + * the lifetime of {appActor}, but would not work if it were invoked + * or if it were destroyed under any conditions during {appActor}'s life + */ +export const authActor = appActor.getSnapshot().context.authActor! +export const useAuthState = () => useSelector(authActor, (state) => state) +export const useToken = () => + useSelector(authActor, (state) => state.context.token) +export const useUser = () => + useSelector(authActor, (state) => state.context.user) + +/** + * GOTCHA: the type coercion of this actor works because it is spawned for + * the lifetime of {appActor}, but would not work if it were invoked + * or if it were destroyed under any conditions during {appActor}'s life + */ +export const settingsActor = appActor.getSnapshot().context.settingsActor! +export const getSettings = () => { + const { currentProject: _, ...settings } = settingsActor.getSnapshot().context + return settings +} +export const useSettings = () => + useSelector(settingsActor, (state) => { + // We have to peel everything that isn't settings off + const { currentProject, ...settings } = state.context + return settings + }) + +export const systemIOActor = appActor.getSnapshot().context.systemIOActor! + +export const engineStreamActor = + appActor.getSnapshot().context.engineStreamActor! + +export const commandBarActor = appActor.getSnapshot().context.commandBarActor! + +const cmdBarStateSelector = (state: SnapshotFrom) => + state +export const useCommandBarState = () => { + return useSelector(commandBarActor, cmdBarStateSelector) +} + +// Initialize global commands +commandBarActor.send({ + type: 'Add commands', + data: { + commands: [ + ...createAuthCommands({ authActor }), + ...createProjectCommands({ systemIOActor }), + ], + }, +}) diff --git a/src/lib/toolbar.ts b/src/lib/toolbar.ts index e027a9e22..f3909d0d7 100644 --- a/src/lib/toolbar.ts +++ b/src/lib/toolbar.ts @@ -4,7 +4,7 @@ import type { EventFrom, StateFrom } from 'xstate' import type { CustomIconName } from '@src/components/CustomIcon' import { createLiteral } from '@src/lang/create' import { isDesktop } from '@src/lib/isDesktop' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' import type { modelingMachine } from '@src/machines/modelingMachine' import { isEditingExistingSketch, @@ -190,7 +190,7 @@ export const toolbarConfig: Record = { data: { name: 'Fillet', groupId: 'modeling' }, }), icon: 'fillet3d', - status: DEV || IS_NIGHTLY_OR_DEBUG ? 'available' : 'kcl-only', + status: 'available', title: 'Fillet', hotkey: 'F', description: 'Round the edges of a 3D solid.', @@ -204,7 +204,7 @@ export const toolbarConfig: Record = { data: { name: 'Chamfer', groupId: 'modeling' }, }), icon: 'chamfer3d', - status: DEV || IS_NIGHTLY_OR_DEBUG ? 'available' : 'kcl-only', + status: 'available', title: 'Chamfer', hotkey: 'C', description: 'Bevel the edges of a 3D solid.', @@ -492,37 +492,6 @@ export const toolbarConfig: Record = { { id: 'arcs', array: [ - { - id: 'tangential-arc', - onClick: ({ modelingState, modelingSend }) => - modelingSend({ - type: 'change tool', - data: { - tool: !modelingState.matches({ Sketch: 'Tangential arc to' }) - ? 'tangentialArc' - : 'none', - }, - }), - icon: 'arc', - status: 'available', - disabled: (state) => - (!isEditingExistingSketch(state.context) && - !state.matches({ Sketch: 'Tangential arc to' })) || - pipeHasCircle(state.context), - disabledReason: (state) => - !isEditingExistingSketch(state.context) && - !state.matches({ Sketch: 'Tangential arc to' }) - ? "Cannot start a tangential arc because there's no previous line to be tangential to. Try drawing a line first or selecting an existing sketch to edit." - : undefined, - title: 'Tangential Arc', - hotkey: (state) => - state.matches({ Sketch: 'Tangential arc to' }) - ? ['Esc', 'A'] - : 'A', - description: 'Start drawing an arc tangent to the current segment', - links: [], - isActive: (state) => state.matches({ Sketch: 'Tangential arc to' }), - }, { id: 'three-point-arc', onClick: ({ modelingState, modelingSend }) => @@ -554,6 +523,37 @@ export const toolbarConfig: Record = { isActive: (state) => state.matches({ Sketch: 'Arc three point tool' }), }, + { + id: 'tangential-arc', + onClick: ({ modelingState, modelingSend }) => + modelingSend({ + type: 'change tool', + data: { + tool: !modelingState.matches({ Sketch: 'Tangential arc to' }) + ? 'tangentialArc' + : 'none', + }, + }), + icon: 'arc', + status: 'available', + disabled: (state) => + (!isEditingExistingSketch(state.context) && + !state.matches({ Sketch: 'Tangential arc to' })) || + pipeHasCircle(state.context), + disabledReason: (state) => + !isEditingExistingSketch(state.context) && + !state.matches({ Sketch: 'Tangential arc to' }) + ? "Cannot start a tangential arc because there's no previous line to be tangential to. Try drawing a line first or selecting an existing sketch to edit." + : undefined, + title: 'Tangential Arc', + hotkey: (state) => + state.matches({ Sketch: 'Tangential arc to' }) + ? ['Esc', 'A'] + : 'A', + description: 'Start drawing an arc tangent to the current segment', + links: [], + isActive: (state) => state.matches({ Sketch: 'Tangential arc to' }), + }, { id: 'arc', onClick: ({ modelingState, modelingSend }) => diff --git a/src/lib/types.ts b/src/lib/types.ts index 230a09381..db1056360 100644 --- a/src/lib/types.ts +++ b/src/lib/types.ts @@ -1,4 +1,15 @@ import type { FileEntry, Project } from '@src/lib/project' +import type CodeManager from '@src/lang/codeManager' +import type { EngineCommandManager } from '@src/lang/std/engineConnection' +import type { KclManager } from '@src/lang/KclSingleton' +import type { SceneInfra } from '@src/clientSideScene/sceneInfra' +import type { SceneEntities } from '@src/clientSideScene/sceneEntities' +import type { engineStreamMachine } from '@src/machines/engineStreamMachine' +import type { authMachine } from '@src/machines/authMachine' +import type { settingsMachine } from '@src/machines/settingsMachine' +import type { systemIOMachine } from '@src/machines/systemIO/systemIOMachine' +import type { ActorRefFrom } from 'xstate' +import type { commandBarMachine } from '@src/machines/commandBarMachine' export type IndexLoaderData = { code: string | null @@ -111,3 +122,16 @@ export type AsyncFn any> = WithReturnType< F, Promise > + +export type AppMachineContext = { + codeManager: CodeManager + kclManager: KclManager + engineCommandManager: EngineCommandManager + sceneInfra: SceneInfra + sceneEntitiesManager: SceneEntities + authActor?: ActorRefFrom + settingsActor?: ActorRefFrom + systemIOActor?: ActorRefFrom + engineStreamActor?: ActorRefFrom + commandBarActor?: ActorRefFrom +} diff --git a/src/machines/appMachine.ts b/src/machines/appMachine.ts deleted file mode 100644 index fc85a1e37..000000000 --- a/src/machines/appMachine.ts +++ /dev/null @@ -1,78 +0,0 @@ -import { useSelector } from '@xstate/react' -import { createActor, setup, spawnChild } from 'xstate' - -import { createSettings } from '@src/lib/settings/initialSettings' -import { authMachine } from '@src/machines/authMachine' -import type { EngineStreamActor } from '@src/machines/engineStreamMachine' -import { - engineStreamContextCreate, - engineStreamMachine, -} from '@src/machines/engineStreamMachine' -import { ACTOR_IDS } from '@src/machines/machineConstants' -import { settingsMachine } from '@src/machines/settingsMachine' - -const { AUTH, SETTINGS, ENGINE_STREAM } = ACTOR_IDS -const appMachineActors = { - [AUTH]: authMachine, - [SETTINGS]: settingsMachine, - [ENGINE_STREAM]: engineStreamMachine, -} as const - -const appMachine = setup({ - types: {} as { - children: { - auth: typeof AUTH - settings: typeof SETTINGS - } - }, - actors: appMachineActors, -}).createMachine({ - id: 'modeling-app', - entry: [ - spawnChild(AUTH, { id: AUTH, systemId: AUTH }), - spawnChild(SETTINGS, { - id: SETTINGS, - systemId: SETTINGS, - input: createSettings(), - }), - spawnChild(ENGINE_STREAM, { - id: ENGINE_STREAM, - systemId: ENGINE_STREAM, - input: engineStreamContextCreate(), - }), - ], -}) - -export const appActor = createActor(appMachine) -/** - * GOTCHA: the type coercion of this actor works because it is spawned for - * the lifetime of {appActor}, but would not work if it were invoked - * or if it were destroyed under any conditions during {appActor}'s life - */ -export const authActor = appActor.getSnapshot().children.auth! -export const useAuthState = () => useSelector(authActor, (state) => state) -export const useToken = () => - useSelector(authActor, (state) => state.context.token) -export const useUser = () => - useSelector(authActor, (state) => state.context.user) - -/** - * GOTCHA: the type coercion of this actor works because it is spawned for - * the lifetime of {appActor}, but would not work if it were invoked - * or if it were destroyed under any conditions during {appActor}'s life - */ -export const settingsActor = appActor.getSnapshot().children.settings! -export const getSettings = () => { - const { currentProject: _, ...settings } = settingsActor.getSnapshot().context - return settings -} -export const useSettings = () => - useSelector(settingsActor, (state) => { - // We have to peel everything that isn't settings off - const { currentProject, ...settings } = state.context - return settings - }) - -export const engineStreamActor = appActor.system.get( - ENGINE_STREAM -) as EngineStreamActor diff --git a/src/machines/authMachine.ts b/src/machines/authMachine.ts index 7061128bf..0f69597e1 100644 --- a/src/machines/authMachine.ts +++ b/src/machines/authMachine.ts @@ -78,9 +78,7 @@ export const authMachine = setup({ getUser: fromPromise(({ input }: { input: { token?: string } }) => getUser(input) ), - logout: fromPromise(async () => - isDesktop() ? writeTokenFile('') : logout() - ), + logout: fromPromise(logout), }, }).createMachine({ /** @xstate-layout N4IgpgJg5mDOIC5QEECuAXAFgOgMabFwGsBJAMwBkB7KGCEgOwGIIqGxsBLBgNyqI75CRALQAbGnRHcA2gAYAuolAAHKrE7pObZSAAeiAIwAWQ9gBspuQCYAnAGYAHPYCsx+4ccAaEAE9E1q7YcoZyxrYR1m7mcrYAvnE+aFh4BMTk1LSQjExgAE55VHnYKmIAhuhkRQC2qcLikpDSDPJKSCBqGlo67QYI9gDs5tge5o6h5vau7oY+-v3mA9jWco4u5iu21ua2YcYJSRg4Eln0zJkABFQYrbqdmtoMun2GA7YjxuPmLqvGNh5zRCfJaOcyLUzuAYuFyGcwHEDJY6NCAAeQwTEuskUd3UDx6oD6Im2wUcAzkMJ2cjBxlMgIWLmwZLWljecjJTjh8IYVAgcF0iJxXUez0QIgGxhJZIpu2ptL8AWwtje1nCW2iq1shns8MRdXSlGRjEFeKevUQjkcy3sqwGHimbg83nlCF22GMytVUWMMUc8USCKO2BOdCN7Xu3VNBKMKsVFp2hm2vu+1id83slkVrgTxhcW0pNJ1geDkDR6GNEZFCAT1kZZLk9cMLltb0WdPMjewjjC1mzOZCtk5CSAA */ @@ -254,7 +252,11 @@ async function getAndSyncStoredToken(input: { async function logout() { localStorage.removeItem(TOKEN_PERSIST_KEY) - if (isDesktop()) return Promise.resolve(null) + if (isDesktop()) { + await writeTokenFile('') + return Promise.resolve(null) + } + return fetch(withBaseUrl('/logout'), { method: 'POST', credentials: 'include', diff --git a/src/machines/commandBarMachine.ts b/src/machines/commandBarMachine.ts index 6149c7d55..088bc547b 100644 --- a/src/machines/commandBarMachine.ts +++ b/src/machines/commandBarMachine.ts @@ -1,10 +1,6 @@ -import { useSelector } from '@xstate/react' import toast from 'react-hot-toast' -import type { SnapshotFrom } from 'xstate' -import { assign, createActor, fromPromise, setup } from 'xstate' - +import { assign, fromPromise, setup } from 'xstate' import type { MachineManager } from '@src/components/MachineManagerProvider' -import { authCommands } from '@src/lib/commandBarConfigs/authCommandConfig' import type { Command, CommandArgument, @@ -658,16 +654,3 @@ function sortCommands(a: Command, b: Command) { if (a.groupId === 'settings' && !(b.groupId === 'settings')) return 1 return a.name.localeCompare(b.name) } - -export const commandBarActor = createActor(commandBarMachine, { - input: { - commands: [...authCommands], - }, -}).start() - -/** Basic state snapshot selector */ -const cmdBarStateSelector = (state: SnapshotFrom) => - state -export const useCommandBarState = () => { - return useSelector(commandBarActor, cmdBarStateSelector) -} diff --git a/src/machines/engineStreamMachine.ts b/src/machines/engineStreamMachine.ts index 04b656bee..69ac02e1a 100644 --- a/src/machines/engineStreamMachine.ts +++ b/src/machines/engineStreamMachine.ts @@ -1,7 +1,7 @@ -import { engineCommandManager, sceneInfra } from '@src/lib/singletons' import type { MutableRefObject } from 'react' import type { ActorRefFrom } from 'xstate' import { assign, fromPromise, setup } from 'xstate' +import type { AppMachineContext } from '@src/lib/types' export enum EngineStreamState { Off = 'off', @@ -79,9 +79,13 @@ export const engineStreamMachine = setup({ actors: { [EngineStreamTransition.Play]: fromPromise( async ({ - input: { context, params }, + input: { context, params, rootContext }, }: { - input: { context: EngineStreamContext; params: { zoomToFit: boolean } } + input: { + context: EngineStreamContext + params: { zoomToFit: boolean } + rootContext: AppMachineContext + } }) => { const canvas = context.canvasRef.current if (!canvas) return false @@ -98,7 +102,7 @@ export const engineStreamMachine = setup({ return } - await sceneInfra.camControls.restoreRemoteCameraStateAndTriggerSync() + await rootContext.sceneInfra.camControls.restoreRemoteCameraStateAndTriggerSync() video.style.display = 'block' canvas.style.display = 'none' @@ -108,9 +112,9 @@ export const engineStreamMachine = setup({ ), [EngineStreamTransition.Pause]: fromPromise( async ({ - input: { context }, + input: { context, rootContext }, }: { - input: { context: EngineStreamContext } + input: { context: EngineStreamContext; rootContext: AppMachineContext } }) => { const video = context.videoRef.current if (!video) return @@ -123,7 +127,7 @@ export const engineStreamMachine = setup({ await holdOntoVideoFrameInCanvas(video, canvas) video.style.display = 'none' - await sceneInfra.camControls.saveRemoteCameraState() + await rootContext.sceneInfra.camControls.saveRemoteCameraState() // Make sure we're on the next frame for no flickering between canvas // and the video elements. @@ -138,16 +142,20 @@ export const engineStreamMachine = setup({ context.mediaStream = null video.srcObject = null - engineCommandManager.tearDown({ idleMode: true }) + rootContext.engineCommandManager.tearDown({ idleMode: true }) })() ) } ), [EngineStreamTransition.StartOrReconfigureEngine]: fromPromise( async ({ - input: { context, event }, + input: { context, event, rootContext }, }: { - input: { context: EngineStreamContext; event: any } + input: { + context: EngineStreamContext + event: any + rootContext: AppMachineContext + } }) => { if (!context.authToken) return @@ -172,10 +180,10 @@ export const engineStreamMachine = setup({ ...event.settings, } - engineCommandManager.settings = settingsNext + rootContext.engineCommandManager.settings = settingsNext window.requestAnimationFrame(() => { - engineCommandManager.start({ + rootContext.engineCommandManager.start({ setMediaStream: event.onMediaStream, setIsStreamReady: (isStreamReady: boolean) => { event.setAppState({ isStreamReady }) @@ -225,7 +233,12 @@ export const engineStreamMachine = setup({ reenter: true, invoke: { src: EngineStreamTransition.StartOrReconfigureEngine, - input: (args) => args, + input: (args) => ({ + context: args.context, + rootContext: args.self.system.get('root').getSnapshot().context, + params: { zoomToFit: args.context.zoomToFit }, + event: args.event, + }), }, on: { // Transition requested by engineConnection @@ -246,6 +259,7 @@ export const engineStreamMachine = setup({ src: EngineStreamTransition.Play, input: (args) => ({ context: args.context, + rootContext: args.self.system.get('root').getSnapshot().context, params: { zoomToFit: args.context.zoomToFit }, }), }, @@ -261,7 +275,11 @@ export const engineStreamMachine = setup({ [EngineStreamState.Reconfiguring]: { invoke: { src: EngineStreamTransition.StartOrReconfigureEngine, - input: (args) => args, + input: (args) => ({ + context: args.context, + rootContext: args.self.system.get('root').getSnapshot().context, + event: args.event, + }), onDone: { target: EngineStreamState.Playing, }, @@ -270,7 +288,10 @@ export const engineStreamMachine = setup({ [EngineStreamState.Paused]: { invoke: { src: EngineStreamTransition.Pause, - input: (args) => args, + input: (args) => ({ + context: args.context, + rootContext: args.self.system.get('root').getSnapshot().context, + }), }, on: { [EngineStreamTransition.StartOrReconfigureEngine]: { @@ -282,7 +303,11 @@ export const engineStreamMachine = setup({ reenter: true, invoke: { src: EngineStreamTransition.StartOrReconfigureEngine, - input: (args) => args, + input: (args) => ({ + context: args.context, + rootContext: args.self.system.get('root').getSnapshot().context, + event: args.event, + }), }, on: { // The stream can be paused as it's resuming. diff --git a/src/machines/featureTreeMachine.ts b/src/machines/featureTreeMachine.ts index f3942c3ca..7310d3bf2 100644 --- a/src/machines/featureTreeMachine.ts +++ b/src/machines/featureTreeMachine.ts @@ -20,7 +20,7 @@ import { } from '@src/lib/operations' import { kclManager } from '@src/lib/singletons' import { err } from '@src/lib/trap' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { commandBarActor } from '@src/lib/singletons' type FeatureTreeEvent = | { diff --git a/src/machines/fileMachine.ts b/src/machines/fileMachine.ts index a200226c5..06616ee1d 100644 --- a/src/machines/fileMachine.ts +++ b/src/machines/fileMachine.ts @@ -10,6 +10,7 @@ type FileMachineContext = { type FileMachineEvents = | { type: 'Open file'; data: { name: string } } + | { type: 'Open file in new window'; data: { name: string } } | { type: 'Rename file' data: { oldName: string; newName: string; isDir: boolean } @@ -95,6 +96,7 @@ export const fileMachine = setup({ }, }), navigateToFile: () => {}, + openFileInNewWindow: () => {}, renameToastSuccess: () => {}, createToastSuccess: () => {}, toastSuccess: () => {}, @@ -213,6 +215,10 @@ export const fileMachine = setup({ target: 'Opening file', }, + 'Open file in new window': { + target: 'Opening file in new window', + }, + 'Set selected directory': { target: 'Has files', actions: ['setSelectedDirectory'], @@ -400,6 +406,10 @@ export const fileMachine = setup({ entry: ['navigateToFile'], }, + 'Opening file in new window': { + entry: ['openFileInNewWindow'], + }, + 'Creating file': { invoke: { src: 'createFile', diff --git a/src/machines/machineConstants.ts b/src/machines/machineConstants.ts index 50986d3a3..399b7961c 100644 --- a/src/machines/machineConstants.ts +++ b/src/machines/machineConstants.ts @@ -1,5 +1,7 @@ export const ACTOR_IDS = { AUTH: 'auth', SETTINGS: 'settings', + SYSTEM_IO: 'systemIO', ENGINE_STREAM: 'engine_stream', + COMMAND_BAR: 'command_bar', } as const diff --git a/src/machines/projectsMachine.ts b/src/machines/projectsMachine.ts deleted file mode 100644 index 1be6a2d71..000000000 --- a/src/machines/projectsMachine.ts +++ /dev/null @@ -1,337 +0,0 @@ -import { assign, fromPromise, setup } from 'xstate' - -import type { ProjectsCommandSchema } from '@src/lib/commandBarConfigs/projectsCommandConfig' -import type { Project } from '@src/lib/project' -import { isArray } from '@src/lib/utils' - -export const projectsMachine = setup({ - types: { - context: {} as { - projects: Project[] - defaultProjectName: string - defaultDirectory: string - hasListedProjects: boolean - }, - events: {} as - | { type: 'Read projects'; data: ProjectsCommandSchema['Read projects'] } - | { type: 'Open project'; data: ProjectsCommandSchema['Open project'] } - | { - type: 'Rename project' - data: ProjectsCommandSchema['Rename project'] - } - | { - type: 'Create project' - data: ProjectsCommandSchema['Create project'] - } - | { - type: 'Delete project' - data: ProjectsCommandSchema['Delete project'] - } - | { - type: 'Import file from URL' - data: ProjectsCommandSchema['Import file from URL'] - } - | { type: 'navigate'; data: { name: string } } - | { - type: 'xstate.done.actor.read-projects' - output: Project[] - } - | { - type: 'xstate.done.actor.delete-project' - output: { message: string; name: string } - } - | { - type: 'xstate.done.actor.create-project' - output: { message: string; name: string } - } - | { - type: 'xstate.done.actor.rename-project' - output: { message: string; oldName: string; newName: string } - } - | { - type: 'xstate.done.actor.create-file' - output: { message: string; projectName: string; fileName: string } - } - | { type: 'assign'; data: { [key: string]: any } }, - input: {} as { - projects: Project[] - defaultProjectName: string - defaultDirectory: string - hasListedProjects: boolean - }, - }, - actions: { - setProjects: assign({ - projects: ({ context, event }) => - 'output' in event && isArray(event.output) - ? event.output - : context.projects, - }), - setHasListedProjects: assign({ - hasListedProjects: () => true, - }), - toastSuccess: () => {}, - toastError: () => {}, - navigateToProject: () => {}, - navigateToProjectIfNeeded: () => {}, - navigateToFile: () => {}, - }, - actors: { - readProjects: fromPromise(() => Promise.resolve([] as Project[])), - createProject: fromPromise( - (_: { input: { name: string; projects: Project[] } }) => - Promise.resolve({ message: '' }) - ), - renameProject: fromPromise( - (_: { - input: { - oldName: string - newName: string - defaultProjectName: string - defaultDirectory: string - projects: Project[] - } - }) => - Promise.resolve({ - message: '', - oldName: '', - newName: '', - }) - ), - deleteProject: fromPromise( - (_: { input: { defaultDirectory: string; name: string } }) => - Promise.resolve({ - message: '', - name: '', - }) - ), - createFile: fromPromise( - (_: { - input: ProjectsCommandSchema['Import file from URL'] & { - projects: Project[] - } - }) => Promise.resolve({ message: '', projectName: '', fileName: '' }) - ), - }, - guards: { - 'Has at least 1 project': ({ event }) => { - if (event.type !== 'xstate.done.actor.read-projects') return false - return event.output.length ? event.output.length >= 1 : false - }, - }, -}).createMachine({ - /** @xstate-layout N4IgpgJg5mDOIC5QAkD2BbMACdBDAxgBYCWAdmAMS6yzFSkDaADALqKgAOqtALsaqXYgAHogAsAJgA0IAJ6IAjAHYAbADoArBJVMFTCQA4mTAMwmxAXwsy0mHARLkKASXRcATjywAzYgBtsb3cMLABVACUAGWY2JBAuXn5BONEESRl5BAUFFQM1HQUxJSYATmyFAyUTKxsMbDwiMjA1ZGosUlQsDmCAKzB8HlgKcLBcCC7e-sGYoQTiPgEhVIUy9SKSiQ0NEwkclRyMxGKJNRKlMRVDU23zpRqQW3qHJpa2jonUPoGhgGF3UZ42G6nymMzicwWyVAyzKJzECg0OiYGjERhK+kOWUMSlOGiUlTEJlMxRKBnuj3sjXIr1gHy+g2Go3GwPpsDBnG48ySS0QGj0+Q0JTOGkqBg2JhKmNRJy2OyUEn0Kml5LqlMczVatJZUyGI1IuDs2oG7PinMhPIQfKYAqFShF+PFkrkRwkYjU8OUShKKhMKg0pUs1geqoa6ppdJ1FD+AKBk2NrFmZu5KV5-L9tvtYokEsxelRagUCoMiMumyUdpVdlDL01Ee+FAAImAAoC6zwTRDk9DU9b08LRY7MWd1AZcoS-aoLiYFJWnlSNW0jQyAPIcMCkNsdpOLFOWtOC-sO7NOzLbGWFbY6acmFESWdql7R3B8UhQNsUCACZpkABuqAA1s0+D-M+YAALRLluiQ7t2WRMGIbryjeBgGBIEglNsCi5sY6hMOchQqEoCLFCh97VtST4vm+S4UGA7jBO4agcH4z7eKg7joGowExhBcbtgm4LblCIiKPBiHZiKqHoZhuaFhoBbGMYBhiPBCgStUQYUuRzR6gaZDUXxH5fmov4Ac0-z6pgvEgvGsQctBwnLGJahIZJaEYdOmKEfJuQSoRRKSRcZHPNSunoPp750QxTEsTwbEcWoFkGuBkECfZXIwSJcEIS5Ekoe5MnOggXqIaUqFiiUhIInemkhiFzRNi2EU0Z+1KmYBagQM2YCAtZ9JQRljmiTlrn5dJnlFShOLwcpZjKBs+KBrUVb1WojU9c1hlRexMWsexnFdS2KV8QN5q7laNqHlmOaTcpmjoWc8L6HoYrBfOagjGMm02QyrXfqQf4dSBEB9Tqp1dllebichUkeVhRXTiU+QecUKhCps4pvWGn0QN9rJGW1ANmYlTKg98DAKHZpoORanoyqh8oImU3pKJifJ5Ohdr7CYqjIvKWMvDjeORttjHMXtCXA2T0xpdTg20+W9MSIzgorIRXlpr6ORbPs+jwQLFEgVRPj+JQf0mUTHXcaBYG+AE4OZcsxbuts5hbCK+zFmImJgSc5zPV6BQVD6RQG80lERXblCi7tcX7VxRvgVHDtDQgaE4vK2gXKiTA6ItubmJo8IoqOylERUVhBh0XXwHEWn1YmNO7mBKg+2KpzK2IpIBUwBhqUtwYre9tbvEutfpWdsFFnkPpVJnQqz15Ozur36FoypREbGH4Zj438u7tO1qIu5qK5PBGyYjebpEkS44e9oVTbxHr5tnvk9ZeXajTuW+zaHNuzYRUmoeCEp-RKlUHJbeYVhYDDfhDVIxR5IGGnISQk6E+6EkxMUBQX9c66EMMg3Q5Zt7rWNkuOBjsjilDUAqQsZQzzgOkJNUk7o7SmF-tiXOUCmQwMGBQ1OF4TgVGzFUG8yIxQGDZiYPI4oqh+mPsrDSy05xhmfm+KO-CLT+iRucEUuwFQqWzMWXM2YTAuTtL6ZBylkRcMrkAA */ - id: 'Home machine', - - initial: 'Reading projects', - - context: ({ input }) => ({ - ...input, - }), - - on: { - assign: { - actions: assign(({ event }) => ({ - ...event.data, - })), - }, - - 'Import file from URL': '.Creating file', - }, - states: { - 'Has no projects': { - on: { - 'Read projects': { - target: 'Reading projects', - }, - 'Create project': { - target: 'Creating project', - }, - }, - }, - - 'Has projects': { - on: { - 'Read projects': { - target: 'Reading projects', - }, - - 'Rename project': { - target: 'Renaming project', - }, - - 'Create project': { - target: 'Creating project', - }, - - 'Delete project': { - target: 'Deleting project', - }, - - 'Open project': { - target: 'Reading projects', - actions: 'navigateToProject', - reenter: true, - }, - }, - }, - - 'Creating project': { - invoke: { - id: 'create-project', - src: 'createProject', - input: ({ event, context }) => { - if ( - event.type !== 'Create project' && - event.type !== 'Import file from URL' - ) { - return { - name: '', - projects: context.projects, - } - } - return { - name: event.data.name, - projects: context.projects, - } - }, - onDone: [ - { - target: 'Reading projects', - actions: ['toastSuccess', 'navigateToProject'], - }, - ], - onError: [ - { - target: 'Reading projects', - actions: ['toastError'], - }, - ], - }, - }, - - 'Renaming project': { - invoke: { - id: 'rename-project', - src: 'renameProject', - input: ({ event, context }) => { - if (event.type !== 'Rename project') { - // This is to make TS happy - return { - defaultProjectName: context.defaultProjectName, - defaultDirectory: context.defaultDirectory, - oldName: '', - newName: '', - projects: context.projects, - } - } - return { - defaultProjectName: context.defaultProjectName, - defaultDirectory: context.defaultDirectory, - oldName: event.data.oldName, - newName: event.data.newName, - projects: context.projects, - } - }, - onDone: [ - { - target: '#Home machine.Reading projects', - actions: ['toastSuccess', 'navigateToProjectIfNeeded'], - }, - ], - onError: [ - { - target: '#Home machine.Reading projects', - actions: ['toastError'], - }, - ], - }, - }, - - 'Deleting project': { - invoke: { - id: 'delete-project', - src: 'deleteProject', - input: ({ event, context }) => { - if (event.type !== 'Delete project') { - // This is to make TS happy - return { - defaultDirectory: context.defaultDirectory, - name: '', - } - } - return { - defaultDirectory: context.defaultDirectory, - name: event.data.name, - } - }, - onDone: [ - { - actions: ['toastSuccess', 'navigateToProjectIfNeeded'], - target: '#Home machine.Reading projects', - }, - ], - onError: { - actions: ['toastError'], - target: '#Home machine.Has projects', - }, - }, - }, - - 'Reading projects': { - invoke: { - id: 'read-projects', - src: 'readProjects', - onDone: [ - { - guard: 'Has at least 1 project', - target: 'Has projects', - actions: ['setProjects', 'setHasListedProjects'], - }, - { - target: 'Has no projects', - actions: ['setProjects', 'setHasListedProjects'], - }, - ], - onError: [ - { - target: 'Has no projects', - actions: ['toastError'], - }, - ], - }, - }, - - 'Creating file': { - invoke: { - id: 'create-file', - src: 'createFile', - input: ({ event, context }) => { - if (event.type !== 'Import file from URL') { - return { - code: '', - name: '', - method: 'existingProject', - projects: context.projects, - } - } - return { - code: event.data.code || '', - name: event.data.name, - method: event.data.method, - projectName: event.data.projectName, - projects: context.projects, - } - }, - onDone: { - target: 'Reading projects', - actions: ['navigateToFile', 'toastSuccess'], - }, - onError: { - target: 'Reading projects', - actions: 'toastError', - }, - }, - }, - }, -}) diff --git a/src/machines/settingsMachine.ts b/src/machines/settingsMachine.ts index b51e96c71..1374a14e3 100644 --- a/src/machines/settingsMachine.ts +++ b/src/machines/settingsMachine.ts @@ -35,13 +35,6 @@ import { saveSettings, setSettingsAtLevel, } from '@src/lib/settings/settingsUtils' -import { - codeManager, - engineCommandManager, - kclManager, - sceneEntitiesManager, - sceneInfra, -} from '@src/lib/singletons' import { Themes, darkModeMatcher, @@ -50,7 +43,6 @@ import { setThemeClass, } from '@src/lib/theme' import { reportRejection } from '@src/lib/trap' -import { commandBarActor } from '@src/machines/commandBarMachine' type SettingsMachineContext = SettingsType & { currentProject?: Project @@ -95,13 +87,14 @@ export const settingsMachine = setup({ doNotPersist: boolean context: SettingsMachineContext toastCallback?: () => void + rootContext: any } >(async ({ input }) => { // Without this, when a user changes the file, it'd // create a detection loop with the file-system watcher. if (input.doNotPersist) return - codeManager.writeCausedByAppCheckedInFileTreeFileSystemWatcher = true + input.rootContext.codeManager.writeCausedByAppCheckedInFileTreeFileSystemWatcher = true const { currentProject, ...settings } = input.context const val = await saveSettings(settings, currentProject?.path) @@ -147,7 +140,9 @@ export const settingsMachine = setup({ registerCommands: fromCallback< { type: 'update' }, { settings: SettingsType; actor: AnyActorRef } - >(({ input, receive }) => { + >(({ input, receive, self }) => { + const commandBarActor = self.system.get('root').getSnapshot() + .context.commandBarActor // If the user wants to hide the settings commands //from the command bar don't add them. if (settings.commandBar.includeSettings.current === false) return @@ -190,20 +185,28 @@ export const settingsMachine = setup({ }), }, actions: { - setEngineTheme: ({ context }) => { + setEngineTheme: ({ context, self }) => { + const rootContext = self.system.get('root').getSnapshot().context + const engineCommandManager = rootContext.engineCommandManager if (engineCommandManager && context.app.theme.current) { engineCommandManager .setTheme(context.app.theme.current) .catch(reportRejection) } }, - setClientTheme: ({ context }) => { + setClientTheme: ({ context, self }) => { + const rootContext = self.system.get('root').getSnapshot().context + const sceneInfra = rootContext.sceneInfra + const sceneEntitiesManager = rootContext.sceneEntitiesManager + if (!sceneInfra || !sceneEntitiesManager) return const opposingTheme = getOppositeTheme(context.app.theme.current) sceneInfra.theme = opposingTheme sceneEntitiesManager.updateSegmentBaseColor(opposingTheme) }, - setAllowOrbitInSketchMode: ({ context }) => { + setAllowOrbitInSketchMode: ({ context, self }) => { + const rootContext = self.system.get('root').getSnapshot().context + const sceneInfra = rootContext.sceneInfra if (!sceneInfra.camControls) return sceneInfra.camControls._setting_allowOrbitInSketchMode = context.app.allowOrbitInSketchMode.current @@ -232,7 +235,9 @@ export const settingsMachine = setup({ id: `${event.type}.success`, }) }, - 'Execute AST': ({ context, event }) => { + 'Execute AST': ({ context, event, self }) => { + const rootContext = self.system.get('root').getSnapshot().context + const kclManager = rootContext.kclManager try { const relevantSetting = (s: typeof settings) => { return ( @@ -345,8 +350,10 @@ export const settingsMachine = setup({ currentTheme === Themes.System ? getSystemTheme() : currentTheme ) }, - setEngineCameraProjection: ({ context }) => { + setEngineCameraProjection: ({ context, self }) => { const newCurrentProjection = context.modeling.cameraProjection.current + const rootContext = self.system.get('root').getSnapshot().context + const sceneInfra = rootContext.sceneInfra sceneInfra.camControls.setEngineCameraProjection(newCurrentProjection) }, sendThemeToWatcher: sendTo('watchSystemTheme', ({ context }) => ({ @@ -532,7 +539,7 @@ export const settingsMachine = setup({ console.error('Error persisting settings') }, }, - input: ({ context, event }) => { + input: ({ context, event, self }) => { if ( event.type === 'set.app.namedViews' && 'toastCallback' in event.data @@ -541,12 +548,14 @@ export const settingsMachine = setup({ doNotPersist: event.doNotPersist ?? false, context, toastCallback: event.data.toastCallback, + rootContext: self.system.get('root').getSnapshot().context, } } return { doNotPersist: event.doNotPersist ?? false, context, + rootContext: self.system.get('root').getSnapshot().context, } }, }, diff --git a/src/machines/systemIO/hooks.ts b/src/machines/systemIO/hooks.ts new file mode 100644 index 000000000..b5917de4c --- /dev/null +++ b/src/machines/systemIO/hooks.ts @@ -0,0 +1,21 @@ +import { systemIOActor } from '@src/lib/singletons' +import { useSelector } from '@xstate/react' +export const useRequestedProjectName = () => + useSelector(systemIOActor, (state) => state.context.requestedProjectName) +export const useRequestedFileName = () => + useSelector(systemIOActor, (state) => state.context.requestedFileName) +export const useProjectDirectoryPath = () => + useSelector(systemIOActor, (state) => state.context.projectDirectoryPath) +export const useFolders = () => + useSelector(systemIOActor, (state) => state.context.folders) +export const useState = () => useSelector(systemIOActor, (state) => state) +export const useCanReadWriteProjectDirectory = () => + useSelector( + systemIOActor, + (state) => state.context.canReadWriteProjectDirectory + ) +export const useHasListedProjects = () => + useSelector(systemIOActor, (state) => state.context.hasListedProjects) + +export const useClearURLParams = () => + useSelector(systemIOActor, (state) => state.context.clearURLParams) diff --git a/src/machines/systemIO/snapshotContext.ts b/src/machines/systemIO/snapshotContext.ts new file mode 100644 index 000000000..b227bed92 --- /dev/null +++ b/src/machines/systemIO/snapshotContext.ts @@ -0,0 +1,11 @@ +import { systemIOActor } from '@src/lib/singletons' + +export const folderSnapshot = () => { + const { folders } = systemIOActor.getSnapshot().context + return folders +} + +export const defaultProjectFolderNameSnapshot = () => { + const { defaultProjectFolderName } = systemIOActor.getSnapshot().context + return defaultProjectFolderName +} diff --git a/src/machines/systemIO/systemIOMachine.test.ts b/src/machines/systemIO/systemIOMachine.test.ts new file mode 100644 index 000000000..36add501d --- /dev/null +++ b/src/machines/systemIO/systemIOMachine.test.ts @@ -0,0 +1,78 @@ +import { DEFAULT_PROJECT_NAME } from '@src/lib/constants' +import { systemIOMachineDesktop } from '@src/machines/systemIO/systemIOMachineDesktop' +import { + NO_PROJECT_DIRECTORY, + SystemIOMachineEvents, + SystemIOMachineStates, +} from '@src/machines/systemIO/utils' +import path from 'node:path' +import { createActor, waitFor } from 'xstate' + +describe('systemIOMachine - XState', () => { + describe('desktop', () => { + describe('when initializied', () => { + it('should contain the default context values', () => { + const actor = createActor(systemIOMachineDesktop).start() + const context = actor.getSnapshot().context + expect(context.folders).toStrictEqual([]) + expect(context.defaultProjectFolderName).toStrictEqual( + DEFAULT_PROJECT_NAME + ) + expect(context.projectDirectoryPath).toBe(NO_PROJECT_DIRECTORY) + expect(context.hasListedProjects).toBe(false) + expect(context.requestedProjectName).toStrictEqual({ + name: NO_PROJECT_DIRECTORY, + }) + expect(context.requestedFileName).toStrictEqual({ + project: NO_PROJECT_DIRECTORY, + file: NO_PROJECT_DIRECTORY, + }) + }) + it('should be in idle state', () => { + const actor = createActor(systemIOMachineDesktop).start() + const state = actor.getSnapshot().value + expect(state).toBe(SystemIOMachineStates.idle) + }) + }) + describe('when reading projects', () => { + it('should exit early when project directory is empty string', async () => { + const actor = createActor(systemIOMachineDesktop).start() + actor.send({ + type: SystemIOMachineEvents.readFoldersFromProjectDirectory, + }) + await waitFor(actor, (state) => + state.matches(SystemIOMachineStates.readingFolders) + ) + await waitFor(actor, (state) => + state.matches(SystemIOMachineStates.idle) + ) + const context = actor.getSnapshot().context + expect(context.folders).toStrictEqual([]) + }) + }) + describe('when setting project directory path', () => { + it('should set new project directory path', async () => { + const kclSamplesPath = path.join('public', 'kcl-samples') + const actor = createActor(systemIOMachineDesktop).start() + actor.send({ + type: SystemIOMachineEvents.setProjectDirectoryPath, + data: { requestedProjectDirectoryPath: kclSamplesPath }, + }) + let context = actor.getSnapshot().context + expect(context.projectDirectoryPath).toBe(kclSamplesPath) + }) + }) + describe('when setting default project folder name', () => { + it('should set a new default project folder name', async () => { + const expected = 'coolcoolcoolProjectName' + const actor = createActor(systemIOMachineDesktop).start() + actor.send({ + type: SystemIOMachineEvents.setDefaultProjectFolderName, + data: { requestedDefaultProjectFolderName: expected }, + }) + let context = actor.getSnapshot().context + expect(context.defaultProjectFolderName).toBe(expected) + }) + }) + }) +}) diff --git a/src/machines/systemIO/systemIOMachine.ts b/src/machines/systemIO/systemIOMachine.ts new file mode 100644 index 000000000..7db087c8f --- /dev/null +++ b/src/machines/systemIO/systemIOMachine.ts @@ -0,0 +1,444 @@ +import { DEFAULT_PROJECT_NAME } from '@src/lib/constants' +import type { Project } from '@src/lib/project' +import type { SystemIOContext } from '@src/machines/systemIO/utils' +import { + NO_PROJECT_DIRECTORY, + SystemIOMachineActions, + SystemIOMachineActors, + SystemIOMachineEvents, + SystemIOMachineStates, +} from '@src/machines/systemIO/utils' +import toast from 'react-hot-toast' +import { assertEvent, assign, fromPromise, setup } from 'xstate' +import type { AppMachineContext } from '@src/lib/types' + +/** + * Handles any system level I/O for folders and files + * This machine will be initializes once within the applications runtime + * and exist for the entire life cycle of the application and able to be access + * at a global level. + */ +export const systemIOMachine = setup({ + types: { + context: {} as SystemIOContext, + events: {} as + | { + type: SystemIOMachineEvents.readFoldersFromProjectDirectory + } + | { + type: SystemIOMachineEvents.done_readFoldersFromProjectDirectory + output: Project[] + } + | { + type: SystemIOMachineEvents.done_checkReadWrite + output: { value: boolean; error: unknown } + } + | { + type: SystemIOMachineEvents.setProjectDirectoryPath + data: { requestedProjectDirectoryPath: string } + } + | { + type: SystemIOMachineEvents.navigateToProject + data: { requestedProjectName: string } + } + | { + type: SystemIOMachineEvents.navigateToFile + data: { requestedProjectName: string; requestedFileName: string } + } + | { + type: SystemIOMachineEvents.createProject + data: { requestedProjectName: string } + } + | { + type: SystemIOMachineEvents.renameProject + data: { requestedProjectName: string; projectName: string } + } + | { + type: SystemIOMachineEvents.deleteProject + data: { requestedProjectName: string } + } + | { + type: SystemIOMachineEvents.createKCLFile + data: { + requestedProjectName: string + requestedFileName: string + requestedCode: string + } + } + | { + type: SystemIOMachineEvents.importFileFromURL + data: { + requestedProjectName: string + requestedFileName: string + requestedCode: string + } + } + | { + type: SystemIOMachineEvents.setDefaultProjectFolderName + data: { requestedDefaultProjectFolderName: string } + }, + }, + actions: { + [SystemIOMachineActions.setFolders]: assign({ + folders: ({ event }) => { + assertEvent( + event, + SystemIOMachineEvents.done_readFoldersFromProjectDirectory + ) + return event.output + }, + }), + [SystemIOMachineActions.setProjectDirectoryPath]: assign({ + projectDirectoryPath: ({ event }) => { + assertEvent(event, SystemIOMachineEvents.setProjectDirectoryPath) + return event.data.requestedProjectDirectoryPath + }, + }), + [SystemIOMachineActions.setRequestedProjectName]: assign({ + requestedProjectName: ({ event }) => { + assertEvent(event, SystemIOMachineEvents.navigateToProject) + return { name: event.data.requestedProjectName } + }, + }), + [SystemIOMachineActions.setRequestedFileName]: assign({ + requestedFileName: ({ event }) => { + assertEvent(event, SystemIOMachineEvents.navigateToFile) + return { + project: event.data.requestedProjectName, + file: event.data.requestedFileName, + } + }, + }), + [SystemIOMachineActions.setDefaultProjectFolderName]: assign({ + defaultProjectFolderName: ({ event }) => { + assertEvent(event, SystemIOMachineEvents.setDefaultProjectFolderName) + return event.data.requestedDefaultProjectFolderName + }, + }), + [SystemIOMachineActions.toastSuccess]: ({ event }) => { + toast.success( + ('data' in event && typeof event.data === 'string' && event.data) || + ('output' in event && + 'message' in event.output && + typeof event.output.message === 'string' && + event.output.message) || + '' + ) + }, + [SystemIOMachineActions.toastError]: ({ event }) => { + toast.error( + ('data' in event && typeof event.data === 'string' && event.data) || + ('output' in event && + typeof event.output === 'string' && + event.output) || + ('error' in event && + event.error instanceof Error && + event.error.message) || + '' + ) + }, + [SystemIOMachineActions.setReadWriteProjectDirectory]: assign({ + canReadWriteProjectDirectory: ({ event }) => { + assertEvent(event, SystemIOMachineEvents.done_checkReadWrite) + return event.output + }, + }), + }, + actors: { + [SystemIOMachineActors.readFoldersFromProjectDirectory]: fromPromise( + async ({ input: context }: { input: SystemIOContext }) => { + const folders: Project[] = [] + return folders + } + ), + [SystemIOMachineActors.createProject]: fromPromise( + async ({ + input: { context, requestedProjectName }, + }: { + input: { context: SystemIOContext; requestedProjectName: string } + }) => { + return { message: '', name: '' } + } + ), + [SystemIOMachineActors.deleteProject]: fromPromise( + async ({ + input: { context, requestedProjectName }, + }: { + input: { context: SystemIOContext; requestedProjectName: string } + }) => { + return { message: '', name: '' } + } + ), + [SystemIOMachineActors.renameProject]: fromPromise( + async ({ + input: { context, requestedProjectName, projectName }, + }: { + input: { + context: SystemIOContext + requestedProjectName: string + projectName: string + } + }): Promise<{ message: string; newName: string; oldName: string }> => { + return { message: '', newName: '', oldName: '' } + } + ), + [SystemIOMachineActors.createKCLFile]: fromPromise( + async ({ + input, + }: { + input: { + context: SystemIOContext + requestedProjectName: string + requestedFileName: string + requestedCode: string + rootContext: AppMachineContext + } + }): Promise<{ + message: string + fileName: string + projectName: string + }> => { + return { message: '', fileName: '', projectName: '' } + } + ), + [SystemIOMachineActors.checkReadWrite]: fromPromise( + async ({ + input: { context, requestedProjectDirectoryPath }, + }: { + input: { + context: SystemIOContext + requestedProjectDirectoryPath: string + } + }): Promise<{ value: boolean; error: unknown }> => { + return { value: true, error: undefined } + } + ), + }, +}).createMachine({ + initial: SystemIOMachineStates.idle, + // Remember, this machine and change its projectDirectory at any point + // '' will be no project directory, aka clear this machine out! + // To be the absolute root of someones computer we should take the string of path.resolve() in node.js which is different for each OS + context: () => ({ + folders: [], + defaultProjectFolderName: DEFAULT_PROJECT_NAME, + projectDirectoryPath: NO_PROJECT_DIRECTORY, + hasListedProjects: false, + requestedProjectName: { name: NO_PROJECT_DIRECTORY }, + requestedFileName: { + project: NO_PROJECT_DIRECTORY, + file: NO_PROJECT_DIRECTORY, + }, + canReadWriteProjectDirectory: { value: true, error: undefined }, + clearURLParams: { value: false }, + }), + states: { + [SystemIOMachineStates.idle]: { + on: { + // on can be an action + [SystemIOMachineEvents.readFoldersFromProjectDirectory]: { + target: SystemIOMachineStates.readingFolders, + }, + [SystemIOMachineEvents.setProjectDirectoryPath]: { + target: SystemIOMachineStates.checkingReadWrite, + actions: [SystemIOMachineActions.setProjectDirectoryPath], + }, + [SystemIOMachineEvents.navigateToProject]: { + actions: [SystemIOMachineActions.setRequestedProjectName], + }, + [SystemIOMachineEvents.navigateToFile]: { + actions: [SystemIOMachineActions.setRequestedFileName], + }, + [SystemIOMachineEvents.createProject]: { + target: SystemIOMachineStates.creatingProject, + }, + [SystemIOMachineEvents.renameProject]: { + target: SystemIOMachineStates.renamingProject, + }, + [SystemIOMachineEvents.deleteProject]: { + target: SystemIOMachineStates.deletingProject, + }, + [SystemIOMachineEvents.createKCLFile]: { + target: SystemIOMachineStates.creatingKCLFile, + }, + [SystemIOMachineEvents.setDefaultProjectFolderName]: { + actions: [SystemIOMachineActions.setDefaultProjectFolderName], + }, + [SystemIOMachineEvents.importFileFromURL]: { + target: SystemIOMachineStates.importFileFromURL, + }, + }, + }, + [SystemIOMachineStates.readingFolders]: { + invoke: { + id: SystemIOMachineActors.readFoldersFromProjectDirectory, + src: SystemIOMachineActors.readFoldersFromProjectDirectory, + input: ({ context }) => { + return context + }, + onDone: { + target: SystemIOMachineStates.idle, + actions: [ + SystemIOMachineActions.setFolders, + assign({ hasListedProjects: true }), + ], + }, + onError: { + target: SystemIOMachineStates.idle, + }, + }, + }, + [SystemIOMachineStates.creatingProject]: { + invoke: { + id: SystemIOMachineActors.createProject, + src: SystemIOMachineActors.createProject, + input: ({ context, event }) => { + assertEvent(event, SystemIOMachineEvents.createProject) + return { + context, + requestedProjectName: event.data.requestedProjectName, + } + }, + onDone: { + target: SystemIOMachineStates.readingFolders, + actions: [ + assign({ + requestedProjectName: ({ event }) => { + return { name: event.output.name } + }, + }), + SystemIOMachineActions.toastSuccess, + ], + }, + onError: { + target: SystemIOMachineStates.idle, + actions: [SystemIOMachineActions.toastError], + }, + }, + }, + [SystemIOMachineStates.renamingProject]: { + invoke: { + id: SystemIOMachineActors.renameProject, + src: SystemIOMachineActors.renameProject, + input: ({ context, event }) => { + assertEvent(event, SystemIOMachineEvents.renameProject) + return { + context, + requestedProjectName: event.data.requestedProjectName, + projectName: event.data.projectName, + } + }, + onDone: { + target: SystemIOMachineStates.readingFolders, + actions: [SystemIOMachineActions.toastSuccess], + }, + onError: { + target: SystemIOMachineStates.idle, + actions: [SystemIOMachineActions.toastError], + }, + }, + }, + [SystemIOMachineStates.deletingProject]: { + invoke: { + id: SystemIOMachineActors.deleteProject, + src: SystemIOMachineActors.deleteProject, + input: ({ context, event }) => { + assertEvent(event, SystemIOMachineEvents.deleteProject) + return { + context, + requestedProjectName: event.data.requestedProjectName, + } + }, + onDone: { + target: SystemIOMachineStates.readingFolders, + actions: [SystemIOMachineActions.toastSuccess], + }, + onError: { + target: SystemIOMachineStates.idle, + actions: [SystemIOMachineActions.toastError], + }, + }, + }, + [SystemIOMachineStates.creatingKCLFile]: { + invoke: { + id: SystemIOMachineActors.createKCLFile, + src: SystemIOMachineActors.createKCLFile, + input: ({ context, event, self }) => { + assertEvent(event, SystemIOMachineEvents.createKCLFile) + return { + context, + requestedProjectName: event.data.requestedProjectName, + requestedFileName: event.data.requestedFileName, + requestedCode: event.data.requestedCode, + rootContext: self.system.get('root').getSnapshot().context, + } + }, + onDone: { + target: SystemIOMachineStates.idle, + }, + onError: { + target: SystemIOMachineStates.idle, + actions: [SystemIOMachineActions.toastError], + }, + }, + }, + [SystemIOMachineStates.importFileFromURL]: { + invoke: { + id: SystemIOMachineActors.importFileFromURL, + src: SystemIOMachineActors.createKCLFile, + input: ({ context, event, self }) => { + assertEvent(event, SystemIOMachineEvents.importFileFromURL) + return { + context, + requestedProjectName: event.data.requestedProjectName, + requestedFileName: event.data.requestedFileName, + requestedCode: event.data.requestedCode, + rootContext: self.system.get('root').getSnapshot().context, + } + }, + onDone: { + target: SystemIOMachineStates.readingFolders, + // Clear on web? not desktop + actions: [ + assign({ + requestedFileName: ({ context, event }) => { + assertEvent(event, SystemIOMachineEvents.done_importFileFromURL) + // Not the entire path + return { + project: event.output.projectName, + file: event.output.fileName + '.kcl', + } + }, + }), + assign({ clearURLParams: { value: true } }), + ], + }, + onError: { + target: SystemIOMachineStates.idle, + actions: [SystemIOMachineActions.toastError], + }, + }, + }, + [SystemIOMachineStates.checkingReadWrite]: { + invoke: { + id: SystemIOMachineActors.checkReadWrite, + src: SystemIOMachineActors.checkReadWrite, + input: ({ context, event }) => { + assertEvent(event, SystemIOMachineEvents.setProjectDirectoryPath) + return { + context, + requestedProjectDirectoryPath: + event.data.requestedProjectDirectoryPath, + } + }, + onDone: { + target: SystemIOMachineStates.readingFolders, + }, + onError: { + target: SystemIOMachineStates.readingFolders, + actions: [SystemIOMachineActions.toastError], + }, + }, + }, + }, +}) diff --git a/src/machines/systemIO/systemIOMachineDesktop.ts b/src/machines/systemIO/systemIOMachineDesktop.ts new file mode 100644 index 000000000..c9443f597 --- /dev/null +++ b/src/machines/systemIO/systemIOMachineDesktop.ts @@ -0,0 +1,232 @@ +import { + createNewProjectDirectory, + getProjectInfo, + mkdirOrNOOP, + readAppSettingsFile, + renameProjectDirectory, +} from '@src/lib/desktop' +import { + doesProjectNameNeedInterpolated, + getNextFileName, + getNextProjectIndex, + getUniqueProjectName, + interpolateProjectNameWithIndex, +} from '@src/lib/desktopFS' +import type { Project } from '@src/lib/project' +import { systemIOMachine } from '@src/machines/systemIO/systemIOMachine' +import type { SystemIOContext } from '@src/machines/systemIO/utils' +import { + NO_PROJECT_DIRECTORY, + SystemIOMachineActors, +} from '@src/machines/systemIO/utils' +import { fromPromise } from 'xstate' +import type { AppMachineContext } from '@src/lib/types' + +export const systemIOMachineDesktop = systemIOMachine.provide({ + actors: { + [SystemIOMachineActors.readFoldersFromProjectDirectory]: fromPromise( + async ({ input: context }: { input: SystemIOContext }) => { + const projects = [] + const projectDirectoryPath = context.projectDirectoryPath + if (projectDirectoryPath === NO_PROJECT_DIRECTORY) { + return [] + } + await mkdirOrNOOP(projectDirectoryPath) + // Gotcha: readdir will list all folders at this project directory even if you do not have readwrite access on the directory path + const entries = await window.electron.readdir(projectDirectoryPath) + const { value: canReadWriteProjectDirectory } = + await window.electron.canReadWriteDirectory(projectDirectoryPath) + + for (let entry of entries) { + // Skip directories that start with a dot + if (entry.startsWith('.')) { + continue + } + const projectPath = window.electron.path.join( + projectDirectoryPath, + entry + ) + + // if it's not a directory ignore. + // Gotcha: statIsDirectory will work even if you do not have read write permissions on the project path + const isDirectory = await window.electron.statIsDirectory(projectPath) + if (!isDirectory) { + continue + } + const project: Project = await getProjectInfo(projectPath) + if ( + project.kcl_file_count === 0 && + project.readWriteAccess && + canReadWriteProjectDirectory + ) { + continue + } + projects.push(project) + } + return projects + } + ), + [SystemIOMachineActors.createProject]: fromPromise( + async ({ + input, + }: { + input: { context: SystemIOContext; requestedProjectName: string } + }) => { + const folders = input.context.folders + const requestedProjectName = input.requestedProjectName + const uniqueName = getUniqueProjectName(requestedProjectName, folders) + await createNewProjectDirectory(uniqueName) + return { + message: `Successfully created "${uniqueName}"`, + name: uniqueName, + } + } + ), + [SystemIOMachineActors.renameProject]: fromPromise( + async ({ + input, + }: { + input: { + context: SystemIOContext + requestedProjectName: string + projectName: string + } + }) => { + const folders = input.context.folders + const requestedProjectName = input.requestedProjectName + const projectName = input.projectName + let newProjectName: string = requestedProjectName + if (doesProjectNameNeedInterpolated(requestedProjectName)) { + const nextIndex = getNextProjectIndex(requestedProjectName, folders) + newProjectName = interpolateProjectNameWithIndex( + requestedProjectName, + nextIndex + ) + } + + // Toast an error if the project name is taken + if (folders.find((p) => p.name === newProjectName)) { + return Promise.reject( + new Error(`Project with name "${newProjectName}" already exists`) + ) + } + + await renameProjectDirectory( + window.electron.path.join( + input.context.projectDirectoryPath, + projectName + ), + newProjectName + ) + + return { + message: `Successfully renamed "${projectName}" to "${newProjectName}"`, + oldName: projectName, + newName: newProjectName, + } + } + ), + [SystemIOMachineActors.deleteProject]: fromPromise( + async ({ + input, + }: { + input: { context: SystemIOContext; requestedProjectName: string } + }) => { + await window.electron.rm( + window.electron.path.join( + input.context.projectDirectoryPath, + input.requestedProjectName + ), + { + recursive: true, + } + ) + + return { + message: `Successfully deleted "${input.requestedProjectName}"`, + name: input.requestedProjectName, + } + } + ), + [SystemIOMachineActors.createKCLFile]: fromPromise( + async ({ + input, + }: { + input: { + context: SystemIOContext + requestedProjectName: string + requestedFileName: string + requestedCode: string + rootContext: AppMachineContext + } + }) => { + const requestedProjectName = input.requestedProjectName + const requestedFileName = input.requestedFileName + const requestedCode = input.requestedCode + const folders = input.context.folders + + let newProjectName = requestedProjectName + + if (!newProjectName) { + newProjectName = getUniqueProjectName( + input.context.defaultProjectFolderName, + input.context.folders + ) + } + + const needsInterpolated = + doesProjectNameNeedInterpolated(newProjectName) + if (needsInterpolated) { + const nextIndex = getNextProjectIndex(newProjectName, folders) + newProjectName = interpolateProjectNameWithIndex( + newProjectName, + nextIndex + ) + } + + const baseDir = window.electron.join( + input.context.projectDirectoryPath, + newProjectName + ) + const { name: newFileName } = getNextFileName({ + entryName: requestedFileName, + baseDir, + }) + const configuration = await readAppSettingsFile() + + // Create the project around the file if newProject + await createNewProjectDirectory( + newProjectName, + requestedCode, + configuration, + newFileName + ) + + return { + message: 'File created successfully', + fileName: input.requestedFileName, + projectName: newProjectName, + } + } + ), + [SystemIOMachineActors.checkReadWrite]: fromPromise( + async ({ + input, + }: { + input: { + context: SystemIOContext + requestedProjectDirectoryPath: string + } + }) => { + const requestProjectDirectoryPath = input.requestedProjectDirectoryPath + if (!requestProjectDirectoryPath) { + return { value: true, error: undefined } + } + const result = await window.electron.canReadWriteDirectory( + requestProjectDirectoryPath + ) + return result + } + ), + }, +}) diff --git a/src/machines/systemIO/systemIOMachineWeb.ts b/src/machines/systemIO/systemIOMachineWeb.ts new file mode 100644 index 000000000..cac7ef9ad --- /dev/null +++ b/src/machines/systemIO/systemIOMachineWeb.ts @@ -0,0 +1,50 @@ +import { systemIOMachine } from '@src/machines/systemIO/systemIOMachine' +import type { SystemIOContext } from '@src/machines/systemIO/utils' +import { SystemIOMachineActors } from '@src/machines/systemIO/utils' +import { fromPromise } from 'xstate' +import { newKclFile } from '@src/lang/project' +import { readLocalStorageProjectSettingsFile } from '@src/lib/settings/settingsUtils' +import { err } from '@src/lib/trap' +import { DEFAULT_DEFAULT_LENGTH_UNIT } from '@src/lib/constants' +import type { AppMachineContext } from '@src/lib/types' + +export const systemIOMachineWeb = systemIOMachine.provide({ + actors: { + [SystemIOMachineActors.createKCLFile]: fromPromise( + async ({ + input, + }: { + input: { + context: SystemIOContext + requestedProjectName: string + requestedFileName: string + requestedCode: string + rootContext: AppMachineContext + } + }) => { + // Browser version doesn't navigate, just overwrites the current file + // clearImportSearchParams() + const projectSettings = readLocalStorageProjectSettingsFile() + if (err(projectSettings)) { + return Promise.reject( + 'Unable to read project settings from local storage' + ) + } + const codeToWrite = newKclFile( + input.requestedCode, + projectSettings?.settings?.modeling?.base_unit || + DEFAULT_DEFAULT_LENGTH_UNIT + ) + if (err(codeToWrite)) return Promise.reject(codeToWrite) + input.rootContext.codeManager.updateCodeStateEditor(codeToWrite) + await input.rootContext.codeManager.writeToFile() + await input.rootContext.kclManager.executeCode() + return { + message: 'File overwritten successfully', + fileName: input.requestedFileName, + projectName: '', + } + } + ), + }, +}) diff --git a/src/machines/systemIO/utils.ts b/src/machines/systemIO/utils.ts new file mode 100644 index 000000000..08e9f9c43 --- /dev/null +++ b/src/machines/systemIO/utils.ts @@ -0,0 +1,73 @@ +import type { Project } from '@src/lib/project' + +export enum SystemIOMachineActors { + readFoldersFromProjectDirectory = 'read folders from project directory', + setProjectDirectoryPath = 'set project directory path', + createProject = 'create project', + renameProject = 'rename project', + deleteProject = 'delete project', + createKCLFile = 'create kcl file', + checkReadWrite = 'check read write', + importFileFromURL = 'import file from URL', +} + +export enum SystemIOMachineStates { + idle = 'idle', + readingFolders = 'readingFolders', + settingProjectDirectoryPath = 'settingProjectDirectoryPath', + creatingProject = 'creatingProject', + renamingProject = 'renamingProject', + deletingProject = 'deletingProject', + creatingKCLFile = 'creatingKCLFile', + checkingReadWrite = 'checkingReadWrite', + importFileFromURL = 'importFileFromURL', +} + +const donePrefix = 'xstate.done.actor.' + +export enum SystemIOMachineEvents { + readFoldersFromProjectDirectory = 'read folders from project directory', + done_readFoldersFromProjectDirectory = donePrefix + + 'read folders from project directory', + setProjectDirectoryPath = 'set project directory path', + navigateToProject = 'navigate to project', + navigateToFile = 'navigate to file', + createProject = 'create project', + renameProject = 'rename project', + deleteProject = 'delete project', + createKCLFile = 'create kcl file', + setDefaultProjectFolderName = 'set default project folder name', + done_checkReadWrite = donePrefix + 'check read write', + importFileFromURL = 'import file from URL', + done_importFileFromURL = donePrefix + 'import file from URL', +} + +export enum SystemIOMachineActions { + setFolders = 'set folders', + setProjectDirectoryPath = 'set project directory path', + setRequestedProjectName = 'set requested project name', + setRequestedFileName = 'set requested file name', + setDefaultProjectFolderName = 'set default project folder name', + toastSuccess = 'toastSuccess', + toastError = 'toastError', + setReadWriteProjectDirectory = 'set read write project directory', +} + +export const NO_PROJECT_DIRECTORY = '' + +export type SystemIOContext = { + /** Only store folders under the projectDirectory, do not maintain folders outside this directory */ + folders: Project[] + /** For this machines runtime, this is the default string when creating a project + * A project is defined by creating a folder at the one level below the working project directory */ + defaultProjectFolderName: string + /** working project directory that stores all the project folders */ + projectDirectoryPath: string + /** has the application gone through the initialization of systemIOMachine at least once. + * this is required to prevent chokidar from spamming invalid events during initialization. */ + hasListedProjects: boolean + requestedProjectName: { name: string } + requestedFileName: { project: string; file: string } + canReadWriteProjectDirectory: { value: boolean; error: unknown } + clearURLParams: { value: boolean } +} diff --git a/src/main.ts b/src/main.ts index 4936effa5..81770b21a 100644 --- a/src/main.ts +++ b/src/main.ts @@ -283,6 +283,10 @@ ipcMain.handle('shell.openExternal', (event, data) => { return shell.openExternal(data) }) +ipcMain.handle('openInNewWindow', (event, data) => { + return createWindow(data) +}) + ipcMain.handle( 'take.screenshot', async (event, data: { width: number; height: number }) => { diff --git a/src/menu/register.ts b/src/menu/register.ts index 854fefe07..8fb1dd78c 100644 --- a/src/menu/register.ts +++ b/src/menu/register.ts @@ -10,8 +10,8 @@ import { } from '@src/lib/singletons' import { reportRejection } from '@src/lib/trap' import { uuidv4 } from '@src/lib/utils' -import { authActor, settingsActor } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { authActor, settingsActor } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' import type { WebContentSendPayload } from '@src/menu/channels' import type { NavigateFunction } from 'react-router-dom' diff --git a/src/preload.ts b/src/preload.ts index 2c0c0f873..f39e28e27 100644 --- a/src/preload.ts +++ b/src/preload.ts @@ -21,6 +21,7 @@ const resizeWindow = (width: number, height: number) => const open = (args: any) => ipcRenderer.invoke('dialog.showOpenDialog', args) const save = (args: any) => ipcRenderer.invoke('dialog.showSaveDialog', args) const openExternal = (url: any) => ipcRenderer.invoke('shell.openExternal', url) +const openInNewWindow = (url: any) => ipcRenderer.invoke('openInNewWindow', url) const takeElectronWindowScreenshot = ({ width, height, @@ -248,6 +249,7 @@ contextBridge.exposeInMainWorld('electron', { save, // opens the URL openExternal, + openInNewWindow, showInFolder, getPath, packageJson, diff --git a/src/routes/Home.tsx b/src/routes/Home.tsx index 473096669..051470756 100644 --- a/src/routes/Home.tsx +++ b/src/routes/Home.tsx @@ -1,5 +1,5 @@ import type { FormEvent } from 'react' -import { useEffect, useRef, useState } from 'react' +import { useEffect, useRef } from 'react' import { toast } from 'react-hot-toast' import { useHotkeys } from 'react-hotkeys-hook' import { Link, useNavigate, useSearchParams } from 'react-router-dom' @@ -15,7 +15,6 @@ import { } from '@src/components/ProjectSearchBar' import { useCreateFileLinkQuery } from '@src/hooks/useCreateFileLinkQueryWatcher' import { useMenuListener } from '@src/hooks/useMenu' -import { useProjectsContext } from '@src/hooks/useProjectsContext' import { isDesktop } from '@src/lib/isDesktop' import { PATHS } from '@src/lib/paths' import { markOnce } from '@src/lib/performance' @@ -27,21 +26,24 @@ import { getSortIcon, } from '@src/lib/sorting' import { reportRejection } from '@src/lib/trap' -import { authActor, useSettings } from '@src/machines/appMachine' -import { commandBarActor } from '@src/machines/commandBarMachine' +import { authActor, systemIOActor, useSettings } from '@src/lib/singletons' +import { commandBarActor } from '@src/lib/singletons' +import { + useCanReadWriteProjectDirectory, + useFolders, + useState as useSystemIOState, +} from '@src/machines/systemIO/hooks' +import { + SystemIOMachineEvents, + SystemIOMachineStates, +} from '@src/machines/systemIO/utils' import type { WebContentSendPayload } from '@src/menu/channels' // This route only opens in the desktop context for now, // as defined in Router.tsx, so we can use the desktop APIs and types. const Home = () => { - const { state, send } = useProjectsContext() - const [readWriteProjectDir, setReadWriteProjectDir] = useState<{ - value: boolean - error: unknown - }>({ - value: true, - error: undefined, - }) + const state = useSystemIOState() + const readWriteProjectDir = useCanReadWriteProjectDirectory() // Only create the native file menus on desktop useEffect(() => { @@ -156,40 +158,13 @@ const Home = () => { } ) const ref = useRef(null) - - const projects = state?.context.projects ?? [] + const projects = useFolders() const [searchParams, setSearchParams] = useSearchParams() const { searchResults, query, setQuery } = useProjectSearch(projects) const sort = searchParams.get('sort_by') ?? 'modified:desc' const isSortByModified = sort?.includes('modified') || !sort || sort === null - // Update the default project name and directory in the home machine - // when the settings change - useEffect(() => { - send({ - type: 'assign', - data: { - defaultProjectName: settings.projects.defaultProjectName.current, - defaultDirectory: settings.app.projectDirectory.current, - }, - }) - - // Must be a truthy string, not '' or null or undefined - if (settings.app.projectDirectory.current) { - window.electron - .canReadWriteDirectory(settings.app.projectDirectory.current) - .then((res) => { - setReadWriteProjectDir(res) - }) - .catch(reportRejection) - } - }, [ - settings.app.projectDirectory.current, - settings.projects.defaultProjectName.current, - send, - ]) - async function handleRenameProject( e: FormEvent, project: Project @@ -204,17 +179,20 @@ const Home = () => { } if (newProjectName !== project.name) { - send({ - type: 'Rename project', - data: { oldName: project.name, newName: newProjectName as string }, + systemIOActor.send({ + type: SystemIOMachineEvents.renameProject, + data: { + requestedProjectName: String(newProjectName), + projectName: project.name, + }, }) } } async function handleDeleteProject(project: Project) { - send({ - type: 'Delete project', - data: { name: project.name || '' }, + systemIOActor.send({ + type: SystemIOMachineEvents.deleteProject, + data: { requestedProjectName: project.name }, }) } /** Type narrowing function of unknown error to a string */ @@ -246,9 +224,6 @@ const Home = () => { data: { groupId: 'projects', name: 'Create project', - argDefaultValues: { - name: settings.projects.defaultProjectName.current, - }, }, }) } @@ -345,7 +320,7 @@ const Home = () => { data-testid="home-section" className="flex-1 overflow-y-auto pr-2 pb-24" > - {state?.matches('Reading projects') ? ( + {state?.matches(SystemIOMachineStates.readingFolders) ? ( Loading your Projects... ) : ( <> diff --git a/src/routes/Onboarding/Camera.tsx b/src/routes/Onboarding/Camera.tsx index 04e7b50fd..211cd8e7b 100644 --- a/src/routes/Onboarding/Camera.tsx +++ b/src/routes/Onboarding/Camera.tsx @@ -1,7 +1,7 @@ import { SettingsSection } from '@src/components/Settings/SettingsSection' import type { CameraSystem } from '@src/lib/cameraControls' import { cameraMouseDragGuards, cameraSystems } from '@src/lib/cameraControls' -import { settingsActor, useSettings } from '@src/machines/appMachine' +import { settingsActor, useSettings } from '@src/lib/singletons' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { diff --git a/src/routes/Onboarding/Introduction.tsx b/src/routes/Onboarding/Introduction.tsx index 80460a05e..ceb0eeb17 100644 --- a/src/routes/Onboarding/Introduction.tsx +++ b/src/routes/Onboarding/Introduction.tsx @@ -13,7 +13,7 @@ import { codeManager, kclManager } from '@src/lib/singletons' import { Themes, getSystemTheme } from '@src/lib/theme' import { reportRejection } from '@src/lib/trap' import type { IndexLoaderData } from '@src/lib/types' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { OnboardingButtons, useDemoCode } from '@src/routes/Onboarding/utils' diff --git a/src/routes/Onboarding/ParametricModeling.tsx b/src/routes/Onboarding/ParametricModeling.tsx index 4360b5067..a3fb6fa24 100644 --- a/src/routes/Onboarding/ParametricModeling.tsx +++ b/src/routes/Onboarding/ParametricModeling.tsx @@ -1,7 +1,7 @@ import { bracketThicknessCalculationLine } from '@src/lib/exampleKcl' import { isDesktop } from '@src/lib/isDesktop' import { Themes, getSystemTheme } from '@src/lib/theme' -import { useSettings } from '@src/machines/appMachine' +import { useSettings } from '@src/lib/singletons' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { OnboardingButtons, useDemoCode } from '@src/routes/Onboarding/utils' diff --git a/src/routes/Onboarding/Units.tsx b/src/routes/Onboarding/Units.tsx index 82e16235b..cbccf8e17 100644 --- a/src/routes/Onboarding/Units.tsx +++ b/src/routes/Onboarding/Units.tsx @@ -3,7 +3,7 @@ import { faArrowRight, faXmark } from '@fortawesome/free-solid-svg-icons' import { ActionButton } from '@src/components/ActionButton' import { SettingsSection } from '@src/components/Settings/SettingsSection' import { type BaseUnit, baseUnitsUnion } from '@src/lib/settings/settingsTypes' -import { settingsActor, useSettings } from '@src/machines/appMachine' +import { settingsActor, useSettings } from '@src/lib/singletons' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { useDismiss, useNextClick } from '@src/routes/Onboarding/utils' diff --git a/src/routes/Onboarding/UserMenu.tsx b/src/routes/Onboarding/UserMenu.tsx index 30849812a..e3e03fee9 100644 --- a/src/routes/Onboarding/UserMenu.tsx +++ b/src/routes/Onboarding/UserMenu.tsx @@ -1,6 +1,6 @@ import { useEffect, useState } from 'react' -import { useUser } from '@src/machines/appMachine' +import { useUser } from '@src/lib/singletons' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { OnboardingButtons } from '@src/routes/Onboarding/utils' diff --git a/src/routes/Onboarding/utils.tsx b/src/routes/Onboarding/utils.tsx index e775e1dda..30309ad87 100644 --- a/src/routes/Onboarding/utils.tsx +++ b/src/routes/Onboarding/utils.tsx @@ -14,7 +14,7 @@ import makeUrlPathRelative from '@src/lib/makeUrlPathRelative' import { PATHS } from '@src/lib/paths' import { codeManager, editorManager, kclManager } from '@src/lib/singletons' import { reportRejection, trap } from '@src/lib/trap' -import { settingsActor } from '@src/machines/appMachine' +import { settingsActor } from '@src/lib/singletons' import { onboardingRoutes } from '@src/routes/Onboarding' import { onboardingPaths } from '@src/routes/Onboarding/paths' import { parse, resultIsOk } from '@src/lang/wasm' diff --git a/src/routes/SignIn.tsx b/src/routes/SignIn.tsx index fb17dd638..67e48f30f 100644 --- a/src/routes/SignIn.tsx +++ b/src/routes/SignIn.tsx @@ -14,7 +14,7 @@ import { PATHS } from '@src/lib/paths' import { Themes, getSystemTheme } from '@src/lib/theme' import { reportRejection } from '@src/lib/trap' import { toSync } from '@src/lib/utils' -import { authActor, useSettings } from '@src/machines/appMachine' +import { authActor, useSettings } from '@src/lib/singletons' import { APP_VERSION, IS_NIGHTLY } from '@src/routes/utils' const subtleBorder = diff --git a/yarn.lock b/yarn.lock new file mode 100644 index 000000000..41ea4c373 --- /dev/null +++ b/yarn.lock @@ -0,0 +1,10706 @@ +# THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. +# yarn lockfile v1 + + +"7zip-bin@~5.2.0": + version "5.2.0" + resolved "https://registry.yarnpkg.com/7zip-bin/-/7zip-bin-5.2.0.tgz#7a03314684dd6572b7dfa89e68ce31d60286854d" + integrity sha512-ukTPVhqG4jNzMro2qA9HSCSSVJN3aN7tlb+hfqYCt3ER0yWroeA2VR38MNrOHLQ/cVj+DaIMad0kFCtWWowh/A== + +"@adobe/css-tools@^4.0.1": + version "4.4.2" + resolved "https://registry.yarnpkg.com/@adobe/css-tools/-/css-tools-4.4.2.tgz#c836b1bd81e6d62cd6cdf3ee4948bcdce8ea79c8" + integrity sha512-baYZExFpsdkBNuvGKTKWCwKH57HRZLVtycZS05WTQNVOiXVSeAki3nU35zlRbToeMW8aHlJfyS+1C4BOv27q0A== + +"@alloc/quick-lru@^5.2.0": + version "5.2.0" + resolved "https://registry.yarnpkg.com/@alloc/quick-lru/-/quick-lru-5.2.0.tgz#7bf68b20c0a350f936915fcae06f58e32007ce30" + integrity sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw== + +"@ampproject/remapping@^2.2.0": + version "2.3.0" + resolved "https://registry.yarnpkg.com/@ampproject/remapping/-/remapping-2.3.0.tgz#ed441b6fa600072520ce18b43d2c8cc8caecc7f4" + integrity sha512-30iZtAPgz+LTIYoeivqYo853f02jBYSd5uGnGpkFV0M3xOt9aN73erkgYAmZU43x4VfqcnLxW9Kpg3R5LC4YYw== + dependencies: + "@jridgewell/gen-mapping" "^0.3.5" + "@jridgewell/trace-mapping" "^0.3.24" + +"@azure/abort-controller@^2.0.0": + version "2.1.2" + resolved "https://registry.yarnpkg.com/@azure/abort-controller/-/abort-controller-2.1.2.tgz#42fe0ccab23841d9905812c58f1082d27784566d" + integrity sha512-nBrLsEWm4J2u5LpAPjxADTlq3trDgVZZXHNKabeXZtpq3d3AbN/KGO82R87rdDz5/lYB024rtEf10/q0urNgsA== + dependencies: + tslib "^2.6.2" + +"@azure/core-auth@^1.4.0", "@azure/core-auth@^1.8.0", "@azure/core-auth@^1.9.0": + version "1.9.0" + resolved "https://registry.yarnpkg.com/@azure/core-auth/-/core-auth-1.9.0.tgz#ac725b03fabe3c892371065ee9e2041bee0fd1ac" + integrity sha512-FPwHpZywuyasDSLMqJ6fhbOK3TqUdviZNF8OqRGA4W5Ewib2lEEZ+pBsYcBa88B2NGO/SEnYPGhyBqNlE8ilSw== + dependencies: + "@azure/abort-controller" "^2.0.0" + "@azure/core-util" "^1.11.0" + tslib "^2.6.2" + +"@azure/core-client@^1.9.2": + version "1.9.3" + resolved "https://registry.yarnpkg.com/@azure/core-client/-/core-client-1.9.3.tgz#9ca8f3bdc730d10d58f65c9c2c9ca992bc15bb67" + integrity sha512-/wGw8fJ4mdpJ1Cum7s1S+VQyXt1ihwKLzfabS1O/RDADnmzVc01dHn44qD0BvGH6KlZNzOMW95tEpKqhkCChPA== + dependencies: + "@azure/abort-controller" "^2.0.0" + "@azure/core-auth" "^1.4.0" + "@azure/core-rest-pipeline" "^1.9.1" + "@azure/core-tracing" "^1.0.0" + "@azure/core-util" "^1.6.1" + "@azure/logger" "^1.0.0" + tslib "^2.6.2" + +"@azure/core-rest-pipeline@^1.17.0", "@azure/core-rest-pipeline@^1.9.1": + version "1.19.1" + resolved "https://registry.yarnpkg.com/@azure/core-rest-pipeline/-/core-rest-pipeline-1.19.1.tgz#e740676444777a04dc55656d8660131dfd926924" + integrity sha512-zHeoI3NCs53lLBbWNzQycjnYKsA1CVKlnzSNuSFcUDwBp8HHVObePxrM7HaX+Ha5Ks639H7chNC9HOaIhNS03w== + dependencies: + "@azure/abort-controller" "^2.0.0" + "@azure/core-auth" "^1.8.0" + "@azure/core-tracing" "^1.0.1" + "@azure/core-util" "^1.11.0" + "@azure/logger" "^1.0.0" + http-proxy-agent "^7.0.0" + https-proxy-agent "^7.0.0" + tslib "^2.6.2" + +"@azure/core-tracing@^1.0.0", "@azure/core-tracing@^1.0.1": + version "1.2.0" + resolved "https://registry.yarnpkg.com/@azure/core-tracing/-/core-tracing-1.2.0.tgz#7be5d53c3522d639cf19042cbcdb19f71bc35ab2" + integrity sha512-UKTiEJPkWcESPYJz3X5uKRYyOcJD+4nYph+KpfdPRnQJVrZfk0KJgdnaAWKfhsBBtAf/D58Az4AvCJEmWgIBAg== + dependencies: + tslib "^2.6.2" + +"@azure/core-util@^1.11.0", "@azure/core-util@^1.6.1": + version "1.11.0" + resolved "https://registry.yarnpkg.com/@azure/core-util/-/core-util-1.11.0.tgz#f530fc67e738aea872fbdd1cc8416e70219fada7" + integrity sha512-DxOSLua+NdpWoSqULhjDyAZTXFdP/LKkqtYuxxz1SCN289zk3OG8UOpnCQAz/tygyACBtWp/BoO72ptK7msY8g== + dependencies: + "@azure/abort-controller" "^2.0.0" + tslib "^2.6.2" + +"@azure/identity@^4.1.0": + version "4.9.1" + resolved "https://registry.yarnpkg.com/@azure/identity/-/identity-4.9.1.tgz#ee4b9435f1b96bea5985e7dec989760a67d9a119" + integrity sha512-986D7Cf1AOwYqSDtO/FnMAyk/Jc8qpftkGsxuehoh4F85MhQ4fICBGX/44+X1y78lN4Sqib3Bsoaoh/FvOGgmg== + dependencies: + "@azure/abort-controller" "^2.0.0" + "@azure/core-auth" "^1.9.0" + "@azure/core-client" "^1.9.2" + "@azure/core-rest-pipeline" "^1.17.0" + "@azure/core-tracing" "^1.0.0" + "@azure/core-util" "^1.11.0" + "@azure/logger" "^1.0.0" + "@azure/msal-browser" "^4.2.0" + "@azure/msal-node" "^3.5.0" + open "^10.1.0" + tslib "^2.2.0" + +"@azure/logger@^1.0.0": + version "1.1.4" + resolved "https://registry.yarnpkg.com/@azure/logger/-/logger-1.1.4.tgz#223cbf2b424dfa66478ce9a4f575f59c6f379768" + integrity sha512-4IXXzcCdLdlXuCG+8UKEwLA1T1NHqUfanhXYHiQTn+6sfWCZXduqbtXDGceg3Ce5QxTGo7EqmbV6Bi+aqKuClQ== + dependencies: + tslib "^2.6.2" + +"@azure/msal-browser@^4.2.0": + version "4.11.0" + resolved "https://registry.yarnpkg.com/@azure/msal-browser/-/msal-browser-4.11.0.tgz#e9d9651d692969e68c78ef873ed9a69e02389a64" + integrity sha512-0p5Ut3wORMP+975AKvaSPIO4UytgsfAvJ7RxaTx+nkP+Hpkmm93AuiMkBWKI2x9tApU/SLgIyPz/ZwLYUIWb5Q== + dependencies: + "@azure/msal-common" "15.5.1" + +"@azure/msal-common@15.5.1": + version "15.5.1" + resolved "https://registry.yarnpkg.com/@azure/msal-common/-/msal-common-15.5.1.tgz#3b34c81013530e1425a1fad40f3ac1238e1780f8" + integrity sha512-oxK0khbc4Bg1bKQnqDr7ikULhVL2OHgSrIq0Vlh4b6+hm4r0lr6zPMQE8ZvmacJuh+ZZGKBM5iIObhF1q1QimQ== + +"@azure/msal-node@^3.5.0": + version "3.5.1" + resolved "https://registry.yarnpkg.com/@azure/msal-node/-/msal-node-3.5.1.tgz#8bb233cbeeda83f64af4cc29569f1b5312c9b9ad" + integrity sha512-dkgMYM5B6tI88r/oqf5bYd93WkenQpaWwiszJDk7avVjso8cmuKRTW97dA1RMi6RhihZFLtY1VtWxU9+sW2T5g== + dependencies: + "@azure/msal-common" "15.5.1" + jsonwebtoken "^9.0.0" + uuid "^8.3.0" + +"@babel/code-frame@^7.10.4", "@babel/code-frame@^7.12.13", "@babel/code-frame@^7.26.2": + version "7.26.2" + resolved "https://registry.yarnpkg.com/@babel/code-frame/-/code-frame-7.26.2.tgz#4b5fab97d33338eff916235055f0ebc21e573a85" + integrity sha512-RJlIHRueQgwWitWgF8OdFYGZX328Ax5BCemNGlqHfplnRT9ESi8JkFlvaVYbS+UubVY6dpv87Fs2u5M29iNFVQ== + dependencies: + "@babel/helper-validator-identifier" "^7.25.9" + js-tokens "^4.0.0" + picocolors "^1.0.0" + +"@babel/compat-data@^7.22.6", "@babel/compat-data@^7.26.8": + version "7.26.8" + resolved "https://registry.yarnpkg.com/@babel/compat-data/-/compat-data-7.26.8.tgz#821c1d35641c355284d4a870b8a4a7b0c141e367" + integrity sha512-oH5UPLMWR3L2wEFLnFJ1TZXqHufiTKAiLfqw5zkhS4dKXLJ10yVztfil/twG8EDTA4F/tvVNw9nOl4ZMslB8rQ== + +"@babel/core@^7.21.4", "@babel/core@^7.26.10": + version "7.26.10" + resolved "https://registry.yarnpkg.com/@babel/core/-/core-7.26.10.tgz#5c876f83c8c4dcb233ee4b670c0606f2ac3000f9" + integrity sha512-vMqyb7XCDMPvJFFOaT9kxtiRh42GwlZEg1/uIgtZshS5a/8OaduUfCi7kynKgc3Tw/6Uo2D+db9qBttghhmxwQ== + dependencies: + "@ampproject/remapping" "^2.2.0" + "@babel/code-frame" "^7.26.2" + "@babel/generator" "^7.26.10" + "@babel/helper-compilation-targets" "^7.26.5" + "@babel/helper-module-transforms" "^7.26.0" + "@babel/helpers" "^7.26.10" + "@babel/parser" "^7.26.10" + "@babel/template" "^7.26.9" + "@babel/traverse" "^7.26.10" + "@babel/types" "^7.26.10" + convert-source-map "^2.0.0" + debug "^4.1.0" + gensync "^1.0.0-beta.2" + json5 "^2.2.3" + semver "^6.3.1" + +"@babel/generator@^7.26.10", "@babel/generator@^7.27.0": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/generator/-/generator-7.27.0.tgz#764382b5392e5b9aff93cadb190d0745866cbc2c" + integrity sha512-VybsKvpiN1gU1sdMZIp7FcqphVVKEwcuj02x73uvcHE0PTihx1nlBcowYWhDwjpoAXRv43+gDzyggGnn1XZhVw== + dependencies: + "@babel/parser" "^7.27.0" + "@babel/types" "^7.27.0" + "@jridgewell/gen-mapping" "^0.3.5" + "@jridgewell/trace-mapping" "^0.3.25" + jsesc "^3.0.2" + +"@babel/helper-annotate-as-pure@^7.18.6", "@babel/helper-annotate-as-pure@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-annotate-as-pure/-/helper-annotate-as-pure-7.25.9.tgz#d8eac4d2dc0d7b6e11fa6e535332e0d3184f06b4" + integrity sha512-gv7320KBUFJz1RnylIg5WWYPRXKZ884AGkYpgpWW02TH66Dl+HaC1t1CKd0z3R4b6hdYEcmrNZHUmfCP+1u3/g== + dependencies: + "@babel/types" "^7.25.9" + +"@babel/helper-compilation-targets@^7.22.6", "@babel/helper-compilation-targets@^7.25.9", "@babel/helper-compilation-targets@^7.26.5": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/helper-compilation-targets/-/helper-compilation-targets-7.27.0.tgz#de0c753b1cd1d9ab55d473c5a5cf7170f0a81880" + integrity sha512-LVk7fbXml0H2xH34dFzKQ7TDZ2G4/rVTOrq9V+icbbadjbVxxeFeDsNHv2SrZeWoA+6ZiTyWYWtScEIW07EAcA== + dependencies: + "@babel/compat-data" "^7.26.8" + "@babel/helper-validator-option" "^7.25.9" + browserslist "^4.24.0" + lru-cache "^5.1.1" + semver "^6.3.1" + +"@babel/helper-create-class-features-plugin@^7.21.0", "@babel/helper-create-class-features-plugin@^7.25.9": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/helper-create-class-features-plugin/-/helper-create-class-features-plugin-7.27.0.tgz#518fad6a307c6a96f44af14912b2c20abe9bfc30" + integrity sha512-vSGCvMecvFCd/BdpGlhpXYNhhC4ccxyvQWpbGL4CWbvfEoLFWUZuSuf7s9Aw70flgQF+6vptvgK2IfOnKlRmBg== + dependencies: + "@babel/helper-annotate-as-pure" "^7.25.9" + "@babel/helper-member-expression-to-functions" "^7.25.9" + "@babel/helper-optimise-call-expression" "^7.25.9" + "@babel/helper-replace-supers" "^7.26.5" + "@babel/helper-skip-transparent-expression-wrappers" "^7.25.9" + "@babel/traverse" "^7.27.0" + semver "^6.3.1" + +"@babel/helper-create-regexp-features-plugin@^7.18.6", "@babel/helper-create-regexp-features-plugin@^7.25.9": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/helper-create-regexp-features-plugin/-/helper-create-regexp-features-plugin-7.27.0.tgz#0e41f7d38c2ebe06ebd9cf0e02fb26019c77cd95" + integrity sha512-fO8l08T76v48BhpNRW/nQ0MxfnSdoSKUJBMjubOAYffsVuGG5qOfMq7N6Es7UJvi7Y8goXXo07EfcHZXDPuELQ== + dependencies: + "@babel/helper-annotate-as-pure" "^7.25.9" + regexpu-core "^6.2.0" + semver "^6.3.1" + +"@babel/helper-define-polyfill-provider@^0.6.3", "@babel/helper-define-polyfill-provider@^0.6.4": + version "0.6.4" + resolved "https://registry.yarnpkg.com/@babel/helper-define-polyfill-provider/-/helper-define-polyfill-provider-0.6.4.tgz#15e8746368bfa671785f5926ff74b3064c291fab" + integrity sha512-jljfR1rGnXXNWnmQg2K3+bvhkxB51Rl32QRaOTuwwjviGrHzIbSc8+x9CpraDtbT7mfyjXObULP4w/adunNwAw== + dependencies: + "@babel/helper-compilation-targets" "^7.22.6" + "@babel/helper-plugin-utils" "^7.22.5" + debug "^4.1.1" + lodash.debounce "^4.0.8" + resolve "^1.14.2" + +"@babel/helper-member-expression-to-functions@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-member-expression-to-functions/-/helper-member-expression-to-functions-7.25.9.tgz#9dfffe46f727005a5ea29051ac835fb735e4c1a3" + integrity sha512-wbfdZ9w5vk0C0oyHqAJbc62+vet5prjj01jjJ8sKn3j9h3MQQlflEdXYvuqRWjHnM12coDEqiC1IRCi0U/EKwQ== + dependencies: + "@babel/traverse" "^7.25.9" + "@babel/types" "^7.25.9" + +"@babel/helper-module-imports@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-module-imports/-/helper-module-imports-7.25.9.tgz#e7f8d20602ebdbf9ebbea0a0751fb0f2a4141715" + integrity sha512-tnUA4RsrmflIM6W6RFTLFSXITtl0wKjgpnLgXyowocVPrbYrLUXSBXDgTs8BlbmIzIdlBySRQjINYs2BAkiLtw== + dependencies: + "@babel/traverse" "^7.25.9" + "@babel/types" "^7.25.9" + +"@babel/helper-module-transforms@^7.25.9", "@babel/helper-module-transforms@^7.26.0": + version "7.26.0" + resolved "https://registry.yarnpkg.com/@babel/helper-module-transforms/-/helper-module-transforms-7.26.0.tgz#8ce54ec9d592695e58d84cd884b7b5c6a2fdeeae" + integrity sha512-xO+xu6B5K2czEnQye6BHA7DolFFmS3LB7stHZFaOLb1pAwO1HWLS8fXA+eh0A2yIvltPVmx3eNNDBJA2SLHXFw== + dependencies: + "@babel/helper-module-imports" "^7.25.9" + "@babel/helper-validator-identifier" "^7.25.9" + "@babel/traverse" "^7.25.9" + +"@babel/helper-optimise-call-expression@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-optimise-call-expression/-/helper-optimise-call-expression-7.25.9.tgz#3324ae50bae7e2ab3c33f60c9a877b6a0146b54e" + integrity sha512-FIpuNaz5ow8VyrYcnXQTDRGvV6tTjkNtCK/RYNDXGSLlUD6cBuQTSw43CShGxjvfBTfcUA/r6UhUCbtYqkhcuQ== + dependencies: + "@babel/types" "^7.25.9" + +"@babel/helper-plugin-utils@^7.0.0", "@babel/helper-plugin-utils@^7.14.5", "@babel/helper-plugin-utils@^7.18.6", "@babel/helper-plugin-utils@^7.20.2", "@babel/helper-plugin-utils@^7.22.5", "@babel/helper-plugin-utils@^7.25.9", "@babel/helper-plugin-utils@^7.26.5": + version "7.26.5" + resolved "https://registry.yarnpkg.com/@babel/helper-plugin-utils/-/helper-plugin-utils-7.26.5.tgz#18580d00c9934117ad719392c4f6585c9333cc35" + integrity sha512-RS+jZcRdZdRFzMyr+wcsaqOmld1/EqTghfaBGQQd/WnRdzdlvSZ//kF7U8VQTxf1ynZ4cjUcYgjVGx13ewNPMg== + +"@babel/helper-remap-async-to-generator@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-remap-async-to-generator/-/helper-remap-async-to-generator-7.25.9.tgz#e53956ab3d5b9fb88be04b3e2f31b523afd34b92" + integrity sha512-IZtukuUeBbhgOcaW2s06OXTzVNJR0ybm4W5xC1opWFFJMZbwRj5LCk+ByYH7WdZPZTt8KnFwA8pvjN2yqcPlgw== + dependencies: + "@babel/helper-annotate-as-pure" "^7.25.9" + "@babel/helper-wrap-function" "^7.25.9" + "@babel/traverse" "^7.25.9" + +"@babel/helper-replace-supers@^7.25.9", "@babel/helper-replace-supers@^7.26.5": + version "7.26.5" + resolved "https://registry.yarnpkg.com/@babel/helper-replace-supers/-/helper-replace-supers-7.26.5.tgz#6cb04e82ae291dae8e72335dfe438b0725f14c8d" + integrity sha512-bJ6iIVdYX1YooY2X7w1q6VITt+LnUILtNk7zT78ykuwStx8BauCzxvFqFaHjOpW1bVnSUM1PN1f0p5P21wHxvg== + dependencies: + "@babel/helper-member-expression-to-functions" "^7.25.9" + "@babel/helper-optimise-call-expression" "^7.25.9" + "@babel/traverse" "^7.26.5" + +"@babel/helper-skip-transparent-expression-wrappers@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-skip-transparent-expression-wrappers/-/helper-skip-transparent-expression-wrappers-7.25.9.tgz#0b2e1b62d560d6b1954893fd2b705dc17c91f0c9" + integrity sha512-K4Du3BFa3gvyhzgPcntrkDgZzQaq6uozzcpGbOO1OEJaI+EJdqWIMTLgFgQf6lrfiDFo5FU+BxKepI9RmZqahA== + dependencies: + "@babel/traverse" "^7.25.9" + "@babel/types" "^7.25.9" + +"@babel/helper-string-parser@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-string-parser/-/helper-string-parser-7.25.9.tgz#1aabb72ee72ed35789b4bbcad3ca2862ce614e8c" + integrity sha512-4A/SCr/2KLd5jrtOMFzaKjVtAei3+2r/NChoBNoZ3EyP/+GlhoaEGoWOZUmFmoITP7zOJyHIMm+DYRd8o3PvHA== + +"@babel/helper-validator-identifier@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-validator-identifier/-/helper-validator-identifier-7.25.9.tgz#24b64e2c3ec7cd3b3c547729b8d16871f22cbdc7" + integrity sha512-Ed61U6XJc3CVRfkERJWDz4dJwKe7iLmmJsbOGu9wSloNSFttHV0I8g6UAgb7qnK5ly5bGLPd4oXZlxCdANBOWQ== + +"@babel/helper-validator-option@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-validator-option/-/helper-validator-option-7.25.9.tgz#86e45bd8a49ab7e03f276577f96179653d41da72" + integrity sha512-e/zv1co8pp55dNdEcCynfj9X7nyUKUXoUEwfXqaZt0omVOmDe9oOTdKStH4GmAw6zxMFs50ZayuMfHDKlO7Tfw== + +"@babel/helper-wrap-function@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/helper-wrap-function/-/helper-wrap-function-7.25.9.tgz#d99dfd595312e6c894bd7d237470025c85eea9d0" + integrity sha512-ETzz9UTjQSTmw39GboatdymDq4XIQbR8ySgVrylRhPOFpsd+JrKHIuF0de7GCWmem+T4uC5z7EZguod7Wj4A4g== + dependencies: + "@babel/template" "^7.25.9" + "@babel/traverse" "^7.25.9" + "@babel/types" "^7.25.9" + +"@babel/helpers@^7.26.10": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/helpers/-/helpers-7.27.0.tgz#53d156098defa8243eab0f32fa17589075a1b808" + integrity sha512-U5eyP/CTFPuNE3qk+WZMxFkp/4zUzdceQlfzf7DdGdhp+Fezd7HD+i8Y24ZuTMKX3wQBld449jijbGq6OdGNQg== + dependencies: + "@babel/template" "^7.27.0" + "@babel/types" "^7.27.0" + +"@babel/parser@^7.1.0", "@babel/parser@^7.20.7", "@babel/parser@^7.21.4", "@babel/parser@^7.26.10", "@babel/parser@^7.27.0": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/parser/-/parser-7.27.0.tgz#3d7d6ee268e41d2600091cbd4e145ffee85a44ec" + integrity sha512-iaepho73/2Pz7w2eMS0Q5f83+0RKI7i4xmiYeBmDzfRVbQtTOG7Ts0S4HzJVsTMGI9keU8rNfuZr8DKfSt7Yyg== + dependencies: + "@babel/types" "^7.27.0" + +"@babel/plugin-bugfix-firefox-class-in-computed-class-key@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-bugfix-firefox-class-in-computed-class-key/-/plugin-bugfix-firefox-class-in-computed-class-key-7.25.9.tgz#cc2e53ebf0a0340777fff5ed521943e253b4d8fe" + integrity sha512-ZkRyVkThtxQ/J6nv3JFYv1RYY+JT5BvU0y3k5bWrmuG4woXypRa4PXmm9RhOwodRkYFWqC0C0cqcJ4OqR7kW+g== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/traverse" "^7.25.9" + +"@babel/plugin-bugfix-safari-class-field-initializer-scope@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-bugfix-safari-class-field-initializer-scope/-/plugin-bugfix-safari-class-field-initializer-scope-7.25.9.tgz#af9e4fb63ccb8abcb92375b2fcfe36b60c774d30" + integrity sha512-MrGRLZxLD/Zjj0gdU15dfs+HH/OXvnw/U4jJD8vpcP2CJQapPEv1IWwjc/qMg7ItBlPwSv1hRBbb7LeuANdcnw== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression/-/plugin-bugfix-safari-id-destructuring-collision-in-function-expression-7.25.9.tgz#e8dc26fcd616e6c5bf2bd0d5a2c151d4f92a9137" + integrity sha512-2qUwwfAFpJLZqxd02YW9btUCZHl+RFvdDkNfZwaIJrvB8Tesjsk8pEQkTvGwZXLqXUx/2oyY3ySRhm6HOXuCug== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining/-/plugin-bugfix-v8-spread-parameters-in-optional-chaining-7.25.9.tgz#807a667f9158acac6f6164b4beb85ad9ebc9e1d1" + integrity sha512-6xWgLZTJXwilVjlnV7ospI3xi+sl8lN8rXXbBD6vYn3UYDlGsag8wrZkKcSI8G6KgqKP7vNFaDgeDnfAABq61g== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/helper-skip-transparent-expression-wrappers" "^7.25.9" + "@babel/plugin-transform-optional-chaining" "^7.25.9" + +"@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly/-/plugin-bugfix-v8-static-class-fields-redefine-readonly-7.25.9.tgz#de7093f1e7deaf68eadd7cc6b07f2ab82543269e" + integrity sha512-aLnMXYPnzwwqhYSCyXfKkIkYgJ8zv9RK+roo9DkTXz38ynIhd9XCbN08s3MGvqL2MYGVUGdRQLL/JqBIeJhJBg== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/traverse" "^7.25.9" + +"@babel/plugin-proposal-private-property-in-object@7.21.0-placeholder-for-preset-env.2": + version "7.21.0-placeholder-for-preset-env.2" + resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.0-placeholder-for-preset-env.2.tgz#7844f9289546efa9febac2de4cfe358a050bd703" + integrity sha512-SOSkfJDddaM7mak6cPEpswyTRnuRltl429hMraQEglW+OkovnCzsiszTmsrlY//qLFjCpQDFRvjdm2wA5pPm9w== + +"@babel/plugin-proposal-private-property-in-object@^7.21.11": + version "7.21.11" + resolved "https://registry.yarnpkg.com/@babel/plugin-proposal-private-property-in-object/-/plugin-proposal-private-property-in-object-7.21.11.tgz#69d597086b6760c4126525cfa154f34631ff272c" + integrity sha512-0QZ8qP/3RLDVBwBFoWAwCtgcDZJVwA5LUJRZU8x2YFfKNuFq161wK3cuGrALu5yiPu+vzwTAg/sMWVNeWeNyaw== + dependencies: + "@babel/helper-annotate-as-pure" "^7.18.6" + "@babel/helper-create-class-features-plugin" "^7.21.0" + "@babel/helper-plugin-utils" "^7.20.2" + "@babel/plugin-syntax-private-property-in-object" "^7.14.5" + +"@babel/plugin-syntax-import-assertions@^7.26.0": + version "7.26.0" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-import-assertions/-/plugin-syntax-import-assertions-7.26.0.tgz#620412405058efa56e4a564903b79355020f445f" + integrity sha512-QCWT5Hh830hK5EQa7XzuqIkQU9tT/whqbDz7kuaZMHFl1inRRg7JnuAEOQ0Ur0QUl0NufCk1msK2BeY79Aj/eg== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-syntax-import-attributes@^7.26.0": + version "7.26.0" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-import-attributes/-/plugin-syntax-import-attributes-7.26.0.tgz#3b1412847699eea739b4f2602c74ce36f6b0b0f7" + integrity sha512-e2dttdsJ1ZTpi3B9UYGLw41hifAubg19AtCu/2I/F1QNVclOBr1dYpTdmdyZ84Xiz43BS/tCUkMAZNLv12Pi+A== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-syntax-private-property-in-object@^7.14.5": + version "7.14.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-private-property-in-object/-/plugin-syntax-private-property-in-object-7.14.5.tgz#0dc6671ec0ea22b6e94a1114f857970cd39de1ad" + integrity sha512-0wVnp9dxJ72ZUJDV27ZfbSj6iHLoytYZmh3rFcxNnvsJF3ktkzLDZPy/mA17HGsaQT3/DQsWYX1f1QGWkCoVUg== + dependencies: + "@babel/helper-plugin-utils" "^7.14.5" + +"@babel/plugin-syntax-unicode-sets-regex@^7.18.6": + version "7.18.6" + resolved "https://registry.yarnpkg.com/@babel/plugin-syntax-unicode-sets-regex/-/plugin-syntax-unicode-sets-regex-7.18.6.tgz#d49a3b3e6b52e5be6740022317580234a6a47357" + integrity sha512-727YkEAPwSIQTv5im8QHz3upqp92JTWhidIC81Tdx4VJYIte/VndKf1qKrfnnhPLiPghStWfvC/iFaMCQu7Nqg== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.18.6" + "@babel/helper-plugin-utils" "^7.18.6" + +"@babel/plugin-transform-arrow-functions@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-arrow-functions/-/plugin-transform-arrow-functions-7.25.9.tgz#7821d4410bee5daaadbb4cdd9a6649704e176845" + integrity sha512-6jmooXYIwn9ca5/RylZADJ+EnSxVUS5sjeJ9UPk6RWRzXCmOJCy6dqItPJFpw2cuCangPK4OYr5uhGKcmrm5Qg== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-async-generator-functions@^7.26.8": + version "7.26.8" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-async-generator-functions/-/plugin-transform-async-generator-functions-7.26.8.tgz#5e3991135e3b9c6eaaf5eff56d1ae5a11df45ff8" + integrity sha512-He9Ej2X7tNf2zdKMAGOsmg2MrFc+hfoAhd3po4cWfo/NWjzEAKa0oQruj1ROVUdl0e6fb6/kE/G3SSxE0lRJOg== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + "@babel/helper-remap-async-to-generator" "^7.25.9" + "@babel/traverse" "^7.26.8" + +"@babel/plugin-transform-async-to-generator@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-async-to-generator/-/plugin-transform-async-to-generator-7.25.9.tgz#c80008dacae51482793e5a9c08b39a5be7e12d71" + integrity sha512-NT7Ejn7Z/LjUH0Gv5KsBCxh7BH3fbLTV0ptHvpeMvrt3cPThHfJfst9Wrb7S8EvJ7vRTFI7z+VAvFVEQn/m5zQ== + dependencies: + "@babel/helper-module-imports" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/helper-remap-async-to-generator" "^7.25.9" + +"@babel/plugin-transform-block-scoped-functions@^7.26.5": + version "7.26.5" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoped-functions/-/plugin-transform-block-scoped-functions-7.26.5.tgz#3dc4405d31ad1cbe45293aa57205a6e3b009d53e" + integrity sha512-chuTSY+hq09+/f5lMj8ZSYgCFpppV2CbYrhNFJ1BFoXpiWPnnAb7R0MqrafCpN8E1+YRrtM1MXZHJdIx8B6rMQ== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + +"@babel/plugin-transform-block-scoping@^7.25.9": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-block-scoping/-/plugin-transform-block-scoping-7.27.0.tgz#acc2c0d98a7439bbde4244588ddbd4904701d47f" + integrity sha512-u1jGphZ8uDI2Pj/HJj6YQ6XQLZCNjOlprjxB5SVz6rq2T6SwAR+CdrWK0CP7F+9rDVMXdB0+r6Am5G5aobOjAQ== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + +"@babel/plugin-transform-class-properties@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-class-properties/-/plugin-transform-class-properties-7.25.9.tgz#a8ce84fedb9ad512549984101fa84080a9f5f51f" + integrity sha512-bbMAII8GRSkcd0h0b4X+36GksxuheLFjP65ul9w6C3KgAamI3JqErNgSrosX6ZPj+Mpim5VvEbawXxJCyEUV3Q== + dependencies: + "@babel/helper-create-class-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-class-static-block@^7.26.0": + version "7.26.0" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-class-static-block/-/plugin-transform-class-static-block-7.26.0.tgz#6c8da219f4eb15cae9834ec4348ff8e9e09664a0" + integrity sha512-6J2APTs7BDDm+UMqP1useWqhcRAXo0WIoVj26N7kPFB6S73Lgvyka4KTZYIxtgYXiN5HTyRObA72N2iu628iTQ== + dependencies: + "@babel/helper-create-class-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-classes@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-classes/-/plugin-transform-classes-7.25.9.tgz#7152457f7880b593a63ade8a861e6e26a4469f52" + integrity sha512-mD8APIXmseE7oZvZgGABDyM34GUmK45Um2TXiBUt7PnuAxrgoSVf123qUzPxEr/+/BHrRn5NMZCdE2m/1F8DGg== + dependencies: + "@babel/helper-annotate-as-pure" "^7.25.9" + "@babel/helper-compilation-targets" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/helper-replace-supers" "^7.25.9" + "@babel/traverse" "^7.25.9" + globals "^11.1.0" + +"@babel/plugin-transform-computed-properties@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-computed-properties/-/plugin-transform-computed-properties-7.25.9.tgz#db36492c78460e534b8852b1d5befe3c923ef10b" + integrity sha512-HnBegGqXZR12xbcTHlJ9HGxw1OniltT26J5YpfruGqtUHlz/xKf/G2ak9e+t0rVqrjXa9WOhvYPz1ERfMj23AA== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/template" "^7.25.9" + +"@babel/plugin-transform-destructuring@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-destructuring/-/plugin-transform-destructuring-7.25.9.tgz#966ea2595c498224340883602d3cfd7a0c79cea1" + integrity sha512-WkCGb/3ZxXepmMiX101nnGiU+1CAdut8oHyEOHxkKuS1qKpU2SMXE2uSvfz8PBuLd49V6LEsbtyPhWC7fnkgvQ== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-dotall-regex@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-dotall-regex/-/plugin-transform-dotall-regex-7.25.9.tgz#bad7945dd07734ca52fe3ad4e872b40ed09bb09a" + integrity sha512-t7ZQ7g5trIgSRYhI9pIJtRl64KHotutUJsh4Eze5l7olJv+mRSg4/MmbZ0tv1eeqRbdvo/+trvJD/Oc5DmW2cA== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-duplicate-keys@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-duplicate-keys/-/plugin-transform-duplicate-keys-7.25.9.tgz#8850ddf57dce2aebb4394bb434a7598031059e6d" + integrity sha512-LZxhJ6dvBb/f3x8xwWIuyiAHy56nrRG3PeYTpBkkzkYRRQ6tJLu68lEF5VIqMUZiAV7a8+Tb78nEoMCMcqjXBw== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-duplicate-named-capturing-groups-regex@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-duplicate-named-capturing-groups-regex/-/plugin-transform-duplicate-named-capturing-groups-regex-7.25.9.tgz#6f7259b4de127721a08f1e5165b852fcaa696d31" + integrity sha512-0UfuJS0EsXbRvKnwcLjFtJy/Sxc5J5jhLHnFhy7u4zih97Hz6tJkLU+O+FMMrNZrosUPxDi6sYxJ/EA8jDiAog== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-dynamic-import@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-dynamic-import/-/plugin-transform-dynamic-import-7.25.9.tgz#23e917de63ed23c6600c5dd06d94669dce79f7b8" + integrity sha512-GCggjexbmSLaFhqsojeugBpeaRIgWNTcgKVq/0qIteFEqY2A+b9QidYadrWlnbWQUrW5fn+mCvf3tr7OeBFTyg== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-exponentiation-operator@^7.26.3": + version "7.26.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-exponentiation-operator/-/plugin-transform-exponentiation-operator-7.26.3.tgz#e29f01b6de302c7c2c794277a48f04a9ca7f03bc" + integrity sha512-7CAHcQ58z2chuXPWblnn1K6rLDnDWieghSOEmqQsrBenH0P9InCUtOJYD89pvngljmZlJcz3fcmgYsXFNGa1ZQ== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-export-namespace-from@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-export-namespace-from/-/plugin-transform-export-namespace-from-7.25.9.tgz#90745fe55053394f554e40584cda81f2c8a402a2" + integrity sha512-2NsEz+CxzJIVOPx2o9UsW1rXLqtChtLoVnwYHHiB04wS5sgn7mrV45fWMBX0Kk+ub9uXytVYfNP2HjbVbCB3Ww== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-for-of@^7.26.9": + version "7.26.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-for-of/-/plugin-transform-for-of-7.26.9.tgz#27231f79d5170ef33b5111f07fe5cafeb2c96a56" + integrity sha512-Hry8AusVm8LW5BVFgiyUReuoGzPUpdHQQqJY5bZnbbf+ngOHWuCuYFKw/BqaaWlvEUrF91HMhDtEaI1hZzNbLg== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + "@babel/helper-skip-transparent-expression-wrappers" "^7.25.9" + +"@babel/plugin-transform-function-name@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-function-name/-/plugin-transform-function-name-7.25.9.tgz#939d956e68a606661005bfd550c4fc2ef95f7b97" + integrity sha512-8lP+Yxjv14Vc5MuWBpJsoUCd3hD6V9DgBon2FVYL4jJgbnVQ9fTgYmonchzZJOVNgzEgbxp4OwAf6xz6M/14XA== + dependencies: + "@babel/helper-compilation-targets" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/traverse" "^7.25.9" + +"@babel/plugin-transform-json-strings@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-json-strings/-/plugin-transform-json-strings-7.25.9.tgz#c86db407cb827cded902a90c707d2781aaa89660" + integrity sha512-xoTMk0WXceiiIvsaquQQUaLLXSW1KJ159KP87VilruQm0LNNGxWzahxSS6T6i4Zg3ezp4vA4zuwiNUR53qmQAw== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-literals@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-literals/-/plugin-transform-literals-7.25.9.tgz#1a1c6b4d4aa59bc4cad5b6b3a223a0abd685c9de" + integrity sha512-9N7+2lFziW8W9pBl2TzaNht3+pgMIRP74zizeCSrtnSKVdUl8mAjjOP2OOVQAfZ881P2cNjDj1uAMEdeD50nuQ== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-logical-assignment-operators@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-logical-assignment-operators/-/plugin-transform-logical-assignment-operators-7.25.9.tgz#b19441a8c39a2fda0902900b306ea05ae1055db7" + integrity sha512-wI4wRAzGko551Y8eVf6iOY9EouIDTtPb0ByZx+ktDGHwv6bHFimrgJM/2T021txPZ2s4c7bqvHbd+vXG6K948Q== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-member-expression-literals@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-member-expression-literals/-/plugin-transform-member-expression-literals-7.25.9.tgz#63dff19763ea64a31f5e6c20957e6a25e41ed5de" + integrity sha512-PYazBVfofCQkkMzh2P6IdIUaCEWni3iYEerAsRWuVd8+jlM1S9S9cz1dF9hIzyoZ8IA3+OwVYIp9v9e+GbgZhA== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-modules-amd@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-amd/-/plugin-transform-modules-amd-7.25.9.tgz#49ba478f2295101544abd794486cd3088dddb6c5" + integrity sha512-g5T11tnI36jVClQlMlt4qKDLlWnG5pP9CSM4GhdRciTNMRgkfpo5cR6b4rGIOYPgRRuFAvwjPQ/Yk+ql4dyhbw== + dependencies: + "@babel/helper-module-transforms" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-modules-commonjs@^7.26.3": + version "7.26.3" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-commonjs/-/plugin-transform-modules-commonjs-7.26.3.tgz#8f011d44b20d02c3de44d8850d971d8497f981fb" + integrity sha512-MgR55l4q9KddUDITEzEFYn5ZsGDXMSsU9E+kh7fjRXTIC3RHqfCo8RPRbyReYJh44HQ/yomFkqbOFohXvDCiIQ== + dependencies: + "@babel/helper-module-transforms" "^7.26.0" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-modules-systemjs@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-systemjs/-/plugin-transform-modules-systemjs-7.25.9.tgz#8bd1b43836269e3d33307151a114bcf3ba6793f8" + integrity sha512-hyss7iIlH/zLHaehT+xwiymtPOpsiwIIRlCAOwBB04ta5Tt+lNItADdlXw3jAWZ96VJ2jlhl/c+PNIQPKNfvcA== + dependencies: + "@babel/helper-module-transforms" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/helper-validator-identifier" "^7.25.9" + "@babel/traverse" "^7.25.9" + +"@babel/plugin-transform-modules-umd@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-modules-umd/-/plugin-transform-modules-umd-7.25.9.tgz#6710079cdd7c694db36529a1e8411e49fcbf14c9" + integrity sha512-bS9MVObUgE7ww36HEfwe6g9WakQ0KF07mQF74uuXdkoziUPfKyu/nIm663kz//e5O1nPInPFx36z7WJmJ4yNEw== + dependencies: + "@babel/helper-module-transforms" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-named-capturing-groups-regex@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-named-capturing-groups-regex/-/plugin-transform-named-capturing-groups-regex-7.25.9.tgz#454990ae6cc22fd2a0fa60b3a2c6f63a38064e6a" + integrity sha512-oqB6WHdKTGl3q/ItQhpLSnWWOpjUJLsOCLVyeFgeTktkBSCiurvPOsyt93gibI9CmuKvTUEtWmG5VhZD+5T/KA== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-new-target@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-new-target/-/plugin-transform-new-target-7.25.9.tgz#42e61711294b105c248336dcb04b77054ea8becd" + integrity sha512-U/3p8X1yCSoKyUj2eOBIx3FOn6pElFOKvAAGf8HTtItuPyB+ZeOqfn+mvTtg9ZlOAjsPdK3ayQEjqHjU/yLeVQ== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-nullish-coalescing-operator@^7.26.6": + version "7.26.6" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-nullish-coalescing-operator/-/plugin-transform-nullish-coalescing-operator-7.26.6.tgz#fbf6b3c92cb509e7b319ee46e3da89c5bedd31fe" + integrity sha512-CKW8Vu+uUZneQCPtXmSBUC6NCAUdya26hWCElAWh5mVSlSRsmiCPUUDKb3Z0szng1hiAJa098Hkhg9o4SE35Qw== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + +"@babel/plugin-transform-numeric-separator@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-numeric-separator/-/plugin-transform-numeric-separator-7.25.9.tgz#bfed75866261a8b643468b0ccfd275f2033214a1" + integrity sha512-TlprrJ1GBZ3r6s96Yq8gEQv82s8/5HnCVHtEJScUj90thHQbwe+E5MLhi2bbNHBEJuzrvltXSru+BUxHDoog7Q== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-object-rest-spread@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-object-rest-spread/-/plugin-transform-object-rest-spread-7.25.9.tgz#0203725025074164808bcf1a2cfa90c652c99f18" + integrity sha512-fSaXafEE9CVHPweLYw4J0emp1t8zYTXyzN3UuG+lylqkvYd7RMrsOQ8TYx5RF231be0vqtFC6jnx3UmpJmKBYg== + dependencies: + "@babel/helper-compilation-targets" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/plugin-transform-parameters" "^7.25.9" + +"@babel/plugin-transform-object-super@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-object-super/-/plugin-transform-object-super-7.25.9.tgz#385d5de135162933beb4a3d227a2b7e52bb4cf03" + integrity sha512-Kj/Gh+Rw2RNLbCK1VAWj2U48yxxqL2x0k10nPtSdRa0O2xnHXalD0s+o1A6a0W43gJ00ANo38jxkQreckOzv5A== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/helper-replace-supers" "^7.25.9" + +"@babel/plugin-transform-optional-catch-binding@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-optional-catch-binding/-/plugin-transform-optional-catch-binding-7.25.9.tgz#10e70d96d52bb1f10c5caaac59ac545ea2ba7ff3" + integrity sha512-qM/6m6hQZzDcZF3onzIhZeDHDO43bkNNlOX0i8n3lR6zLbu0GN2d8qfM/IERJZYauhAHSLHy39NF0Ctdvcid7g== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-optional-chaining@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-optional-chaining/-/plugin-transform-optional-chaining-7.25.9.tgz#e142eb899d26ef715435f201ab6e139541eee7dd" + integrity sha512-6AvV0FsLULbpnXeBjrY4dmWF8F7gf8QnvTEoO/wX/5xm/xE1Xo8oPuD3MPS+KS9f9XBEAWN7X1aWr4z9HdOr7A== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/helper-skip-transparent-expression-wrappers" "^7.25.9" + +"@babel/plugin-transform-parameters@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-parameters/-/plugin-transform-parameters-7.25.9.tgz#b856842205b3e77e18b7a7a1b94958069c7ba257" + integrity sha512-wzz6MKwpnshBAiRmn4jR8LYz/g8Ksg0o80XmwZDlordjwEk9SxBzTWC7F5ef1jhbrbOW2DJ5J6ayRukrJmnr0g== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-private-methods@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-private-methods/-/plugin-transform-private-methods-7.25.9.tgz#847f4139263577526455d7d3223cd8bda51e3b57" + integrity sha512-D/JUozNpQLAPUVusvqMxyvjzllRaF8/nSrP1s2YGQT/W4LHK4xxsMcHjhOGTS01mp9Hda8nswb+FblLdJornQw== + dependencies: + "@babel/helper-create-class-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-private-property-in-object@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-private-property-in-object/-/plugin-transform-private-property-in-object-7.25.9.tgz#9c8b73e64e6cc3cbb2743633885a7dd2c385fe33" + integrity sha512-Evf3kcMqzXA3xfYJmZ9Pg1OvKdtqsDMSWBDzZOPLvHiTt36E75jLDQo5w1gtRU95Q4E5PDttrTf25Fw8d/uWLw== + dependencies: + "@babel/helper-annotate-as-pure" "^7.25.9" + "@babel/helper-create-class-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-property-literals@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-property-literals/-/plugin-transform-property-literals-7.25.9.tgz#d72d588bd88b0dec8b62e36f6fda91cedfe28e3f" + integrity sha512-IvIUeV5KrS/VPavfSM/Iu+RE6llrHrYIKY1yfCzyO/lMXHQ+p7uGhonmGVisv6tSBSVgWzMBohTcvkC9vQcQFA== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-react-jsx-self@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-react-jsx-self/-/plugin-transform-react-jsx-self-7.25.9.tgz#c0b6cae9c1b73967f7f9eb2fca9536ba2fad2858" + integrity sha512-y8quW6p0WHkEhmErnfe58r7x0A70uKphQm8Sp8cV7tjNQwK56sNVK0M73LK3WuYmsuyrftut4xAkjjgU0twaMg== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-react-jsx-source@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-react-jsx-source/-/plugin-transform-react-jsx-source-7.25.9.tgz#4c6b8daa520b5f155b5fb55547d7c9fa91417503" + integrity sha512-+iqjT8xmXhhYv4/uiYd8FNQsraMFZIfxVSqxxVSZP0WbbSAWvBXAul0m/zu+7Vv4O/3WtApy9pmaTMiumEZgfg== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-regenerator@^7.25.9": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-regenerator/-/plugin-transform-regenerator-7.27.0.tgz#822feebef43d6a59a81f696b2512df5b1682db31" + integrity sha512-LX/vCajUJQDqE7Aum/ELUMZAY19+cDpghxrnyt5I1tV6X5PyC86AOoWXWFYFeIvauyeSA6/ktn4tQVn/3ZifsA== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + regenerator-transform "^0.15.2" + +"@babel/plugin-transform-regexp-modifiers@^7.26.0": + version "7.26.0" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-regexp-modifiers/-/plugin-transform-regexp-modifiers-7.26.0.tgz#2f5837a5b5cd3842a919d8147e9903cc7455b850" + integrity sha512-vN6saax7lrA2yA/Pak3sCxuD6F5InBjn9IcrIKQPjpsLvuHYLVroTxjdlVRHjjBWxKOqIwpTXDkOssYT4BFdRw== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-reserved-words@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-reserved-words/-/plugin-transform-reserved-words-7.25.9.tgz#0398aed2f1f10ba3f78a93db219b27ef417fb9ce" + integrity sha512-7DL7DKYjn5Su++4RXu8puKZm2XBPHyjWLUidaPEkCUBbE7IPcsrkRHggAOOKydH1dASWdcUBxrkOGNxUv5P3Jg== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-shorthand-properties@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-shorthand-properties/-/plugin-transform-shorthand-properties-7.25.9.tgz#bb785e6091f99f826a95f9894fc16fde61c163f2" + integrity sha512-MUv6t0FhO5qHnS/W8XCbHmiRWOphNufpE1IVxhK5kuN3Td9FT1x4rx4K42s3RYdMXCXpfWkGSbCSd0Z64xA7Ng== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-spread@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-spread/-/plugin-transform-spread-7.25.9.tgz#24a35153931b4ba3d13cec4a7748c21ab5514ef9" + integrity sha512-oNknIB0TbURU5pqJFVbOOFspVlrpVwo2H1+HUIsVDvp5VauGGDP1ZEvO8Nn5xyMEs3dakajOxlmkNW7kNgSm6A== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + "@babel/helper-skip-transparent-expression-wrappers" "^7.25.9" + +"@babel/plugin-transform-sticky-regex@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-sticky-regex/-/plugin-transform-sticky-regex-7.25.9.tgz#c7f02b944e986a417817b20ba2c504dfc1453d32" + integrity sha512-WqBUSgeVwucYDP9U/xNRQam7xV8W5Zf+6Eo7T2SRVUFlhRiMNFdFz58u0KZmCVVqs2i7SHgpRnAhzRNmKfi2uA== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-template-literals@^7.26.8": + version "7.26.8" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-template-literals/-/plugin-transform-template-literals-7.26.8.tgz#966b15d153a991172a540a69ad5e1845ced990b5" + integrity sha512-OmGDL5/J0CJPJZTHZbi2XpO0tyT2Ia7fzpW5GURwdtp2X3fMmN8au/ej6peC/T33/+CRiIpA8Krse8hFGVmT5Q== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + +"@babel/plugin-transform-typeof-symbol@^7.26.7": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-typeof-symbol/-/plugin-transform-typeof-symbol-7.27.0.tgz#044a0890f3ca694207c7826d0c7a65e5ac008aae" + integrity sha512-+LLkxA9rKJpNoGsbLnAgOCdESl73vwYn+V6b+5wHbrE7OGKVDPHIQvbFSzqE6rwqaCw2RE+zdJrlLkcf8YOA0w== + dependencies: + "@babel/helper-plugin-utils" "^7.26.5" + +"@babel/plugin-transform-unicode-escapes@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-escapes/-/plugin-transform-unicode-escapes-7.25.9.tgz#a75ef3947ce15363fccaa38e2dd9bc70b2788b82" + integrity sha512-s5EDrE6bW97LtxOcGj1Khcx5AaXwiMmi4toFWRDP9/y0Woo6pXC+iyPu/KuhKtfSrNFd7jJB+/fkOtZy6aIC6Q== + dependencies: + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-unicode-property-regex@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-property-regex/-/plugin-transform-unicode-property-regex-7.25.9.tgz#a901e96f2c1d071b0d1bb5dc0d3c880ce8f53dd3" + integrity sha512-Jt2d8Ga+QwRluxRQ307Vlxa6dMrYEMZCgGxoPR8V52rxPyldHu3hdlHspxaqYmE7oID5+kB+UKUB/eWS+DkkWg== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-unicode-regex@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-regex/-/plugin-transform-unicode-regex-7.25.9.tgz#5eae747fe39eacf13a8bd006a4fb0b5d1fa5e9b1" + integrity sha512-yoxstj7Rg9dlNn9UQxzk4fcNivwv4nUYz7fYXBaKxvw/lnmPuOm/ikoELygbYq68Bls3D/D+NBPHiLwZdZZ4HA== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/plugin-transform-unicode-sets-regex@^7.25.9": + version "7.25.9" + resolved "https://registry.yarnpkg.com/@babel/plugin-transform-unicode-sets-regex/-/plugin-transform-unicode-sets-regex-7.25.9.tgz#65114c17b4ffc20fa5b163c63c70c0d25621fabe" + integrity sha512-8BYqO3GeVNHtx69fdPshN3fnzUNLrWdHhk/icSwigksJGczKSizZ+Z6SBCxTs723Fr5VSNorTIK7a+R2tISvwQ== + dependencies: + "@babel/helper-create-regexp-features-plugin" "^7.25.9" + "@babel/helper-plugin-utils" "^7.25.9" + +"@babel/preset-env@^7.26.9": + version "7.26.9" + resolved "https://registry.yarnpkg.com/@babel/preset-env/-/preset-env-7.26.9.tgz#2ec64e903d0efe743699f77a10bdf7955c2123c3" + integrity sha512-vX3qPGE8sEKEAZCWk05k3cpTAE3/nOYca++JA+Rd0z2NCNzabmYvEiSShKzm10zdquOIAVXsy2Ei/DTW34KlKQ== + dependencies: + "@babel/compat-data" "^7.26.8" + "@babel/helper-compilation-targets" "^7.26.5" + "@babel/helper-plugin-utils" "^7.26.5" + "@babel/helper-validator-option" "^7.25.9" + "@babel/plugin-bugfix-firefox-class-in-computed-class-key" "^7.25.9" + "@babel/plugin-bugfix-safari-class-field-initializer-scope" "^7.25.9" + "@babel/plugin-bugfix-safari-id-destructuring-collision-in-function-expression" "^7.25.9" + "@babel/plugin-bugfix-v8-spread-parameters-in-optional-chaining" "^7.25.9" + "@babel/plugin-bugfix-v8-static-class-fields-redefine-readonly" "^7.25.9" + "@babel/plugin-proposal-private-property-in-object" "7.21.0-placeholder-for-preset-env.2" + "@babel/plugin-syntax-import-assertions" "^7.26.0" + "@babel/plugin-syntax-import-attributes" "^7.26.0" + "@babel/plugin-syntax-unicode-sets-regex" "^7.18.6" + "@babel/plugin-transform-arrow-functions" "^7.25.9" + "@babel/plugin-transform-async-generator-functions" "^7.26.8" + "@babel/plugin-transform-async-to-generator" "^7.25.9" + "@babel/plugin-transform-block-scoped-functions" "^7.26.5" + "@babel/plugin-transform-block-scoping" "^7.25.9" + "@babel/plugin-transform-class-properties" "^7.25.9" + "@babel/plugin-transform-class-static-block" "^7.26.0" + "@babel/plugin-transform-classes" "^7.25.9" + "@babel/plugin-transform-computed-properties" "^7.25.9" + "@babel/plugin-transform-destructuring" "^7.25.9" + "@babel/plugin-transform-dotall-regex" "^7.25.9" + "@babel/plugin-transform-duplicate-keys" "^7.25.9" + "@babel/plugin-transform-duplicate-named-capturing-groups-regex" "^7.25.9" + "@babel/plugin-transform-dynamic-import" "^7.25.9" + "@babel/plugin-transform-exponentiation-operator" "^7.26.3" + "@babel/plugin-transform-export-namespace-from" "^7.25.9" + "@babel/plugin-transform-for-of" "^7.26.9" + "@babel/plugin-transform-function-name" "^7.25.9" + "@babel/plugin-transform-json-strings" "^7.25.9" + "@babel/plugin-transform-literals" "^7.25.9" + "@babel/plugin-transform-logical-assignment-operators" "^7.25.9" + "@babel/plugin-transform-member-expression-literals" "^7.25.9" + "@babel/plugin-transform-modules-amd" "^7.25.9" + "@babel/plugin-transform-modules-commonjs" "^7.26.3" + "@babel/plugin-transform-modules-systemjs" "^7.25.9" + "@babel/plugin-transform-modules-umd" "^7.25.9" + "@babel/plugin-transform-named-capturing-groups-regex" "^7.25.9" + "@babel/plugin-transform-new-target" "^7.25.9" + "@babel/plugin-transform-nullish-coalescing-operator" "^7.26.6" + "@babel/plugin-transform-numeric-separator" "^7.25.9" + "@babel/plugin-transform-object-rest-spread" "^7.25.9" + "@babel/plugin-transform-object-super" "^7.25.9" + "@babel/plugin-transform-optional-catch-binding" "^7.25.9" + "@babel/plugin-transform-optional-chaining" "^7.25.9" + "@babel/plugin-transform-parameters" "^7.25.9" + "@babel/plugin-transform-private-methods" "^7.25.9" + "@babel/plugin-transform-private-property-in-object" "^7.25.9" + "@babel/plugin-transform-property-literals" "^7.25.9" + "@babel/plugin-transform-regenerator" "^7.25.9" + "@babel/plugin-transform-regexp-modifiers" "^7.26.0" + "@babel/plugin-transform-reserved-words" "^7.25.9" + "@babel/plugin-transform-shorthand-properties" "^7.25.9" + "@babel/plugin-transform-spread" "^7.25.9" + "@babel/plugin-transform-sticky-regex" "^7.25.9" + "@babel/plugin-transform-template-literals" "^7.26.8" + "@babel/plugin-transform-typeof-symbol" "^7.26.7" + "@babel/plugin-transform-unicode-escapes" "^7.25.9" + "@babel/plugin-transform-unicode-property-regex" "^7.25.9" + "@babel/plugin-transform-unicode-regex" "^7.25.9" + "@babel/plugin-transform-unicode-sets-regex" "^7.25.9" + "@babel/preset-modules" "0.1.6-no-external-plugins" + babel-plugin-polyfill-corejs2 "^0.4.10" + babel-plugin-polyfill-corejs3 "^0.11.0" + babel-plugin-polyfill-regenerator "^0.6.1" + core-js-compat "^3.40.0" + semver "^6.3.1" + +"@babel/preset-modules@0.1.6-no-external-plugins": + version "0.1.6-no-external-plugins" + resolved "https://registry.yarnpkg.com/@babel/preset-modules/-/preset-modules-0.1.6-no-external-plugins.tgz#ccb88a2c49c817236861fee7826080573b8a923a" + integrity sha512-HrcgcIESLm9aIR842yhJ5RWan/gebQUJ6E/E5+rf0y9o6oj7w0Br+sWuL6kEQ/o/AdfvR1Je9jG18/gnpwjEyA== + dependencies: + "@babel/helper-plugin-utils" "^7.0.0" + "@babel/types" "^7.4.4" + esutils "^2.0.2" + +"@babel/runtime@^7.12.5", "@babel/runtime@^7.20.13", "@babel/runtime@^7.8.4", "@babel/runtime@^7.9.2": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/runtime/-/runtime-7.27.0.tgz#fbee7cf97c709518ecc1f590984481d5460d4762" + integrity sha512-VtPOkrdPHZsKc/clNqyi9WUA8TINkZ4cGk63UUE3u4pmB2k+ZMQRDuIOagv8UVd6j7k0T3+RRIb7beKTebNbcw== + dependencies: + regenerator-runtime "^0.14.0" + +"@babel/template@^7.25.9", "@babel/template@^7.26.9", "@babel/template@^7.27.0": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/template/-/template-7.27.0.tgz#b253e5406cc1df1c57dcd18f11760c2dbf40c0b4" + integrity sha512-2ncevenBqXI6qRMukPlXwHKHchC7RyMuu4xv5JBXRfOGVcTy1mXCD12qrp7Jsoxll1EV3+9sE4GugBVRjT2jFA== + dependencies: + "@babel/code-frame" "^7.26.2" + "@babel/parser" "^7.27.0" + "@babel/types" "^7.27.0" + +"@babel/traverse@^7.21.4", "@babel/traverse@^7.25.9", "@babel/traverse@^7.26.10", "@babel/traverse@^7.26.5", "@babel/traverse@^7.26.8", "@babel/traverse@^7.27.0": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/traverse/-/traverse-7.27.0.tgz#11d7e644779e166c0442f9a07274d02cd91d4a70" + integrity sha512-19lYZFzYVQkkHkl4Cy4WrAVcqBkgvV2YM2TU3xG6DIwO7O3ecbDPfW3yM3bjAGcqcQHi+CCtjMR3dIEHxsd6bA== + dependencies: + "@babel/code-frame" "^7.26.2" + "@babel/generator" "^7.27.0" + "@babel/parser" "^7.27.0" + "@babel/template" "^7.27.0" + "@babel/types" "^7.27.0" + debug "^4.3.1" + globals "^11.1.0" + +"@babel/types@^7.0.0", "@babel/types@^7.20.7", "@babel/types@^7.21.4", "@babel/types@^7.25.9", "@babel/types@^7.26.10", "@babel/types@^7.27.0", "@babel/types@^7.4.4": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@babel/types/-/types-7.27.0.tgz#ef9acb6b06c3173f6632d993ecb6d4ae470b4559" + integrity sha512-H45s8fVLYjbhFH62dIJ3WtmJ6RSPt/3DRO0ZcT2SUiYiQyz3BLVb9ADEnLl91m74aQPS3AzzeajZHYOalWe3bg== + dependencies: + "@babel/helper-string-parser" "^7.25.9" + "@babel/helper-validator-identifier" "^7.25.9" + +"@biomejs/biome@^1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/biome/-/biome-1.9.4.tgz#89766281cbc3a0aae865a7ff13d6aaffea2842bf" + integrity sha512-1rkd7G70+o9KkTn5KLmDYXihGoTaIGO9PIIN2ZB7UJxFrWw04CZHPYiMRjYsaDvVV7hP1dYNRLxSANLaBFGpog== + optionalDependencies: + "@biomejs/cli-darwin-arm64" "1.9.4" + "@biomejs/cli-darwin-x64" "1.9.4" + "@biomejs/cli-linux-arm64" "1.9.4" + "@biomejs/cli-linux-arm64-musl" "1.9.4" + "@biomejs/cli-linux-x64" "1.9.4" + "@biomejs/cli-linux-x64-musl" "1.9.4" + "@biomejs/cli-win32-arm64" "1.9.4" + "@biomejs/cli-win32-x64" "1.9.4" + +"@biomejs/cli-darwin-arm64@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-darwin-arm64/-/cli-darwin-arm64-1.9.4.tgz#dfa376d23a54a2d8f17133c92f23c1bf2e62509f" + integrity sha512-bFBsPWrNvkdKrNCYeAp+xo2HecOGPAy9WyNyB/jKnnedgzl4W4Hb9ZMzYNbf8dMCGmUdSavlYHiR01QaYR58cw== + +"@biomejs/cli-darwin-x64@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-darwin-x64/-/cli-darwin-x64-1.9.4.tgz#eafc2ce3849d385fc02238aad1ca4a73395a64d9" + integrity sha512-ngYBh/+bEedqkSevPVhLP4QfVPCpb+4BBe2p7Xs32dBgs7rh9nY2AIYUL6BgLw1JVXV8GlpKmb/hNiuIxfPfZg== + +"@biomejs/cli-linux-arm64-musl@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-linux-arm64-musl/-/cli-linux-arm64-musl-1.9.4.tgz#d780c3e01758fc90f3268357e3f19163d1f84fca" + integrity sha512-v665Ct9WCRjGa8+kTr0CzApU0+XXtRgwmzIf1SeKSGAv+2scAlW6JR5PMFo6FzqqZ64Po79cKODKf3/AAmECqA== + +"@biomejs/cli-linux-arm64@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-linux-arm64/-/cli-linux-arm64-1.9.4.tgz#8ed1dd0e89419a4b66a47f95aefb8c46ae6041c9" + integrity sha512-fJIW0+LYujdjUgJJuwesP4EjIBl/N/TcOX3IvIHJQNsAqvV2CHIogsmA94BPG6jZATS4Hi+xv4SkBBQSt1N4/g== + +"@biomejs/cli-linux-x64-musl@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-linux-x64-musl/-/cli-linux-x64-musl-1.9.4.tgz#f36982b966bd671a36671e1de4417963d7db15fb" + integrity sha512-gEhi/jSBhZ2m6wjV530Yy8+fNqG8PAinM3oV7CyO+6c3CEh16Eizm21uHVsyVBEB6RIM8JHIl6AGYCv6Q6Q9Tg== + +"@biomejs/cli-linux-x64@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-linux-x64/-/cli-linux-x64-1.9.4.tgz#a0a7f56680c76b8034ddc149dbf398bdd3a462e8" + integrity sha512-lRCJv/Vi3Vlwmbd6K+oQ0KhLHMAysN8lXoCI7XeHlxaajk06u7G+UsFSO01NAs5iYuWKmVZjmiOzJ0OJmGsMwg== + +"@biomejs/cli-win32-arm64@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-win32-arm64/-/cli-win32-arm64-1.9.4.tgz#e2ef4e0084e76b7e26f0fc887c5ef1265ea56200" + integrity sha512-tlbhLk+WXZmgwoIKwHIHEBZUwxml7bRJgk0X2sPyNR3S93cdRq6XulAZRQJ17FYGGzWne0fgrXBKpl7l4M87Hg== + +"@biomejs/cli-win32-x64@1.9.4": + version "1.9.4" + resolved "https://registry.yarnpkg.com/@biomejs/cli-win32-x64/-/cli-win32-x64-1.9.4.tgz#4c7afa90e3970213599b4095e62f87e5972b2340" + integrity sha512-8Y5wMhVIPaWe6jw2H+KlEm4wP/f7EW3810ZLmDlrEEy5KvBsb9ECEfu/kMWD484ijfQ8+nIi0giMgu9g1UAuuA== + +"@codemirror/autocomplete@6.18.6", "@codemirror/autocomplete@^6.0.0", "@codemirror/autocomplete@^6.18.6": + version "6.18.6" + resolved "https://registry.yarnpkg.com/@codemirror/autocomplete/-/autocomplete-6.18.6.tgz#de26e864a1ec8192a1b241eb86addbb612964ddb" + integrity sha512-PHHBXFomUs5DF+9tCOM/UoW6XQ4R44lLNNhRaW9PKPTU0D7lIjRg3ElxaJnTwsl/oHiR93WSXDBrekhoUGCPtg== + dependencies: + "@codemirror/language" "^6.0.0" + "@codemirror/state" "^6.0.0" + "@codemirror/view" "^6.17.0" + "@lezer/common" "^1.0.0" + +"@codemirror/commands@^6.0.0", "@codemirror/commands@^6.8.1": + version "6.8.1" + resolved "https://registry.yarnpkg.com/@codemirror/commands/-/commands-6.8.1.tgz#639f5559d2f33f2582a2429c58cb0c1b925c7a30" + integrity sha512-KlGVYufHMQzxbdQONiLyGQDUW0itrLZwq3CcY7xpv9ZLRHqzkBSoteocBHtMCoY7/Ci4xhzSrToIeLg7FxHuaw== + dependencies: + "@codemirror/language" "^6.0.0" + "@codemirror/state" "^6.4.0" + "@codemirror/view" "^6.27.0" + "@lezer/common" "^1.1.0" + +"@codemirror/language@^6.0.0", "@codemirror/language@^6.10.3", "@codemirror/language@^6.11.0": + version "6.11.0" + resolved "https://registry.yarnpkg.com/@codemirror/language/-/language-6.11.0.tgz#5ae90972601497f4575f30811519d720bf7232c9" + integrity sha512-A7+f++LodNNc1wGgoRDTt78cOwWm9KVezApgjOMp1W4hM0898nsqBXwF+sbePE7ZRcjN7Sa1Z5m2oN27XkmEjQ== + dependencies: + "@codemirror/state" "^6.0.0" + "@codemirror/view" "^6.23.0" + "@lezer/common" "^1.1.0" + "@lezer/highlight" "^1.0.0" + "@lezer/lr" "^1.0.0" + style-mod "^4.0.0" + +"@codemirror/lint@^6.0.0", "@codemirror/lint@^6.8.5": + version "6.8.5" + resolved "https://registry.yarnpkg.com/@codemirror/lint/-/lint-6.8.5.tgz#9edaa808e764e28e07665b015951934c8ec3a418" + integrity sha512-s3n3KisH7dx3vsoeGMxsbRAgKe4O1vbrnKBClm99PU0fWxmxsx5rR2PfqQgIt+2MMJBHbiJ5rfIdLYfB9NNvsA== + dependencies: + "@codemirror/state" "^6.0.0" + "@codemirror/view" "^6.35.0" + crelt "^1.0.5" + +"@codemirror/search@^6.0.0", "@codemirror/search@^6.5.10": + version "6.5.10" + resolved "https://registry.yarnpkg.com/@codemirror/search/-/search-6.5.10.tgz#7367bfc88094d078b91c752bc74140fb565b55ee" + integrity sha512-RMdPdmsrUf53pb2VwflKGHEe1XVM07hI7vV2ntgw1dmqhimpatSJKva4VA9h4TLUDOD4EIF02201oZurpnEFsg== + dependencies: + "@codemirror/state" "^6.0.0" + "@codemirror/view" "^6.0.0" + crelt "^1.0.5" + +"@codemirror/state@^6.0.0", "@codemirror/state@^6.2.1", "@codemirror/state@^6.4.0", "@codemirror/state@^6.5.0", "@codemirror/state@^6.5.2": + version "6.5.2" + resolved "https://registry.yarnpkg.com/@codemirror/state/-/state-6.5.2.tgz#8eca3a64212a83367dc85475b7d78d5c9b7076c6" + integrity sha512-FVqsPqtPWKVVL3dPSxy8wEF/ymIEuVzF1PK3VbUgrxXpJUSHQWWZz4JMToquRxnkw+36LTamCZG2iua2Ptq0fA== + dependencies: + "@marijn/find-cluster-break" "^1.0.0" + +"@codemirror/theme-one-dark@^6.1.2": + version "6.1.2" + resolved "https://registry.yarnpkg.com/@codemirror/theme-one-dark/-/theme-one-dark-6.1.2.tgz#fcef9f9cfc17a07836cb7da17c9f6d7231064df8" + integrity sha512-F+sH0X16j/qFLMAfbciKTxVOwkdAS336b7AXTKOZhy8BR3eH/RelsnLgLFINrpST63mmN2OuwUt0W2ndUgYwUA== + dependencies: + "@codemirror/language" "^6.0.0" + "@codemirror/state" "^6.0.0" + "@codemirror/view" "^6.0.0" + "@lezer/highlight" "^1.0.0" + +"@codemirror/view@^6.0.0", "@codemirror/view@^6.17.0", "@codemirror/view@^6.23.0", "@codemirror/view@^6.27.0", "@codemirror/view@^6.35.0": + version "6.36.6" + resolved "https://registry.yarnpkg.com/@codemirror/view/-/view-6.36.6.tgz#735a6431caed0c2c7d26c645066b02f10e802812" + integrity sha512-uxugGLet+Nzp0Jcit8Hn3LypM8ioMLKTsdf8FRoT3HWvZtb9GhaWMe0Cc15rz90Ljab4YFJiAulmIVB74OY0IQ== + dependencies: + "@codemirror/state" "^6.5.0" + style-mod "^4.1.0" + w3c-keyname "^2.2.4" + +"@cspotcode/source-map-support@^0.8.0": + version "0.8.1" + resolved "https://registry.yarnpkg.com/@cspotcode/source-map-support/-/source-map-support-0.8.1.tgz#00629c35a688e05a88b1cda684fb9d5e73f000a1" + integrity sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw== + dependencies: + "@jridgewell/trace-mapping" "0.3.9" + +"@csstools/color-helpers@^5.0.2": + version "5.0.2" + resolved "https://registry.yarnpkg.com/@csstools/color-helpers/-/color-helpers-5.0.2.tgz#82592c9a7c2b83c293d9161894e2a6471feb97b8" + integrity sha512-JqWH1vsgdGcw2RR6VliXXdA0/59LttzlU8UlRT/iUUsEeWfYq8I+K0yhihEUTTHLRm1EXvpsCx3083EU15ecsA== + +"@csstools/css-calc@^2.1.3": + version "2.1.3" + resolved "https://registry.yarnpkg.com/@csstools/css-calc/-/css-calc-2.1.3.tgz#6f68affcb569a86b91965e8622d644be35a08423" + integrity sha512-XBG3talrhid44BY1x3MHzUx/aTG8+x/Zi57M4aTKK9RFB4aLlF3TTSzfzn8nWVHWL3FgAXAxmupmDd6VWww+pw== + +"@csstools/css-color-parser@^3.0.9": + version "3.0.9" + resolved "https://registry.yarnpkg.com/@csstools/css-color-parser/-/css-color-parser-3.0.9.tgz#8d81b77d6f211495b5100ec4cad4c8828de49f6b" + integrity sha512-wILs5Zk7BU86UArYBJTPy/FMPPKVKHMj1ycCEyf3VUptol0JNRLFU/BZsJ4aiIHJEbSLiizzRrw8Pc1uAEDrXw== + dependencies: + "@csstools/color-helpers" "^5.0.2" + "@csstools/css-calc" "^2.1.3" + +"@csstools/css-parser-algorithms@^3.0.4": + version "3.0.4" + resolved "https://registry.yarnpkg.com/@csstools/css-parser-algorithms/-/css-parser-algorithms-3.0.4.tgz#74426e93bd1c4dcab3e441f5cc7ba4fb35d94356" + integrity sha512-Up7rBoV77rv29d3uKHUIVubz1BTcgyUK72IvCQAbfbMv584xHcGKCKbWh7i8hPrRJ7qU4Y8IO3IY9m+iTB7P3A== + +"@csstools/css-tokenizer@^3.0.3": + version "3.0.3" + resolved "https://registry.yarnpkg.com/@csstools/css-tokenizer/-/css-tokenizer-3.0.3.tgz#a5502c8539265fecbd873c1e395a890339f119c2" + integrity sha512-UJnjoFsmxfKUdNYdWgOB0mWUypuLvAfQPH1+pyvRJs6euowbFkFC6P13w1l8mJyi3vxYMxc9kld5jZEGRQs6bw== + +"@csstools/postcss-oklab-function@^4.0.9": + version "4.0.9" + resolved "https://registry.yarnpkg.com/@csstools/postcss-oklab-function/-/postcss-oklab-function-4.0.9.tgz#21d8846b3fe632d663b8cfe5d1fa2ccdd3f0709a" + integrity sha512-UHrnujimwtdDw8BYDcWJtBXuJ13uc/BjAddPdfMc/RsWxhg8gG8UbvTF0tnMtHrZ4i7lwy85fPEzK1AiykMyRA== + dependencies: + "@csstools/css-color-parser" "^3.0.9" + "@csstools/css-parser-algorithms" "^3.0.4" + "@csstools/css-tokenizer" "^3.0.3" + "@csstools/postcss-progressive-custom-properties" "^4.0.1" + "@csstools/utilities" "^2.0.0" + +"@csstools/postcss-progressive-custom-properties@^4.0.1": + version "4.0.1" + resolved "https://registry.yarnpkg.com/@csstools/postcss-progressive-custom-properties/-/postcss-progressive-custom-properties-4.0.1.tgz#487e619d2577358cb08ed7319dd203d4d5db5bb6" + integrity sha512-Ofz81HaY8mmbP8/Qr3PZlUzjsyV5WuxWmvtYn+jhYGvvjFazTmN9R2io5W5znY1tyk2CA9uM0IPWyY4ygDytCw== + dependencies: + postcss-value-parser "^4.2.0" + +"@csstools/utilities@^2.0.0": + version "2.0.0" + resolved "https://registry.yarnpkg.com/@csstools/utilities/-/utilities-2.0.0.tgz#f7ff0fee38c9ffb5646d47b6906e0bc8868bde60" + integrity sha512-5VdOr0Z71u+Yp3ozOx8T11N703wIFGVRgOWbOZMKgglPJsWA54MRIoMNVMa7shUToIhx5J8vX4sOZgD2XiihiQ== + +"@develar/schema-utils@~2.6.5": + version "2.6.5" + resolved "https://registry.yarnpkg.com/@develar/schema-utils/-/schema-utils-2.6.5.tgz#3ece22c5838402419a6e0425f85742b961d9b6c6" + integrity sha512-0cp4PsWQ/9avqTVMCtZ+GirikIA36ikvjtHweU4/j8yLtgObI0+JUPhYFScgwlteveGB1rt3Cm8UhN04XayDig== + dependencies: + ajv "^6.12.0" + ajv-keywords "^3.4.1" + +"@electron-forge/cli@^7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/cli/-/cli-7.8.0.tgz#871e6f5e491cf7c41b11847f3176369176c2e073" + integrity sha512-XZ+Hg7pxeE9pgrahqcpMlND+VH0l0UTZLyO5wkI+YfanNyBQksB2mw24XeEtCA6x8F2IaEYdIGgijmPF6qpjzA== + dependencies: + "@electron-forge/core" "7.8.0" + "@electron-forge/core-utils" "7.8.0" + "@electron-forge/shared-types" "7.8.0" + "@electron/get" "^3.0.0" + chalk "^4.0.0" + commander "^11.1.0" + debug "^4.3.1" + fs-extra "^10.0.0" + listr2 "^7.0.2" + log-symbols "^4.0.0" + semver "^7.2.1" + +"@electron-forge/core-utils@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/core-utils/-/core-utils-7.8.0.tgz#e38682c892ac260d9f60ac11c3f9f7c150e56fde" + integrity sha512-ZioRzqkXVOGuwkfvXN/FPZxcssJ9AkOZx6RvxomQn90F77G2KfEbw4ZwAxVTQ+jWNUzydTic5qavWle++Y5IeA== + dependencies: + "@electron-forge/shared-types" "7.8.0" + "@electron/rebuild" "^3.7.0" + "@malept/cross-spawn-promise" "^2.0.0" + chalk "^4.0.0" + debug "^4.3.1" + find-up "^5.0.0" + fs-extra "^10.0.0" + log-symbols "^4.0.0" + semver "^7.2.1" + +"@electron-forge/core@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/core/-/core-7.8.0.tgz#45991916312c6ee76cb627653c565182a3dd189b" + integrity sha512-7byf660ECZND+irOhGxvpmRXjk1bMrsTWh5J2AZMEvaXI8tub9OrZY9VSbi5fcDt0lpHPKmgVk7NRf/ZjJ+beQ== + dependencies: + "@electron-forge/core-utils" "7.8.0" + "@electron-forge/maker-base" "7.8.0" + "@electron-forge/plugin-base" "7.8.0" + "@electron-forge/publisher-base" "7.8.0" + "@electron-forge/shared-types" "7.8.0" + "@electron-forge/template-base" "7.8.0" + "@electron-forge/template-vite" "7.8.0" + "@electron-forge/template-vite-typescript" "7.8.0" + "@electron-forge/template-webpack" "7.8.0" + "@electron-forge/template-webpack-typescript" "7.8.0" + "@electron-forge/tracer" "7.8.0" + "@electron/get" "^3.0.0" + "@electron/packager" "^18.3.5" + "@electron/rebuild" "^3.7.0" + "@malept/cross-spawn-promise" "^2.0.0" + chalk "^4.0.0" + debug "^4.3.1" + fast-glob "^3.2.7" + filenamify "^4.1.0" + find-up "^5.0.0" + fs-extra "^10.0.0" + global-dirs "^3.0.0" + got "^11.8.5" + interpret "^3.1.1" + listr2 "^7.0.2" + lodash "^4.17.20" + log-symbols "^4.0.0" + node-fetch "^2.6.7" + rechoir "^0.8.0" + semver "^7.2.1" + source-map-support "^0.5.13" + sudo-prompt "^9.1.1" + username "^5.1.0" + +"@electron-forge/maker-base@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/maker-base/-/maker-base-7.8.0.tgz#4790a4cd477bd5952aba783e9ff433d15a56217c" + integrity sha512-yGRvz70w+NnKO7PhzNFRgYM+x6kxYFgpbChJIQBs3WChd9bGjL+MZLrwYqmxOFLpWNwRAJ6PEi4E/8U5GgV6AQ== + dependencies: + "@electron-forge/shared-types" "7.8.0" + fs-extra "^10.0.0" + which "^2.0.2" + +"@electron-forge/plugin-base@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/plugin-base/-/plugin-base-7.8.0.tgz#bb617781c28b99d2b65d6f2e66679b245d388e78" + integrity sha512-rDeeChRWIp5rQVo3Uc1q0ncUvA+kWWURW7tMuQjPvy2qVSgX+jIf5krk+T1Dp06+D4YZzEIrkibRaamAaIcR1w== + dependencies: + "@electron-forge/shared-types" "7.8.0" + +"@electron-forge/plugin-fuses@^7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/plugin-fuses/-/plugin-fuses-7.8.0.tgz#8c526146a8c5b14584b1844e81af13c26514498d" + integrity sha512-ZxFtol3aHNY+oYrZWa7EDBLl4uk/+NlOCJmqC7C32R/3S/Kn2ebVRxpLwrFM12KtHeD+Z3gmZNBhwOe0TECgOA== + dependencies: + "@electron-forge/plugin-base" "7.8.0" + "@electron-forge/shared-types" "7.8.0" + +"@electron-forge/plugin-vite@^7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/plugin-vite/-/plugin-vite-7.8.0.tgz#da57830c604a183ef4ae9b3ca01ac458e501bab9" + integrity sha512-qopX6DU51mUD4bnGYklo5nr0U+hmwATKQavUpncg1i+R0pyYSUrYSVYu2HVFNj8F9QXDyXhf1I2AwwZe9STYug== + dependencies: + "@electron-forge/core-utils" "7.8.0" + "@electron-forge/plugin-base" "7.8.0" + "@electron-forge/shared-types" "7.8.0" + "@electron-forge/web-multi-logger" "7.8.0" + chalk "^4.0.0" + debug "^4.3.1" + fs-extra "^10.0.0" + listr2 "^7.0.2" + +"@electron-forge/publisher-base@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/publisher-base/-/publisher-base-7.8.0.tgz#5b86f7dec28aecf19448a54b7f65371e18bcc9c7" + integrity sha512-wrZyptJ0Uqvlh2wYzDZfIu2HgCQ+kdGiBlcucmLY4W+GUqf043O8cbYso3D9NXQxOow55QC/1saCQkgLphprPA== + dependencies: + "@electron-forge/shared-types" "7.8.0" + +"@electron-forge/shared-types@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/shared-types/-/shared-types-7.8.0.tgz#dbc9b20ecd547d0758d7abc90587983c7f2fd3e8" + integrity sha512-Ul+7HPvAZiAirqpZm0vc9YvlkAE+2bcrI10p3t50mEtuxn5VO/mB72NXiEKfWzHm8F31JySIe9bUV6s1MHQcCw== + dependencies: + "@electron-forge/tracer" "7.8.0" + "@electron/packager" "^18.3.5" + "@electron/rebuild" "^3.7.0" + listr2 "^7.0.2" + +"@electron-forge/template-base@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/template-base/-/template-base-7.8.0.tgz#9721d6c42120db094c83436404a458a2cb6e5412" + integrity sha512-hc8NwoDqEEmZFH/p0p3MK/7xygMmI+cm8Gavoj2Mr2xS7VUUu4r3b5PwIGKvkLfPG34uwsiVwtid2t1rWGF4UA== + dependencies: + "@electron-forge/core-utils" "7.8.0" + "@electron-forge/shared-types" "7.8.0" + "@malept/cross-spawn-promise" "^2.0.0" + debug "^4.3.1" + fs-extra "^10.0.0" + username "^5.1.0" + +"@electron-forge/template-vite-typescript@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/template-vite-typescript/-/template-vite-typescript-7.8.0.tgz#106558f6e3af263629e730fd2a60e9eaffefb7cf" + integrity sha512-kW3CaVxKHUYuVfY+rT3iepeZ69frBRGh3YZOngLY2buCvGIqNEx+VCgrFBRDDbOKGmwQtwO1E9wp2rtC8q6Ztg== + dependencies: + "@electron-forge/shared-types" "7.8.0" + "@electron-forge/template-base" "7.8.0" + fs-extra "^10.0.0" + +"@electron-forge/template-vite@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/template-vite/-/template-vite-7.8.0.tgz#8906aa82cb4c0efc3501da0141a9dd382715dd86" + integrity sha512-bf/jd8WzD0gU7Jet+WSi0Lm0SQmseb08WY27ZfJYEs2EVNMiwDfPicgQnOaqP++2yTrXhj1OY/rolZCP9CUyVw== + dependencies: + "@electron-forge/shared-types" "7.8.0" + "@electron-forge/template-base" "7.8.0" + fs-extra "^10.0.0" + +"@electron-forge/template-webpack-typescript@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/template-webpack-typescript/-/template-webpack-typescript-7.8.0.tgz#26731829d8fa18196ae65bf3db46fb6659b1eac7" + integrity sha512-Pl8l+gv3HzqCfFIMLxlEsoAkNd0VEWeZZ675SYyqs0/kBQUifn0bKNhVE4gUZwKGgQCcG1Gvb23KdVGD3H3XmA== + dependencies: + "@electron-forge/shared-types" "7.8.0" + "@electron-forge/template-base" "7.8.0" + fs-extra "^10.0.0" + +"@electron-forge/template-webpack@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/template-webpack/-/template-webpack-7.8.0.tgz#b85b2827193142c6c836ceec8a1776471fdd28c4" + integrity sha512-AdLGC6NVgrd7Q0SaaeiwJKmSBjN6C2EHxZgLMy1yxNSpazU9m3DtYQilDjXqmCWfxkeNzdke0NaeDvLgdJSw5A== + dependencies: + "@electron-forge/shared-types" "7.8.0" + "@electron-forge/template-base" "7.8.0" + fs-extra "^10.0.0" + +"@electron-forge/tracer@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/tracer/-/tracer-7.8.0.tgz#19a9a8164efd2b7a9fa144d424d5e714f115cf77" + integrity sha512-t4fIATZEX6/7PJNfyh6tLzKEsNMpO01Nz/rgHWBxeRvjCw5UNul9OOxoM7b43vfFAO9Jv++34oI3VJ09LeVQ2Q== + dependencies: + chrome-trace-event "^1.0.3" + +"@electron-forge/web-multi-logger@7.8.0": + version "7.8.0" + resolved "https://registry.yarnpkg.com/@electron-forge/web-multi-logger/-/web-multi-logger-7.8.0.tgz#634bcc98e841d67f252a8e9a52821bddf0821dca" + integrity sha512-2nUP7O9auXDsoa185AsZPlIbpargj1lNFweNH1Lch1MCwLlJOI9ZJHiCTAB4qviS4usRs00WeebWg/uN/zOWvA== + dependencies: + express "^4.17.1" + express-ws "^5.0.2" + xterm "^4.9.0" + xterm-addon-fit "^0.5.0" + xterm-addon-search "^0.8.0" + +"@electron/asar@3.2.18": + version "3.2.18" + resolved "https://registry.yarnpkg.com/@electron/asar/-/asar-3.2.18.tgz#fa607f829209bab8b9e0ce6658d3fe81b2cba517" + integrity sha512-2XyvMe3N3Nrs8cV39IKELRHTYUWFKrmqqSY1U+GMlc0jvqjIVnoxhNd2H4JolWQncbJi1DCvb5TNxZuI2fEjWg== + dependencies: + commander "^5.0.0" + glob "^7.1.6" + minimatch "^3.0.4" + +"@electron/asar@^3.2.13", "@electron/asar@^3.2.7", "@electron/asar@^3.3.1": + version "3.4.1" + resolved "https://registry.yarnpkg.com/@electron/asar/-/asar-3.4.1.tgz#4e9196a4b54fba18c56cd8d5cac67c5bdc588065" + integrity sha512-i4/rNPRS84t0vSRa2HorerGRXWyF4vThfHesw0dmcWHp+cspK743UanA0suA5Q5y8kzY2y6YKrvbIUn69BCAiA== + dependencies: + commander "^5.0.0" + glob "^7.1.6" + minimatch "^3.0.4" + +"@electron/fuses@^1.8.0": + version "1.8.0" + resolved "https://registry.yarnpkg.com/@electron/fuses/-/fuses-1.8.0.tgz#ad34d3cc4703b1258b83f6989917052cfc1490a0" + integrity sha512-zx0EIq78WlY/lBb1uXlziZmDZI4ubcCXIMJ4uGjXzZW0nS19TjSPeXPAjzzTmKQlJUZm0SbmZhPKP7tuQ1SsEw== + dependencies: + chalk "^4.1.1" + fs-extra "^9.0.1" + minimist "^1.2.5" + +"@electron/get@^2.0.0": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@electron/get/-/get-2.0.3.tgz#fba552683d387aebd9f3fcadbcafc8e12ee4f960" + integrity sha512-Qkzpg2s9GnVV2I2BjRksUi43U5e6+zaQMcjoJy0C+C5oxaKl+fmckGDQFtRpZpZV0NQekuZZ+tGz7EA9TVnQtQ== + dependencies: + debug "^4.1.1" + env-paths "^2.2.0" + fs-extra "^8.1.0" + got "^11.8.5" + progress "^2.0.3" + semver "^6.2.0" + sumchecker "^3.0.1" + optionalDependencies: + global-agent "^3.0.0" + +"@electron/get@^3.0.0": + version "3.1.0" + resolved "https://registry.yarnpkg.com/@electron/get/-/get-3.1.0.tgz#22c5a0bd917ab201badeb77bc4ad18cba54cb4ec" + integrity sha512-F+nKc0xW+kVbBRhFzaMgPy3KwmuNTYX1fx6+FxxoSnNgwYX6LD7AKBTWkU0MQ6IBoe7dz069CNkR673sPAgkCQ== + dependencies: + debug "^4.1.1" + env-paths "^2.2.0" + fs-extra "^8.1.0" + got "^11.8.5" + progress "^2.0.3" + semver "^6.2.0" + sumchecker "^3.0.1" + optionalDependencies: + global-agent "^3.0.0" + +"@electron/node-gyp@git+https://github.com/electron/node-gyp.git#06b29aafb7708acef8b3669835c8a7857ebc92d2": + version "10.2.0-electron.1" + resolved "git+https://github.com/electron/node-gyp.git#06b29aafb7708acef8b3669835c8a7857ebc92d2" + dependencies: + env-paths "^2.2.0" + exponential-backoff "^3.1.1" + glob "^8.1.0" + graceful-fs "^4.2.6" + make-fetch-happen "^10.2.1" + nopt "^6.0.0" + proc-log "^2.0.1" + semver "^7.3.5" + tar "^6.2.1" + which "^2.0.2" + +"@electron/notarize@2.5.0", "@electron/notarize@^2.1.0", "@electron/notarize@^2.5.0": + version "2.5.0" + resolved "https://registry.yarnpkg.com/@electron/notarize/-/notarize-2.5.0.tgz#d4d25356adfa29df4a76bd64a8bd347237cd251e" + integrity sha512-jNT8nwH1f9X5GEITXaQ8IF/KdskvIkOFfB2CvwumsveVidzpSc+mvhhTMdAGSYF3O+Nq49lJ7y+ssODRXu06+A== + dependencies: + debug "^4.1.1" + fs-extra "^9.0.1" + promise-retry "^2.0.1" + +"@electron/osx-sign@1.3.1": + version "1.3.1" + resolved "https://registry.yarnpkg.com/@electron/osx-sign/-/osx-sign-1.3.1.tgz#faf7eeca7ca004a6be541dc4cf7a1bd59ec59b1c" + integrity sha512-BAfviURMHpmb1Yb50YbCxnOY0wfwaLXH5KJ4+80zS0gUkzDX3ec23naTlEqKsN+PwYn+a1cCzM7BJ4Wcd3sGzw== + dependencies: + compare-version "^0.1.2" + debug "^4.3.4" + fs-extra "^10.0.0" + isbinaryfile "^4.0.8" + minimist "^1.2.6" + plist "^3.0.5" + +"@electron/osx-sign@^1.0.5": + version "1.3.3" + resolved "https://registry.yarnpkg.com/@electron/osx-sign/-/osx-sign-1.3.3.tgz#af751510488318d9f7663694af85819690d75583" + integrity sha512-KZ8mhXvWv2rIEgMbWZ4y33bDHyUKMXnx4M0sTyPNK/vcB81ImdeY9Ggdqy0SWbMDgmbqyQ+phgejh6V3R2QuSg== + dependencies: + compare-version "^0.1.2" + debug "^4.3.4" + fs-extra "^10.0.0" + isbinaryfile "^4.0.8" + minimist "^1.2.6" + plist "^3.0.5" + +"@electron/packager@^18.3.5": + version "18.3.6" + resolved "https://registry.yarnpkg.com/@electron/packager/-/packager-18.3.6.tgz#3596399334866737bc33ab663e90376d7aad89e4" + integrity sha512-1eXHB5t+SQKvUiDpWGpvr90ZSSbXj+isrh3YbjCTjKT4bE4SQrKSBfukEAaBvp67+GXHFtCHjQgN9qSTFIge+Q== + dependencies: + "@electron/asar" "^3.2.13" + "@electron/get" "^3.0.0" + "@electron/notarize" "^2.1.0" + "@electron/osx-sign" "^1.0.5" + "@electron/universal" "^2.0.1" + "@electron/windows-sign" "^1.0.0" + debug "^4.0.1" + extract-zip "^2.0.0" + filenamify "^4.1.0" + fs-extra "^11.1.0" + galactus "^1.0.0" + get-package-info "^1.0.0" + junk "^3.1.0" + parse-author "^2.0.0" + plist "^3.0.0" + resedit "^2.0.0" + resolve "^1.1.6" + semver "^7.1.3" + yargs-parser "^21.1.1" + +"@electron/rebuild@3.7.0": + version "3.7.0" + resolved "https://registry.yarnpkg.com/@electron/rebuild/-/rebuild-3.7.0.tgz#82e20c467ddedbb295d7f641592c52e68c141e9f" + integrity sha512-VW++CNSlZwMYP7MyXEbrKjpzEwhB5kDNbzGtiPEjwYysqyTCF+YbNJ210Dj3AjWsGSV4iEEwNkmJN9yGZmVvmw== + dependencies: + "@electron/node-gyp" "https://github.com/electron/node-gyp#06b29aafb7708acef8b3669835c8a7857ebc92d2" + "@malept/cross-spawn-promise" "^2.0.0" + chalk "^4.0.0" + debug "^4.1.1" + detect-libc "^2.0.1" + fs-extra "^10.0.0" + got "^11.7.0" + node-abi "^3.45.0" + node-api-version "^0.2.0" + ora "^5.1.0" + read-binary-file-arch "^1.0.6" + semver "^7.3.5" + tar "^6.0.5" + yargs "^17.0.1" + +"@electron/rebuild@^3.7.0": + version "3.7.2" + resolved "https://registry.yarnpkg.com/@electron/rebuild/-/rebuild-3.7.2.tgz#8d808b29159c50086d27a5dec72b40bf16b4b582" + integrity sha512-19/KbIR/DAxbsCkiaGMXIdPnMCJLkcf8AvGnduJtWBs/CBwiAjY1apCqOLVxrXg+rtXFCngbXhBanWjxLUt1Mg== + dependencies: + "@electron/node-gyp" "https://github.com/electron/node-gyp#06b29aafb7708acef8b3669835c8a7857ebc92d2" + "@malept/cross-spawn-promise" "^2.0.0" + chalk "^4.0.0" + debug "^4.1.1" + detect-libc "^2.0.1" + fs-extra "^10.0.0" + got "^11.7.0" + node-abi "^3.45.0" + node-api-version "^0.2.0" + ora "^5.1.0" + read-binary-file-arch "^1.0.6" + semver "^7.3.5" + tar "^6.0.5" + yargs "^17.0.1" + +"@electron/universal@2.0.1": + version "2.0.1" + resolved "https://registry.yarnpkg.com/@electron/universal/-/universal-2.0.1.tgz#7b070ab355e02957388f3dbd68e2c3cd08c448ae" + integrity sha512-fKpv9kg4SPmt+hY7SVBnIYULE9QJl8L3sCfcBsnqbJwwBwAeTLokJ9TRt9y7bK0JAzIW2y78TVVjvnQEms/yyA== + dependencies: + "@electron/asar" "^3.2.7" + "@malept/cross-spawn-promise" "^2.0.0" + debug "^4.3.1" + dir-compare "^4.2.0" + fs-extra "^11.1.1" + minimatch "^9.0.3" + plist "^3.1.0" + +"@electron/universal@^2.0.1": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@electron/universal/-/universal-2.0.2.tgz#834466e6b117fdb94048bcbde5c91d2d4571e718" + integrity sha512-mqY1szx5/d5YLvfCDWWoJdkSIjIz+NdWN4pN0r78lYiE7De+slLpuF3lVxIT+hlJnwk5sH2wFRMl6/oUgUVO3A== + dependencies: + "@electron/asar" "^3.3.1" + "@malept/cross-spawn-promise" "^2.0.0" + debug "^4.3.1" + dir-compare "^4.2.0" + fs-extra "^11.1.1" + minimatch "^9.0.3" + plist "^3.1.0" + +"@electron/windows-sign@^1.0.0": + version "1.2.1" + resolved "https://registry.yarnpkg.com/@electron/windows-sign/-/windows-sign-1.2.1.tgz#8e4674cfdeb7cd58e14b58e624500d9f03f2873d" + integrity sha512-YfASnrhJ+ve6Q43ZiDwmpBgYgi2u0bYjeAVi2tDfN7YWAKO8X9EEOuPGtqbJpPLM6TfAHimghICjWe2eaJ8BAg== + dependencies: + cross-dirname "^0.1.0" + debug "^4.3.4" + fs-extra "^11.1.1" + minimist "^1.2.8" + postject "^1.0.0-alpha.6" + +"@esbuild/aix-ppc64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.21.5.tgz#c7184a326533fcdf1b8ee0733e21c713b975575f" + integrity sha512-1SDgH6ZSPTlggy1yI6+Dbkiz8xzpHJEVAlF/AM1tHPLsf5STom9rwtjE4hKAF20FfXXNTFqEYXyJNWh1GiZedQ== + +"@esbuild/aix-ppc64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/aix-ppc64/-/aix-ppc64-0.25.3.tgz#014180d9a149cffd95aaeead37179433f5ea5437" + integrity sha512-W8bFfPA8DowP8l//sxjJLSLkD8iEjMc7cBVyP+u4cEv9sM7mdUCkgsj+t0n/BWPFtv7WWCN5Yzj0N6FJNUUqBQ== + +"@esbuild/android-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.21.5.tgz#09d9b4357780da9ea3a7dfb833a1f1ff439b4052" + integrity sha512-c0uX9VAUBQ7dTDCjq+wdyGLowMdtR/GoC2U5IYk/7D1H1JYC0qseD7+11iMP2mRLN9RcCMRcjC4YMclCzGwS/A== + +"@esbuild/android-arm64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm64/-/android-arm64-0.25.3.tgz#649e47e04ddb24a27dc05c395724bc5f4c55cbfe" + integrity sha512-XelR6MzjlZuBM4f5z2IQHK6LkK34Cvv6Rj2EntER3lwCBFdg6h2lKbtRjpTTsdEjD/WSe1q8UyPBXP1x3i/wYQ== + +"@esbuild/android-arm@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.21.5.tgz#9b04384fb771926dfa6d7ad04324ecb2ab9b2e28" + integrity sha512-vCPvzSjpPHEi1siZdlvAlsPxXl7WbOVUBBAowWug4rJHb68Ox8KualB+1ocNvT5fjv6wpkX6o/iEpbDrf68zcg== + +"@esbuild/android-arm@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/android-arm/-/android-arm-0.25.3.tgz#8a0f719c8dc28a4a6567ef7328c36ea85f568ff4" + integrity sha512-PuwVXbnP87Tcff5I9ngV0lmiSu40xw1At6i3GsU77U7cjDDB4s0X2cyFuBiDa1SBk9DnvWwnGvVaGBqoFWPb7A== + +"@esbuild/android-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.21.5.tgz#29918ec2db754cedcb6c1b04de8cd6547af6461e" + integrity sha512-D7aPRUUNHRBwHxzxRvp856rjUHRFW1SdQATKXH2hqA0kAZb1hKmi02OpYRacl0TxIGz/ZmXWlbZgjwWYaCakTA== + +"@esbuild/android-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/android-x64/-/android-x64-0.25.3.tgz#e2ab182d1fd06da9bef0784a13c28a7602d78009" + integrity sha512-ogtTpYHT/g1GWS/zKM0cc/tIebFjm1F9Aw1boQ2Y0eUQ+J89d0jFY//s9ei9jVIlkYi8AfOjiixcLJSGNSOAdQ== + +"@esbuild/darwin-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.21.5.tgz#e495b539660e51690f3928af50a76fb0a6ccff2a" + integrity sha512-DwqXqZyuk5AiWWf3UfLiRDJ5EDd49zg6O9wclZ7kUMv2WRFr4HKjXp/5t8JZ11QbQfUS6/cRCKGwYhtNAY88kQ== + +"@esbuild/darwin-arm64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-arm64/-/darwin-arm64-0.25.3.tgz#c7f3166fcece4d158a73dcfe71b2672ca0b1668b" + integrity sha512-eESK5yfPNTqpAmDfFWNsOhmIOaQA59tAcF/EfYvo5/QWQCzXn5iUSOnqt3ra3UdzBv073ykTtmeLJZGt3HhA+w== + +"@esbuild/darwin-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.21.5.tgz#c13838fa57372839abdddc91d71542ceea2e1e22" + integrity sha512-se/JjF8NlmKVG4kNIuyWMV/22ZaerB+qaSi5MdrXtd6R08kvs2qCN4C09miupktDitvh8jRFflwGFBQcxZRjbw== + +"@esbuild/darwin-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/darwin-x64/-/darwin-x64-0.25.3.tgz#d8c5342ec1a4bf4b1915643dfe031ba4b173a87a" + integrity sha512-Kd8glo7sIZtwOLcPbW0yLpKmBNWMANZhrC1r6K++uDR2zyzb6AeOYtI6udbtabmQpFaxJ8uduXMAo1gs5ozz8A== + +"@esbuild/freebsd-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.21.5.tgz#646b989aa20bf89fd071dd5dbfad69a3542e550e" + integrity sha512-5JcRxxRDUJLX8JXp/wcBCy3pENnCgBR9bN6JsY4OmhfUtIHe3ZW0mawA7+RDAcMLrMIZaf03NlQiX9DGyB8h4g== + +"@esbuild/freebsd-arm64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-arm64/-/freebsd-arm64-0.25.3.tgz#9f7d789e2eb7747d4868817417cc968ffa84f35b" + integrity sha512-EJiyS70BYybOBpJth3M0KLOus0n+RRMKTYzhYhFeMwp7e/RaajXvP+BWlmEXNk6uk+KAu46j/kaQzr6au+JcIw== + +"@esbuild/freebsd-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.21.5.tgz#aa615cfc80af954d3458906e38ca22c18cf5c261" + integrity sha512-J95kNBj1zkbMXtHVH29bBriQygMXqoVQOQYA+ISs0/2l3T9/kj42ow2mpqerRBxDJnmkUDCaQT/dfNXWX/ZZCQ== + +"@esbuild/freebsd-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/freebsd-x64/-/freebsd-x64-0.25.3.tgz#8ad35c51d084184a8e9e76bb4356e95350a64709" + integrity sha512-Q+wSjaLpGxYf7zC0kL0nDlhsfuFkoN+EXrx2KSB33RhinWzejOd6AvgmP5JbkgXKmjhmpfgKZq24pneodYqE8Q== + +"@esbuild/linux-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.21.5.tgz#70ac6fa14f5cb7e1f7f887bcffb680ad09922b5b" + integrity sha512-ibKvmyYzKsBeX8d8I7MH/TMfWDXBF3db4qM6sy+7re0YXya+K1cem3on9XgdT2EQGMu4hQyZhan7TeQ8XkGp4Q== + +"@esbuild/linux-arm64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm64/-/linux-arm64-0.25.3.tgz#3af0da3d9186092a9edd4e28fa342f57d9e3cd30" + integrity sha512-xCUgnNYhRD5bb1C1nqrDV1PfkwgbswTTBRbAd8aH5PhYzikdf/ddtsYyMXFfGSsb/6t6QaPSzxtbfAZr9uox4A== + +"@esbuild/linux-arm@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.21.5.tgz#fc6fd11a8aca56c1f6f3894f2bea0479f8f626b9" + integrity sha512-bPb5AHZtbeNGjCKVZ9UGqGwo8EUu4cLq68E95A53KlxAPRmUyYv2D6F0uUI65XisGOL1hBP5mTronbgo+0bFcA== + +"@esbuild/linux-arm@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-arm/-/linux-arm-0.25.3.tgz#e91cafa95e4474b3ae3d54da12e006b782e57225" + integrity sha512-dUOVmAUzuHy2ZOKIHIKHCm58HKzFqd+puLaS424h6I85GlSDRZIA5ycBixb3mFgM0Jdh+ZOSB6KptX30DD8YOQ== + +"@esbuild/linux-ia32@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.21.5.tgz#3271f53b3f93e3d093d518d1649d6d68d346ede2" + integrity sha512-YvjXDqLRqPDl2dvRODYmmhz4rPeVKYvppfGYKSNGdyZkA01046pLWyRKKI3ax8fbJoK5QbxblURkwK/MWY18Tg== + +"@esbuild/linux-ia32@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ia32/-/linux-ia32-0.25.3.tgz#81025732d85b68ee510161b94acdf7e3007ea177" + integrity sha512-yplPOpczHOO4jTYKmuYuANI3WhvIPSVANGcNUeMlxH4twz/TeXuzEP41tGKNGWJjuMhotpGabeFYGAOU2ummBw== + +"@esbuild/linux-loong64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.21.5.tgz#ed62e04238c57026aea831c5a130b73c0f9f26df" + integrity sha512-uHf1BmMG8qEvzdrzAqg2SIG/02+4/DHB6a9Kbya0XDvwDEKCoC8ZRWI5JJvNdUjtciBGFQ5PuBlpEOXQj+JQSg== + +"@esbuild/linux-loong64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-loong64/-/linux-loong64-0.25.3.tgz#3c744e4c8d5e1148cbe60a71a11b58ed8ee5deb8" + integrity sha512-P4BLP5/fjyihmXCELRGrLd793q/lBtKMQl8ARGpDxgzgIKJDRJ/u4r1A/HgpBpKpKZelGct2PGI4T+axcedf6g== + +"@esbuild/linux-mips64el@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.21.5.tgz#e79b8eb48bf3b106fadec1ac8240fb97b4e64cbe" + integrity sha512-IajOmO+KJK23bj52dFSNCMsz1QP1DqM6cwLUv3W1QwyxkyIWecfafnI555fvSGqEKwjMXVLokcV5ygHW5b3Jbg== + +"@esbuild/linux-mips64el@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-mips64el/-/linux-mips64el-0.25.3.tgz#1dfe2a5d63702db9034cc6b10b3087cc0424ec26" + integrity sha512-eRAOV2ODpu6P5divMEMa26RRqb2yUoYsuQQOuFUexUoQndm4MdpXXDBbUoKIc0iPa4aCO7gIhtnYomkn2x+bag== + +"@esbuild/linux-ppc64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.21.5.tgz#5f2203860a143b9919d383ef7573521fb154c3e4" + integrity sha512-1hHV/Z4OEfMwpLO8rp7CvlhBDnjsC3CttJXIhBi+5Aj5r+MBvy4egg7wCbe//hSsT+RvDAG7s81tAvpL2XAE4w== + +"@esbuild/linux-ppc64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-ppc64/-/linux-ppc64-0.25.3.tgz#2e85d9764c04a1ebb346dc0813ea05952c9a5c56" + integrity sha512-ZC4jV2p7VbzTlnl8nZKLcBkfzIf4Yad1SJM4ZMKYnJqZFD4rTI+pBG65u8ev4jk3/MPwY9DvGn50wi3uhdaghg== + +"@esbuild/linux-riscv64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.21.5.tgz#07bcafd99322d5af62f618cb9e6a9b7f4bb825dc" + integrity sha512-2HdXDMd9GMgTGrPWnJzP2ALSokE/0O5HhTUvWIbD3YdjME8JwvSCnNGBnTThKGEB91OZhzrJ4qIIxk/SBmyDDA== + +"@esbuild/linux-riscv64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-riscv64/-/linux-riscv64-0.25.3.tgz#a9ea3334556b09f85ccbfead58c803d305092415" + integrity sha512-LDDODcFzNtECTrUUbVCs6j9/bDVqy7DDRsuIXJg6so+mFksgwG7ZVnTruYi5V+z3eE5y+BJZw7VvUadkbfg7QA== + +"@esbuild/linux-s390x@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.21.5.tgz#b7ccf686751d6a3e44b8627ababc8be3ef62d8de" + integrity sha512-zus5sxzqBJD3eXxwvjN1yQkRepANgxE9lgOW2qLnmr8ikMTphkjgXu1HR01K4FJg8h1kEEDAqDcZQtbrRnB41A== + +"@esbuild/linux-s390x@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-s390x/-/linux-s390x-0.25.3.tgz#f6a7cb67969222b200974de58f105dfe8e99448d" + integrity sha512-s+w/NOY2k0yC2p9SLen+ymflgcpRkvwwa02fqmAwhBRI3SC12uiS10edHHXlVWwfAagYSY5UpmT/zISXPMW3tQ== + +"@esbuild/linux-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.21.5.tgz#6d8f0c768e070e64309af8004bb94e68ab2bb3b0" + integrity sha512-1rYdTpyv03iycF1+BhzrzQJCdOuAOtaqHTWJZCWvijKD2N5Xu0TtVC8/+1faWqcP9iBCWOmjmhoH94dH82BxPQ== + +"@esbuild/linux-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/linux-x64/-/linux-x64-0.25.3.tgz#a237d3578ecdd184a3066b1f425e314ade0f8033" + integrity sha512-nQHDz4pXjSDC6UfOE1Fw9Q8d6GCAd9KdvMZpfVGWSJztYCarRgSDfOVBY5xwhQXseiyxapkiSJi/5/ja8mRFFA== + +"@esbuild/netbsd-arm64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/netbsd-arm64/-/netbsd-arm64-0.25.3.tgz#4c15c68d8149614ddb6a56f9c85ae62ccca08259" + integrity sha512-1QaLtOWq0mzK6tzzp0jRN3eccmN3hezey7mhLnzC6oNlJoUJz4nym5ZD7mDnS/LZQgkrhEbEiTn515lPeLpgWA== + +"@esbuild/netbsd-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.21.5.tgz#bbe430f60d378ecb88decb219c602667387a6047" + integrity sha512-Woi2MXzXjMULccIwMnLciyZH4nCIMpWQAs049KEeMvOcNADVxo0UBIQPfSmxB3CWKedngg7sWZdLvLczpe0tLg== + +"@esbuild/netbsd-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/netbsd-x64/-/netbsd-x64-0.25.3.tgz#12f6856f8c54c2d7d0a8a64a9711c01a743878d5" + integrity sha512-i5Hm68HXHdgv8wkrt+10Bc50zM0/eonPb/a/OFVfB6Qvpiirco5gBA5bz7S2SHuU+Y4LWn/zehzNX14Sp4r27g== + +"@esbuild/openbsd-arm64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/openbsd-arm64/-/openbsd-arm64-0.25.3.tgz#ca078dad4a34df192c60233b058db2ca3d94bc5c" + integrity sha512-zGAVApJEYTbOC6H/3QBr2mq3upG/LBEXr85/pTtKiv2IXcgKV0RT0QA/hSXZqSvLEpXeIxah7LczB4lkiYhTAQ== + +"@esbuild/openbsd-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.21.5.tgz#99d1cf2937279560d2104821f5ccce220cb2af70" + integrity sha512-HLNNw99xsvx12lFBUwoT8EVCsSvRNDVxNpjZ7bPn947b8gJPzeHWyNVhFsaerc0n3TsbOINvRP2byTZ5LKezow== + +"@esbuild/openbsd-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/openbsd-x64/-/openbsd-x64-0.25.3.tgz#c9178adb60e140e03a881d0791248489c79f95b2" + integrity sha512-fpqctI45NnCIDKBH5AXQBsD0NDPbEFczK98hk/aa6HJxbl+UtLkJV2+Bvy5hLSLk3LHmqt0NTkKNso1A9y1a4w== + +"@esbuild/sunos-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.21.5.tgz#08741512c10d529566baba837b4fe052c8f3487b" + integrity sha512-6+gjmFpfy0BHU5Tpptkuh8+uw3mnrvgs+dSPQXQOv3ekbordwnzTVEb4qnIvQcYXq6gzkyTnoZ9dZG+D4garKg== + +"@esbuild/sunos-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/sunos-x64/-/sunos-x64-0.25.3.tgz#03765eb6d4214ff27e5230af779e80790d1ee09f" + integrity sha512-ROJhm7d8bk9dMCUZjkS8fgzsPAZEjtRJqCAmVgB0gMrvG7hfmPmz9k1rwO4jSiblFjYmNvbECL9uhaPzONMfgA== + +"@esbuild/win32-arm64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.21.5.tgz#675b7385398411240735016144ab2e99a60fc75d" + integrity sha512-Z0gOTd75VvXqyq7nsl93zwahcTROgqvuAcYDUr+vOv8uHhNSKROyU961kgtCD1e95IqPKSQKH7tBTslnS3tA8A== + +"@esbuild/win32-arm64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/win32-arm64/-/win32-arm64-0.25.3.tgz#f1c867bd1730a9b8dfc461785ec6462e349411ea" + integrity sha512-YWcow8peiHpNBiIXHwaswPnAXLsLVygFwCB3A7Bh5jRkIBFWHGmNQ48AlX4xDvQNoMZlPYzjVOQDYEzWCqufMQ== + +"@esbuild/win32-ia32@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.21.5.tgz#1bfc3ce98aa6ca9a0969e4d2af72144c59c1193b" + integrity sha512-SWXFF1CL2RVNMaVs+BBClwtfZSvDgtL//G/smwAc5oVK/UPu2Gu9tIaRgFmYFFKrmg3SyAjSrElf0TiJ1v8fYA== + +"@esbuild/win32-ia32@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/win32-ia32/-/win32-ia32-0.25.3.tgz#77491f59ef6c9ddf41df70670d5678beb3acc322" + integrity sha512-qspTZOIGoXVS4DpNqUYUs9UxVb04khS1Degaw/MnfMe7goQ3lTfQ13Vw4qY/Nj0979BGvMRpAYbs/BAxEvU8ew== + +"@esbuild/win32-x64@0.21.5": + version "0.21.5" + resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.21.5.tgz#acad351d582d157bb145535db2a6ff53dd514b5c" + integrity sha512-tQd/1efJuzPC6rCFwEvLtci/xNFcTZknmXs98FYDfGE4wP9ClFV98nyKrzJKVPMhdDnjzLhdUyMX4PsQAPjwIw== + +"@esbuild/win32-x64@0.25.3": + version "0.25.3" + resolved "https://registry.yarnpkg.com/@esbuild/win32-x64/-/win32-x64-0.25.3.tgz#b17a2171f9074df9e91bfb07ef99a892ac06412a" + integrity sha512-ICgUR+kPimx0vvRzf+N/7L7tVSQeE3BYY+NhHRHXS1kBuPO7z2+7ea2HbhDyZdTephgvNvKrlDDKUexuCVBVvg== + +"@eslint-community/eslint-utils@^4.2.0", "@eslint-community/eslint-utils@^4.4.0": + version "4.6.1" + resolved "https://registry.yarnpkg.com/@eslint-community/eslint-utils/-/eslint-utils-4.6.1.tgz#e4c58fdcf0696e7a5f19c30201ed43123ab15abc" + integrity sha512-KTsJMmobmbrFLe3LDh0PC2FXpcSYJt/MLjlkh/9LEnmKYLSYmT/0EW9JWANjeoemiuZrmogti0tW5Ch+qNUYDw== + dependencies: + eslint-visitor-keys "^3.4.3" + +"@eslint-community/regexpp@^4.10.0", "@eslint-community/regexpp@^4.6.1": + version "4.12.1" + resolved "https://registry.yarnpkg.com/@eslint-community/regexpp/-/regexpp-4.12.1.tgz#cfc6cffe39df390a3841cde2abccf92eaa7ae0e0" + integrity sha512-CCZCDJuduB9OUkFkY2IgppNZMi2lBQgD2qzwXkEia16cge2pijY/aXi96CJMquDMn3nJdlPV1A5KrJEXwfLNzQ== + +"@eslint/eslintrc@^2.1.4": + version "2.1.4" + resolved "https://registry.yarnpkg.com/@eslint/eslintrc/-/eslintrc-2.1.4.tgz#388a269f0f25c1b6adc317b5a2c55714894c70ad" + integrity sha512-269Z39MS6wVJtsoUl10L60WdkhJVdPG24Q4eZTH3nnF6lpvSShEK3wQjDX9JRWAUPvPh7COouPpU9IrqaZFvtQ== + dependencies: + ajv "^6.12.4" + debug "^4.3.2" + espree "^9.6.0" + globals "^13.19.0" + ignore "^5.2.0" + import-fresh "^3.2.1" + js-yaml "^4.1.0" + minimatch "^3.1.2" + strip-json-comments "^3.1.1" + +"@eslint/js@8.57.1": + version "8.57.1" + resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.57.1.tgz#de633db3ec2ef6a3c89e2f19038063e8a122e2c2" + integrity sha512-d9zaMRSTIKDLhctzH12MtXvJKSSUhaHcjV+2Z+GK+EEY7XKpP5yR4x+N3TAcHTcu963nIr+TMcCb4DBCYX1z6Q== + +"@fortawesome/fontawesome-common-types@6.7.2": + version "6.7.2" + resolved "https://registry.yarnpkg.com/@fortawesome/fontawesome-common-types/-/fontawesome-common-types-6.7.2.tgz#7123d74b0c1e726794aed1184795dbce12186470" + integrity sha512-Zs+YeHUC5fkt7Mg1l6XTniei3k4bwG/yo3iFUtZWd/pMx9g3fdvkSK9E0FOC+++phXOka78uJcYb8JaFkW52Xg== + +"@fortawesome/fontawesome-svg-core@^6.7.2": + version "6.7.2" + resolved "https://registry.yarnpkg.com/@fortawesome/fontawesome-svg-core/-/fontawesome-svg-core-6.7.2.tgz#0ac6013724d5cc327c1eb81335b91300a4fce2f2" + integrity sha512-yxtOBWDrdi5DD5o1pmVdq3WMCvnobT0LU6R8RyyVXPvFRd2o79/0NCuQoCjNTeZz9EzA9xS3JxNWfv54RIHFEA== + dependencies: + "@fortawesome/fontawesome-common-types" "6.7.2" + +"@fortawesome/free-brands-svg-icons@^6.7.2": + version "6.7.2" + resolved "https://registry.yarnpkg.com/@fortawesome/free-brands-svg-icons/-/free-brands-svg-icons-6.7.2.tgz#4ebee8098f31da5446dda81edc344025eb59b27e" + integrity sha512-zu0evbcRTgjKfrr77/2XX+bU+kuGfjm0LbajJHVIgBWNIDzrhpRxiCPNT8DW5AdmSsq7Mcf9D1bH0aSeSUSM+Q== + dependencies: + "@fortawesome/fontawesome-common-types" "6.7.2" + +"@fortawesome/free-solid-svg-icons@^6.7.2": + version "6.7.2" + resolved "https://registry.yarnpkg.com/@fortawesome/free-solid-svg-icons/-/free-solid-svg-icons-6.7.2.tgz#fe25883b5eb8464a82918599950d283c465b57f6" + integrity sha512-GsBrnOzU8uj0LECDfD5zomZJIjrPhIlWU82AHwa2s40FKH+kcxQaBvBo3Z4TxyZHIyX8XTDxsyA33/Vx9eFuQA== + dependencies: + "@fortawesome/fontawesome-common-types" "6.7.2" + +"@fortawesome/react-fontawesome@^0.2.0": + version "0.2.2" + resolved "https://registry.yarnpkg.com/@fortawesome/react-fontawesome/-/react-fontawesome-0.2.2.tgz#68b058f9132b46c8599875f6a636dad231af78d4" + integrity sha512-EnkrprPNqI6SXJl//m29hpaNzOp1bruISWaOiRtkMi/xSvHJlzc2j2JAYS7egxt/EbjSNV/k6Xy0AQI6vB2+1g== + dependencies: + prop-types "^15.8.1" + +"@gar/promisify@^1.1.3": + version "1.1.3" + resolved "https://registry.yarnpkg.com/@gar/promisify/-/promisify-1.1.3.tgz#555193ab2e3bb3b6adc3d551c9c030d9e860daf6" + integrity sha512-k2Ty1JcVojjJFwrg/ThKi2ujJ7XNLYaFGNB/bWT9wGR+oSMJHMa5w+CUq6p/pVrKeNNgA7pCqEcjSnHVoqJQFw== + +"@headlessui/react@^1.7.19": + version "1.7.19" + resolved "https://registry.yarnpkg.com/@headlessui/react/-/react-1.7.19.tgz#91c78cf5fcb254f4a0ebe96936d48421caf75f40" + integrity sha512-Ll+8q3OlMJfJbAKM/+/Y2q6PPYbryqNTXDbryx7SXLIDamkF6iQFbriYHga0dY44PvDhvvBWCx1Xj4U5+G4hOw== + dependencies: + "@tanstack/react-virtual" "^3.0.0-beta.60" + client-only "^0.0.1" + +"@headlessui/tailwindcss@^0.2.2": + version "0.2.2" + resolved "https://registry.yarnpkg.com/@headlessui/tailwindcss/-/tailwindcss-0.2.2.tgz#8ebde73fabca72d48636ea56ae790209dc5f0d49" + integrity sha512-xNe42KjdyA4kfUKLLPGzME9zkH7Q3rOZ5huFihWNWOQFxnItxPB3/67yBI8/qBfY8nwBRx5GHn4VprsoluVMGw== + +"@humanwhocodes/config-array@^0.13.0": + version "0.13.0" + resolved "https://registry.yarnpkg.com/@humanwhocodes/config-array/-/config-array-0.13.0.tgz#fb907624df3256d04b9aa2df50d7aa97ec648748" + integrity sha512-DZLEEqFWQFiyK6h5YIeynKx7JlvCYWL0cImfSRXZ9l4Sg2efkFGTuFf6vzXjK1cq6IYkU+Eg/JizXw+TD2vRNw== + dependencies: + "@humanwhocodes/object-schema" "^2.0.3" + debug "^4.3.1" + minimatch "^3.0.5" + +"@humanwhocodes/module-importer@^1.0.1": + version "1.0.1" + resolved "https://registry.yarnpkg.com/@humanwhocodes/module-importer/-/module-importer-1.0.1.tgz#af5b2691a22b44be847b0ca81641c5fb6ad0172c" + integrity sha512-bxveV4V8v5Yb4ncFTT3rPSgZBOpCkjfK0y4oVVVJwIuDVBRMDXrPyXRL988i5ap9m9bnyEEjWfm5WkBmtffLfA== + +"@humanwhocodes/object-schema@^2.0.3": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@humanwhocodes/object-schema/-/object-schema-2.0.3.tgz#4a2868d75d6d6963e423bcf90b7fd1be343409d3" + integrity sha512-93zYdMES/c1D69yZiKDBj0V24vqNzB/koF26KPaagAfd3P/4gUlh3Dys5ogAK+Exi9QyzlD8x/08Zt7wIKcDcA== + +"@iarna/toml@^2.2.5": + version "2.2.5" + resolved "https://registry.yarnpkg.com/@iarna/toml/-/toml-2.2.5.tgz#b32366c89b43c6f8cefbdefac778b9c828e3ba8c" + integrity sha512-trnsAYxU3xnS1gPHPyU961coFyLkh4gAD/0zQ5mymY4yOZ+CYvsPqUbOFSw0aDM4y0tV7tiFxL/1XfXPNC6IPg== + +"@isaacs/cliui@^8.0.2": + version "8.0.2" + resolved "https://registry.yarnpkg.com/@isaacs/cliui/-/cliui-8.0.2.tgz#b37667b7bc181c168782259bab42474fbf52b550" + integrity sha512-O8jcjabXaleOG9DQ0+ARXWZBTfnP4WNAqzuiJK7ll44AmxGKv/J2M4TPjxjY3znBCfvBXFzucm1twdyFybFqEA== + dependencies: + string-width "^5.1.2" + string-width-cjs "npm:string-width@^4.2.0" + strip-ansi "^7.0.1" + strip-ansi-cjs "npm:strip-ansi@^6.0.1" + wrap-ansi "^8.1.0" + wrap-ansi-cjs "npm:wrap-ansi@^7.0.0" + +"@jest/expect-utils@^29.7.0": + version "29.7.0" + resolved "https://registry.yarnpkg.com/@jest/expect-utils/-/expect-utils-29.7.0.tgz#023efe5d26a8a70f21677d0a1afc0f0a44e3a1c6" + integrity sha512-GlsNBWiFQFCVi9QVSx7f5AgMeLxe9YCCs5PuP2O2LdjDAA8Jh9eX7lA1Jq/xdXw3Wb3hyvlFNfZIfcRetSzYcA== + dependencies: + jest-get-type "^29.6.3" + +"@jest/schemas@^29.6.3": + version "29.6.3" + resolved "https://registry.yarnpkg.com/@jest/schemas/-/schemas-29.6.3.tgz#430b5ce8a4e0044a7e3819663305a7b3091c8e03" + integrity sha512-mo5j5X+jIZmJQveBKeS/clAueipV7KgiX1vMgCxam1RNYiqE1w62n0/tJJnHtjW8ZHcQco5gY85jA3mi0L+nSA== + dependencies: + "@sinclair/typebox" "^0.27.8" + +"@jest/types@^29.6.3": + version "29.6.3" + resolved "https://registry.yarnpkg.com/@jest/types/-/types-29.6.3.tgz#1131f8cf634e7e84c5e77bab12f052af585fba59" + integrity sha512-u3UPsIilWKOM3F9CXtrG8LEJmNxwoCQC/XVj4IKYXvvpx7QIi/Kg1LI5uDmDpKlac62NUtX7eLjRh+jVZcLOzw== + dependencies: + "@jest/schemas" "^29.6.3" + "@types/istanbul-lib-coverage" "^2.0.0" + "@types/istanbul-reports" "^3.0.0" + "@types/node" "*" + "@types/yargs" "^17.0.8" + chalk "^4.0.0" + +"@jridgewell/gen-mapping@^0.3.2", "@jridgewell/gen-mapping@^0.3.5": + version "0.3.8" + resolved "https://registry.yarnpkg.com/@jridgewell/gen-mapping/-/gen-mapping-0.3.8.tgz#4f0e06362e01362f823d348f1872b08f666d8142" + integrity sha512-imAbBGkb+ebQyxKgzv5Hu2nmROxoDOXHh80evxdoXNOrvAnVx7zimzc1Oo5h9RlfV4vPXaE2iM5pOFbvOCClWA== + dependencies: + "@jridgewell/set-array" "^1.2.1" + "@jridgewell/sourcemap-codec" "^1.4.10" + "@jridgewell/trace-mapping" "^0.3.24" + +"@jridgewell/resolve-uri@^3.0.3", "@jridgewell/resolve-uri@^3.1.0": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz#7a0ee601f60f99a20c7c7c5ff0c80388c1189bd6" + integrity sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw== + +"@jridgewell/set-array@^1.2.1": + version "1.2.1" + resolved "https://registry.yarnpkg.com/@jridgewell/set-array/-/set-array-1.2.1.tgz#558fb6472ed16a4c850b889530e6b36438c49280" + integrity sha512-R8gLRTZeyp03ymzP/6Lil/28tGeGEzhx1q2k703KGWRAI1VdvPIXdG70VJc2pAMw3NA6JKL5hhFu1sJX0Mnn/A== + +"@jridgewell/sourcemap-codec@^1.4.10", "@jridgewell/sourcemap-codec@^1.4.14", "@jridgewell/sourcemap-codec@^1.5.0": + version "1.5.0" + resolved "https://registry.yarnpkg.com/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.0.tgz#3188bcb273a414b0d215fd22a58540b989b9409a" + integrity sha512-gv3ZRaISU3fjPAgNsriBRqGWQL6quFx04YMPW/zD8XMLsU32mhCCbfbO6KZFLjvYpCZ8zyDEgqsgf+PwPaM7GQ== + +"@jridgewell/trace-mapping@0.3.9": + version "0.3.9" + resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.9.tgz#6534fd5933a53ba7cbf3a17615e273a0d1273ff9" + integrity sha512-3Belt6tdc8bPgAtbcmdtNJlirVoTmEb5e2gC94PnkwEW9jI6CAHUeoG85tjWP5WquqfavoMtMwiG4P926ZKKuQ== + dependencies: + "@jridgewell/resolve-uri" "^3.0.3" + "@jridgewell/sourcemap-codec" "^1.4.10" + +"@jridgewell/trace-mapping@^0.3.24", "@jridgewell/trace-mapping@^0.3.25": + version "0.3.25" + resolved "https://registry.yarnpkg.com/@jridgewell/trace-mapping/-/trace-mapping-0.3.25.tgz#15f190e98895f3fc23276ee14bc76b675c2e50f0" + integrity sha512-vNk6aEwybGtawWmy/PzwnGDOjCkLWSD2wqvjGGAgOAwCGWySYXfYoxt00IJkTF+8Lb57DwOb3Aa0o9CApepiYQ== + dependencies: + "@jridgewell/resolve-uri" "^3.1.0" + "@jridgewell/sourcemap-codec" "^1.4.14" + +"@kittycad/lib@2.0.28": + version "2.0.28" + resolved "https://registry.yarnpkg.com/@kittycad/lib/-/lib-2.0.28.tgz#884d561f0d5440f97a3368c7d9f088667447a654" + integrity sha512-DTxUJgrTWk/IB8xSAPsaT3U34PVlWR3Mh/HwVIGWWpd8v/TKebO2vMdfkfn5XcsW1GwMawH/xZo9Su6OCuLXHg== + dependencies: + openapi-types "^12.0.0" + ts-node "^10.9.1" + tslib "~2.8" + +"@leichtgewicht/ip-codec@^2.0.1": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@leichtgewicht/ip-codec/-/ip-codec-2.0.5.tgz#4fc56c15c580b9adb7dc3c333a134e540b44bfb1" + integrity sha512-Vo+PSpZG2/fmgmiNzYK9qWRh8h/CHrwD0mo1h1DzL4yzHNSfWYujGTYsWGreD000gcgmZ7K4Ys6Tx9TxtsKdDw== + +"@lezer/common@^1.0.0", "@lezer/common@^1.1.0": + version "1.2.3" + resolved "https://registry.yarnpkg.com/@lezer/common/-/common-1.2.3.tgz#138fcddab157d83da557554851017c6c1e5667fd" + integrity sha512-w7ojc8ejBqr2REPsWxJjrMFsA/ysDCFICn8zEOR9mrqzOu2amhITYuLD8ag6XZf0CFXDrhKqw7+tW8cX66NaDA== + +"@lezer/generator@^1.7.3": + version "1.7.3" + resolved "https://registry.yarnpkg.com/@lezer/generator/-/generator-1.7.3.tgz#8bf9e99c7690dd31327425ca46706a380413f54d" + integrity sha512-vAI2O1tPF8QMMgp+bdUeeJCneJNkOZvqsrtyb4ohnFVFdboSqPwBEacnt0HH4E+5h+qsIwTHUSAhffU4hzKl1A== + dependencies: + "@lezer/common" "^1.1.0" + "@lezer/lr" "^1.3.0" + +"@lezer/highlight@^1.0.0", "@lezer/highlight@^1.2.1": + version "1.2.1" + resolved "https://registry.yarnpkg.com/@lezer/highlight/-/highlight-1.2.1.tgz#596fa8f9aeb58a608be0a563e960c373cbf23f8b" + integrity sha512-Z5duk4RN/3zuVO7Jq0pGLJ3qynpxUVsh7IbUbGj88+uV2ApSAn6kWg2au3iJb+0Zi7kKtqffIESgNcRXWZWmSA== + dependencies: + "@lezer/common" "^1.0.0" + +"@lezer/lr@^1.0.0", "@lezer/lr@^1.3.0", "@lezer/lr@^1.4.1": + version "1.4.2" + resolved "https://registry.yarnpkg.com/@lezer/lr/-/lr-1.4.2.tgz#931ea3dea8e9de84e90781001dae30dea9ff1727" + integrity sha512-pu0K1jCIdnQ12aWNaAVU5bzi7Bd1w54J3ECgANPmYLtQKP0HBj2cE/5coBD66MT10xbtIuUr7tg0Shbsvk0mDA== + dependencies: + "@lezer/common" "^1.0.0" + +"@malept/cross-spawn-promise@^2.0.0": + version "2.0.0" + resolved "https://registry.yarnpkg.com/@malept/cross-spawn-promise/-/cross-spawn-promise-2.0.0.tgz#d0772de1aa680a0bfb9ba2f32b4c828c7857cb9d" + integrity sha512-1DpKU0Z5ThltBwjNySMC14g0CkbyhCaz9FkhxqNsZI6uAPJXFS8cMXlBKo26FJ8ZuW6S9GCMcR9IO5k2X5/9Fg== + dependencies: + cross-spawn "^7.0.1" + +"@malept/flatpak-bundler@^0.4.0": + version "0.4.0" + resolved "https://registry.yarnpkg.com/@malept/flatpak-bundler/-/flatpak-bundler-0.4.0.tgz#e8a32c30a95d20c2b1bb635cc580981a06389858" + integrity sha512-9QOtNffcOF/c1seMCDnjckb3R9WHcG34tky+FHpNKKCW0wc/scYLwMtO+ptyGUfMW0/b/n4qRiALlaFHc9Oj7Q== + dependencies: + debug "^4.1.1" + fs-extra "^9.0.0" + lodash "^4.17.15" + tmp-promise "^3.0.2" + +"@marijn/find-cluster-break@^1.0.0": + version "1.0.2" + resolved "https://registry.yarnpkg.com/@marijn/find-cluster-break/-/find-cluster-break-1.0.2.tgz#775374306116d51c0c500b8c4face0f9a04752d8" + integrity sha512-l0h88YhZFyKdXIFNfSWpyjStDjGHwZ/U7iobcK1cQQD8sejsONdQtTVU+1wVN1PBw40PiiHB1vA5S7VTfQiP9g== + +"@nabla/vite-plugin-eslint@^2.0.5": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@nabla/vite-plugin-eslint/-/vite-plugin-eslint-2.0.5.tgz#f99333a9fe26442e497fd6af269803453b8c4c68" + integrity sha512-m6Q8ZVM0ASZfYyfFbG661mDklhZQZEeBMQgtB26NhdadxPSctHzHsUbF87msdSb0V4Z8R1p/vRkB5SHNOpANAQ== + dependencies: + "@types/eslint" "*" + chalk "^4" + debug "^4" + +"@nodelib/fs.scandir@2.1.5": + version "2.1.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz#7619c2eb21b25483f6d167548b4cfd5a7488c3d5" + integrity sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g== + dependencies: + "@nodelib/fs.stat" "2.0.5" + run-parallel "^1.1.9" + +"@nodelib/fs.stat@2.0.5", "@nodelib/fs.stat@^2.0.2": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz#5bd262af94e9d25bd1e71b05deed44876a222e8b" + integrity sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A== + +"@nodelib/fs.walk@^1.2.3", "@nodelib/fs.walk@^1.2.8": + version "1.2.8" + resolved "https://registry.yarnpkg.com/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz#e95737e8bb6746ddedf69c556953494f196fe69a" + integrity sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg== + dependencies: + "@nodelib/fs.scandir" "2.1.5" + fastq "^1.6.0" + +"@npmcli/fs@^2.1.0": + version "2.1.2" + resolved "https://registry.yarnpkg.com/@npmcli/fs/-/fs-2.1.2.tgz#a9e2541a4a2fec2e69c29b35e6060973da79b865" + integrity sha512-yOJKRvohFOaLqipNtwYB9WugyZKhC/DZC4VYPmpaCzDBrA8YpK3qHZ8/HGscMnE4GqbkLNuVcCnxkeQEdGt6LQ== + dependencies: + "@gar/promisify" "^1.1.3" + semver "^7.3.5" + +"@npmcli/move-file@^2.0.0": + version "2.0.1" + resolved "https://registry.yarnpkg.com/@npmcli/move-file/-/move-file-2.0.1.tgz#26f6bdc379d87f75e55739bab89db525b06100e4" + integrity sha512-mJd2Z5TjYWq/ttPLLGqArdtnC74J6bOzg4rMDnN+p1xTacZ2yPRCk2y0oSWQtygLR9YVQXgOcONrwtnk3JupxQ== + dependencies: + mkdirp "^1.0.4" + rimraf "^3.0.2" + +"@pkgjs/parseargs@^0.11.0": + version "0.11.0" + resolved "https://registry.yarnpkg.com/@pkgjs/parseargs/-/parseargs-0.11.0.tgz#a77ea742fab25775145434eb1d2328cf5013ac33" + integrity sha512-+1VkjdD0QBLPodGrJUeqarH8VAIvQODIbwh9XpP5Syisf7YoQgsJKPNFoqqLQlu+VQ/tVSshMR6loPMn8U+dPg== + +"@playwright/test@^1.52.0": + version "1.52.0" + resolved "https://registry.yarnpkg.com/@playwright/test/-/test-1.52.0.tgz#267ec595b43a8f4fa5e444ea503689629e91a5b8" + integrity sha512-uh6W7sb55hl7D6vsAeA+V2p5JnlAqzhqFyF0VcJkKZXkgnFcVG9PziERRHQfPLfNGx1C292a4JqbWzhR8L4R1g== + dependencies: + playwright "1.52.0" + +"@react-hook/latest@^1.0.2": + version "1.0.3" + resolved "https://registry.yarnpkg.com/@react-hook/latest/-/latest-1.0.3.tgz#c2d1d0b0af8b69ec6e2b3a2412ba0768ac82db80" + integrity sha512-dy6duzl+JnAZcDbNTfmaP3xHiKtbXYOaz3G51MGVljh548Y8MWzTr+PHLOfvpypEVW9zwvl+VyKjbWKEVbV1Rg== + +"@react-hook/passive-layout-effect@^1.2.0": + version "1.2.1" + resolved "https://registry.yarnpkg.com/@react-hook/passive-layout-effect/-/passive-layout-effect-1.2.1.tgz#c06dac2d011f36d61259aa1c6df4f0d5e28bc55e" + integrity sha512-IwEphTD75liO8g+6taS+4oqz+nnroocNfWVHWz7j+N+ZO2vYrc6PV1q7GQhuahL0IOR7JccFTsFKQ/mb6iZWAg== + +"@react-hook/resize-observer@^2.0.1": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@react-hook/resize-observer/-/resize-observer-2.0.2.tgz#f49fe4e6b9de86c583d136df7fae430684528092" + integrity sha512-tzKKzxNpfE5TWmxuv+5Ae3IF58n0FQgQaWJmcbYkjXTRZATXxClnTprQ2uuYygYTpu1pqbBskpwMpj6jpT1djA== + dependencies: + "@react-hook/latest" "^1.0.2" + "@react-hook/passive-layout-effect" "^1.2.0" + +"@redocly/ajv@^8.11.2": + version "8.11.2" + resolved "https://registry.yarnpkg.com/@redocly/ajv/-/ajv-8.11.2.tgz#46e1bf321ec0ac1e0fd31dea41a3d1fcbdcda0b5" + integrity sha512-io1JpnwtIcvojV7QKDUSIuMN/ikdOUd1ReEnUnMKGfDVridQZ31J0MmIuqwuRjWDZfmvr+Q0MqCcfHM2gTivOg== + dependencies: + fast-deep-equal "^3.1.1" + json-schema-traverse "^1.0.0" + require-from-string "^2.0.2" + uri-js-replace "^1.0.1" + +"@redocly/config@^0.22.0": + version "0.22.2" + resolved "https://registry.yarnpkg.com/@redocly/config/-/config-0.22.2.tgz#9a05e694816d53a5236cf8768d3cad0e49d8b116" + integrity sha512-roRDai8/zr2S9YfmzUfNhKjOF0NdcOIqF7bhf4MVC5UxpjIysDjyudvlAiVbpPHp3eDRWbdzUgtkK1a7YiDNyQ== + +"@redocly/openapi-core@^1.28.0": + version "1.34.2" + resolved "https://registry.yarnpkg.com/@redocly/openapi-core/-/openapi-core-1.34.2.tgz#859de74a3261fb09a3dc695be4a8d9b1f2059d75" + integrity sha512-glfkQFJizLdq2fBkNvc2FJW0sxDb5exd0wIXhFk+WHaFLMREBC3CxRo2Zq7uJIdfV9U3YTceMbXJklpDfmmwFQ== + dependencies: + "@redocly/ajv" "^8.11.2" + "@redocly/config" "^0.22.0" + colorette "^1.2.0" + https-proxy-agent "^7.0.5" + js-levenshtein "^1.1.6" + js-yaml "^4.1.0" + minimatch "^5.0.1" + pluralize "^8.0.0" + yaml-ast-parser "0.0.43" + +"@remix-run/router@1.23.0": + version "1.23.0" + resolved "https://registry.yarnpkg.com/@remix-run/router/-/router-1.23.0.tgz#35390d0e7779626c026b11376da6789eb8389242" + integrity sha512-O3rHJzAQKamUz1fvE0Qaw0xSFqsA/yafi2iqeE0pvdFtCO1viYx8QL6f3Ln/aCCTLxs68SLf0KPM9eSeM8yBnA== + +"@replit/codemirror-interact@^6.3.1": + version "6.3.1" + resolved "https://registry.yarnpkg.com/@replit/codemirror-interact/-/codemirror-interact-6.3.1.tgz#45f6ce8c7e4ff952c698599bd9899f22739fe966" + integrity sha512-3kGCrOSo7tRGrmvThRdMRCjrUo0BqFilC+MiucSznFD6W4snyxaqElZTNWMEfFsj/rWHykcUHlyfLOoxNWjvAw== + dependencies: + "@codemirror/state" "^6.2.1" + +"@rollup/plugin-typescript@^12.1.2": + version "12.1.2" + resolved "https://registry.yarnpkg.com/@rollup/plugin-typescript/-/plugin-typescript-12.1.2.tgz#ebaeec2e7376faa889030ccd7cb485a649e63118" + integrity sha512-cdtSp154H5sv637uMr1a8OTWB0L1SWDSm1rDGiyfcGcvQ6cuTs4MDk2BVEBGysUWago4OJN4EQZqOTl/QY3Jgg== + dependencies: + "@rollup/pluginutils" "^5.1.0" + resolve "^1.22.1" + +"@rollup/plugin-virtual@^3.0.2": + version "3.0.2" + resolved "https://registry.yarnpkg.com/@rollup/plugin-virtual/-/plugin-virtual-3.0.2.tgz#17e17eeecb4c9fa1c0a6e72c9e5f66382fddbb82" + integrity sha512-10monEYsBp3scM4/ND4LNH5Rxvh3e/cVeL3jWTgZ2SrQ+BmUoQcopVQvnaMcOnykb1VkxUFuDAN+0FnpTFRy2A== + +"@rollup/pluginutils@^5.1.0": + version "5.1.4" + resolved "https://registry.yarnpkg.com/@rollup/pluginutils/-/pluginutils-5.1.4.tgz#bb94f1f9eaaac944da237767cdfee6c5b2262d4a" + integrity sha512-USm05zrsFxYLPdWWq+K3STlWiT/3ELn3RcV5hJMghpeAIhxfsUIg6mt12CBJBInWMV4VneoV7SfGv8xIwo2qNQ== + dependencies: + "@types/estree" "^1.0.0" + estree-walker "^2.0.2" + picomatch "^4.0.2" + +"@rollup/rollup-android-arm-eabi@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.40.0.tgz#d964ee8ce4d18acf9358f96adc408689b6e27fe3" + integrity sha512-+Fbls/diZ0RDerhE8kyC6hjADCXA1K4yVNlH0EYfd2XjyH0UGgzaQ8MlT0pCXAThfxv3QUAczHaL+qSv1E4/Cg== + +"@rollup/rollup-android-arm64@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.40.0.tgz#9b5e130ecc32a5fc1e96c09ff371743ee71a62d3" + integrity sha512-PPA6aEEsTPRz+/4xxAmaoWDqh67N7wFbgFUJGMnanCFs0TV99M0M8QhhaSCks+n6EbQoFvLQgYOGXxlMGQe/6w== + +"@rollup/rollup-darwin-arm64@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.40.0.tgz#ef439182c739b20b3c4398cfc03e3c1249ac8903" + integrity sha512-GwYOcOakYHdfnjjKwqpTGgn5a6cUX7+Ra2HeNj/GdXvO2VJOOXCiYYlRFU4CubFM67EhbmzLOmACKEfvp3J1kQ== + +"@rollup/rollup-darwin-x64@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.40.0.tgz#d7380c1531ab0420ca3be16f17018ef72dd3d504" + integrity sha512-CoLEGJ+2eheqD9KBSxmma6ld01czS52Iw0e2qMZNpPDlf7Z9mj8xmMemxEucinev4LgHalDPczMyxzbq+Q+EtA== + +"@rollup/rollup-freebsd-arm64@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.40.0.tgz#cbcbd7248823c6b430ce543c59906dd3c6df0936" + integrity sha512-r7yGiS4HN/kibvESzmrOB/PxKMhPTlz+FcGvoUIKYoTyGd5toHp48g1uZy1o1xQvybwwpqpe010JrcGG2s5nkg== + +"@rollup/rollup-freebsd-x64@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.40.0.tgz#96bf6ff875bab5219c3472c95fa6eb992586a93b" + integrity sha512-mVDxzlf0oLzV3oZOr0SMJ0lSDd3xC4CmnWJ8Val8isp9jRGl5Dq//LLDSPFrasS7pSm6m5xAcKaw3sHXhBjoRw== + +"@rollup/rollup-linux-arm-gnueabihf@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.40.0.tgz#d80cd62ce6d40f8e611008d8dbf03b5e6bbf009c" + integrity sha512-y/qUMOpJxBMy8xCXD++jeu8t7kzjlOCkoxxajL58G62PJGBZVl/Gwpm7JK9+YvlB701rcQTzjUZ1JgUoPTnoQA== + +"@rollup/rollup-linux-arm-musleabihf@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.40.0.tgz#75440cfc1e8d0f87a239b4c31dfeaf4719b656b7" + integrity sha512-GoCsPibtVdJFPv/BOIvBKO/XmwZLwaNWdyD8TKlXuqp0veo2sHE+A/vpMQ5iSArRUz/uaoj4h5S6Pn0+PdhRjg== + +"@rollup/rollup-linux-arm64-gnu@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.40.0.tgz#ac527485ecbb619247fb08253ec8c551a0712e7c" + integrity sha512-L5ZLphTjjAD9leJzSLI7rr8fNqJMlGDKlazW2tX4IUF9P7R5TMQPElpH82Q7eNIDQnQlAyiNVfRPfP2vM5Avvg== + +"@rollup/rollup-linux-arm64-musl@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.40.0.tgz#74d2b5cb11cf714cd7d1682e7c8b39140e908552" + integrity sha512-ATZvCRGCDtv1Y4gpDIXsS+wfFeFuLwVxyUBSLawjgXK2tRE6fnsQEkE4csQQYWlBlsFztRzCnBvWVfcae/1qxQ== + +"@rollup/rollup-linux-loongarch64-gnu@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-loongarch64-gnu/-/rollup-linux-loongarch64-gnu-4.40.0.tgz#a0a310e51da0b5fea0e944b0abd4be899819aef6" + integrity sha512-wG9e2XtIhd++QugU5MD9i7OnpaVb08ji3P1y/hNbxrQ3sYEelKJOq1UJ5dXczeo6Hj2rfDEL5GdtkMSVLa/AOg== + +"@rollup/rollup-linux-powerpc64le-gnu@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-powerpc64le-gnu/-/rollup-linux-powerpc64le-gnu-4.40.0.tgz#4077e2862b0ac9f61916d6b474d988171bd43b83" + integrity sha512-vgXfWmj0f3jAUvC7TZSU/m/cOE558ILWDzS7jBhiCAFpY2WEBn5jqgbqvmzlMjtp8KlLcBlXVD2mkTSEQE6Ixw== + +"@rollup/rollup-linux-riscv64-gnu@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.40.0.tgz#5812a1a7a2f9581cbe12597307cc7ba3321cf2f3" + integrity sha512-uJkYTugqtPZBS3Z136arevt/FsKTF/J9dEMTX/cwR7lsAW4bShzI2R0pJVw+hcBTWF4dxVckYh72Hk3/hWNKvA== + +"@rollup/rollup-linux-riscv64-musl@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.40.0.tgz#973aaaf4adef4531375c36616de4e01647f90039" + integrity sha512-rKmSj6EXQRnhSkE22+WvrqOqRtk733x3p5sWpZilhmjnkHkpeCgWsFFo0dGnUGeA+OZjRl3+VYq+HyCOEuwcxQ== + +"@rollup/rollup-linux-s390x-gnu@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.40.0.tgz#9bad59e907ba5bfcf3e9dbd0247dfe583112f70b" + integrity sha512-SpnYlAfKPOoVsQqmTFJ0usx0z84bzGOS9anAC0AZ3rdSo3snecihbhFTlJZ8XMwzqAcodjFU4+/SM311dqE5Sw== + +"@rollup/rollup-linux-x64-gnu@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.40.0.tgz#68b045a720bd9b4d905f462b997590c2190a6de0" + integrity sha512-RcDGMtqF9EFN8i2RYN2W+64CdHruJ5rPqrlYw+cgM3uOVPSsnAQps7cpjXe9be/yDp8UC7VLoCoKC8J3Kn2FkQ== + +"@rollup/rollup-linux-x64-musl@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.40.0.tgz#8e703e2c2ad19ba7b2cb3d8c3a4ad11d4ee3a282" + integrity sha512-HZvjpiUmSNx5zFgwtQAV1GaGazT2RWvqeDi0hV+AtC8unqqDSsaFjPxfsO6qPtKRRg25SisACWnJ37Yio8ttaw== + +"@rollup/rollup-win32-arm64-msvc@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.40.0.tgz#c5bee19fa670ff5da5f066be6a58b4568e9c650b" + integrity sha512-UtZQQI5k/b8d7d3i9AZmA/t+Q4tk3hOC0tMOMSq2GlMYOfxbesxG4mJSeDp0EHs30N9bsfwUvs3zF4v/RzOeTQ== + +"@rollup/rollup-win32-ia32-msvc@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.40.0.tgz#846e02c17044bd922f6f483a3b4d36aac6e2b921" + integrity sha512-+m03kvI2f5syIqHXCZLPVYplP8pQch9JHyXKZ3AGMKlg8dCyr2PKHjwRLiW53LTrN/Nc3EqHOKxUxzoSPdKddA== + +"@rollup/rollup-win32-x64-msvc@4.40.0": + version "4.40.0" + resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.40.0.tgz#fd92d31a2931483c25677b9c6698106490cbbc76" + integrity sha512-lpPE1cLfP5oPzVjKMx10pgBmKELQnFJXHgvtHCtuJWOv8MxqdEIMNtgHgBFf7Ea2/7EuVwa9fodWUfXAlXZLZQ== + +"@rtsao/scc@^1.1.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@rtsao/scc/-/scc-1.1.0.tgz#927dd2fae9bc3361403ac2c7a00c32ddce9ad7e8" + integrity sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g== + +"@sinclair/typebox@^0.27.8": + version "0.27.8" + resolved "https://registry.yarnpkg.com/@sinclair/typebox/-/typebox-0.27.8.tgz#6667fac16c436b5434a387a34dedb013198f6e6e" + integrity sha512-+Fj43pSMwJs4KRrH/938Uf+uAELIgVBmQzg/q1YG10djyfA3TnrU8N8XzqCh/okZdszqBQTZf96idMfE5lnwTA== + +"@sindresorhus/is@^4.0.0": + version "4.6.0" + resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-4.6.0.tgz#3c7c9c46e678feefe7a2e5bb609d3dbd665ffb3f" + integrity sha512-t09vSN3MdfsyCHoFcTRCH/iUtG7OJ0CsjzB8cjAmKc/va/kIgeDI/TxsigdncE/4be734m0cvIYwNaV4i2XqAw== + +"@swc/core-darwin-arm64@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-darwin-arm64/-/core-darwin-arm64-1.11.22.tgz#61d07fd1a86d1f5035bc3cc180bf29d8a1f6cabd" + integrity sha512-upSiFQfo1TE2QM3+KpBcp5SrOdKKjoc+oUoD1mmBDU2Wv4Bjjv16Z2I5ADvIqMV+b87AhYW+4Qu6iVrQD7j96Q== + +"@swc/core-darwin-x64@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-darwin-x64/-/core-darwin-x64-1.11.22.tgz#e4004c5cc80362d6f2ad482961558d1640878d50" + integrity sha512-8PEuF/gxIMJVK21DjuCOtzdqstn2DqnxVhpAYfXEtm3WmMqLIOIZBypF/xafAozyaHws4aB/5xmz8/7rPsjavw== + +"@swc/core-linux-arm-gnueabihf@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-linux-arm-gnueabihf/-/core-linux-arm-gnueabihf-1.11.22.tgz#48078c84c71a6edc45884fb9f106f1a8da1b8311" + integrity sha512-NIPTXvqtn9e7oQHgdaxM9Z/anHoXC3Fg4ZAgw5rSGa1OlnKKupt5sdfJamNggSi+eAtyoFcyfkgqHnfe2u63HA== + +"@swc/core-linux-arm64-gnu@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-linux-arm64-gnu/-/core-linux-arm64-gnu-1.11.22.tgz#03ce14d829441c37a3b217040196e0fad9bf79a2" + integrity sha512-xZ+bgS60c5r8kAeYsLNjJJhhQNkXdidQ277pUabSlu5GjR0CkQUPQ+L9hFeHf8DITEqpPBPRiAiiJsWq5eqMBg== + +"@swc/core-linux-arm64-musl@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-linux-arm64-musl/-/core-linux-arm64-musl-1.11.22.tgz#16ff1a87f24d79ac330eb1b243b4e2d0005cce34" + integrity sha512-JhrP/q5VqQl2eJR0xKYIkKTPjgf8CRsAmRnjJA2PtZhfQ543YbYvUqxyXSRyBOxdyX8JwzuAxIPEAlKlT7PPuQ== + +"@swc/core-linux-x64-gnu@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-linux-x64-gnu/-/core-linux-x64-gnu-1.11.22.tgz#d249e3ae8b48459e0845aa5fc3099d210833390b" + integrity sha512-htmAVL+U01gk9GyziVUP0UWYaUQBgrsiP7Ytf6uDffrySyn/FclUS3MDPocNydqYsOpj3OpNKPxkaHK+F+X5fg== + +"@swc/core-linux-x64-musl@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-linux-x64-musl/-/core-linux-x64-musl-1.11.22.tgz#f65d4a75ef829f9da7cd5425d21ddf1d6da5cd46" + integrity sha512-PL0VHbduWPX+ANoyOzr58jBiL2VnD0xGSFwPy7NRZ1Pr6SNWm4jw3x2u6RjLArGhS5EcWp64BSk9ZxqmTV3FEg== + +"@swc/core-win32-arm64-msvc@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-win32-arm64-msvc/-/core-win32-arm64-msvc-1.11.22.tgz#a67dd652d752bd1e6c2244a267c9f179480296fd" + integrity sha512-moJvFhhTVGoMeEThtdF7hQog80Q00CS06v5uB+32VRuv+I31+4WPRyGlTWHO+oY4rReNcXut/mlDHPH7p0LdFg== + +"@swc/core-win32-ia32-msvc@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-win32-ia32-msvc/-/core-win32-ia32-msvc-1.11.22.tgz#3cd02e4d1b759fa35820a58a45cd55f6ebe43ac3" + integrity sha512-/jnsPJJz89F1aKHIb5ScHkwyzBciz2AjEq2m9tDvQdIdVufdJ4SpEDEN9FqsRNRLcBHjtbLs6bnboA+B+pRFXw== + +"@swc/core-win32-x64-msvc@1.11.22": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core-win32-x64-msvc/-/core-win32-x64-msvc-1.11.22.tgz#971610deb596d420042592d76791273e3921628f" + integrity sha512-lc93Y8Mku7LCFGqIxJ91coXZp2HeoDcFZSHCL90Wttg5xhk5xVM9uUCP+OdQsSsEixLF34h5DbT9ObzP8rAdRw== + +"@swc/core@^1.10.16": + version "1.11.22" + resolved "https://registry.yarnpkg.com/@swc/core/-/core-1.11.22.tgz#3d77cf84a8cbbe6903f1fe4aa988c50b46b27d82" + integrity sha512-mjPYbqq8XjwqSE0hEPT9CzaJDyxql97LgK4iyvYlwVSQhdN1uK0DBG4eP9PxYzCS2MUGAXB34WFLegdUj5HGpg== + dependencies: + "@swc/counter" "^0.1.3" + "@swc/types" "^0.1.21" + optionalDependencies: + "@swc/core-darwin-arm64" "1.11.22" + "@swc/core-darwin-x64" "1.11.22" + "@swc/core-linux-arm-gnueabihf" "1.11.22" + "@swc/core-linux-arm64-gnu" "1.11.22" + "@swc/core-linux-arm64-musl" "1.11.22" + "@swc/core-linux-x64-gnu" "1.11.22" + "@swc/core-linux-x64-musl" "1.11.22" + "@swc/core-win32-arm64-msvc" "1.11.22" + "@swc/core-win32-ia32-msvc" "1.11.22" + "@swc/core-win32-x64-msvc" "1.11.22" + +"@swc/counter@^0.1.3": + version "0.1.3" + resolved "https://registry.yarnpkg.com/@swc/counter/-/counter-0.1.3.tgz#cc7463bd02949611c6329596fccd2b0ec782b0e9" + integrity sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ== + +"@swc/types@^0.1.21": + version "0.1.21" + resolved "https://registry.yarnpkg.com/@swc/types/-/types-0.1.21.tgz#6fcadbeca1d8bc89e1ab3de4948cef12344a38c0" + integrity sha512-2YEtj5HJVbKivud9N4bpPBAyZhj4S2Ipe5LkUG94alTpr7in/GU/EARgPAd3BwU+YOmFVJC2+kjqhGRi3r0ZpQ== + dependencies: + "@swc/counter" "^0.1.3" + +"@szmarczak/http-timer@^4.0.5": + version "4.0.6" + resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-4.0.6.tgz#b4a914bb62e7c272d4e5989fe4440f812ab1d807" + integrity sha512-4BAffykYOgO+5nzBWYwE3W90sBgLJoUPRWWcL8wlyiM8IB8ipJz3UMJ9KXQd1RKQXpKp8Tutn80HZtWsu2u76w== + dependencies: + defer-to-connect "^2.0.0" + +"@tanstack/react-virtual@^3.0.0-beta.60": + version "3.13.6" + resolved "https://registry.yarnpkg.com/@tanstack/react-virtual/-/react-virtual-3.13.6.tgz#30243c8c3166673caf66bfbf5352e1b314a3a4cd" + integrity sha512-WT7nWs8ximoQ0CDx/ngoFP7HbQF9Q2wQe4nh2NB+u2486eX3nZRE40P9g6ccCVq7ZfTSH5gFOuCoVH5DLNS/aA== + dependencies: + "@tanstack/virtual-core" "3.13.6" + +"@tanstack/virtual-core@3.13.6": + version "3.13.6" + resolved "https://registry.yarnpkg.com/@tanstack/virtual-core/-/virtual-core-3.13.6.tgz#329f962f1596b3280736c266a982897ed2112157" + integrity sha512-cnQUeWnhNP8tJ4WsGcYiX24Gjkc9ALstLbHcBj1t3E7EimN6n6kHH+DPV4PpDnuw00NApQp+ViojMj1GRdwYQg== + +"@testing-library/dom@^10.0.0": + version "10.4.0" + resolved "https://registry.yarnpkg.com/@testing-library/dom/-/dom-10.4.0.tgz#82a9d9462f11d240ecadbf406607c6ceeeff43a8" + integrity sha512-pemlzrSESWbdAloYml3bAJMEfNh1Z7EduzqPKprCH5S341frlpYnUEW0H72dLxa6IsYr+mPno20GiSm+h9dEdQ== + dependencies: + "@babel/code-frame" "^7.10.4" + "@babel/runtime" "^7.12.5" + "@types/aria-query" "^5.0.1" + aria-query "5.3.0" + chalk "^4.1.0" + dom-accessibility-api "^0.5.9" + lz-string "^1.5.0" + pretty-format "^27.0.2" + +"@testing-library/jest-dom@^5.14.1": + version "5.17.0" + resolved "https://registry.yarnpkg.com/@testing-library/jest-dom/-/jest-dom-5.17.0.tgz#5e97c8f9a15ccf4656da00fecab505728de81e0c" + integrity sha512-ynmNeT7asXyH3aSVv4vvX4Rb+0qjOhdNHnO/3vuZNqPmhDpV/+rCSGwQ7bLcmU2cJ4dvoheIO85LQj0IbJHEtg== + dependencies: + "@adobe/css-tools" "^4.0.1" + "@babel/runtime" "^7.9.2" + "@types/testing-library__jest-dom" "^5.9.1" + 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" + +"@testing-library/react@^15.0.2": + version "15.0.7" + resolved "https://registry.yarnpkg.com/@testing-library/react/-/react-15.0.7.tgz#ff733ce0893c875cb5a47672e8e772897128f4ae" + integrity sha512-cg0RvEdD1TIhhkm1IeYMQxrzy0MtUNfa3minv4MjbgcYzJAZ7yD0i0lwoPOTPr+INtiXFezt2o8xMSnyHhEn2Q== + dependencies: + "@babel/runtime" "^7.12.5" + "@testing-library/dom" "^10.0.0" + "@types/react-dom" "^18.0.0" + +"@tootallnate/once@2": + version "2.0.0" + resolved "https://registry.yarnpkg.com/@tootallnate/once/-/once-2.0.0.tgz#f544a148d3ab35801c1f633a7441fd87c2e484bf" + integrity sha512-XCuKFP5PS55gnMVu3dty8KPatLqUoy/ZYzDzAGCQ8JNFCkLXzmI7vNHCR+XpbZaMWQK/vQubr7PkYq8g470J/A== + +"@ts-stack/markdown@^1.5.0": + version "1.5.0" + resolved "https://registry.yarnpkg.com/@ts-stack/markdown/-/markdown-1.5.0.tgz#5dc298a20dc3dc040143c5a5948201eb6bf5419d" + integrity sha512-ntVX2Kmb2jyTdH94plJohokvDVPvp6CwXHqsa9NVZTK8cOmHDCYNW0j6thIadUVRTStJhxhfdeovLd0owqDxLw== + dependencies: + tslib "^2.3.0" + +"@tsconfig/node10@^1.0.7": + version "1.0.11" + resolved "https://registry.yarnpkg.com/@tsconfig/node10/-/node10-1.0.11.tgz#6ee46400685f130e278128c7b38b7e031ff5b2f2" + integrity sha512-DcRjDCujK/kCk/cUe8Xz8ZSpm8mS3mNNpta+jGCA6USEDfktlNvm1+IuZ9eTcDbNk41BHwpHHeW+N1lKCz4zOw== + +"@tsconfig/node12@^1.0.7": + version "1.0.11" + resolved "https://registry.yarnpkg.com/@tsconfig/node12/-/node12-1.0.11.tgz#ee3def1f27d9ed66dac6e46a295cffb0152e058d" + integrity sha512-cqefuRsh12pWyGsIoBKJA9luFu3mRxCA+ORZvA4ktLSzIuCUtWVxGIuXigEwO5/ywWFMZ2QEGKWvkZG1zDMTag== + +"@tsconfig/node14@^1.0.0": + version "1.0.3" + resolved "https://registry.yarnpkg.com/@tsconfig/node14/-/node14-1.0.3.tgz#e4386316284f00b98435bf40f72f75a09dabf6c1" + integrity sha512-ysT8mhdixWK6Hw3i1V2AeRqZ5WfXg1G43mqoYlM2nc6388Fq5jcXyr5mRsqViLx/GJYdoL0bfXD8nmF+Zn/Iow== + +"@tsconfig/node16@^1.0.2": + version "1.0.4" + resolved "https://registry.yarnpkg.com/@tsconfig/node16/-/node16-1.0.4.tgz#0b92dcc0cc1c81f6f306a381f28e31b1a56536e9" + integrity sha512-vxhUy4J8lyeyinH7Azl1pdd43GJhZH/tP2weN8TntQblOY+A0XbT8DJk1/oCPuOOyg/Ja757rG0CgHcWC8OfMA== + +"@tsconfig/strictest@^2.0.5": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@tsconfig/strictest/-/strictest-2.0.5.tgz#2cbc67f207ba87fdec2a84ad79b1708cf4edd93b" + integrity sha512-ec4tjL2Rr0pkZ5hww65c+EEPYwxOi4Ryv+0MtjeaSQRJyq322Q27eOQiFbuNgw2hpL4hB1/W/HBGk3VKS43osg== + +"@tweenjs/tween.js@^23.1.1", "@tweenjs/tween.js@~23.1.3": + version "23.1.3" + resolved "https://registry.yarnpkg.com/@tweenjs/tween.js/-/tween.js-23.1.3.tgz#eff0245735c04a928bb19c026b58c2a56460539d" + integrity sha512-vJmvvwFxYuGnF2axRtPYocag6Clbb5YS7kLL+SO/TeVFzHqDIWrNKYtcsPMibjDx9O+bu+psAy9NKfWklassUA== + +"@types/aria-query@^5.0.1": + version "5.0.4" + resolved "https://registry.yarnpkg.com/@types/aria-query/-/aria-query-5.0.4.tgz#1a31c3d378850d2778dabb6374d036dcba4ba708" + integrity sha512-rfT93uj5s0PRL7EzccGMs3brplhcrghnDoV26NqKhCAS1hVo+WdNsPvE/yb6ilfr5hi2MEk6d5EWJTKdxg8jVw== + +"@types/babel__core@^7.20.5": + version "7.20.5" + resolved "https://registry.yarnpkg.com/@types/babel__core/-/babel__core-7.20.5.tgz#3df15f27ba85319caa07ba08d0721889bb39c017" + integrity sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA== + dependencies: + "@babel/parser" "^7.20.7" + "@babel/types" "^7.20.7" + "@types/babel__generator" "*" + "@types/babel__template" "*" + "@types/babel__traverse" "*" + +"@types/babel__generator@*": + version "7.27.0" + resolved "https://registry.yarnpkg.com/@types/babel__generator/-/babel__generator-7.27.0.tgz#b5819294c51179957afaec341442f9341e4108a9" + integrity sha512-ufFd2Xi92OAVPYsy+P4n7/U7e68fex0+Ee8gSG9KX7eo084CWiQ4sdxktvdl0bOPupXtVJPY19zk6EwWqUQ8lg== + dependencies: + "@babel/types" "^7.0.0" + +"@types/babel__template@*": + version "7.4.4" + resolved "https://registry.yarnpkg.com/@types/babel__template/-/babel__template-7.4.4.tgz#5672513701c1b2199bc6dad636a9d7491586766f" + integrity sha512-h/NUaSyG5EyxBIp8YRxo4RMe2/qQgvyowRwVMzhYhBCONbW8PUsg4lkFMrhgZhUe5z3L3MiLDuvyJ/CaPa2A8A== + dependencies: + "@babel/parser" "^7.1.0" + "@babel/types" "^7.0.0" + +"@types/babel__traverse@*": + version "7.20.7" + resolved "https://registry.yarnpkg.com/@types/babel__traverse/-/babel__traverse-7.20.7.tgz#968cdc2366ec3da159f61166428ee40f370e56c2" + integrity sha512-dkO5fhS7+/oos4ciWxyEyjWe48zmG6wbCheo/G2ZnHx4fs3EU6YC6UM8rk56gAjNJ9P3MTH2jo5jb92/K6wbng== + dependencies: + "@babel/types" "^7.20.7" + +"@types/cacheable-request@^6.0.1": + version "6.0.3" + resolved "https://registry.yarnpkg.com/@types/cacheable-request/-/cacheable-request-6.0.3.tgz#a430b3260466ca7b5ca5bfd735693b36e7a9d183" + integrity sha512-IQ3EbTzGxIigb1I3qPZc1rWJnH0BmSKv5QYTalEwweFvyBDLSAe24zP0le/hyi7ecGfZVlIVAg4BZqb8WBwKqw== + dependencies: + "@types/http-cache-semantics" "*" + "@types/keyv" "^3.1.4" + "@types/node" "*" + "@types/responselike" "^1.0.0" + +"@types/debug@^4.1.6": + version "4.1.12" + resolved "https://registry.yarnpkg.com/@types/debug/-/debug-4.1.12.tgz#a155f21690871953410df4b6b6f53187f0500917" + integrity sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ== + dependencies: + "@types/ms" "*" + +"@types/diff@^7.0.2": + version "7.0.2" + resolved "https://registry.yarnpkg.com/@types/diff/-/diff-7.0.2.tgz#d638edebf3c97aa4962b6f1164a7921ab3de9f83" + integrity sha512-JSWRMozjFKsGlEjiiKajUjIJVKuKdE3oVy2DNtK+fUo8q82nhFZ2CPQwicAIkXrofahDXrWJ7mjelvZphMS98Q== + +"@types/electron@^1.6.10": + version "1.6.12" + resolved "https://registry.yarnpkg.com/@types/electron/-/electron-1.6.12.tgz#0921813f95b92fdce9f642f85cb561318fcf0a04" + integrity sha512-NIJokDkGv9h+MStCL1IuiL1FOHYVkszoWeNxJtSI5dcEKRGbX83JcVYNAgk019qOQgJkHtz9WdP0CDXvrArrGg== + dependencies: + electron "*" + +"@types/eslint@*": + version "9.6.1" + resolved "https://registry.yarnpkg.com/@types/eslint/-/eslint-9.6.1.tgz#d5795ad732ce81715f27f75da913004a56751584" + integrity sha512-FXx2pKgId/WyYo2jXw63kk7/+TY7u7AziEJxJAnSFzHlqTAS3Ync6SvgYAN/k4/PQpnnVuzoMuVnByKK2qp0ag== + dependencies: + "@types/estree" "*" + "@types/json-schema" "*" + +"@types/estree@*", "@types/estree@1.0.7", "@types/estree@^1.0.0": + version "1.0.7" + resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.7.tgz#4158d3105276773d5b7695cd4834b1722e4f37a8" + integrity sha512-w28IoSUCJpidD/TGviZwwMJckNESJZXFu7NBZ5YJ4mEUnNraUn9Pm8HSZm/jDF1pDWYKspWE7oVphigUPRakIQ== + +"@types/fs-extra@9.0.13", "@types/fs-extra@^9.0.11": + version "9.0.13" + resolved "https://registry.yarnpkg.com/@types/fs-extra/-/fs-extra-9.0.13.tgz#7594fbae04fe7f1918ce8b3d213f74ff44ac1f45" + integrity sha512-nEnwB++1u5lVDM2UI4c1+5R+FYaKfaAzS4OococimjVm3nQw3TuzH5UNsocrcTBbhnerblyHj4A49qXbIiZdpA== + dependencies: + "@types/node" "*" + +"@types/glob@^8.1.0": + version "8.1.0" + resolved "https://registry.yarnpkg.com/@types/glob/-/glob-8.1.0.tgz#b63e70155391b0584dce44e7ea25190bbc38f2fc" + integrity sha512-IO+MJPVhoqz+28h1qLAcBEH2+xHMK6MTyHJc7MTnnYb6wsoLR29POVGJ7LycmVXIqyy/4/2ShP5sUwTXuOwb/w== + dependencies: + "@types/minimatch" "^5.1.2" + "@types/node" "*" + +"@types/http-cache-semantics@*": + version "4.0.4" + resolved "https://registry.yarnpkg.com/@types/http-cache-semantics/-/http-cache-semantics-4.0.4.tgz#b979ebad3919799c979b17c72621c0bc0a31c6c4" + integrity sha512-1m0bIFVc7eJWyve9S0RnuRgcQqF/Xd5QsUZAZeQFr1Q3/p9JWoQQEqmVy+DPTNpGXwhgIetAoYF8JSc33q29QA== + +"@types/isomorphic-fetch@^0.0.39": + version "0.0.39" + resolved "https://registry.yarnpkg.com/@types/isomorphic-fetch/-/isomorphic-fetch-0.0.39.tgz#889573a72ca637bc1a665910a41ff1cb3b52011f" + integrity sha512-I0gou/ZdA1vMG7t7gMzL7VYu2xAKU78rW9U1l10MI0nn77pEHq3tQqHQ8hMmXdMpBlkxZOorjI4sO594Z3kKJw== + +"@types/istanbul-lib-coverage@*", "@types/istanbul-lib-coverage@^2.0.0": + version "2.0.6" + resolved "https://registry.yarnpkg.com/@types/istanbul-lib-coverage/-/istanbul-lib-coverage-2.0.6.tgz#7739c232a1fee9b4d3ce8985f314c0c6d33549d7" + integrity sha512-2QF/t/auWm0lsy8XtKVPG19v3sSOQlJe/YHZgfjb/KBBHOGSV+J2q/S671rcq9uTBrLAXmZpqJiaQbMT+zNU1w== + +"@types/istanbul-lib-report@*": + version "3.0.3" + resolved "https://registry.yarnpkg.com/@types/istanbul-lib-report/-/istanbul-lib-report-3.0.3.tgz#53047614ae72e19fc0401d872de3ae2b4ce350bf" + integrity sha512-NQn7AHQnk/RSLOxrBbGyJM/aVQ+pjj5HCgasFxc0K/KhoATfQ/47AyUl15I2yBUpihjmas+a+VJBOqecrFH+uA== + dependencies: + "@types/istanbul-lib-coverage" "*" + +"@types/istanbul-reports@^3.0.0": + version "3.0.4" + resolved "https://registry.yarnpkg.com/@types/istanbul-reports/-/istanbul-reports-3.0.4.tgz#0f03e3d2f670fbdac586e34b433783070cc16f54" + integrity sha512-pk2B1NWalF9toCRu6gjBzR69syFjP4Od8WRAX+0mmf9lAjCRicLOWc+ZrxZHx/0XRjotgkF9t6iaMJ+aXcOdZQ== + dependencies: + "@types/istanbul-lib-report" "*" + +"@types/jest@*": + version "29.5.14" + resolved "https://registry.yarnpkg.com/@types/jest/-/jest-29.5.14.tgz#2b910912fa1d6856cadcd0c1f95af7df1d6049e5" + integrity sha512-ZN+4sdnLUbo8EVvVc2ao0GFW6oVrQRPn4K2lglySj7APvSrgzxHiNNK99us4WDMi57xxA2yggblIAMNhXOotLQ== + dependencies: + expect "^29.0.0" + pretty-format "^29.0.0" + +"@types/json-schema@*": + version "7.0.15" + resolved "https://registry.yarnpkg.com/@types/json-schema/-/json-schema-7.0.15.tgz#596a1747233694d50f6ad8a7869fcb6f56cf5841" + integrity sha512-5+fP8P8MFNC+AyZCDxrB2pkZFPGzqQWUzpSeuuVLvm8VMcorNYavBqoFcxK8bQz4Qsbn4oUEEem4wDLfcysGHA== + +"@types/json5@^0.0.29": + version "0.0.29" + resolved "https://registry.yarnpkg.com/@types/json5/-/json5-0.0.29.tgz#ee28707ae94e11d2b827bcbe5270bcea7f3e71ee" + integrity sha512-dRLjCWHYg4oaA77cxO64oO+7JwCwnIzkZPdrrC71jQmQtlhM556pwKo5bUzqvZndkVbeFLIIi+9TC40JNF5hNQ== + +"@types/keyv@^3.1.4": + version "3.1.4" + resolved "https://registry.yarnpkg.com/@types/keyv/-/keyv-3.1.4.tgz#3ccdb1c6751b0c7e52300bcdacd5bcbf8faa75b6" + integrity sha512-BQ5aZNSCpj7D6K2ksrRCTmKRLEpnPvWDiLPfoGyhZ++8YtiK9d/3DBKPJgry359X/P1PfruyYwvnvwFjuEiEIg== + dependencies: + "@types/node" "*" + +"@types/minimatch@^5.1.2": + version "5.1.2" + resolved "https://registry.yarnpkg.com/@types/minimatch/-/minimatch-5.1.2.tgz#07508b45797cb81ec3f273011b054cd0755eddca" + integrity sha512-K0VQKziLUWkVKiRVrx4a40iPaxTUefQmjtkQofBkYRcoaaL/8rhwDWww9qWbrgicNOgnpIsMxyNIUM4+n6dUIA== + +"@types/minimist@^1.2.5": + version "1.2.5" + resolved "https://registry.yarnpkg.com/@types/minimist/-/minimist-1.2.5.tgz#ec10755e871497bcd83efe927e43ec46e8c0747e" + integrity sha512-hov8bUuiLiyFPGyFPE1lwWhmzYbirOXQNNo40+y3zow8aFVTeyn3VWL0VFFfdNddA8S4Vf0Tc062rzyNr7Paag== + +"@types/mocha@^10.0.10": + version "10.0.10" + resolved "https://registry.yarnpkg.com/@types/mocha/-/mocha-10.0.10.tgz#91f62905e8d23cbd66225312f239454a23bebfa0" + integrity sha512-xPyYSz1cMPnJQhl0CLMH68j3gprKZaTjG3s5Vi+fDgx+uhG9NOXwbVt52eFS8ECyXhyKcjDLCBEqBExKuiZb7Q== + +"@types/ms@*": + version "2.1.0" + resolved "https://registry.yarnpkg.com/@types/ms/-/ms-2.1.0.tgz#052aa67a48eccc4309d7f0191b7e41434b90bb78" + integrity sha512-GsCCIZDE/p3i96vtEqx+7dBUGXrc7zeSK3wwPHIaRThS+9OhWIXRqzs4d6k1SVU8g91DrNRWxWUGhp5KXQb2VA== + +"@types/node@*", "@types/node@^22.14.1", "@types/node@^22.7.7": + version "22.14.1" + resolved "https://registry.yarnpkg.com/@types/node/-/node-22.14.1.tgz#53b54585cec81c21eee3697521e31312d6ca1e6f" + integrity sha512-u0HuPQwe/dHrItgHHpmw3N2fYCR6x4ivMNbPHRkBVP4CvN+kiRrKHWk3i8tXiO/joPwXLMYvF9TTF0eqgHIuOw== + dependencies: + undici-types "~6.21.0" + +"@types/node@^20.9.0": + version "20.17.30" + resolved "https://registry.yarnpkg.com/@types/node/-/node-20.17.30.tgz#1d93f656d3b869dbef7b796568ac457606ba58d0" + integrity sha512-7zf4YyHA+jvBNfVrk2Gtvs6x7E8V+YDW05bNfG2XkWDJfYRXrTiP/DsB2zSYTaHX0bGIujTBQdMVAhb+j7mwpg== + dependencies: + undici-types "~6.19.2" + +"@types/pixelmatch@^5.2.6": + version "5.2.6" + resolved "https://registry.yarnpkg.com/@types/pixelmatch/-/pixelmatch-5.2.6.tgz#fba6de304ac958495f27d85989f5c6bb7499a686" + integrity sha512-wC83uexE5KGuUODn6zkm9gMzTwdY5L0chiK+VrKcDfEjzxh1uadlWTvOmAbCpnM9zx/Ww3f8uKlYQVnO/TrqVg== + dependencies: + "@types/node" "*" + +"@types/plist@^3.0.1": + version "3.0.5" + resolved "https://registry.yarnpkg.com/@types/plist/-/plist-3.0.5.tgz#9a0c49c0f9886c8c8696a7904dd703f6284036e0" + integrity sha512-E6OCaRmAe4WDmWNsL/9RMqdkkzDCY1etutkflWk4c+AcjDU07Pcz1fQwTX0TQz+Pxqn9i4L1TU3UFpjnrcDgxA== + dependencies: + "@types/node" "*" + xmlbuilder ">=11.0.1" + +"@types/pngjs@^6.0.4": + version "6.0.5" + resolved "https://registry.yarnpkg.com/@types/pngjs/-/pngjs-6.0.5.tgz#6dec2f7eb8284543ca4e423f3c09b119fa939ea3" + integrity sha512-0k5eKfrA83JOZPppLtS2C7OUtyNAl2wKNxfyYl9Q5g9lPkgBl/9hNyAu6HuEH2J4XmIv2znEpkDd0SaZVxW6iQ== + dependencies: + "@types/node" "*" + +"@types/prop-types@*": + version "15.7.14" + resolved "https://registry.yarnpkg.com/@types/prop-types/-/prop-types-15.7.14.tgz#1433419d73b2a7ebfc6918dcefd2ec0d5cd698f2" + integrity sha512-gNMvNH49DJ7OJYv+KAKn0Xp45p8PLl6zo2YnvDIbTd4J6MER2BmWN49TG7n9LvkyihINxeKW8+3bfS2yDC9dzQ== + +"@types/react-dom@^18.0.0", "@types/react-dom@^18.3.1": + version "18.3.6" + resolved "https://registry.yarnpkg.com/@types/react-dom/-/react-dom-18.3.6.tgz#fa59a5e9a33499a792af6c1130f55921ef49d268" + integrity sha512-nf22//wEbKXusP6E9pfOCDwFdHAX4u172eaJI4YkDRQEZiorm6KfYnSC2SWLDMVWUOWPERmJnN0ujeAfTBLvrw== + +"@types/react-modal@^3.16.3": + version "3.16.3" + resolved "https://registry.yarnpkg.com/@types/react-modal/-/react-modal-3.16.3.tgz#250f32c07f1de28e2bcf9c3e84b56adaa6897013" + integrity sha512-xXuGavyEGaFQDgBv4UVm8/ZsG+qxeQ7f77yNrW3n+1J6XAstUy5rYHeIHPh1KzsGc6IkCIdu6lQ2xWzu1jBTLg== + dependencies: + "@types/react" "*" + +"@types/react@*": + version "19.1.2" + resolved "https://registry.yarnpkg.com/@types/react/-/react-19.1.2.tgz#11df86f66f188f212c90ecb537327ec68bfd593f" + integrity sha512-oxLPMytKchWGbnQM9O7D67uPa9paTNxO7jVoNMXgkkErULBPhPARCfkKL9ytcIJJRGjbsVwW4ugJzyFFvm/Tiw== + dependencies: + csstype "^3.0.2" + +"@types/react@^18.3.4": + version "18.3.20" + resolved "https://registry.yarnpkg.com/@types/react/-/react-18.3.20.tgz#b0dccda9d2f1bc24d2a04b1d0cb5d0b9a3576ad3" + integrity sha512-IPaCZN7PShZK/3t6Q87pfTkRm6oLTd4vztyoj+cbHUF1g3FfVb2tFIL79uCRKEfv16AhqDMBywP2VW3KIZUvcg== + dependencies: + "@types/prop-types" "*" + csstype "^3.0.2" + +"@types/responselike@^1.0.0": + version "1.0.3" + resolved "https://registry.yarnpkg.com/@types/responselike/-/responselike-1.0.3.tgz#cc29706f0a397cfe6df89debfe4bf5cea159db50" + integrity sha512-H/+L+UkTV33uf49PH5pCAUBVPNj2nDBXTN+qS1dOwyyg24l3CcicicCA7ca+HMvJBZcFgl5r8e+RR6elsb4Lyw== + dependencies: + "@types/node" "*" + +"@types/stack-utils@^2.0.0": + version "2.0.3" + resolved "https://registry.yarnpkg.com/@types/stack-utils/-/stack-utils-2.0.3.tgz#6209321eb2c1712a7e7466422b8cb1fc0d9dd5d8" + integrity sha512-9aEbYZ3TbYMznPdcdr3SmIrLXwC/AKZXQeCf9Pgao5CKb8CyHuEX5jzWPTkvregvhRJHcpRO6BFoGW9ycaOkYw== + +"@types/stats.js@*": + version "0.17.3" + resolved "https://registry.yarnpkg.com/@types/stats.js/-/stats.js-0.17.3.tgz#705446e12ce0fad618557dd88236f51148b7a935" + integrity sha512-pXNfAD3KHOdif9EQXZ9deK82HVNaXP5ZIF5RP2QG6OQFNTaY2YIetfrE9t528vEreGQvEPRDDc8muaoYeK0SxQ== + +"@types/testing-library__jest-dom@^5.9.1": + version "5.14.9" + resolved "https://registry.yarnpkg.com/@types/testing-library__jest-dom/-/testing-library__jest-dom-5.14.9.tgz#0fb1e6a0278d87b6737db55af5967570b67cb466" + integrity sha512-FSYhIjFlfOpGSRyVoMBMuS3ws5ehFQODymf3vlI7U1K8c7PHwWwFY7VREfmsuzHSOnoKs/9/Y983ayOs7eRzqw== + dependencies: + "@types/jest" "*" + +"@types/three@^0.175.0": + version "0.175.0" + resolved "https://registry.yarnpkg.com/@types/three/-/three-0.175.0.tgz#dfb15b3e2d09a16c6e2fd8caeaf5c35f7b6d7e6a" + integrity sha512-ldMSBgtZOZ3g9kJ3kOZSEtZIEITmJOzu8eKVpkhf036GuNkM4mt0NXecrjCn5tMm1OblOF7dZehlaDypBfNokw== + dependencies: + "@tweenjs/tween.js" "~23.1.3" + "@types/stats.js" "*" + "@types/webxr" "*" + "@webgpu/types" "*" + fflate "~0.8.2" + meshoptimizer "~0.18.1" + +"@types/ua-parser-js@^0.7.39": + version "0.7.39" + resolved "https://registry.yarnpkg.com/@types/ua-parser-js/-/ua-parser-js-0.7.39.tgz#832c58e460c9435e4e34bb866e85e9146e12cdbb" + integrity sha512-P/oDfpofrdtF5xw433SPALpdSchtJmY7nsJItf8h3KXqOslkbySh8zq4dSWXH2oTjRvJ5PczVEoCZPow6GicLg== + +"@types/uuid@^9.0.8": + version "9.0.8" + resolved "https://registry.yarnpkg.com/@types/uuid/-/uuid-9.0.8.tgz#7545ba4fc3c003d6c756f651f3bf163d8f0f29ba" + integrity sha512-jg+97EGIcY9AGHJJRaaPVgetKDsrTgbRjQ5Msgjh/DQKEFl0DtyRr/VCOyD1T2R1MNeWPK/u7JoGhlDZnKBAfA== + +"@types/verror@^1.10.3": + version "1.10.11" + resolved "https://registry.yarnpkg.com/@types/verror/-/verror-1.10.11.tgz#d3d6b418978c8aa202d41e5bb3483227b6ecc1bb" + integrity sha512-RlDm9K7+o5stv0Co8i8ZRGxDbrTxhJtgjqjFyVh/tXQyl/rYtTKlnTvZ88oSTeYREWurwx20Js4kTuKCsFkUtg== + +"@types/vscode@^1.97.0": + version "1.99.1" + resolved "https://registry.yarnpkg.com/@types/vscode/-/vscode-1.99.1.tgz#bde6e2d9ccbe0493fded98ad639bf2671b8ec9ee" + integrity sha512-cQlqxHZ040ta6ovZXnXRxs3fJiTmlurkIWOfZVcLSZPcm9J4ikFpXuB7gihofGn5ng+kDVma5EmJIclfk0trPQ== + +"@types/webxr@*": + version "0.5.22" + resolved "https://registry.yarnpkg.com/@types/webxr/-/webxr-0.5.22.tgz#d8a14c12bbfaaa4a13de21ec2d4a8197b3e1b532" + integrity sha512-Vr6Stjv5jPRqH690f5I5GLjVk8GSsoQSYJ2FVd/3jJF7KaqfwPi3ehfBS96mlQ2kPCwZaX6U0rG2+NGHBKkA/A== + +"@types/wicg-file-system-access@^2023.10.6": + version "2023.10.6" + resolved "https://registry.yarnpkg.com/@types/wicg-file-system-access/-/wicg-file-system-access-2023.10.6.tgz#cd8149b14c31b73a53478e656790873945e73ba1" + integrity sha512-YO/183gNRzZFSdKu+ikkD7ambAj4PhgjFAF2A/Mw/7wroSF6ne8r804RkpZzqrJ/F6DO2/IYlQF/ULOZ/bhKyA== + +"@types/ws@^8.18.1": + version "8.18.1" + resolved "https://registry.yarnpkg.com/@types/ws/-/ws-8.18.1.tgz#48464e4bf2ddfd17db13d845467f6070ffea4aa9" + integrity sha512-ThVF6DCVhA8kUGy+aazFQ4kXQ7E1Ty7A3ypFOe0IcJV8O/M511G99AW24irKrW56Wt44yG9+ij8FaqoBGkuBXg== + dependencies: + "@types/node" "*" + +"@types/yargs-parser@*": + version "21.0.3" + resolved "https://registry.yarnpkg.com/@types/yargs-parser/-/yargs-parser-21.0.3.tgz#815e30b786d2e8f0dcd85fd5bcf5e1a04d008f15" + integrity sha512-I4q9QU9MQv4oEOz4tAHJtNz1cwuLxn2F3xcc2iV5WdqLPpUnj30aUuxt1mAxYTG+oe8CZMV/+6rU4S4gRDzqtQ== + +"@types/yargs@^17.0.8": + version "17.0.33" + resolved "https://registry.yarnpkg.com/@types/yargs/-/yargs-17.0.33.tgz#8c32303da83eec050a84b3c7ae7b9f922d13e32d" + integrity sha512-WpxBCKWPLr4xSsHgz511rFJAM+wS28w2zEO1QDNY5zM/S8ok70NNfztH0xwhqKyaK0OHCbN98LDAZuy1ctxDkA== + dependencies: + "@types/yargs-parser" "*" + +"@types/yauzl@^2.9.1": + version "2.10.3" + resolved "https://registry.yarnpkg.com/@types/yauzl/-/yauzl-2.10.3.tgz#e9b2808b4f109504a03cda958259876f61017999" + integrity sha512-oJoftv0LSuaDZE3Le4DbKX+KS9G36NzOeSap90UIK0yMA/NhKJhqlSGtNDORNRaIbQfzjXDrQa0ytJ6mNRGz/Q== + dependencies: + "@types/node" "*" + +"@typescript-eslint/eslint-plugin@8.31.0", "@typescript-eslint/eslint-plugin@^8.27.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/eslint-plugin/-/eslint-plugin-8.31.0.tgz#ef3ece95406a80026f82a19a2984c1e375981711" + integrity sha512-evaQJZ/J/S4wisevDvC1KFZkPzRetH8kYZbkgcTRyql3mcKsf+ZFDV1BVWUGTCAW5pQHoqn5gK5b8kn7ou9aFQ== + dependencies: + "@eslint-community/regexpp" "^4.10.0" + "@typescript-eslint/scope-manager" "8.31.0" + "@typescript-eslint/type-utils" "8.31.0" + "@typescript-eslint/utils" "8.31.0" + "@typescript-eslint/visitor-keys" "8.31.0" + graphemer "^1.4.0" + ignore "^5.3.1" + natural-compare "^1.4.0" + ts-api-utils "^2.0.1" + +"@typescript-eslint/parser@8.31.0", "@typescript-eslint/parser@^8.27.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/parser/-/parser-8.31.0.tgz#5ec28823d06dd20ed5f67b61224823f12ccde095" + integrity sha512-67kYYShjBR0jNI5vsf/c3WG4u+zDnCTHTPqVMQguffaWWFs7artgwKmfwdifl+r6XyM5LYLas/dInj2T0SgJyw== + dependencies: + "@typescript-eslint/scope-manager" "8.31.0" + "@typescript-eslint/types" "8.31.0" + "@typescript-eslint/typescript-estree" "8.31.0" + "@typescript-eslint/visitor-keys" "8.31.0" + debug "^4.3.4" + +"@typescript-eslint/scope-manager@8.31.0", "@typescript-eslint/scope-manager@^8.15.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/scope-manager/-/scope-manager-8.31.0.tgz#48c7f7d729ea038e36cae0ff511e48c2412fb11c" + integrity sha512-knO8UyF78Nt8O/B64i7TlGXod69ko7z6vJD9uhSlm0qkAbGeRUSudcm0+K/4CrRjrpiHfBCjMWlc08Vav1xwcw== + dependencies: + "@typescript-eslint/types" "8.31.0" + "@typescript-eslint/visitor-keys" "8.31.0" + +"@typescript-eslint/type-utils@8.31.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/type-utils/-/type-utils-8.31.0.tgz#01536a993fae23e2def885b006aaa991cbfbe9e7" + integrity sha512-DJ1N1GdjI7IS7uRlzJuEDCgDQix3ZVYVtgeWEyhyn4iaoitpMBX6Ndd488mXSx0xah/cONAkEaYyylDyAeHMHg== + dependencies: + "@typescript-eslint/typescript-estree" "8.31.0" + "@typescript-eslint/utils" "8.31.0" + debug "^4.3.4" + ts-api-utils "^2.0.1" + +"@typescript-eslint/types@8.31.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/types/-/types-8.31.0.tgz#c48e20ec47a43b72747714f49ea9f7b38a4fa6c1" + integrity sha512-Ch8oSjVyYyJxPQk8pMiP2FFGYatqXQfQIaMp+TpuuLlDachRWpUAeEu1u9B/v/8LToehUIWyiKcA/w5hUFRKuQ== + +"@typescript-eslint/typescript-estree@8.31.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/typescript-estree/-/typescript-estree-8.31.0.tgz#9c7f84eff6ad23d63cf086c6e93af571cd561270" + integrity sha512-xLmgn4Yl46xi6aDSZ9KkyfhhtnYI15/CvHbpOy/eR5NWhK/BK8wc709KKwhAR0m4ZKRP7h07bm4BWUYOCuRpQQ== + dependencies: + "@typescript-eslint/types" "8.31.0" + "@typescript-eslint/visitor-keys" "8.31.0" + debug "^4.3.4" + fast-glob "^3.3.2" + is-glob "^4.0.3" + minimatch "^9.0.4" + semver "^7.6.0" + ts-api-utils "^2.0.1" + +"@typescript-eslint/utils@8.31.0", "@typescript-eslint/utils@^6.0.0 || ^7.0.0 || ^8.0.0", "@typescript-eslint/utils@^8.15.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/utils/-/utils-8.31.0.tgz#6fb52471a29fdd16fc253d568c5ad4b048f78ba4" + integrity sha512-qi6uPLt9cjTFxAb1zGNgTob4x9ur7xC6mHQJ8GwEzGMGE9tYniublmJaowOJ9V2jUzxrltTPfdG2nKlWsq0+Ww== + dependencies: + "@eslint-community/eslint-utils" "^4.4.0" + "@typescript-eslint/scope-manager" "8.31.0" + "@typescript-eslint/types" "8.31.0" + "@typescript-eslint/typescript-estree" "8.31.0" + +"@typescript-eslint/visitor-keys@8.31.0": + version "8.31.0" + resolved "https://registry.yarnpkg.com/@typescript-eslint/visitor-keys/-/visitor-keys-8.31.0.tgz#9a1a97ed16c60d4d1e7399b41c11a6d94ebc1ce5" + integrity sha512-QcGHmlRHWOl93o64ZUMNewCdwKGU6WItOU52H0djgNmn1EOrhVudrDzXz4OycCRSCPwFCDrE2iIt5vmuUdHxuQ== + dependencies: + "@typescript-eslint/types" "8.31.0" + eslint-visitor-keys "^4.2.0" + +"@ungap/structured-clone@^1.2.0": + version "1.3.0" + resolved "https://registry.yarnpkg.com/@ungap/structured-clone/-/structured-clone-1.3.0.tgz#d06bbb384ebcf6c505fde1c3d0ed4ddffe0aaff8" + integrity sha512-WmoN8qaIAo7WTYWbAZuG8PYEhn5fkz7dZrqTBZ7dtt//lL2Gwms1IcnQ5yHqjDfX8Ft5j4YzDM23f87zBfDe9g== + +"@vitejs/plugin-react@^4.4.1": + version "4.4.1" + resolved "https://registry.yarnpkg.com/@vitejs/plugin-react/-/plugin-react-4.4.1.tgz#d7d1e9c9616d7536b0953637edfee7c6cbe2fe0f" + integrity sha512-IpEm5ZmeXAP/osiBXVVP5KjFMzbWOonMs0NaQQl+xYnUAcq4oHUBsF2+p4MgKWG4YMmFYJU8A6sxRPuowllm6w== + dependencies: + "@babel/core" "^7.26.10" + "@babel/plugin-transform-react-jsx-self" "^7.25.9" + "@babel/plugin-transform-react-jsx-source" "^7.25.9" + "@types/babel__core" "^7.20.5" + react-refresh "^0.17.0" + +"@vitest/expect@1.6.1": + version "1.6.1" + resolved "https://registry.yarnpkg.com/@vitest/expect/-/expect-1.6.1.tgz#b90c213f587514a99ac0bf84f88cff9042b0f14d" + integrity sha512-jXL+9+ZNIJKruofqXuuTClf44eSpcHlgj3CiuNihUF3Ioujtmc0zIa3UJOW5RjDK1YLBJZnWBlPuqhYycLioog== + dependencies: + "@vitest/spy" "1.6.1" + "@vitest/utils" "1.6.1" + chai "^4.3.10" + +"@vitest/expect@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vitest/expect/-/expect-3.1.2.tgz#b203a7ad2efa6af96c85f6c116216bda259d2bc8" + integrity sha512-O8hJgr+zREopCAqWl3uCVaOdqJwZ9qaDwUP7vy3Xigad0phZe9APxKhPcDNqYYi0rX5oMvwJMSCAXY2afqeTSA== + dependencies: + "@vitest/spy" "3.1.2" + "@vitest/utils" "3.1.2" + chai "^5.2.0" + tinyrainbow "^2.0.0" + +"@vitest/mocker@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vitest/mocker/-/mocker-3.1.2.tgz#1ff239036072feb543ab56825ada09b12a075af2" + integrity sha512-kOtd6K2lc7SQ0mBqYv/wdGedlqPdM/B38paPY+OwJ1XiNi44w3Fpog82UfOibmHaV9Wod18A09I9SCKLyDMqgw== + dependencies: + "@vitest/spy" "3.1.2" + estree-walker "^3.0.3" + magic-string "^0.30.17" + +"@vitest/pretty-format@3.1.2", "@vitest/pretty-format@^3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vitest/pretty-format/-/pretty-format-3.1.2.tgz#689b0604c0b73fdccb144f11b64d70c9233b23b8" + integrity sha512-R0xAiHuWeDjTSB3kQ3OQpT8Rx3yhdOAIm/JM4axXxnG7Q/fS8XUwggv/A4xzbQA+drYRjzkMnpYnOGAc4oeq8w== + dependencies: + tinyrainbow "^2.0.0" + +"@vitest/runner@1.6.1": + version "1.6.1" + resolved "https://registry.yarnpkg.com/@vitest/runner/-/runner-1.6.1.tgz#10f5857c3e376218d58c2bfacfea1161e27e117f" + integrity sha512-3nSnYXkVkf3mXFfE7vVyPmi3Sazhb/2cfZGGs0JRzFsPFvAMBEcrweV1V1GsrstdXeKCTXlJbvnQwGWgEIHmOA== + dependencies: + "@vitest/utils" "1.6.1" + p-limit "^5.0.0" + pathe "^1.1.1" + +"@vitest/runner@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vitest/runner/-/runner-3.1.2.tgz#ffeba74618046221e944e94f09b565af772170cf" + integrity sha512-bhLib9l4xb4sUMPXnThbnhX2Yi8OutBMA8Yahxa7yavQsFDtwY/jrUZwpKp2XH9DhRFJIeytlyGpXCqZ65nR+g== + dependencies: + "@vitest/utils" "3.1.2" + pathe "^2.0.3" + +"@vitest/snapshot@1.6.1": + version "1.6.1" + resolved "https://registry.yarnpkg.com/@vitest/snapshot/-/snapshot-1.6.1.tgz#90414451a634bb36cd539ccb29ae0d048a8c0479" + integrity sha512-WvidQuWAzU2p95u8GAKlRMqMyN1yOJkGHnx3M1PL9Raf7AQ1kwLKg04ADlCa3+OXUZE7BceOhVZiuWAbzCKcUQ== + dependencies: + magic-string "^0.30.5" + pathe "^1.1.1" + pretty-format "^29.7.0" + +"@vitest/snapshot@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vitest/snapshot/-/snapshot-3.1.2.tgz#46c52a417afbf1fe94fba0a5735cbedf9cfc60f6" + integrity sha512-Q1qkpazSF/p4ApZg1vfZSQ5Yw6OCQxVMVrLjslbLFA1hMDrT2uxtqMaw8Tc/jy5DLka1sNs1Y7rBcftMiaSH/Q== + dependencies: + "@vitest/pretty-format" "3.1.2" + magic-string "^0.30.17" + pathe "^2.0.3" + +"@vitest/spy@1.6.1": + version "1.6.1" + resolved "https://registry.yarnpkg.com/@vitest/spy/-/spy-1.6.1.tgz#33376be38a5ed1ecd829eb986edaecc3e798c95d" + integrity sha512-MGcMmpGkZebsMZhbQKkAf9CX5zGvjkBTqf8Zx3ApYWXr3wG+QvEu2eXWfnIIWYSJExIp4V9FCKDEeygzkYrXMw== + dependencies: + tinyspy "^2.2.0" + +"@vitest/spy@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vitest/spy/-/spy-3.1.2.tgz#3a5be04d71c4a458c8d6859503626e2aed61bcbf" + integrity sha512-OEc5fSXMws6sHVe4kOFyDSj/+4MSwst0ib4un0DlcYgQvRuYQ0+M2HyqGaauUMnjq87tmUaMNDxKQx7wNfVqPA== + dependencies: + tinyspy "^3.0.2" + +"@vitest/utils@1.6.1": + version "1.6.1" + resolved "https://registry.yarnpkg.com/@vitest/utils/-/utils-1.6.1.tgz#6d2f36cb6d866f2bbf59da854a324d6bf8040f17" + integrity sha512-jOrrUvXM4Av9ZWiG1EajNto0u96kWAhJ1LmPmJhXXQx/32MecEKd10pOLYgS2BQx1TgkGhloPU1ArDW2vvaY6g== + dependencies: + diff-sequences "^29.6.3" + estree-walker "^3.0.3" + loupe "^2.3.7" + pretty-format "^29.7.0" + +"@vitest/utils@3.1.2": + version "3.1.2" + resolved "https://registry.yarnpkg.com/@vitest/utils/-/utils-3.1.2.tgz#f3ae55b3a205c88c346a2a8dcde7c89210364932" + integrity sha512-5GGd0ytZ7BH3H6JTj9Kw7Prn1Nbg0wZVrIvou+UWxm54d+WoXXgAgjFJ8wn3LdagWLFSEfpPeyYrByZaGEZHLg== + dependencies: + "@vitest/pretty-format" "3.1.2" + loupe "^3.1.3" + tinyrainbow "^2.0.0" + +"@vitest/web-worker@^1.5.0": + version "1.6.1" + resolved "https://registry.yarnpkg.com/@vitest/web-worker/-/web-worker-1.6.1.tgz#86c9d43736997daef035679ffef745fea063b03e" + integrity sha512-T3zLS/sga4z8ZqgA+iDKyiKOcL7OGG3EJ8xqXEEDuvtQELxCITehwPbDe7cGKQ+YspFsLqdNC9td30uQ1uStFg== + dependencies: + debug "^4.3.4" + +"@vscode/test-electron@^2.4.1": + version "2.5.2" + resolved "https://registry.yarnpkg.com/@vscode/test-electron/-/test-electron-2.5.2.tgz#f7d4078e8230ce9c94322f2a29cc16c17954085d" + integrity sha512-8ukpxv4wYe0iWMRQU18jhzJOHkeGKbnw7xWRX3Zw1WJA4cEKbHcmmLPdPrPtL6rhDcrlCZN+xKRpv09n4gRHYg== + dependencies: + http-proxy-agent "^7.0.2" + https-proxy-agent "^7.0.5" + jszip "^3.10.1" + ora "^8.1.0" + semver "^7.6.2" + +"@vscode/vsce-sign-alpine-arm64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-alpine-arm64/-/vsce-sign-alpine-arm64-2.0.2.tgz#4accc485e55aa6ff04b195b47f722ead57daa58e" + integrity sha512-E80YvqhtZCLUv3YAf9+tIbbqoinWLCO/B3j03yQPbjT3ZIHCliKZlsy1peNc4XNZ5uIb87Jn0HWx/ZbPXviuAQ== + +"@vscode/vsce-sign-alpine-x64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-alpine-x64/-/vsce-sign-alpine-x64-2.0.2.tgz#4a4b7b505b4cc0f58596394897c49a0bce0e540c" + integrity sha512-n1WC15MSMvTaeJ5KjWCzo0nzjydwxLyoHiMJHu1Ov0VWTZiddasmOQHekA47tFRycnt4FsQrlkSCTdgHppn6bw== + +"@vscode/vsce-sign-darwin-arm64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-darwin-arm64/-/vsce-sign-darwin-arm64-2.0.2.tgz#10aa69feb7f81a3dc68c242038ca03eaff19c12e" + integrity sha512-rz8F4pMcxPj8fjKAJIfkUT8ycG9CjIp888VY/6pq6cuI2qEzQ0+b5p3xb74CJnBbSC0p2eRVoe+WgNCAxCLtzQ== + +"@vscode/vsce-sign-darwin-x64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-darwin-x64/-/vsce-sign-darwin-x64-2.0.2.tgz#3315528f3ea1007a648b3320bff36a33a9e07aa5" + integrity sha512-MCjPrQ5MY/QVoZ6n0D92jcRb7eYvxAujG/AH2yM6lI0BspvJQxp0o9s5oiAM9r32r9tkLpiy5s2icsbwefAQIw== + +"@vscode/vsce-sign-linux-arm64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-linux-arm64/-/vsce-sign-linux-arm64-2.0.2.tgz#ce5c5cfc99e3454b4fb770405812b46bd6dca870" + integrity sha512-Ybeu7cA6+/koxszsORXX0OJk9N0GgfHq70Wqi4vv2iJCZvBrOWwcIrxKjvFtwyDgdeQzgPheH5nhLVl5eQy7WA== + +"@vscode/vsce-sign-linux-arm@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-linux-arm/-/vsce-sign-linux-arm-2.0.2.tgz#4142fda83e7130b31aedd8aa81e4daa6334323c2" + integrity sha512-Fkb5jpbfhZKVw3xwR6t7WYfwKZktVGNXdg1m08uEx1anO0oUPUkoQRsNm4QniL3hmfw0ijg00YA6TrxCRkPVOQ== + +"@vscode/vsce-sign-linux-x64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-linux-x64/-/vsce-sign-linux-x64-2.0.2.tgz#59ab93f322efb3cf49166d4e2e812789c3117428" + integrity sha512-NsPPFVtLaTlVJKOiTnO8Cl78LZNWy0Q8iAg+LlBiCDEgC12Gt4WXOSs2pmcIjDYzj2kY4NwdeN1mBTaujYZaPg== + +"@vscode/vsce-sign-win32-arm64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-win32-arm64/-/vsce-sign-win32-arm64-2.0.2.tgz#d095704a14b0404c0b6f696e9889e9a51b31a86c" + integrity sha512-wPs848ymZ3Ny+Y1Qlyi7mcT6VSigG89FWQnp2qRYCyMhdJxOpA4lDwxzlpL8fG6xC8GjQjGDkwbkWUcCobvksQ== + +"@vscode/vsce-sign-win32-x64@2.0.2": + version "2.0.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign-win32-x64/-/vsce-sign-win32-x64-2.0.2.tgz#294ea72b44fedd694d49f5cef4c55bf3876dc257" + integrity sha512-pAiRN6qSAhDM5SVOIxgx+2xnoVUePHbRNC7OD2aOR3WltTKxxF25OfpK8h8UQ7A0BuRkSgREbB59DBlFk4iAeg== + +"@vscode/vsce-sign@^2.0.0": + version "2.0.5" + resolved "https://registry.yarnpkg.com/@vscode/vsce-sign/-/vsce-sign-2.0.5.tgz#8850036476dc0d4e080d9c2d8325e3e97eff5193" + integrity sha512-GfYWrsT/vypTMDMgWDm75iDmAOMe7F71sZECJ+Ws6/xyIfmB3ELVnVN+LwMFAvmXY+e6eWhR2EzNGF/zAhWY3Q== + optionalDependencies: + "@vscode/vsce-sign-alpine-arm64" "2.0.2" + "@vscode/vsce-sign-alpine-x64" "2.0.2" + "@vscode/vsce-sign-darwin-arm64" "2.0.2" + "@vscode/vsce-sign-darwin-x64" "2.0.2" + "@vscode/vsce-sign-linux-arm" "2.0.2" + "@vscode/vsce-sign-linux-arm64" "2.0.2" + "@vscode/vsce-sign-linux-x64" "2.0.2" + "@vscode/vsce-sign-win32-arm64" "2.0.2" + "@vscode/vsce-sign-win32-x64" "2.0.2" + +"@vscode/vsce@^3.3.2": + version "3.3.2" + resolved "https://registry.yarnpkg.com/@vscode/vsce/-/vsce-3.3.2.tgz#1bb86222987814dbb3217c3f8befd63f249c8101" + integrity sha512-XQ4IhctYalSTMwLnMS8+nUaGbU7v99Qm2sOoGfIEf2QC7jpiLXZZMh7NwArEFsKX4gHTJLx0/GqAUlCdC3gKCw== + dependencies: + "@azure/identity" "^4.1.0" + "@vscode/vsce-sign" "^2.0.0" + azure-devops-node-api "^12.5.0" + chalk "^2.4.2" + cheerio "^1.0.0-rc.9" + cockatiel "^3.1.2" + commander "^12.1.0" + form-data "^4.0.0" + glob "^11.0.0" + hosted-git-info "^4.0.2" + jsonc-parser "^3.2.0" + leven "^3.1.0" + markdown-it "^14.1.0" + mime "^1.3.4" + minimatch "^3.0.3" + parse-semver "^1.1.1" + read "^1.0.7" + semver "^7.5.2" + tmp "^0.2.3" + typed-rest-client "^1.8.4" + url-join "^4.0.1" + xml2js "^0.5.0" + yauzl "^2.3.1" + yazl "^2.2.2" + optionalDependencies: + keytar "^7.7.0" + +"@webgpu/types@*": + version "0.1.60" + resolved "https://registry.yarnpkg.com/@webgpu/types/-/types-0.1.60.tgz#4f58e763ce060a81b03ab7ee1ad879c7024923fd" + integrity sha512-8B/tdfRFKdrnejqmvq95ogp8tf52oZ51p3f4QD5m5Paey/qlX4Rhhy5Y8tgFMi7Ms70HzcMMw3EQjH/jdhTwlA== + +"@xmldom/xmldom@^0.8.8": + version "0.8.10" + resolved "https://registry.yarnpkg.com/@xmldom/xmldom/-/xmldom-0.8.10.tgz#a1337ca426aa61cef9fe15b5b28e340a72f6fa99" + integrity sha512-2WALfTl4xo2SkGCYRt6rDTFfk9R1czmBvUQy12gK2KuRKIpWEhcbbzy8EZXtz/jkRqHX8bFEc6FC1HjX4TUWYw== + +"@xstate/cli@^0.5.17": + version "0.5.17" + resolved "https://registry.yarnpkg.com/@xstate/cli/-/cli-0.5.17.tgz#169ece0192680fea59fbc592b14d0d5cf4eeec7e" + integrity sha512-t/R0PeKamiyshT4SYlo4RwBRpuwzYlUuVesMeIxLZ01C2wfbB1mtFb5kMMN35opkDpyYh72fxi8Pt8HxpLZm+w== + dependencies: + "@babel/core" "^7.21.4" + "@xstate/machine-extractor" "^0.16.0" + "@xstate/tools-shared" "^4.1.0" + chokidar "^3.5.3" + commander "^8.0.0" + dotenv "^16.0.3" + isomorphic-fetch "^3.0.0" + prettier "^2.8.8" + xstate "^4.33.4" + xstate-beta "npm:xstate@beta" + +"@xstate/inspect@^0.8.0": + version "0.8.0" + resolved "https://registry.yarnpkg.com/@xstate/inspect/-/inspect-0.8.0.tgz#f99d3706cd823d4922c47ce4f4376eecac502cc7" + integrity sha512-wSkFeOnp+7dhn+zTThO0M4D2FEqZN9lGIWowJu5JLa2ojjtlzRwK8SkjcHZ4rLX8VnMev7kGjgQLrGs8kxy+hw== + dependencies: + fast-safe-stringify "^2.1.1" + +"@xstate/machine-extractor@^0.16.0": + version "0.16.0" + resolved "https://registry.yarnpkg.com/@xstate/machine-extractor/-/machine-extractor-0.16.0.tgz#cceff7365dedb681ecd867be8d1ce9e8fb417c25" + integrity sha512-oYtHWSH/4jSZ46o2qBy2EJqZE6uaP+VYx9mdYYdzw6JT0uMY1PLePWaWAuh3jhi1qupXatNpbfbGeJtetZJCFQ== + dependencies: + "@babel/parser" "^7.21.4" + "@babel/traverse" "^7.21.4" + "@babel/types" "^7.21.4" + recast "^0.23.1" + +"@xstate/react@^4.1.1": + version "4.1.3" + resolved "https://registry.yarnpkg.com/@xstate/react/-/react-4.1.3.tgz#d3db7102ad950584d15f5a07fc17d52a127f3c68" + integrity sha512-zhE+ZfrcCR87bu71Rkh5Z5ruZBivR/7uD/dkelzJqjQdI45IZc9DqTI8lL4Cg5+VN2p5k86KxDsusqW1kW11Tg== + dependencies: + use-isomorphic-layout-effect "^1.1.2" + use-sync-external-store "^1.2.0" + +"@xstate/tools-shared@^4.1.0": + version "4.1.0" + resolved "https://registry.yarnpkg.com/@xstate/tools-shared/-/tools-shared-4.1.0.tgz#ccbc3145a439bbbf2c7b1794d9b6bc4098552354" + integrity sha512-AVqlHFlGjoceexIK33D8CDm8zev17gdRFqGRPiDLE9yvjCSzlzPcc65/VlWCV2EvQt99gUHR1bLMw6ONafkF6g== + dependencies: + "@xstate/machine-extractor" "^0.16.0" + +abbrev@^1.0.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.1.tgz#f8f2c887ad10bf67f634f005b6987fed3179aac8" + integrity sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q== + +accepts@~1.3.8: + version "1.3.8" + resolved "https://registry.yarnpkg.com/accepts/-/accepts-1.3.8.tgz#0bf0be125b67014adcb0b0921e62db7bffe16b2e" + integrity sha512-PYAthTa2m2VKxuvSD3DPC/Gy+U+sOA1LAuT8mkmRuvw+NACSaeXEQ+NHcVF7rONl6qcaxV3Uuemwawk+7+SJLw== + dependencies: + mime-types "~2.1.34" + negotiator "0.6.3" + +acorn-jsx@^5.3.2: + version "5.3.2" + resolved "https://registry.yarnpkg.com/acorn-jsx/-/acorn-jsx-5.3.2.tgz#7ed5bb55908b3b2f1bc55c6af1653bada7f07937" + integrity sha512-rq9s+JNhf0IChjtDXxllJ7g41oZk5SlXtp0LHwyA5cejwn7vKmKp4pPri6YEePv2PU65sAsegbXtIinmDFDXgQ== + +acorn-walk@^8.1.1, acorn-walk@^8.3.2: + version "8.3.4" + resolved "https://registry.yarnpkg.com/acorn-walk/-/acorn-walk-8.3.4.tgz#794dd169c3977edf4ba4ea47583587c5866236b7" + integrity sha512-ueEepnujpqee2o5aIYnvHU6C0A42MNdsIDeqy5BydrkuC5R1ZuUFnm27EeFJGoEHJQgn3uleRvmTXaJgfXbt4g== + dependencies: + acorn "^8.11.0" + +acorn@^8.11.0, acorn@^8.14.0, acorn@^8.4.1, acorn@^8.9.0: + version "8.14.1" + resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.14.1.tgz#721d5dc10f7d5b5609a891773d47731796935dfb" + integrity sha512-OvQ/2pUDKmgfCg++xsTX1wGxfTaszcHVcTctW4UJB4hibJx2HXxxO5UmVgyjMa+ZDsiaf5wWLXYpRWMmBI0QHg== + +agent-base@6, agent-base@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-6.0.2.tgz#49fff58577cfee3f37176feab4c22e00f86d7f77" + integrity sha512-RZNwNclF7+MS/8bDg70amg32dyeZGZxiDuQmZxKLAlQjr3jGyLx+4Kkk58UO7D2QdgFIQCovuSuZESne6RG6XQ== + dependencies: + debug "4" + +agent-base@^7.1.0, agent-base@^7.1.2: + version "7.1.3" + resolved "https://registry.yarnpkg.com/agent-base/-/agent-base-7.1.3.tgz#29435eb821bc4194633a5b89e5bc4703bafc25a1" + integrity sha512-jRR5wdylq8CkOe6hei19GGZnxM6rBGwFl3Bg0YItGDimvjGtAvdZk4Pu6Cl4u4Igsws4a1fd1Vq3ezrhn4KmFw== + +agentkeepalive@^4.2.1: + version "4.6.0" + resolved "https://registry.yarnpkg.com/agentkeepalive/-/agentkeepalive-4.6.0.tgz#35f73e94b3f40bf65f105219c623ad19c136ea6a" + integrity sha512-kja8j7PjmncONqaTsB8fQ+wE2mSU2DJ9D4XKoJ5PFWIdRMa6SLSN1ff4mOr4jCbfRSsxR4keIiySJU0N9T5hIQ== + dependencies: + humanize-ms "^1.2.1" + +aggregate-error@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/aggregate-error/-/aggregate-error-3.1.0.tgz#92670ff50f5359bdb7a3e0d40d0ec30c5737687a" + integrity sha512-4I7Td01quW/RpocfNayFdFVk1qSuoh0E7JrbRJ16nH01HhKFQ88INq9Sd+nd72zqRySlr9BmDA8xlEJ6vJMrYA== + dependencies: + clean-stack "^2.0.0" + indent-string "^4.0.0" + +ajv-keywords@^3.4.1: + version "3.5.2" + resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d" + integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== + +ajv@^6.10.0, ajv@^6.12.0, ajv@^6.12.4: + version "6.12.6" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" + integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== + dependencies: + fast-deep-equal "^3.1.1" + fast-json-stable-stringify "^2.0.0" + json-schema-traverse "^0.4.1" + uri-js "^4.2.2" + +ansi-colors@^4.1.3: + version "4.1.3" + resolved "https://registry.yarnpkg.com/ansi-colors/-/ansi-colors-4.1.3.tgz#37611340eb2243e70cc604cad35d63270d48781b" + integrity sha512-/6w/C21Pm1A7aZitlI5Ni/2J6FFQN8i1Cvz3kHABAAbw93v/NlvKdVOqz7CCWz/3iv/JplRSEEZ83XION15ovw== + +ansi-escapes@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/ansi-escapes/-/ansi-escapes-5.0.0.tgz#b6a0caf0eef0c41af190e9a749e0c00ec04bb2a6" + integrity sha512-5GFMVX8HqE/TB+FuBJGuO5XG0WrsA6ptUqoODaT/n9mmUaZFkqnBueB4leqGBCmrUHnCnC4PCZTCd0E7QQ83bA== + dependencies: + type-fest "^1.0.2" + +ansi-regex@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304" + integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ== + +ansi-regex@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-6.1.0.tgz#95ec409c69619d6cb1b8b34f14b660ef28ebd654" + integrity sha512-7HSX4QQb4CspciLpVFwyRe79O3xsIZDDLER21kERQ71oaPodF8jL725AgJMFAYbooIqolJoRLuM81SpeUkpkvA== + +ansi-styles@^3.2.1: + version "3.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d" + integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA== + dependencies: + color-convert "^1.9.0" + +ansi-styles@^4.0.0, ansi-styles@^4.1.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" + integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== + dependencies: + color-convert "^2.0.1" + +ansi-styles@^5.0.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-5.2.0.tgz#07449690ad45777d1924ac2abb2fc8895dba836b" + integrity sha512-Cxwpt2SfTzTtXcfOlzGEee8O+c+MmUgGrNiBcXnuWxuFJHe6a5Hz7qwhwe5OgaSYI0IJvkLqWX1ASG+cJOkEiA== + +ansi-styles@^6.0.0, ansi-styles@^6.1.0: + version "6.2.1" + resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-6.2.1.tgz#0e62320cf99c21afff3b3012192546aacbfb05c5" + integrity sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug== + +any-promise@^1.0.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/any-promise/-/any-promise-1.3.0.tgz#abc6afeedcea52e809cdc0376aed3ce39635d17f" + integrity sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A== + +anymatch@~3.1.2: + version "3.1.3" + resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.3.tgz#790c58b19ba1720a84205b57c618d5ad8524973e" + integrity sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw== + dependencies: + normalize-path "^3.0.0" + picomatch "^2.0.4" + +app-builder-bin@5.0.0-alpha.12: + version "5.0.0-alpha.12" + resolved "https://registry.yarnpkg.com/app-builder-bin/-/app-builder-bin-5.0.0-alpha.12.tgz#2daf82f8badc698e0adcc95ba36af4ff0650dc80" + integrity sha512-j87o0j6LqPL3QRr8yid6c+Tt5gC7xNfYo6uQIQkorAC6MpeayVMZrEDzKmJJ/Hlv7EnOQpaRm53k6ktDYZyB6w== + +app-builder-lib@26.0.12: + version "26.0.12" + resolved "https://registry.yarnpkg.com/app-builder-lib/-/app-builder-lib-26.0.12.tgz#2e33df936e0f78d4266b058ece90308ea981eefb" + integrity sha512-+/CEPH1fVKf6HowBUs6LcAIoRcjeqgvAeoSE+cl7Y7LndyQ9ViGPYibNk7wmhMHzNgHIuIbw4nWADPO+4mjgWw== + dependencies: + "@develar/schema-utils" "~2.6.5" + "@electron/asar" "3.2.18" + "@electron/fuses" "^1.8.0" + "@electron/notarize" "2.5.0" + "@electron/osx-sign" "1.3.1" + "@electron/rebuild" "3.7.0" + "@electron/universal" "2.0.1" + "@malept/flatpak-bundler" "^0.4.0" + "@types/fs-extra" "9.0.13" + async-exit-hook "^2.0.1" + builder-util "26.0.11" + builder-util-runtime "9.3.1" + chromium-pickle-js "^0.2.0" + config-file-ts "0.2.8-rc1" + debug "^4.3.4" + dotenv "^16.4.5" + dotenv-expand "^11.0.6" + ejs "^3.1.8" + electron-publish "26.0.11" + fs-extra "^10.1.0" + hosted-git-info "^4.1.0" + is-ci "^3.0.0" + isbinaryfile "^5.0.0" + js-yaml "^4.1.0" + json5 "^2.2.3" + lazy-val "^1.0.5" + minimatch "^10.0.0" + plist "3.1.0" + resedit "^1.7.0" + semver "^7.3.8" + tar "^6.1.12" + temp-file "^3.4.0" + tiny-async-pool "1.3.0" + +arg@^4.1.0: + version "4.1.3" + resolved "https://registry.yarnpkg.com/arg/-/arg-4.1.3.tgz#269fc7ad5b8e42cb63c896d5666017261c144089" + integrity sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA== + +arg@^5.0.2: + version "5.0.2" + resolved "https://registry.yarnpkg.com/arg/-/arg-5.0.2.tgz#c81433cc427c92c4dcf4865142dbca6f15acd59c" + integrity sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg== + +argparse@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/argparse/-/argparse-2.0.1.tgz#246f50f3ca78a3240f6c997e8a9bd1eac49e4b38" + integrity sha512-8+9WqebbFzpX9OR+Wa6O29asIogeRMzcGtAINdpMHHyAg10f05aSFVBbcEqGf/PXw1EjAZ+q2/bEBg3DvurK3Q== + +aria-query@5.3.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/aria-query/-/aria-query-5.3.0.tgz#650c569e41ad90b51b3d7df5e5eed1c7549c103e" + integrity sha512-b0P0sZPKtyu8HkeRAfCq0IfURZK+SuwMjY1UXGBU27wpAiTwQAIlq56IbIO+ytk/JjS1fMR14ee5WBBfKi5J6A== + dependencies: + dequal "^2.0.3" + +aria-query@^5.0.0, aria-query@^5.3.2: + version "5.3.2" + resolved "https://registry.yarnpkg.com/aria-query/-/aria-query-5.3.2.tgz#93f81a43480e33a338f19163a3d10a50c01dcd59" + integrity sha512-COROpnaoap1E2F000S62r6A60uHZnmlvomhfyT2DlTcrY1OrBKn2UhH7qn5wTC9zMvD0AY7csdPSNwKP+7WiQw== + +array-buffer-byte-length@^1.0.1, array-buffer-byte-length@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/array-buffer-byte-length/-/array-buffer-byte-length-1.0.2.tgz#384d12a37295aec3769ab022ad323a18a51ccf8b" + integrity sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw== + dependencies: + call-bound "^1.0.3" + is-array-buffer "^3.0.5" + +array-flatten@1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2" + integrity sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg== + +array-includes@^3.1.6, array-includes@^3.1.8: + version "3.1.8" + resolved "https://registry.yarnpkg.com/array-includes/-/array-includes-3.1.8.tgz#5e370cbe172fdd5dd6530c1d4aadda25281ba97d" + integrity sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ== + dependencies: + call-bind "^1.0.7" + define-properties "^1.2.1" + es-abstract "^1.23.2" + es-object-atoms "^1.0.0" + get-intrinsic "^1.2.4" + is-string "^1.0.7" + +array.prototype.findlast@^1.2.5: + version "1.2.5" + resolved "https://registry.yarnpkg.com/array.prototype.findlast/-/array.prototype.findlast-1.2.5.tgz#3e4fbcb30a15a7f5bf64cf2faae22d139c2e4904" + integrity sha512-CVvd6FHg1Z3POpBLxO6E6zr+rSKEQ9L6rZHAaY7lLfhKsWYUBBOuMs0e9o24oopj6H+geRCX0YJ+TJLBK2eHyQ== + dependencies: + call-bind "^1.0.7" + define-properties "^1.2.1" + es-abstract "^1.23.2" + es-errors "^1.3.0" + es-object-atoms "^1.0.0" + es-shim-unscopables "^1.0.2" + +array.prototype.findlastindex@^1.2.5: + version "1.2.6" + resolved "https://registry.yarnpkg.com/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.6.tgz#cfa1065c81dcb64e34557c9b81d012f6a421c564" + integrity sha512-F/TKATkzseUExPlfvmwQKGITM3DGTK+vkAsCZoDc5daVygbJBnjEUCbgkAvVFsgfXfX4YIqZ/27G3k3tdXrTxQ== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.4" + define-properties "^1.2.1" + es-abstract "^1.23.9" + es-errors "^1.3.0" + es-object-atoms "^1.1.1" + es-shim-unscopables "^1.1.0" + +array.prototype.flat@^1.3.1, array.prototype.flat@^1.3.2: + version "1.3.3" + resolved "https://registry.yarnpkg.com/array.prototype.flat/-/array.prototype.flat-1.3.3.tgz#534aaf9e6e8dd79fb6b9a9917f839ef1ec63afe5" + integrity sha512-rwG/ja1neyLqCuGZ5YYrznA62D4mZXg0i1cIskIUKSiqF3Cje9/wXAls9B9s1Wa2fomMsIv8czB8jZcPmxCXFg== + dependencies: + call-bind "^1.0.8" + define-properties "^1.2.1" + es-abstract "^1.23.5" + es-shim-unscopables "^1.0.2" + +array.prototype.flatmap@^1.3.2, array.prototype.flatmap@^1.3.3: + version "1.3.3" + resolved "https://registry.yarnpkg.com/array.prototype.flatmap/-/array.prototype.flatmap-1.3.3.tgz#712cc792ae70370ae40586264629e33aab5dd38b" + integrity sha512-Y7Wt51eKJSyi80hFrJCePGGNo5ktJCslFuboqJsbf57CCPcm5zztluPlc4/aD8sWsKvlwatezpV4U1efk8kpjg== + dependencies: + call-bind "^1.0.8" + define-properties "^1.2.1" + es-abstract "^1.23.5" + es-shim-unscopables "^1.0.2" + +array.prototype.tosorted@^1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/array.prototype.tosorted/-/array.prototype.tosorted-1.1.4.tgz#fe954678ff53034e717ea3352a03f0b0b86f7ffc" + integrity sha512-p6Fx8B7b7ZhL/gmUsAy0D15WhvDccw3mnGNbZpi3pmeJdxtWsj2jEaI4Y6oo3XiHfzuSgPwKc04MYt6KgvC/wA== + dependencies: + call-bind "^1.0.7" + define-properties "^1.2.1" + es-abstract "^1.23.3" + es-errors "^1.3.0" + es-shim-unscopables "^1.0.2" + +arraybuffer.prototype.slice@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/arraybuffer.prototype.slice/-/arraybuffer.prototype.slice-1.0.4.tgz#9d760d84dbdd06d0cbf92c8849615a1a7ab3183c" + integrity sha512-BNoCY6SXXPQ7gF2opIP4GBE+Xw7U+pHMYKuzjgCN3GwiaIR09UUeKfheyIry77QtrCBlC0KK0q5/TER/tYh3PQ== + dependencies: + array-buffer-byte-length "^1.0.1" + call-bind "^1.0.8" + define-properties "^1.2.1" + es-abstract "^1.23.5" + es-errors "^1.3.0" + get-intrinsic "^1.2.6" + is-array-buffer "^3.0.4" + +asap@~2.0.3: + version "2.0.6" + resolved "https://registry.yarnpkg.com/asap/-/asap-2.0.6.tgz#e50347611d7e690943208bbdafebcbc2fb866d46" + integrity sha512-BSHWgDSAiKs50o2Re8ppvp3seVHXSRM44cdSsT9FfNEUUZLOGWVCsiWaRPWM1Znn+mqZ1OfVZ3z3DWEzSp7hRA== + +assert-plus@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525" + integrity sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw== + +assertion-error@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-1.1.0.tgz#e60b6b0e8f301bd97e5375215bda406c85118c0b" + integrity sha512-jgsaNduz+ndvGyFt3uSuWqvy4lCnIJiovtouQN5JZHOKCS2QuhEdbcQHFhVksz2N2U9hXJo8odG7ETyWlEeuDw== + +assertion-error@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-2.0.1.tgz#f641a196b335690b1070bf00b6e7593fec190bf7" + integrity sha512-Izi8RQcffqCeNVgFigKli1ssklIbpHnCYc6AknXGYoB6grJqyeby7jv12JUQgmTAnIDnbck1uxksT4dzN3PWBA== + +ast-types-flow@^0.0.8: + version "0.0.8" + resolved "https://registry.yarnpkg.com/ast-types-flow/-/ast-types-flow-0.0.8.tgz#0a85e1c92695769ac13a428bb653e7538bea27d6" + integrity sha512-OH/2E5Fg20h2aPrbe+QL8JZQFko0YZaF+j4mnQ7BGhfavO7OpSLa8a0y9sBwomHdSbkhTS8TQNayBfnW5DwbvQ== + +ast-types@^0.16.1: + version "0.16.1" + resolved "https://registry.yarnpkg.com/ast-types/-/ast-types-0.16.1.tgz#7a9da1617c9081bc121faafe91711b4c8bb81da2" + integrity sha512-6t10qk83GOG8p0vKmaCr8eiilZwO171AvbROMtvvNiwrTly62t+7XkA8RdIIVbpMhCASAsxgAzdRSwh6nw/5Dg== + dependencies: + tslib "^2.0.1" + +astral-regex@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31" + integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ== + +async-exit-hook@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/async-exit-hook/-/async-exit-hook-2.0.1.tgz#8bd8b024b0ec9b1c01cccb9af9db29bd717dfaf3" + integrity sha512-NW2cX8m1Q7KPA7a5M2ULQeZ2wR5qI5PAbw5L0UOMxdioVk9PMZ0h1TmyZEkPYrCvYjDlFICusOu1dlEKAAeXBw== + +async-function@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/async-function/-/async-function-1.0.0.tgz#509c9fca60eaf85034c6829838188e4e4c8ffb2b" + integrity sha512-hsU18Ae8CDTR6Kgu9DYf0EbCr/a5iGL0rytQDobUcdpYOKokk8LEjVphnXkDkgpi0wYVsqrXuP0bZxJaTqdgoA== + +async@^3.2.3, async@^3.2.6: + version "3.2.6" + resolved "https://registry.yarnpkg.com/async/-/async-3.2.6.tgz#1b0728e14929d51b85b449b7f06e27c1145e38ce" + integrity sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA== + +asynckit@^0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" + integrity sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q== + +at-least-node@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/at-least-node/-/at-least-node-1.0.0.tgz#602cd4b46e844ad4effc92a8011a3c46e0238dc2" + integrity sha512-+q/t7Ekv1EDY2l6Gda6LLiX14rU9TV20Wa3ofeQmwPFZbOMo9DXrLbOjFaaclkXKWidIaopwAObQDqwWtGUjqg== + +author-regex@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/author-regex/-/author-regex-1.0.0.tgz#d08885be6b9bbf9439fe087c76287245f0a81450" + integrity sha512-KbWgR8wOYRAPekEmMXrYYdc7BRyhn2Ftk7KWfMUnQ43hFdojWEFRxhhRUm3/OFEdPa1r0KAvTTg9YQK57xTe0g== + +autoprefixer@^10.4.21: + version "10.4.21" + resolved "https://registry.yarnpkg.com/autoprefixer/-/autoprefixer-10.4.21.tgz#77189468e7a8ad1d9a37fbc08efc9f480cf0a95d" + integrity sha512-O+A6LWV5LDHSJD3LjHYoNi4VLsj/Whi7k6zG12xTYaU4cQ8oxQGckXNX8cRHK5yOZ/ppVHe0ZBXGzSV9jXdVbQ== + dependencies: + browserslist "^4.24.4" + caniuse-lite "^1.0.30001702" + fraction.js "^4.3.7" + normalize-range "^0.1.2" + picocolors "^1.1.1" + postcss-value-parser "^4.2.0" + +available-typed-arrays@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/available-typed-arrays/-/available-typed-arrays-1.0.7.tgz#a5cc375d6a03c2efc87a553f3e0b1522def14846" + integrity sha512-wvUjBtSGN7+7SjNpq/9M2Tg350UZD3q62IFZLbRAR1bSMlCo1ZaeW+BJ+D090e4hIIZLBcTDWe4Mh4jvUDajzQ== + dependencies: + possible-typed-array-names "^1.0.0" + +axe-core@^4.10.0: + version "4.10.3" + resolved "https://registry.yarnpkg.com/axe-core/-/axe-core-4.10.3.tgz#04145965ac7894faddbac30861e5d8f11bfd14fc" + integrity sha512-Xm7bpRXnDSX2YE2YFfBk2FnF0ep6tmG7xPh8iHee8MIcrgq762Nkce856dYtJYLkuIoYZvGfTs/PbZhideTcEg== + +axobject-query@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/axobject-query/-/axobject-query-4.1.0.tgz#28768c76d0e3cff21bc62a9e2d0b6ac30042a1ee" + integrity sha512-qIj0G9wZbMGNLjLmg1PT6v2mE9AH2zlnADJD/2tC6E00hgmhUOfEB6greHPAfLRSufHqROIUTkw6E+M3lH0PTQ== + +azure-devops-node-api@^12.5.0: + version "12.5.0" + resolved "https://registry.yarnpkg.com/azure-devops-node-api/-/azure-devops-node-api-12.5.0.tgz#38b9efd7c5ac74354fe4e8dbe42697db0b8e85a5" + integrity sha512-R5eFskGvOm3U/GzeAuxRkUsAl0hrAwGgWn6zAd2KrZmrEhWZVqLew4OOupbQlXUuojUzpGtq62SmdhJ06N88og== + dependencies: + tunnel "0.0.6" + typed-rest-client "^1.8.4" + +babel-plugin-polyfill-corejs2@^0.4.10: + version "0.4.13" + resolved "https://registry.yarnpkg.com/babel-plugin-polyfill-corejs2/-/babel-plugin-polyfill-corejs2-0.4.13.tgz#7d445f0e0607ebc8fb6b01d7e8fb02069b91dd8b" + integrity sha512-3sX/eOms8kd3q2KZ6DAhKPc0dgm525Gqq5NtWKZ7QYYZEv57OQ54KtblzJzH1lQF/eQxO8KjWGIK9IPUJNus5g== + dependencies: + "@babel/compat-data" "^7.22.6" + "@babel/helper-define-polyfill-provider" "^0.6.4" + semver "^6.3.1" + +babel-plugin-polyfill-corejs3@^0.11.0: + version "0.11.1" + resolved "https://registry.yarnpkg.com/babel-plugin-polyfill-corejs3/-/babel-plugin-polyfill-corejs3-0.11.1.tgz#4e4e182f1bb37c7ba62e2af81d8dd09df31344f6" + integrity sha512-yGCqvBT4rwMczo28xkH/noxJ6MZ4nJfkVYdoDaC/utLtWrXxv27HVrzAeSbqR8SxDsp46n0YF47EbHoixy6rXQ== + dependencies: + "@babel/helper-define-polyfill-provider" "^0.6.3" + core-js-compat "^3.40.0" + +babel-plugin-polyfill-regenerator@^0.6.1: + version "0.6.4" + resolved "https://registry.yarnpkg.com/babel-plugin-polyfill-regenerator/-/babel-plugin-polyfill-regenerator-0.6.4.tgz#428c615d3c177292a22b4f93ed99e358d7906a9b" + integrity sha512-7gD3pRadPrbjhjLyxebmx/WrFYcuSjZ0XbdUujQMZ/fcE9oeewk2U/7PCvez84UeuK3oSjmPZ0Ch0dlupQvGzw== + dependencies: + "@babel/helper-define-polyfill-provider" "^0.6.4" + +balanced-match@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" + integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== + +base16@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/base16/-/base16-1.0.0.tgz#e297f60d7ec1014a7a971a39ebc8a98c0b681e70" + integrity sha512-pNdYkNPiJUnEhnfXV56+sQy8+AaPcG3POZAUnwr4EeqCUZFz4u2PePbo3e5Gj4ziYPCWGUZT9RHisvJKnwFuBQ== + +base64-arraybuffer@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/base64-arraybuffer/-/base64-arraybuffer-1.0.2.tgz#1c37589a7c4b0746e34bd1feb951da2df01c1bdc" + integrity sha512-I3yl4r9QB5ZRY3XuJVEPfc2XhZO6YweFPI+UovAzn+8/hb3oJ6lnysaFcjVpkCPfVWFUDvoZ8kmVDP7WyRtYtQ== + +base64-js@^1.3.1, base64-js@^1.5.1: + version "1.5.1" + resolved "https://registry.yarnpkg.com/base64-js/-/base64-js-1.5.1.tgz#1b1b440160a5bf7ad40b650f095963481903930a" + integrity sha512-AKpaYlHn8t4SVbOHCy+b5+KKgvR4vrsD8vbvrbiQJps7fKDTkjkDry6ji0rUJjC0kzbNePLwzxq8iypo41qeWA== + +basic-auth@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/basic-auth/-/basic-auth-2.0.1.tgz#b998279bf47ce38344b4f3cf916d4679bbf51e3a" + integrity sha512-NF+epuEdnUYVlGuhaxbbq+dvJttwLnGY+YixlXlME5KpQ5W3CnXA5cVTneY3SPbPDRkcjMbifrwmFYcClgOZeg== + dependencies: + safe-buffer "5.1.2" + +binary-extensions@^2.0.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.3.0.tgz#f6e14a97858d327252200242d4ccfe522c445522" + integrity sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw== + +bl@^4.0.3, bl@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/bl/-/bl-4.1.0.tgz#451535264182bec2fbbc83a62ab98cf11d9f7b3a" + integrity sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w== + dependencies: + buffer "^5.5.0" + inherits "^2.0.4" + readable-stream "^3.4.0" + +bluebird@^3.1.1: + version "3.7.2" + resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.7.2.tgz#9f229c15be272454ffa973ace0dbee79a1b0c36f" + integrity sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg== + +body-parser@1.20.3: + version "1.20.3" + resolved "https://registry.yarnpkg.com/body-parser/-/body-parser-1.20.3.tgz#1953431221c6fb5cd63c4b36d53fab0928e548c6" + integrity sha512-7rAxByjUMqQ3/bHJy7D6OGXvx/MMc4IqBn/X0fcM1QUcAItpZrBEYhWGem+tzXH90c+G01ypMcYJBO9Y30203g== + dependencies: + bytes "3.1.2" + content-type "~1.0.5" + debug "2.6.9" + depd "2.0.0" + destroy "1.2.0" + http-errors "2.0.0" + iconv-lite "0.4.24" + on-finished "2.4.1" + qs "6.13.0" + raw-body "2.5.2" + type-is "~1.6.18" + unpipe "1.0.0" + +bonjour-service@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/bonjour-service/-/bonjour-service-1.3.0.tgz#80d867430b5a0da64e82a8047fc1e355bdb71722" + integrity sha512-3YuAUiSkWykd+2Azjgyxei8OWf8thdn8AITIog2M4UICzoqfjlqr64WIjEXZllf/W6vK1goqleSR6brGomxQqA== + dependencies: + fast-deep-equal "^3.1.3" + multicast-dns "^7.2.5" + +boolbase@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/boolbase/-/boolbase-1.0.0.tgz#68dff5fbe60c51eb37725ea9e3ed310dcc1e776e" + integrity sha512-JZOSA7Mo9sNGB8+UjSgzdLtokWAky1zbztM3WRLCbZ70/3cTANmQmOdR7y2g+J0e2WXywy1yS468tY+IruqEww== + +boolean@^3.0.1: + version "3.2.0" + resolved "https://registry.yarnpkg.com/boolean/-/boolean-3.2.0.tgz#9e5294af4e98314494cbb17979fa54ca159f116b" + integrity sha512-d0II/GO9uf9lfUHH2BQsjxzRJZBdsjgsBiW4BvhWk/3qoKwQFjIDVN19PfX8F2D/r9PCMTtLWjYVCFrpeYUzsw== + +brace-expansion@^1.1.7: + version "1.1.11" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" + integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== + dependencies: + balanced-match "^1.0.0" + concat-map "0.0.1" + +brace-expansion@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-2.0.1.tgz#1edc459e0f0c548486ecf9fc99f2221364b9a0ae" + integrity sha512-XnAIvQ8eM+kC6aULx6wuQiwVsnzsi9d3WxzV3FpWTGA19F621kwdbsAcFKXgKUHZWsy+mY6iL1sHTxWEFCytDA== + dependencies: + balanced-match "^1.0.0" + +braces@^3.0.3, braces@~3.0.2: + version "3.0.3" + resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.3.tgz#490332f40919452272d55a8480adc0c441358789" + integrity sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA== + dependencies: + fill-range "^7.1.1" + +browser-stdout@^1.3.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/browser-stdout/-/browser-stdout-1.3.1.tgz#baa559ee14ced73452229bad7326467c61fabd60" + integrity sha512-qhAVI1+Av2X7qelOfAIYwXONood6XlZE/fXaBSmW/T5SzLAmCgzi+eiWE7fUvbHaeNBQH13UftjpXxsfLkMpgw== + +browserslist@^4.24.0, browserslist@^4.24.4: + version "4.24.4" + resolved "https://registry.yarnpkg.com/browserslist/-/browserslist-4.24.4.tgz#c6b2865a3f08bcb860a0e827389003b9fe686e4b" + integrity sha512-KDi1Ny1gSePi1vm0q4oxSF8b4DR44GF4BbmS2YdhPLOEqd8pDviZOGH/GsmRwoWJ2+5Lr085X7naowMwKHDG1A== + dependencies: + caniuse-lite "^1.0.30001688" + electron-to-chromium "^1.5.73" + node-releases "^2.0.19" + update-browserslist-db "^1.1.1" + +bson@^6.10.3: + version "6.10.3" + resolved "https://registry.yarnpkg.com/bson/-/bson-6.10.3.tgz#5f9a463af6b83e264bedd08b236d1356a30eda47" + integrity sha512-MTxGsqgYTwfshYWTRdmZRC+M7FnG1b4y7RO7p2k3X24Wq0yv1m77Wsj0BzlPzd/IowgESfsruQCUToa7vbOpPQ== + +buffer-crc32@~0.2.3: + version "0.2.13" + resolved "https://registry.yarnpkg.com/buffer-crc32/-/buffer-crc32-0.2.13.tgz#0d333e3f00eac50aa1454abd30ef8c2a5d9a7242" + integrity sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ== + +buffer-equal-constant-time@1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/buffer-equal-constant-time/-/buffer-equal-constant-time-1.0.1.tgz#f8e71132f7ffe6e01a5c9697a4c6f3e48d5cc819" + integrity sha512-zRpUiDwd/xk6ADqPMATG8vc9VPrkck7T07OIx0gnjmJAnHnTVXNQG3vfvWNuiZIkwu9KrKdA1iJKfsfTVxE6NA== + +buffer-from@^1.0.0: + version "1.1.2" + resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.2.tgz#2b146a6fd72e80b4f55d255f35ed59a3a9a41bd5" + integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ== + +buffer@^5.1.0, buffer@^5.5.0: + version "5.7.1" + resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0" + integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ== + dependencies: + base64-js "^1.3.1" + ieee754 "^1.1.13" + +builder-util-runtime@9.3.1: + version "9.3.1" + resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-9.3.1.tgz#0daedde0f6d381f2a00a50a407b166fe7dca1a67" + integrity sha512-2/egrNDDnRaxVwK3A+cJq6UOlqOdedGA7JPqCeJjN2Zjk1/QB/6QUi3b714ScIGS7HafFXTyzJEOr5b44I3kvQ== + dependencies: + debug "^4.3.4" + sax "^1.2.4" + +builder-util@26.0.11: + version "26.0.11" + resolved "https://registry.yarnpkg.com/builder-util/-/builder-util-26.0.11.tgz#ad85b92c93f2b976b973e1d87337e0c6813fcb8f" + integrity sha512-xNjXfsldUEe153h1DraD0XvDOpqGR0L5eKFkdReB7eFW5HqysDZFfly4rckda6y9dF39N3pkPlOblcfHKGw+uA== + dependencies: + "7zip-bin" "~5.2.0" + "@types/debug" "^4.1.6" + app-builder-bin "5.0.0-alpha.12" + builder-util-runtime "9.3.1" + chalk "^4.1.2" + cross-spawn "^7.0.6" + debug "^4.3.4" + fs-extra "^10.1.0" + http-proxy-agent "^7.0.0" + https-proxy-agent "^7.0.0" + is-ci "^3.0.0" + js-yaml "^4.1.0" + sanitize-filename "^1.6.3" + source-map-support "^0.5.19" + stat-mode "^1.0.0" + temp-file "^3.4.0" + tiny-async-pool "1.3.0" + +bundle-name@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/bundle-name/-/bundle-name-4.1.0.tgz#f3b96b34160d6431a19d7688135af7cfb8797889" + integrity sha512-tjwM5exMg6BGRI+kNmTntNsvdZS1X8BFYS6tnJ2hdH0kVxM6/eVZ2xy+FqStSWvYmtfFMDLIxurorHwDKfDz5Q== + dependencies: + run-applescript "^7.0.0" + +bytes@3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.2.tgz#8b0beeb98605adf1b128fa4386403c009e0221a5" + integrity sha512-/Nf7TyzTx6S3yRJObOAV7956r8cr2+Oj8AC5dt8wSP3BQAoeX58NoHyCU8P8zGkNXStjTSi6fzO6F0pBdcYbEg== + +cac@^6.7.14: + version "6.7.14" + resolved "https://registry.yarnpkg.com/cac/-/cac-6.7.14.tgz#804e1e6f506ee363cb0e3ccbb09cad5dd9870959" + integrity sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ== + +cacache@^16.1.0: + version "16.1.3" + resolved "https://registry.yarnpkg.com/cacache/-/cacache-16.1.3.tgz#a02b9f34ecfaf9a78c9f4bc16fceb94d5d67a38e" + integrity sha512-/+Emcj9DAXxX4cwlLmRI9c166RuL3w30zp4R7Joiv2cQTtTtA+jeuCAjH3ZlGnYS3tKENSrKhAzVVP9GVyzeYQ== + dependencies: + "@npmcli/fs" "^2.1.0" + "@npmcli/move-file" "^2.0.0" + chownr "^2.0.0" + fs-minipass "^2.1.0" + glob "^8.0.1" + infer-owner "^1.0.4" + lru-cache "^7.7.1" + minipass "^3.1.6" + minipass-collect "^1.0.2" + minipass-flush "^1.0.5" + minipass-pipeline "^1.2.4" + mkdirp "^1.0.4" + p-map "^4.0.0" + promise-inflight "^1.0.1" + rimraf "^3.0.2" + ssri "^9.0.0" + tar "^6.1.11" + unique-filename "^2.0.0" + +cacheable-lookup@^5.0.3: + version "5.0.4" + resolved "https://registry.yarnpkg.com/cacheable-lookup/-/cacheable-lookup-5.0.4.tgz#5a6b865b2c44357be3d5ebc2a467b032719a7005" + integrity sha512-2/kNscPhpcxrOigMZzbiWF7dz8ilhb/nIHU3EyZiXWXpeq/au8qJ8VhdftMkty3n7Gj6HIGalQG8oiBNB3AJgA== + +cacheable-request@^7.0.2: + version "7.0.4" + resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-7.0.4.tgz#7a33ebf08613178b403635be7b899d3e69bbe817" + integrity sha512-v+p6ongsrp0yTGbJXjgxPow2+DL93DASP4kXCDKb8/bwRtt9OEF3whggkkDkGNzgcWy2XaF4a8nZglC7uElscg== + dependencies: + clone-response "^1.0.2" + get-stream "^5.1.0" + http-cache-semantics "^4.0.0" + keyv "^4.0.0" + lowercase-keys "^2.0.0" + normalize-url "^6.0.1" + responselike "^2.0.0" + +call-bind-apply-helpers@^1.0.0, call-bind-apply-helpers@^1.0.1, call-bind-apply-helpers@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/call-bind-apply-helpers/-/call-bind-apply-helpers-1.0.2.tgz#4b5428c222be985d79c3d82657479dbe0b59b2d6" + integrity sha512-Sp1ablJ0ivDkSzjcaJdxEunN5/XvksFJ2sMBFfq6x0ryhQV/2b/KwFe21cMpmHtPOSij8K99/wSfoEuTObmuMQ== + dependencies: + es-errors "^1.3.0" + function-bind "^1.1.2" + +call-bind@^1.0.7, call-bind@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/call-bind/-/call-bind-1.0.8.tgz#0736a9660f537e3388826f440d5ec45f744eaa4c" + integrity sha512-oKlSFMcMwpUg2ednkhQ454wfWiU/ul3CkJe/PEHcTKuiX6RpbehUiFMXu13HalGZxfUwCQzZG747YXBn1im9ww== + dependencies: + call-bind-apply-helpers "^1.0.0" + es-define-property "^1.0.0" + get-intrinsic "^1.2.4" + set-function-length "^1.2.2" + +call-bound@^1.0.2, call-bound@^1.0.3, call-bound@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/call-bound/-/call-bound-1.0.4.tgz#238de935d2a2a692928c538c7ccfa91067fd062a" + integrity sha512-+ys997U96po4Kx/ABpBCqhA9EuxJaQWDQg7295H4hBphv3IZg0boBKuwYpt4YXp6MZ5AmZQnU/tyMTlRpaSejg== + dependencies: + call-bind-apply-helpers "^1.0.2" + get-intrinsic "^1.3.0" + +callsites@^3.0.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/callsites/-/callsites-3.1.0.tgz#b3630abd8943432f54b3f0519238e33cd7df2f73" + integrity sha512-P8BjAsXvZS+VIDUI11hHCQEv74YT67YUi5JJFNWIqL235sBmjX4+qx9Muvls5ivyNENctx46xQLQ3aTuE7ssaQ== + +camelcase-css@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/camelcase-css/-/camelcase-css-2.0.1.tgz#ee978f6947914cc30c6b44741b6ed1df7f043fd5" + integrity sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA== + +camelcase@^6.0.0: + version "6.3.0" + resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.3.0.tgz#5685b95eb209ac9c0c177467778c9c84df58ba9a" + integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA== + +caniuse-lite@^1.0.30001688, caniuse-lite@^1.0.30001702: + version "1.0.30001715" + resolved "https://registry.yarnpkg.com/caniuse-lite/-/caniuse-lite-1.0.30001715.tgz#bd325a37ad366e3fe90827d74062807a34fbaeb2" + integrity sha512-7ptkFGMm2OAOgvZpwgA4yjQ5SQbrNVGdRjzH0pBdy1Fasvcr+KAeECmbCAECzTuDuoX0FCY8KzUxjf9+9kfZEw== + +chai@^4.3.10: + version "4.5.0" + resolved "https://registry.yarnpkg.com/chai/-/chai-4.5.0.tgz#707e49923afdd9b13a8b0b47d33d732d13812fd8" + integrity sha512-RITGBfijLkBddZvnn8jdqoTypxvqbOLYQkGGxXzeFjVHvudaPw0HNFD9x928/eUwYWd2dPCugVqspGALTZZQKw== + dependencies: + assertion-error "^1.1.0" + check-error "^1.0.3" + deep-eql "^4.1.3" + get-func-name "^2.0.2" + loupe "^2.3.6" + pathval "^1.1.1" + type-detect "^4.1.0" + +chai@^5.2.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/chai/-/chai-5.2.0.tgz#1358ee106763624114addf84ab02697e411c9c05" + integrity sha512-mCuXncKXk5iCLhfhwTc0izo0gtEmpz5CtG2y8GiOINBlMVS6v8TMRc5TaLWKS6692m9+dVVfzgeVxR5UxWHTYw== + dependencies: + assertion-error "^2.0.1" + check-error "^2.1.1" + deep-eql "^5.0.1" + loupe "^3.1.0" + pathval "^2.0.0" + +chalk@^2.4.2: + version "2.4.2" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424" + integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ== + dependencies: + ansi-styles "^3.2.1" + escape-string-regexp "^1.0.5" + supports-color "^5.3.0" + +chalk@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-3.0.0.tgz#3f73c2bf526591f574cc492c51e2456349f844e4" + integrity sha512-4D3B6Wf41KOYRFdszmDqMCGq5VV/uMAB273JILmO+3jAlh8X4qDtdtgCR3fxtbLEMzSx22QdhnDcJvu2u1fVwg== + dependencies: + ansi-styles "^4.1.0" + supports-color "^7.1.0" + +chalk@^4, chalk@^4.0.0, chalk@^4.0.2, chalk@^4.1.0, chalk@^4.1.1, chalk@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" + integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== + dependencies: + ansi-styles "^4.1.0" + supports-color "^7.1.0" + +chalk@^5.3.0: + version "5.4.1" + resolved "https://registry.yarnpkg.com/chalk/-/chalk-5.4.1.tgz#1b48bf0963ec158dce2aacf69c093ae2dd2092d8" + integrity sha512-zgVZuo2WcZgfUEmsn6eO3kINexW8RAE4maiQ8QNs8CtpPCSyMiYsULR3HQYkm3w8FIA3SberyMJMSldGsW+U3w== + +change-case@^5.4.4: + version "5.4.4" + resolved "https://registry.yarnpkg.com/change-case/-/change-case-5.4.4.tgz#0d52b507d8fb8f204343432381d1a6d7bff97a02" + integrity sha512-HRQyTk2/YPEkt9TnUPbOpr64Uw3KOicFWPVBb+xiHvd6eBx/qPr9xqfBFDT8P2vWsvvz4jbEkfDe71W3VyNu2w== + +check-error@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/check-error/-/check-error-1.0.3.tgz#a6502e4312a7ee969f646e83bb3ddd56281bd694" + integrity sha512-iKEoDYaRmd1mxM90a2OEfWhjsjPpYPuQ+lMYsoxB126+t8fw7ySEO48nmDg5COTjxDI65/Y2OWpeEHk3ZOe8zg== + dependencies: + get-func-name "^2.0.2" + +check-error@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/check-error/-/check-error-2.1.1.tgz#87eb876ae71ee388fa0471fe423f494be1d96ccc" + integrity sha512-OAlb+T7V4Op9OwdkjmguYRqncdlx5JiofwOAUkmTF+jNdHwzTaTs4sRAGpzLF3oOz5xAyDGrPgeIDFQmDOTiJw== + +cheerio-select@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/cheerio-select/-/cheerio-select-2.1.0.tgz#4d8673286b8126ca2a8e42740d5e3c4884ae21b4" + integrity sha512-9v9kG0LvzrlcungtnJtpGNxY+fzECQKhK4EGJX2vByejiMX84MFNQw4UxPJl3bFbTMw+Dfs37XaIkCwTZfLh4g== + 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" + +cheerio@^1.0.0-rc.9: + version "1.0.0" + resolved "https://registry.yarnpkg.com/cheerio/-/cheerio-1.0.0.tgz#1ede4895a82f26e8af71009f961a9b8cb60d6a81" + integrity sha512-quS9HgjQpdaXOvsZz82Oz7uxtXiy6UIsIQcpBj7HRw2M63Skasm9qlDocAM7jNuaxdhpPU7c4kJN+gA5MCu4ww== + dependencies: + cheerio-select "^2.1.0" + dom-serializer "^2.0.0" + domhandler "^5.0.3" + domutils "^3.1.0" + encoding-sniffer "^0.2.0" + htmlparser2 "^9.1.0" + parse5 "^7.1.2" + parse5-htmlparser2-tree-adapter "^7.0.0" + parse5-parser-stream "^7.1.2" + undici "^6.19.5" + whatwg-mimetype "^4.0.0" + +chokidar@^3.5.3, chokidar@^3.6.0: + version "3.6.0" + resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.6.0.tgz#197c6cc669ef2a8dc5e7b4d97ee4e092c3eb0d5b" + integrity sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw== + dependencies: + anymatch "~3.1.2" + braces "~3.0.2" + glob-parent "~5.1.2" + is-binary-path "~2.1.0" + is-glob "~4.0.1" + normalize-path "~3.0.0" + readdirp "~3.6.0" + optionalDependencies: + fsevents "~2.3.2" + +chokidar@^4.0.3: + version "4.0.3" + resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-4.0.3.tgz#7be37a4c03c9aee1ecfe862a4a23b2c70c205d30" + integrity sha512-Qgzu8kfBvo+cA4962jnP1KkS6Dop5NS6g7R5LFYJr4b8Ub94PPQXUksCw9PvXoeXPRRddRNC5C1JQUR2SMGtnA== + dependencies: + readdirp "^4.0.1" + +chownr@^1.1.1: + version "1.1.4" + resolved "https://registry.yarnpkg.com/chownr/-/chownr-1.1.4.tgz#6fc9d7b42d32a583596337666e7d08084da2cc6b" + integrity sha512-jJ0bqzaylmJtVnNgzTeSOs8DPavpbYgEr/b0YL8/2GO3xJEhInFmhKMUnEJQjZumK7KXGFhUy89PrsJWlakBVg== + +chownr@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/chownr/-/chownr-2.0.0.tgz#15bfbe53d2eab4cf70f18a8cd68ebe5b3cb1dece" + integrity sha512-bIomtDF5KGpdogkLd9VspvFzk9KfpyyGlS8YFVZl7TGPBHL5snIOnxeshwVgPteQ9b4Eydl+pVbIyE1DcvCWgQ== + +chrome-trace-event@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/chrome-trace-event/-/chrome-trace-event-1.0.4.tgz#05bffd7ff928465093314708c93bdfa9bd1f0f5b" + integrity sha512-rNjApaLzuwaOTjCiT8lSDdGN1APCiqkChLMJxJPWLunPAt5fy8xgU9/jNOchV84wfIxrA0lRQB7oCT8jrn/wrQ== + +chromium-pickle-js@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/chromium-pickle-js/-/chromium-pickle-js-0.2.0.tgz#04a106672c18b085ab774d983dfa3ea138f22205" + integrity sha512-1R5Fho+jBq0DDydt+/vHWj5KJNJCKdARKOCwZUen84I5BreWoLqRLANH1U87eJy1tiASPtMnGqJJq0ZsLoRPOw== + +ci-info@^3.2.0: + version "3.9.0" + resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-3.9.0.tgz#4279a62028a7b1f262f3473fc9605f5e218c59b4" + integrity sha512-NIxF55hv4nSqQswkAeiOi1r83xy8JldOFDTWiug55KBu9Jnblncd2U6ViHmYgHf01TPZS77NJBhBMKdWj9HQMQ== + +clean-stack@^2.0.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/clean-stack/-/clean-stack-2.2.0.tgz#ee8472dbb129e727b31e8a10a427dee9dfe4008b" + integrity sha512-4diC9HaTE+KRAMWhDhrGOECgWZxoevMc5TlkObMqNSsVU62PYzXZ/SMTjzyGAFF1YusgxGcSWTEXBhp0CPwQ1A== + +cli-cursor@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-3.1.0.tgz#264305a7ae490d1d03bf0c9ba7c925d1753af307" + integrity sha512-I/zHAwsKf9FqGoXM4WWRACob9+SNukZTd94DWF57E4toouRulbCxcUh6RKUEOQlYTHJnzkPMySvPNaaSLNfLZw== + dependencies: + restore-cursor "^3.1.0" + +cli-cursor@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-4.0.0.tgz#3cecfe3734bf4fe02a8361cbdc0f6fe28c6a57ea" + integrity sha512-VGtlMu3x/4DOtIUwEkRezxUZ2lBacNJCHash0N0WeZDBS+7Ux1dm3XWAgWYxLJFMMdOeXMHXorshEFhbMSGelg== + dependencies: + restore-cursor "^4.0.0" + +cli-cursor@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/cli-cursor/-/cli-cursor-5.0.0.tgz#24a4831ecf5a6b01ddeb32fb71a4b2088b0dce38" + integrity sha512-aCj4O5wKyszjMmDT4tZj93kxyydN/K5zPWSCe6/0AV/AA1pqe5ZBIw0a2ZfPQV7lL5/yb5HsUreJ6UFAF1tEQw== + dependencies: + restore-cursor "^5.0.0" + +cli-spinners@^2.5.0, cli-spinners@^2.9.2: + version "2.9.2" + resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-2.9.2.tgz#1773a8f4b9c4d6ac31563df53b3fc1d79462fe41" + integrity sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg== + +cli-truncate@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-2.1.0.tgz#c39e28bf05edcde5be3b98992a22deed5a2b93c7" + integrity sha512-n8fOixwDD6b/ObinzTrp1ZKFzbgvKZvuz/TvejnLn1aQfC6r52XEx85FmuC+3HI+JM7coBRXUvNqEU2PHVrHpg== + dependencies: + slice-ansi "^3.0.0" + string-width "^4.2.0" + +cli-truncate@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-3.1.0.tgz#3f23ab12535e3d73e839bb43e73c9de487db1389" + integrity sha512-wfOBkjXteqSnI59oPcJkcPl/ZmwvMMOj340qUIY1SKZCv0B9Cf4D4fAucRkIKQmsIuYK3x1rrgU7MeGRruiuiA== + dependencies: + slice-ansi "^5.0.0" + string-width "^5.0.0" + +client-only@^0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/client-only/-/client-only-0.0.1.tgz#38bba5d403c41ab150bff64a95c85013cf73bca1" + integrity sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA== + +cliui@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/cliui/-/cliui-8.0.1.tgz#0c04b075db02cbfe60dc8e6cf2f5486b1a3608aa" + integrity sha512-BSeNnyus75C4//NQ9gQt1/csTXyo/8Sb+afLAkzAptFuMsod9HFokGNudZpi/oQV73hnVK+sR+5PVRMd+Dr7YQ== + dependencies: + string-width "^4.2.0" + strip-ansi "^6.0.1" + wrap-ansi "^7.0.0" + +clone-response@^1.0.2: + version "1.0.3" + resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.3.tgz#af2032aa47816399cf5f0a1d0db902f517abb8c3" + integrity sha512-ROoL94jJH2dUVML2Y/5PEDNaSHgeOdSDicUyS7izcF63G6sTc/FTjLub4b8Il9S8S0beOfYt0TaA5qvFK+w0wA== + dependencies: + mimic-response "^1.0.0" + +clone@^1.0.2: + version "1.0.4" + resolved "https://registry.yarnpkg.com/clone/-/clone-1.0.4.tgz#da309cc263df15994c688ca902179ca3c7cd7c7e" + integrity sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg== + +cockatiel@^3.1.2: + version "3.2.1" + resolved "https://registry.yarnpkg.com/cockatiel/-/cockatiel-3.2.1.tgz#575f937bc4040a20ae27352a6d07c9c5a741981f" + integrity sha512-gfrHV6ZPkquExvMh9IOkKsBzNDk6sDuZ6DdBGUBkvFnTCqCxzpuq48RySgP0AnaqQkw2zynOFj9yly6T1Q2G5Q== + +codemirror@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/codemirror/-/codemirror-6.0.1.tgz#62b91142d45904547ee3e0e0e4c1a79158035a29" + integrity sha512-J8j+nZ+CdWmIeFIGXEFbFPtpiYacFMDR8GlHK3IyHQJMCaVRfGx9NT+Hxivv1ckLWPvNdZqndbr/7lVhrf/Svg== + dependencies: + "@codemirror/autocomplete" "^6.0.0" + "@codemirror/commands" "^6.0.0" + "@codemirror/language" "^6.0.0" + "@codemirror/lint" "^6.0.0" + "@codemirror/search" "^6.0.0" + "@codemirror/state" "^6.0.0" + "@codemirror/view" "^6.0.0" + +color-convert@^1.9.0: + version "1.9.3" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8" + integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg== + dependencies: + color-name "1.1.3" + +color-convert@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" + integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== + dependencies: + color-name "~1.1.4" + +color-convert@~0.5.0: + version "0.5.3" + resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-0.5.3.tgz#bdb6c69ce660fadffe0b0007cc447e1b9f7282bd" + integrity sha512-RwBeO/B/vZR3dfKL1ye/vx8MHZ40ugzpyfeVG5GsiuGnrlMWe2o8wxBbLCpw9CsxV+wHuzYlCiWnybrIA0ling== + +color-name@1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25" + integrity sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw== + +color-name@~1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" + integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== + +colorette@^1.2.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/colorette/-/colorette-1.4.0.tgz#5190fbb87276259a86ad700bff2c6d6faa3fca40" + integrity sha512-Y2oEozpomLn7Q3HFP7dpww7AtMJplbM9lGZP6RDfHqmbeRjiwRg4n6VM6j4KLmRke85uWEI7JqF17f3pqdRA0g== + +colorette@^2.0.20: + version "2.0.20" + resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.20.tgz#9eb793e6833067f7235902fcd3b09917a000a95a" + integrity sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w== + +combined-stream@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/combined-stream/-/combined-stream-1.0.8.tgz#c3d45a8b34fd730631a110a8a2520682b31d5a7f" + integrity sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg== + dependencies: + delayed-stream "~1.0.0" + +commander@^11.1.0: + version "11.1.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-11.1.0.tgz#62fdce76006a68e5c1ab3314dc92e800eb83d906" + integrity sha512-yPVavfyCcRhmorC7rWlkHn15b4wDVgVmBA7kV4QVBsF7kv/9TKJAbAXVTxvTnwP8HHKjRCJDClKbciiYS7p0DQ== + +commander@^12.1.0: + version "12.1.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-12.1.0.tgz#01423b36f501259fdaac4d0e4d60c96c991585d3" + integrity sha512-Vw8qHK3bZM9y/P10u3Vib8o/DdkvA2OtPtZvD871QKjy74Wj1WSKFILMPRPSdUSx5RFK1arlJzEtA4PkFgnbuA== + +commander@^4.0.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/commander/-/commander-4.1.1.tgz#9fd602bd936294e9e9ef46a3f4d6964044b18068" + integrity sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA== + +commander@^5.0.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-5.1.0.tgz#46abbd1652f8e059bddaef99bbdcb2ad9cf179ae" + integrity sha512-P0CysNDQ7rtVw4QIQtm+MRxV66vKFSvlsQvGYXZWR3qFU0jlMKHZZZgw8e+8DSah4UDKMqnknRDQz+xuQXQ/Zg== + +commander@^8.0.0: + version "8.3.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-8.3.0.tgz#4837ea1b2da67b9c616a67afbb0fafee567bca66" + integrity sha512-OkTL9umf+He2DZkUq8f8J9of7yL6RJKI24dVITBmNfZBmri9zYZQrKkuXiKhyfPSu8tUhnVBB1iKXevvnlR4Ww== + +commander@^9.4.0: + version "9.5.0" + resolved "https://registry.yarnpkg.com/commander/-/commander-9.5.0.tgz#bc08d1eb5cedf7ccb797a96199d41c7bc3e60d30" + integrity sha512-KRs7WVDKg86PWiuAqhDrAQnTXZKraVcCc6vFdL14qrZ/DcWwuRo7VoiYXalXO7S5GKpqYiVEwCbgFDfxNHKJBQ== + +compare-version@^0.1.2: + version "0.1.2" + resolved "https://registry.yarnpkg.com/compare-version/-/compare-version-0.1.2.tgz#0162ec2d9351f5ddd59a9202cba935366a725080" + integrity sha512-pJDh5/4wrEnXX/VWRZvruAGHkzKdr46z11OlTPN+VrATlWWhSKewNCJ1futCO5C7eJB3nPMFZA1LeYtcFboZ2A== + +concat-map@0.0.1: + version "0.0.1" + resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" + integrity sha512-/Srv4dswyQNBfohGpz9o6Yb3Gz3SrUDqBH5rTuhGR7ahtlbYKnVxw2bCFMRljaA7EXHaXZ8wsHdodFvbkhKmqg== + +confbox@^0.1.8: + version "0.1.8" + resolved "https://registry.yarnpkg.com/confbox/-/confbox-0.1.8.tgz#820d73d3b3c82d9bd910652c5d4d599ef8ff8b06" + integrity sha512-RMtmw0iFkeR4YV+fUOSucriAQNb9g8zFR52MWCtl+cCZOFRNL6zeB395vPzFhEjjn4fMxXudmELnl/KF/WrK6w== + +config-file-ts@0.2.8-rc1: + version "0.2.8-rc1" + resolved "https://registry.yarnpkg.com/config-file-ts/-/config-file-ts-0.2.8-rc1.tgz#fb7fc6ccb2e313f69dbeb78f1db0b00038049de0" + integrity sha512-GtNECbVI82bT4RiDIzBSVuTKoSHufnU7Ce7/42bkWZJZFLjmDF2WBpVsvRkhKCfKBnTBb3qZrBwPpFBU/Myvhg== + dependencies: + glob "^10.3.12" + typescript "^5.4.3" + +content-disposition@0.5.4: + version "0.5.4" + resolved "https://registry.yarnpkg.com/content-disposition/-/content-disposition-0.5.4.tgz#8b82b4efac82512a02bb0b1dcec9d2c5e8eb5bfe" + integrity sha512-FveZTNuGw04cxlAiWbzi6zTAL/lhehaWbTtgluJh4/E95DqMwTmha3KZN1aAWA8cFIhHzMZUvLevkw5Rqk+tSQ== + dependencies: + safe-buffer "5.2.1" + +content-type@~1.0.4, content-type@~1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/content-type/-/content-type-1.0.5.tgz#8b773162656d1d1086784c8f23a54ce6d73d7918" + integrity sha512-nTjqfcBFEipKdXCv4YDQWCfmcLZKm81ldF0pAopTvyrFGVbcR6P/VAAd5G7N+0tTr8QqiU0tFadD6FK4NtJwOA== + +convert-source-map@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/convert-source-map/-/convert-source-map-2.0.0.tgz#4b560f649fc4e918dd0ab75cf4961e8bc882d82a" + integrity sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg== + +cookie-signature@1.0.6: + version "1.0.6" + resolved "https://registry.yarnpkg.com/cookie-signature/-/cookie-signature-1.0.6.tgz#e303a882b342cc3ee8ca513a79999734dab3ae2c" + integrity sha512-QADzlaHc8icV8I7vbaJXJwod9HWYp8uCqf1xa4OfNu1T7JVxQIrUgOWtHdNDtPiywmFbiS12VjotIXLrKM3orQ== + +cookie@0.7.1: + version "0.7.1" + resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.7.1.tgz#2f73c42142d5d5cf71310a74fc4ae61670e5dbc9" + integrity sha512-6DnInpx7SJ2AK3+CTUE/ZM0vWTUboZCegxhC2xiIydHR9jNuTAASBrfEpHhiGOZw/nX51bHt6YQl8jsGo4y/0w== + +core-js-compat@^3.40.0: + version "3.41.0" + resolved "https://registry.yarnpkg.com/core-js-compat/-/core-js-compat-3.41.0.tgz#4cdfce95f39a8f27759b667cf693d96e5dda3d17" + integrity sha512-RFsU9LySVue9RTwdDVX/T0e2Y6jRYWXERKElIjpuEOEnxaXffI0X7RUwVzfYLfzuLXSNJDYoRYUAmRUcyln20A== + dependencies: + browserslist "^4.24.4" + +core-util-is@1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" + integrity sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ== + +core-util-is@~1.0.0: + version "1.0.3" + resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.3.tgz#a6042d3634c2b27e9328f837b965fac83808db85" + integrity sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ== + +corser@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/corser/-/corser-2.0.1.tgz#8eda252ecaab5840dcd975ceb90d9370c819ff87" + integrity sha512-utCYNzRSQIZNPIcGZdQc92UVJYAhtGAteCFg0yRaFm8f0P+CPtyGyHXJcGXnffjCybUCEx3FQ2G7U3/o9eIkVQ== + +crc@^3.8.0: + version "3.8.0" + resolved "https://registry.yarnpkg.com/crc/-/crc-3.8.0.tgz#ad60269c2c856f8c299e2c4cc0de4556914056c6" + integrity sha512-iX3mfgcTMIq3ZKLIsVFAbv7+Mc10kxabAGQb8HvjA1o3T1PIYprbakQ65d3I+2HGHt6nSKkM9PYjgoJO2KcFBQ== + dependencies: + buffer "^5.1.0" + +create-require@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/create-require/-/create-require-1.1.1.tgz#c1d7e8f1e5f6cfc9ff65f9cd352d37348756c333" + integrity sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ== + +crelt@^1.0.5: + version "1.0.6" + resolved "https://registry.yarnpkg.com/crelt/-/crelt-1.0.6.tgz#7cc898ea74e190fb6ef9dae57f8f81cf7302df72" + integrity sha512-VQ2MBenTq1fWZUH9DJNGti7kKv6EeAuYr3cLwxUWhIu1baTaXh4Ib5W2CqHVqib4/MqbYGJqiL3Zb8GJZr3l4g== + +cross-dirname@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/cross-dirname/-/cross-dirname-0.1.0.tgz#b899599f30a5389f59e78c150e19f957ad16a37c" + integrity sha512-+R08/oI0nl3vfPcqftZRpytksBXDzOUveBq/NBVx0sUp1axwzPQrKinNx5yd5sxPu8j1wIy8AfnVQ+5eFdha6Q== + +cross-env@^7.0.3: + version "7.0.3" + resolved "https://registry.yarnpkg.com/cross-env/-/cross-env-7.0.3.tgz#865264b29677dc015ba8418918965dd232fc54cf" + integrity sha512-+/HKd6EgcQCJGh2PSjZuUitQBQynKor4wrFbRg4DtAgS1aWO+gU52xpH7M9ScGgXSYmAVS9bIJ8EzuaGw0oNAw== + dependencies: + cross-spawn "^7.0.1" + +cross-fetch@^3.1.5: + version "3.2.0" + resolved "https://registry.yarnpkg.com/cross-fetch/-/cross-fetch-3.2.0.tgz#34e9192f53bc757d6614304d9e5e6fb4edb782e3" + integrity sha512-Q+xVJLoGOeIMXZmbUK4HYk+69cQH6LudR0Vu/pRm2YlU/hDV9CiS0gKUMaWY5f2NeUH9C1nV3bsTlCo0FsTV1Q== + dependencies: + node-fetch "^2.7.0" + +cross-spawn@^6.0.0: + version "6.0.6" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-6.0.6.tgz#30d0efa0712ddb7eb5a76e1e8721bffafa6b5d57" + integrity sha512-VqCUuhcd1iB+dsv8gxPttb5iZh/D0iubSP21g36KXdEuf6I5JiioesUVjpCdHV9MZRUfVFlvwtIUyPfxo5trtw== + dependencies: + nice-try "^1.0.4" + path-key "^2.0.1" + semver "^5.5.0" + shebang-command "^1.2.0" + which "^1.2.9" + +cross-spawn@^7.0.1, cross-spawn@^7.0.2, cross-spawn@^7.0.3, cross-spawn@^7.0.6: + version "7.0.6" + resolved "https://registry.yarnpkg.com/cross-spawn/-/cross-spawn-7.0.6.tgz#8a58fe78f00dcd70c370451759dfbfaf03e8ee9f" + integrity sha512-uV2QOWP2nWzsy2aMp8aRibhi9dlzF5Hgh5SHaB9OiTGEyDTiJJyx0uy51QXdyWbtAHNua4XJzUKca3OzKUd3vA== + dependencies: + path-key "^3.1.0" + shebang-command "^2.0.0" + which "^2.0.1" + +css-line-break@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/css-line-break/-/css-line-break-2.1.0.tgz#bfef660dfa6f5397ea54116bb3cb4873edbc4fa0" + integrity sha512-FHcKFCZcAha3LwfVBhCQbW2nCNbkZXn7KVUJcsT5/P8YmfsVja0FMPJr0B903j/E69HUphKiV9iQArX8SDYA4w== + dependencies: + utrie "^1.0.2" + +css-select@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/css-select/-/css-select-5.1.0.tgz#b8ebd6554c3637ccc76688804ad3f6a6fdaea8a6" + integrity sha512-nwoRF1rvRRnnCqqY7updORDsuqKzqYJ28+oSMaJMMgOauh3fvwHqMS7EZpIPqK8GL+g9mKxF1vP/ZjSeNjEVHg== + dependencies: + boolbase "^1.0.0" + css-what "^6.1.0" + domhandler "^5.0.2" + domutils "^3.0.1" + nth-check "^2.0.1" + +css-what@^6.1.0: + version "6.1.0" + resolved "https://registry.yarnpkg.com/css-what/-/css-what-6.1.0.tgz#fb5effcf76f1ddea2c81bdfaa4de44e79bac70f4" + integrity sha512-HTUrgRJ7r4dsZKU6GjmpfRK1O76h97Z8MfS1G0FozR+oF2kG6Vfe8JE6zwrkbxigziPHinCJ+gCPjA9EaBDtRw== + +css.escape@^1.5.1: + version "1.5.1" + resolved "https://registry.yarnpkg.com/css.escape/-/css.escape-1.5.1.tgz#42e27d4fa04ae32f931a4b4d4191fa9cddee97cb" + integrity sha512-YUifsXXuknHlUsmlgyY0PKzgPOr7/FjCePfHNt0jxm83wHZi44VDMQ7/fGNkjY3/jV1MC+1CmZbaHzugyeRtpg== + +cssesc@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/cssesc/-/cssesc-3.0.0.tgz#37741919903b868565e1c09ea747445cd18983ee" + integrity sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg== + +cssfontparser@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/cssfontparser/-/cssfontparser-1.2.1.tgz#f4022fc8f9700c68029d542084afbaf425a3f3e3" + integrity sha512-6tun4LoZnj7VN6YeegOVb67KBX/7JJsqvj+pv3ZA7F878/eN33AbGa5b/S/wXxS/tcp8nc40xRUrsPlxIyNUPg== + +csstype@^3.0.2, csstype@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/csstype/-/csstype-3.1.3.tgz#d80ff294d114fb0e6ac500fbf85b60137d7eff81" + integrity sha512-M1uQkMl8rQK/szD0LNhtqxIPLpimGm8sOBwU7lLnCpSbTyY3yeU1Vc7l4KT5zT4s/yOxHH5O7tIuuLOCnLADRw== + +damerau-levenshtein@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/damerau-levenshtein/-/damerau-levenshtein-1.0.8.tgz#b43d286ccbd36bc5b2f7ed41caf2d0aba1f8a6e7" + integrity sha512-sdQSFB7+llfUcQHUQO3+B8ERRj0Oa4w9POWMI/puGtuf7gFywGmkaLCElnudfTiKZV+NvHqL0ifzdrI8Ro7ESA== + +data-uri-to-buffer@^4.0.0: + version "4.0.1" + resolved "https://registry.yarnpkg.com/data-uri-to-buffer/-/data-uri-to-buffer-4.0.1.tgz#d8feb2b2881e6a4f58c2e08acfd0e2834e26222e" + integrity sha512-0R9ikRb668HB7QDxT1vkpuUBtqc53YyAwMwGeUFKRojY/NWKvdZ+9UYtRfGmhqNbRkTSVpMbmyhXipFFv2cb/A== + +data-view-buffer@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/data-view-buffer/-/data-view-buffer-1.0.2.tgz#211a03ba95ecaf7798a8c7198d79536211f88570" + integrity sha512-EmKO5V3OLXh1rtK2wgXRansaK1/mtVdTUEiEI0W8RkvgT05kfxaH29PliLnpLP73yYO6142Q72QNa8Wx/A5CqQ== + dependencies: + call-bound "^1.0.3" + es-errors "^1.3.0" + is-data-view "^1.0.2" + +data-view-byte-length@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/data-view-byte-length/-/data-view-byte-length-1.0.2.tgz#9e80f7ca52453ce3e93d25a35318767ea7704735" + integrity sha512-tuhGbE6CfTM9+5ANGf+oQb72Ky/0+s3xKUpHvShfiz2RxMFgFPjsXuRLBVMtvMs15awe45SRb83D6wH4ew6wlQ== + dependencies: + call-bound "^1.0.3" + es-errors "^1.3.0" + is-data-view "^1.0.2" + +data-view-byte-offset@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/data-view-byte-offset/-/data-view-byte-offset-1.0.1.tgz#068307f9b71ab76dbbe10291389e020856606191" + integrity sha512-BS8PfmtDGnrgYdOonGZQdLZslWIeCGFP9tpan0hi1Co2Zr2NKADsvGYA8XxuG/4UWgJ6Cjtv+YJnB6MM69QGlQ== + dependencies: + call-bound "^1.0.2" + es-errors "^1.3.0" + is-data-view "^1.0.1" + +debug@2.6.9, debug@^2.2.0: + version "2.6.9" + resolved "https://registry.yarnpkg.com/debug/-/debug-2.6.9.tgz#5d128515df134ff327e90a4c93f4e077a536341f" + integrity sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA== + dependencies: + ms "2.0.0" + +debug@4, debug@^4, debug@^4.0.1, debug@^4.1.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2, debug@^4.3.3, debug@^4.3.4, debug@^4.3.5, debug@^4.3.6, debug@^4.4.0: + version "4.4.0" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.4.0.tgz#2b3f2aea2ffeb776477460267377dc8710faba8a" + integrity sha512-6WTZ/IxCY/T6BALoZHaE4ctp9xm+Z5kY/pzYaCHRFeyVhojxlrm+46y68HA6hr0TcwEssoxNiDEUJQjfPZ/RYA== + dependencies: + ms "^2.1.3" + +debug@^3.2.7: + version "3.2.7" + resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.7.tgz#72580b7e9145fb39b6676f9c5e5fb100b934179a" + integrity sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ== + dependencies: + ms "^2.1.1" + +decamelize@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-4.0.0.tgz#aa472d7bf660eb15f3494efd531cab7f2a709837" + integrity sha512-9iE1PgSik9HeIIw2JO94IidnE3eBoQrFJ3w7sFuzSX4DpmZ3v5sZpUiV5Swcf6mQEF+Y0ru8Neo+p+nyh2J+hQ== + +decamelize@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/decamelize/-/decamelize-6.0.0.tgz#8cad4d916fde5c41a264a43d0ecc56fe3d31749e" + integrity sha512-Fv96DCsdOgB6mdGl67MT5JaTNKRzrzill5OH5s8bjYJXVlcXyPYGyPsUkWyGV5p1TXI5esYIYMMeDJL0hEIwaA== + +decompress-response@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-6.0.0.tgz#ca387612ddb7e104bd16d85aab00d5ecf09c66fc" + integrity sha512-aW35yZM6Bb/4oJlZncMH2LCoZtJXTRxES17vE3hoRiowU2kWHaJKFkSBDnDR+cm9J+9QhXmREyIfv0pji9ejCQ== + dependencies: + mimic-response "^3.1.0" + +deep-eql@^4.1.3: + version "4.1.4" + resolved "https://registry.yarnpkg.com/deep-eql/-/deep-eql-4.1.4.tgz#d0d3912865911bb8fac5afb4e3acfa6a28dc72b7" + integrity sha512-SUwdGfqdKOwxCPeVYjwSyRpJ7Z+fhpwIAtmCUdZIWZ/YP5R9WAsyuSgpLVDi9bjWoN2LXHNss/dk3urXtdQxGg== + dependencies: + type-detect "^4.0.0" + +deep-eql@^5.0.1: + version "5.0.2" + resolved "https://registry.yarnpkg.com/deep-eql/-/deep-eql-5.0.2.tgz#4b756d8d770a9257300825d52a2c2cff99c3a341" + integrity sha512-h5k/5U50IJJFpzfL6nO9jaaumfjO/f2NjK/oYB2Djzm4p9L+3T9qWpZqZ2hAbLPuuYq9wrU08WQyBTL5GbPk5Q== + +deep-extend@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz#c4fa7c95404a17a9c3e8ca7e1537312b736330ac" + integrity sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA== + +deep-is@^0.1.3: + version "0.1.4" + resolved "https://registry.yarnpkg.com/deep-is/-/deep-is-0.1.4.tgz#a6f2dce612fadd2ef1f519b73551f17e85199831" + integrity sha512-oIPzksmTg4/MriiaYGO+okXDT7ztn/w3Eptv/+gSIdMdKsJo0u4CfYNFJPy+4SKMuCqGw2wxnA+URMg3t8a/bQ== + +default-browser-id@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/default-browser-id/-/default-browser-id-5.0.0.tgz#a1d98bf960c15082d8a3fa69e83150ccccc3af26" + integrity sha512-A6p/pu/6fyBcA1TRz/GqWYPViplrftcW2gZC9q79ngNCKAeR/X3gcEdXQHl4KNXV+3wgIJ1CPkJQ3IHM6lcsyA== + +default-browser@^5.2.1: + version "5.2.1" + resolved "https://registry.yarnpkg.com/default-browser/-/default-browser-5.2.1.tgz#7b7ba61204ff3e425b556869ae6d3e9d9f1712cf" + integrity sha512-WY/3TUME0x3KPYdRRxEJJvXRHV4PyPoUsxtZa78lwItwRQRHhd2U9xOscaT/YTf8uCXIAjeJOFBVEh/7FtD8Xg== + dependencies: + bundle-name "^4.1.0" + default-browser-id "^5.0.0" + +defaults@^1.0.3: + version "1.0.4" + resolved "https://registry.yarnpkg.com/defaults/-/defaults-1.0.4.tgz#b0b02062c1e2aa62ff5d9528f0f98baa90978d7a" + integrity sha512-eFuaLoy/Rxalv2kr+lqMlUnrDWV+3j4pljOIJgLIhI058IQfWJ7vXhyEIHu+HtC738klGALYxOKDO0bQP3tg8A== + dependencies: + clone "^1.0.2" + +defer-to-connect@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-2.0.1.tgz#8016bdb4143e4632b77a3449c6236277de520587" + integrity sha512-4tvttepXG1VaYGrRibk5EwJd1t4udunSOVMdLSAL6mId1ix438oPwPZMALY41FCijukO1L0twNcGsdzS7dHgDg== + +define-data-property@^1.0.1, define-data-property@^1.1.4: + version "1.1.4" + resolved "https://registry.yarnpkg.com/define-data-property/-/define-data-property-1.1.4.tgz#894dc141bb7d3060ae4366f6a0107e68fbe48c5e" + integrity sha512-rBMvIzlpA8v6E+SJZoo++HAYqsLrkg7MSfIinMPFhmkorw7X+dOXVJQs+QT69zGkzMyfDnIMN2Wid1+NbL3T+A== + dependencies: + es-define-property "^1.0.0" + es-errors "^1.3.0" + gopd "^1.0.1" + +define-lazy-prop@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/define-lazy-prop/-/define-lazy-prop-3.0.0.tgz#dbb19adfb746d7fc6d734a06b72f4a00d021255f" + integrity sha512-N+MeXYoqr3pOgn8xfyRPREN7gHakLYjhsHhWGT3fWAiL4IkAt0iDw14QiiEm2bE30c5XX5q0FtAA3CK5f9/BUg== + +define-properties@^1.1.3, define-properties@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/define-properties/-/define-properties-1.2.1.tgz#10781cc616eb951a80a034bafcaa7377f6af2b6c" + integrity sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg== + dependencies: + define-data-property "^1.0.1" + has-property-descriptors "^1.0.0" + object-keys "^1.1.1" + +delayed-stream@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/delayed-stream/-/delayed-stream-1.0.0.tgz#df3ae199acadfb7d440aaae0b29e2272b24ec619" + integrity sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ== + +depd@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/depd/-/depd-2.0.0.tgz#b696163cc757560d09cf22cc8fad1571b79e76df" + integrity sha512-g7nH6P6dyDioJogAAGprGpCtVImJhpPk/roCzdb3fIh61/s/nPsfR6onyMwkCAR/OlC3yBC0lESvUoQEAssIrw== + +dequal@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/dequal/-/dequal-2.0.3.tgz#2644214f1997d39ed0ee0ece72335490a7ac67be" + integrity sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA== + +destroy@1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/destroy/-/destroy-1.2.0.tgz#4803735509ad8be552934c67df614f94e66fa015" + integrity sha512-2sJGJTaXIIaR1w4iJSNoN0hnMY7Gpc/n8D4qSCJw8QqFWXf7cuAgnEHxBpweaVcPevC2l3KpjYCx3NypQQgaJg== + +detect-libc@^2.0.0, detect-libc@^2.0.1: + version "2.0.4" + resolved "https://registry.yarnpkg.com/detect-libc/-/detect-libc-2.0.4.tgz#f04715b8ba815e53b4d8109655b6508a6865a7e8" + integrity sha512-3UDv+G9CsCKO1WKMGw9fwq/SWJYbI0c5Y7LU1AXYoDdbhE2AHQ6N6Nb34sG8Fj7T5APy8qXDCKuuIHd1BR0tVA== + +detect-node@^2.0.4: + version "2.1.0" + resolved "https://registry.yarnpkg.com/detect-node/-/detect-node-2.1.0.tgz#c9c70775a49c3d03bc2c06d9a73be550f978f8b1" + integrity sha512-T0NIuQpnTvFDATNuHN5roPwSBG83rFsuO+MXXH9/3N1eFbn4wcPjttvjMLEPWJ0RGUYgQE7cGgS3tNxbqCGM7g== + +didyoumean@^1.2.2: + version "1.2.2" + resolved "https://registry.yarnpkg.com/didyoumean/-/didyoumean-1.2.2.tgz#989346ffe9e839b4555ecf5666edea0d3e8ad037" + integrity sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw== + +diff-sequences@^29.6.3: + version "29.6.3" + resolved "https://registry.yarnpkg.com/diff-sequences/-/diff-sequences-29.6.3.tgz#4deaf894d11407c51efc8418012f9e70b84ea921" + integrity sha512-EjePK1srD3P08o2j4f0ExnylqRs5B9tJjcp9t1krH2qRi8CCdsYfwe9JgSLurFBWwq4uOlipzfk5fHNvwFKr8Q== + +diff@^4.0.1: + version "4.0.2" + resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d" + integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== + +diff@^5.2.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/diff/-/diff-5.2.0.tgz#26ded047cd1179b78b9537d5ef725503ce1ae531" + integrity sha512-uIFDxqpRZGZ6ThOk84hEfqWoHx2devRFvpTZcTHur85vImfaxUbTW9Ryh4CpCuDnToOP1CEtXKIgytHBPVff5A== + +diff@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/diff/-/diff-7.0.0.tgz#3fb34d387cd76d803f6eebea67b921dab0182a9a" + integrity sha512-PJWHUb1RFevKCwaFA9RlG5tCd+FO5iRh9A8HEtkmBH2Li03iJriB6m6JIN4rGz3K3JLawI7/veA1xzRKP6ISBw== + +dir-compare@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/dir-compare/-/dir-compare-4.2.0.tgz#d1d4999c14fbf55281071fdae4293b3b9ce86f19" + integrity sha512-2xMCmOoMrdQIPHdsTawECdNPwlVFB9zGcz3kuhmBO6U3oU+UQjsue0i8ayLKpgBcm+hcXPMVSGUN9d+pvJ6+VQ== + dependencies: + minimatch "^3.0.5" + p-limit "^3.1.0 " + +dlv@^1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/dlv/-/dlv-1.1.3.tgz#5c198a8a11453596e751494d49874bc7732f2e79" + integrity sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA== + +dmg-builder@26.0.12: + version "26.0.12" + resolved "https://registry.yarnpkg.com/dmg-builder/-/dmg-builder-26.0.12.tgz#6996ad0bab80a861c9a7b33ee9734d4f60566b46" + integrity sha512-59CAAjAhTaIMCN8y9kD573vDkxbs1uhDcrFLHSgutYdPcGOU35Rf95725snvzEOy4BFB7+eLJ8djCNPmGwG67w== + dependencies: + app-builder-lib "26.0.12" + builder-util "26.0.11" + builder-util-runtime "9.3.1" + fs-extra "^10.1.0" + iconv-lite "^0.6.2" + js-yaml "^4.1.0" + optionalDependencies: + dmg-license "^1.0.11" + +dmg-license@^1.0.11: + version "1.0.11" + resolved "https://registry.yarnpkg.com/dmg-license/-/dmg-license-1.0.11.tgz#7b3bc3745d1b52be7506b4ee80cb61df6e4cd79a" + integrity sha512-ZdzmqwKmECOWJpqefloC5OJy1+WZBBse5+MR88z9g9Zn4VY+WYUkAyojmhzJckH5YbbZGcYIuGAkY5/Ys5OM2Q== + dependencies: + "@types/plist" "^3.0.1" + "@types/verror" "^1.10.3" + ajv "^6.10.0" + crc "^3.8.0" + iconv-corefoundation "^1.1.7" + plist "^3.0.4" + smart-buffer "^4.0.2" + verror "^1.10.0" + +dns-packet@^5.2.2: + version "5.6.1" + resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-5.6.1.tgz#ae888ad425a9d1478a0674256ab866de1012cf2f" + integrity sha512-l4gcSouhcgIKRvyy99RNVOgxXiicE+2jZoNmaNmZ6JXiGajBOJAesk1OBlJuM5k2c+eudGdLxDqXuPCKIj6kpw== + dependencies: + "@leichtgewicht/ip-codec" "^2.0.1" + +doctrine@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-2.1.0.tgz#5cd01fc101621b42c4cd7f5d1a66243716d3f39d" + integrity sha512-35mSku4ZXK0vfCuHEDAwt55dg2jNajHZ1odvF+8SSr82EsZY4QmXfuWso8oEd8zRhVObSN18aM0CjSdoBX7zIw== + dependencies: + esutils "^2.0.2" + +doctrine@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/doctrine/-/doctrine-3.0.0.tgz#addebead72a6574db783639dc87a121773973961" + integrity sha512-yS+Q5i3hBf7GBkd4KG8a7eBNNWNGLTaEwwYWUijIYM7zrlYDM0BFXHjjPWlWZ1Rg7UaddZeIDmi9jF3HmqiQ2w== + dependencies: + esutils "^2.0.2" + +dom-accessibility-api@^0.5.6, dom-accessibility-api@^0.5.9: + version "0.5.16" + resolved "https://registry.yarnpkg.com/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz#5a7429e6066eb3664d911e33fb0e45de8eb08453" + integrity sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg== + +dom-serializer@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/dom-serializer/-/dom-serializer-2.0.0.tgz#e41b802e1eedf9f6cae183ce5e622d789d7d8e53" + integrity sha512-wIkAryiqt/nV5EQKqQpo3SToSOV9J0DnbJqwK7Wv/Trc92zIAYZ4FlMu+JPFW1DfGFt81ZTCGgDEabffXeLyJg== + dependencies: + domelementtype "^2.3.0" + domhandler "^5.0.2" + entities "^4.2.0" + +domelementtype@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/domelementtype/-/domelementtype-2.3.0.tgz#5c45e8e869952626331d7aab326d01daf65d589d" + integrity sha512-OLETBj6w0OsagBwdXnPdN0cnMfF9opN69co+7ZrbfPGrdpPVNBUj02spi6B1N7wChLQiPn4CSH/zJvXw56gmHw== + +domhandler@^5.0.2, domhandler@^5.0.3: + version "5.0.3" + resolved "https://registry.yarnpkg.com/domhandler/-/domhandler-5.0.3.tgz#cc385f7f751f1d1fc650c21374804254538c7d31" + integrity sha512-cgwlv/1iFQiFnU96XXgROh8xTeetsnJiDsTc7TYCLFd9+/WNkIqPTxiM/8pSd8VIrhXGTf1Ny1q1hquVqDJB5w== + dependencies: + domelementtype "^2.3.0" + +domutils@^3.0.1, domutils@^3.1.0: + version "3.2.2" + resolved "https://registry.yarnpkg.com/domutils/-/domutils-3.2.2.tgz#edbfe2b668b0c1d97c24baf0f1062b132221bc78" + integrity sha512-6kZKyUajlDuqlHKVX1w7gyslj9MPIXzIFiz/rGu35uC1wMi+kMhQwGhl4lt9unC9Vb9INnY9Z3/ZA3+FhASLaw== + dependencies: + dom-serializer "^2.0.0" + domelementtype "^2.3.0" + domhandler "^5.0.3" + +dotenv-expand@^11.0.6: + version "11.0.7" + resolved "https://registry.yarnpkg.com/dotenv-expand/-/dotenv-expand-11.0.7.tgz#af695aea007d6fdc84c86cd8d0ad7beb40a0bd08" + integrity sha512-zIHwmZPRshsCdpMDyVsqGmgyP0yT8GAgXUnkdAoJisxvf33k7yO6OuoKmcTGuXPWSsm8Oh88nZicRLA9Y0rUeA== + dependencies: + dotenv "^16.4.5" + +dotenv@^16.0.3, dotenv@^16.4.5: + version "16.5.0" + resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-16.5.0.tgz#092b49f25f808f020050051d1ff258e404c78692" + integrity sha512-m/C+AwOAr9/W1UOIZUo232ejMNnJAJtYQjUbHoNTBNTJSvqzzDh7vnrei3o3r3m9blf6ZoDkvcw0VmozNRFJxg== + +dpdm@^3.14.0: + version "3.14.0" + resolved "https://registry.yarnpkg.com/dpdm/-/dpdm-3.14.0.tgz#12a60a2d88b23981c91239b86e7462a5c203e5e9" + integrity sha512-YJzsFSyEtj88q5eTELg3UWU7TVZkG1dpbF4JDQ3t1b07xuzXmdoGeSz9TKOke1mUuOpWlk4q+pBh+aHzD6GBTg== + dependencies: + chalk "^4.1.2" + fs-extra "^11.1.1" + glob "^10.3.4" + ora "^5.4.1" + tslib "^2.6.2" + typescript "^5.2.2" + yargs "^17.7.2" + +dunder-proto@^1.0.0, dunder-proto@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/dunder-proto/-/dunder-proto-1.0.1.tgz#d7ae667e1dc83482f8b70fd0f6eefc50da30f58a" + integrity sha512-KIN/nDJBQRcXw0MLVhZE9iQHmG68qAVIBg9CqmUYjmQIhgij9U5MFvrqkUL5FbtyyzZuOeOt0zdeRe4UY7ct+A== + dependencies: + call-bind-apply-helpers "^1.0.1" + es-errors "^1.3.0" + gopd "^1.2.0" + +eastasianwidth@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/eastasianwidth/-/eastasianwidth-0.2.0.tgz#696ce2ec0aa0e6ea93a397ffcf24aa7840c827cb" + integrity sha512-I88TYZWc9XiYHRQ4/3c5rjjfgkjhLyW2luGIheGERbNQ6OY7yTybanSpDXZa8y7VUP9YmDcYa+eyq4ca7iLqWA== + +ecdsa-sig-formatter@1.0.11: + version "1.0.11" + resolved "https://registry.yarnpkg.com/ecdsa-sig-formatter/-/ecdsa-sig-formatter-1.0.11.tgz#ae0f0fa2d85045ef14a817daa3ce9acd0489e5bf" + integrity sha512-nagl3RYrbNv6kQkeJIpt6NJZy8twLB/2vtz6yN9Z4vRKHN4/QZJIEbqohALSgwKdnksuY3k5Addp5lg8sVoVcQ== + dependencies: + safe-buffer "^5.0.1" + +ee-first@1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" + integrity sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow== + +ejs@^3.1.8: + version "3.1.10" + resolved "https://registry.yarnpkg.com/ejs/-/ejs-3.1.10.tgz#69ab8358b14e896f80cc39e62087b88500c3ac3b" + integrity sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA== + dependencies: + jake "^10.8.5" + +electron-builder@^26.0.12: + version "26.0.12" + resolved "https://registry.yarnpkg.com/electron-builder/-/electron-builder-26.0.12.tgz#797af2e70efdd96c9ea5d8a8164b8728c90d65ff" + integrity sha512-cD1kz5g2sgPTMFHjLxfMjUK5JABq3//J4jPswi93tOPFz6btzXYtK5NrDt717NRbukCUDOrrvmYVOWERlqoiXA== + dependencies: + app-builder-lib "26.0.12" + builder-util "26.0.11" + builder-util-runtime "9.3.1" + chalk "^4.1.2" + dmg-builder "26.0.12" + fs-extra "^10.1.0" + is-ci "^3.0.0" + lazy-val "^1.0.5" + simple-update-notifier "2.0.0" + yargs "^17.6.2" + +electron-publish@26.0.11: + version "26.0.11" + resolved "https://registry.yarnpkg.com/electron-publish/-/electron-publish-26.0.11.tgz#92c9329a101af2836d9d228c82966eca1eee9a7b" + integrity sha512-a8QRH0rAPIWH9WyyS5LbNvW9Ark6qe63/LqDB7vu2JXYpi0Gma5Q60Dh4tmTqhOBQt0xsrzD8qE7C+D7j+B24A== + dependencies: + "@types/fs-extra" "^9.0.11" + builder-util "26.0.11" + builder-util-runtime "9.3.1" + chalk "^4.1.2" + form-data "^4.0.0" + fs-extra "^10.1.0" + lazy-val "^1.0.5" + mime "^2.5.2" + +electron-to-chromium@^1.5.73: + version "1.5.142" + resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.5.142.tgz#1de55d0d19b24b07768c4bfc90f41bd7f248d043" + integrity sha512-Ah2HgkTu/9RhTDNThBtzu2Wirdy4DC9b0sMT1pUhbkZQ5U/iwmE+PHZX1MpjD5IkJCc2wSghgGG/B04szAx07w== + +electron-updater@^6.6.2: + version "6.6.2" + resolved "https://registry.yarnpkg.com/electron-updater/-/electron-updater-6.6.2.tgz#3e65e044f1a99b00d61e200e24de8e709c69ce99" + integrity sha512-Cr4GDOkbAUqRHP5/oeOmH/L2Bn6+FQPxVLZtPbcmKZC63a1F3uu5EefYOssgZXG3u/zBlubbJ5PJdITdMVggbw== + dependencies: + builder-util-runtime "9.3.1" + fs-extra "^10.1.0" + js-yaml "^4.1.0" + lazy-val "^1.0.5" + lodash.escaperegexp "^4.1.2" + lodash.isequal "^4.5.0" + semver "^7.6.3" + tiny-typed-emitter "^2.1.0" + +electron@*: + version "35.2.1" + resolved "https://registry.yarnpkg.com/electron/-/electron-35.2.1.tgz#9aa91840e6f72c58cbccf7d17d52da0f479fc35c" + integrity sha512-LO4xXLpzkPPUVLvAbdHMlW7N9Z+Qqz+7QsmSWXluTIQMeJk+v7o36nUTZghyA2I1s//tlMvuaCtle6Qo2W0Ktg== + dependencies: + "@electron/get" "^2.0.0" + "@types/node" "^22.7.7" + extract-zip "^2.0.1" + +electron@^34.1.1: + version "34.5.3" + resolved "https://registry.yarnpkg.com/electron/-/electron-34.5.3.tgz#adfb40aeaf7b6d5bceff3bd35960db9371d11808" + integrity sha512-3hGXL4Hzzt4tadPKaoHHDvOzNrUYqhKkLJpZE2PFoNdDL1HDZwL9sb+qldDoMSbgIVfJpo2/7i6PubK3jMIFLA== + dependencies: + "@electron/get" "^2.0.0" + "@types/node" "^20.9.0" + extract-zip "^2.0.1" + +emoji-regex@^10.3.0: + version "10.4.0" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-10.4.0.tgz#03553afea80b3975749cfcb36f776ca268e413d4" + integrity sha512-EC+0oUMY1Rqm4O6LLrgjtYDvcVYTy7chDnM4Q7030tP4Kwj3u/pR6gP9ygnp2CJMK5Gq+9Q2oqmrFJAz01DXjw== + +emoji-regex@^8.0.0: + version "8.0.0" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" + integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== + +emoji-regex@^9.2.2: + version "9.2.2" + resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-9.2.2.tgz#840c8803b0d8047f4ff0cf963176b32d4ef3ed72" + integrity sha512-L18DaJsXSUk2+42pv8mLs5jJT2hqFkFE4j21wOmgbUqsZ2hL72NsUU785g9RXgo3s0ZNgVl42TiHp3ZtOv/Vyg== + +encodeurl@~1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-1.0.2.tgz#ad3ff4c86ec2d029322f5a02c3a9a606c95b3f59" + integrity sha512-TPJXq8JqFaVYm2CWmPvnP2Iyo4ZSM7/QKcSmuMLDObfpH5fi7RUGmd/rTDf+rut/saiDiQEeVTNgAmJEdAOx0w== + +encodeurl@~2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/encodeurl/-/encodeurl-2.0.0.tgz#7b8ea898077d7e409d3ac45474ea38eaf0857a58" + integrity sha512-Q0n9HRi4m6JuGIV1eFlmvJB7ZEVxu93IrMyiMsGC0lrMJMWzRgx6WGquyfQgZVb31vhGgXnfmPNNXmxnOkRBrg== + +encoding-sniffer@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/encoding-sniffer/-/encoding-sniffer-0.2.0.tgz#799569d66d443babe82af18c9f403498365ef1d5" + integrity sha512-ju7Wq1kg04I3HtiYIOrUrdfdDvkyO9s5XM8QAj/bN61Yo/Vb4vgJxy5vi4Yxk01gWHbrofpPtpxM8bKger9jhg== + dependencies: + iconv-lite "^0.6.3" + whatwg-encoding "^3.1.1" + +encoding@^0.1.13: + version "0.1.13" + resolved "https://registry.yarnpkg.com/encoding/-/encoding-0.1.13.tgz#56574afdd791f54a8e9b2785c0582a2d26210fa9" + integrity sha512-ETBauow1T35Y/WZMkio9jiM0Z5xjHHmJ4XmjZOq1l/dXz3lr2sRn87nJy20RupqSh1F2m3HHPSp8ShIPQJrJ3A== + dependencies: + iconv-lite "^0.6.2" + +end-of-stream@^1.1.0, end-of-stream@^1.4.1: + version "1.4.4" + resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0" + integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q== + dependencies: + once "^1.4.0" + +entities@^4.2.0, entities@^4.4.0, entities@^4.5.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/entities/-/entities-4.5.0.tgz#5d268ea5e7113ec74c4d033b79ea5a35a488fb48" + integrity sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw== + +entities@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/entities/-/entities-6.0.0.tgz#09c9e29cb79b0a6459a9b9db9efb418ac5bb8e51" + integrity sha512-aKstq2TDOndCn4diEyp9Uq/Flu2i1GlLkc6XIDQSDMuaFE3OPW5OphLCyQ5SpSJZTb4reN+kTcYru5yIfXoRPw== + +env-paths@^2.2.0: + version "2.2.1" + resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz#420399d416ce1fbe9bc0a07c62fa68d67fd0f8f2" + integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A== + +err-code@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/err-code/-/err-code-2.0.3.tgz#23c2f3b756ffdfc608d30e27c9a941024807e7f9" + integrity sha512-2bmlRpNKBxT/CRmPOlyISQpNj+qSeYvcym/uT0Jx2bMOlKLtSy1ZmLuVxSEKKyor/N5yhvp/ZiG1oE3DEYMSFA== + +error-ex@^1.2.0: + version "1.3.2" + resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf" + integrity sha512-7dFHNmqeFSEt2ZBsCriorKnn3Z2pj+fd9kmI6QoWw4//DL+icEBfc0U7qJCisqrTsKTjw4fNFy2pW9OqStD84g== + dependencies: + is-arrayish "^0.2.1" + +es-abstract@^1.17.5, es-abstract@^1.23.2, es-abstract@^1.23.3, es-abstract@^1.23.5, es-abstract@^1.23.6, es-abstract@^1.23.9: + version "1.23.9" + resolved "https://registry.yarnpkg.com/es-abstract/-/es-abstract-1.23.9.tgz#5b45994b7de78dada5c1bebf1379646b32b9d606" + integrity sha512-py07lI0wjxAC/DcfK1S6G7iANonniZwTISvdPzk9hzeH0IZIshbuuFxLIU96OyF89Yb9hiqWn8M/bY83KY5vzA== + dependencies: + array-buffer-byte-length "^1.0.2" + arraybuffer.prototype.slice "^1.0.4" + available-typed-arrays "^1.0.7" + call-bind "^1.0.8" + call-bound "^1.0.3" + data-view-buffer "^1.0.2" + data-view-byte-length "^1.0.2" + data-view-byte-offset "^1.0.1" + es-define-property "^1.0.1" + es-errors "^1.3.0" + es-object-atoms "^1.0.0" + es-set-tostringtag "^2.1.0" + es-to-primitive "^1.3.0" + function.prototype.name "^1.1.8" + get-intrinsic "^1.2.7" + get-proto "^1.0.0" + get-symbol-description "^1.1.0" + globalthis "^1.0.4" + gopd "^1.2.0" + has-property-descriptors "^1.0.2" + has-proto "^1.2.0" + has-symbols "^1.1.0" + hasown "^2.0.2" + internal-slot "^1.1.0" + is-array-buffer "^3.0.5" + is-callable "^1.2.7" + is-data-view "^1.0.2" + is-regex "^1.2.1" + is-shared-array-buffer "^1.0.4" + is-string "^1.1.1" + is-typed-array "^1.1.15" + is-weakref "^1.1.0" + math-intrinsics "^1.1.0" + object-inspect "^1.13.3" + object-keys "^1.1.1" + object.assign "^4.1.7" + own-keys "^1.0.1" + regexp.prototype.flags "^1.5.3" + safe-array-concat "^1.1.3" + safe-push-apply "^1.0.0" + safe-regex-test "^1.1.0" + set-proto "^1.0.0" + string.prototype.trim "^1.2.10" + string.prototype.trimend "^1.0.9" + string.prototype.trimstart "^1.0.8" + typed-array-buffer "^1.0.3" + typed-array-byte-length "^1.0.3" + typed-array-byte-offset "^1.0.4" + typed-array-length "^1.0.7" + unbox-primitive "^1.1.0" + which-typed-array "^1.1.18" + +es-define-property@^1.0.0, es-define-property@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/es-define-property/-/es-define-property-1.0.1.tgz#983eb2f9a6724e9303f61addf011c72e09e0b0fa" + integrity sha512-e3nRfgfUZ4rNGL232gUgX06QNyyez04KdjFrF+LTRoOXmrOgFKDg4BCdsjW8EnT69eqdYGmRpJwiPVYNrCaW3g== + +es-errors@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/es-errors/-/es-errors-1.3.0.tgz#05f75a25dab98e4fb1dcd5e1472c0546d5057c8f" + integrity sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw== + +es-iterator-helpers@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/es-iterator-helpers/-/es-iterator-helpers-1.2.1.tgz#d1dd0f58129054c0ad922e6a9a1e65eef435fe75" + integrity sha512-uDn+FE1yrDzyC0pCo961B2IHbdM8y/ACZsKD4dG6WqrjV53BADjwa7D+1aom2rsNVfLyDgU/eigvlJGJ08OQ4w== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.3" + define-properties "^1.2.1" + es-abstract "^1.23.6" + es-errors "^1.3.0" + es-set-tostringtag "^2.0.3" + function-bind "^1.1.2" + get-intrinsic "^1.2.6" + globalthis "^1.0.4" + gopd "^1.2.0" + has-property-descriptors "^1.0.2" + has-proto "^1.2.0" + has-symbols "^1.1.0" + internal-slot "^1.1.0" + iterator.prototype "^1.1.4" + safe-array-concat "^1.1.3" + +es-module-lexer@^1.6.0: + version "1.7.0" + resolved "https://registry.yarnpkg.com/es-module-lexer/-/es-module-lexer-1.7.0.tgz#9159601561880a85f2734560a9099b2c31e5372a" + integrity sha512-jEQoCwk8hyb2AZziIOLhDqpm5+2ww5uIE6lkO/6jcOCusfk6LhMHpXXfBLXTZ7Ydyt0j4VoUQv6uGNYbdW+kBA== + +es-object-atoms@^1.0.0, es-object-atoms@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/es-object-atoms/-/es-object-atoms-1.1.1.tgz#1c4f2c4837327597ce69d2ca190a7fdd172338c1" + integrity sha512-FGgH2h8zKNim9ljj7dankFPcICIK9Cp5bm+c2gQSYePhpaG5+esrLODihIorn+Pe6FGJzWhXQotPv73jTaldXA== + dependencies: + es-errors "^1.3.0" + +es-set-tostringtag@^2.0.3, es-set-tostringtag@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/es-set-tostringtag/-/es-set-tostringtag-2.1.0.tgz#f31dbbe0c183b00a6d26eb6325c810c0fd18bd4d" + integrity sha512-j6vWzfrGVfyXxge+O0x5sh6cvxAog0a/4Rdd2K36zCMV5eJ+/+tOAngRO8cODMNWbVRdVlmGZQL2YS3yR8bIUA== + dependencies: + es-errors "^1.3.0" + get-intrinsic "^1.2.6" + has-tostringtag "^1.0.2" + hasown "^2.0.2" + +es-shim-unscopables@^1.0.2, es-shim-unscopables@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/es-shim-unscopables/-/es-shim-unscopables-1.1.0.tgz#438df35520dac5d105f3943d927549ea3b00f4b5" + integrity sha512-d9T8ucsEhh8Bi1woXCf+TIKDIROLG5WCkxg8geBCbvk22kzwC5G2OnXVMO6FUsvQlgUUXQ2itephWDLqDzbeCw== + dependencies: + hasown "^2.0.2" + +es-to-primitive@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/es-to-primitive/-/es-to-primitive-1.3.0.tgz#96c89c82cc49fd8794a24835ba3e1ff87f214e18" + integrity sha512-w+5mJ3GuFL+NjVtJlvydShqE1eN3h3PbI7/5LAsYJP/2qtuMXjfL2LpHSRqo4b4eSF5K/DH1JXKUAHSB2UW50g== + dependencies: + is-callable "^1.2.7" + is-date-object "^1.0.5" + is-symbol "^1.0.4" + +es6-error@^4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/es6-error/-/es6-error-4.1.1.tgz#9e3af407459deed47e9a91f9b885a84eb05c561d" + integrity sha512-Um/+FxMr9CISWh0bi5Zv0iOD+4cFh5qLeks1qhAopKVAJw3drgKbKySikp7wGhDL0HPeaja0P5ULZrxLkniUVg== + +esbuild@^0.21.3: + version "0.21.5" + resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.21.5.tgz#9ca301b120922959b766360d8ac830da0d02997d" + integrity sha512-mg3OPMV4hXywwpoDxu3Qda5xCKQi+vCTZq8S9J/EpkhB2HzKXq4SNFZE3+NK93JYxc8VMSep+lOUSC/RVKaBqw== + optionalDependencies: + "@esbuild/aix-ppc64" "0.21.5" + "@esbuild/android-arm" "0.21.5" + "@esbuild/android-arm64" "0.21.5" + "@esbuild/android-x64" "0.21.5" + "@esbuild/darwin-arm64" "0.21.5" + "@esbuild/darwin-x64" "0.21.5" + "@esbuild/freebsd-arm64" "0.21.5" + "@esbuild/freebsd-x64" "0.21.5" + "@esbuild/linux-arm" "0.21.5" + "@esbuild/linux-arm64" "0.21.5" + "@esbuild/linux-ia32" "0.21.5" + "@esbuild/linux-loong64" "0.21.5" + "@esbuild/linux-mips64el" "0.21.5" + "@esbuild/linux-ppc64" "0.21.5" + "@esbuild/linux-riscv64" "0.21.5" + "@esbuild/linux-s390x" "0.21.5" + "@esbuild/linux-x64" "0.21.5" + "@esbuild/netbsd-x64" "0.21.5" + "@esbuild/openbsd-x64" "0.21.5" + "@esbuild/sunos-x64" "0.21.5" + "@esbuild/win32-arm64" "0.21.5" + "@esbuild/win32-ia32" "0.21.5" + "@esbuild/win32-x64" "0.21.5" + +esbuild@^0.25.0, esbuild@^0.25.2: + version "0.25.3" + resolved "https://registry.yarnpkg.com/esbuild/-/esbuild-0.25.3.tgz#371f7cb41283e5b2191a96047a7a89562965a285" + integrity sha512-qKA6Pvai73+M2FtftpNKRxJ78GIjmFXFxd/1DVBqGo/qNhLSfv+G12n9pNoWdytJC8U00TrViOwpjT0zgqQS8Q== + optionalDependencies: + "@esbuild/aix-ppc64" "0.25.3" + "@esbuild/android-arm" "0.25.3" + "@esbuild/android-arm64" "0.25.3" + "@esbuild/android-x64" "0.25.3" + "@esbuild/darwin-arm64" "0.25.3" + "@esbuild/darwin-x64" "0.25.3" + "@esbuild/freebsd-arm64" "0.25.3" + "@esbuild/freebsd-x64" "0.25.3" + "@esbuild/linux-arm" "0.25.3" + "@esbuild/linux-arm64" "0.25.3" + "@esbuild/linux-ia32" "0.25.3" + "@esbuild/linux-loong64" "0.25.3" + "@esbuild/linux-mips64el" "0.25.3" + "@esbuild/linux-ppc64" "0.25.3" + "@esbuild/linux-riscv64" "0.25.3" + "@esbuild/linux-s390x" "0.25.3" + "@esbuild/linux-x64" "0.25.3" + "@esbuild/netbsd-arm64" "0.25.3" + "@esbuild/netbsd-x64" "0.25.3" + "@esbuild/openbsd-arm64" "0.25.3" + "@esbuild/openbsd-x64" "0.25.3" + "@esbuild/sunos-x64" "0.25.3" + "@esbuild/win32-arm64" "0.25.3" + "@esbuild/win32-ia32" "0.25.3" + "@esbuild/win32-x64" "0.25.3" + +escalade@^3.1.1, escalade@^3.2.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/escalade/-/escalade-3.2.0.tgz#011a3f69856ba189dffa7dc8fcce99d2a87903e5" + integrity sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA== + +escape-html@~1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/escape-html/-/escape-html-1.0.3.tgz#0258eae4d3d0c0974de1c169188ef0051d1d1988" + integrity sha512-NiSupZ4OeuGwr68lGIeym/ksIZMJodUGOSCZ/FSnTxcrekbvqrgdUxlJOMpijaKZVjAJrWrGs/6Jy8OMuyj9ow== + +escape-string-regexp@^1.0.2, escape-string-regexp@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" + integrity sha512-vbRorB5FUQWvla16U8R/qgaFIya2qGzwDrNmCZuYKrbdSUMG6I1ZCGQRefkRVhuOkIGVne7BQ35DSfo1qvJqFg== + +escape-string-regexp@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-2.0.0.tgz#a30304e99daa32e23b2fd20f51babd07cffca344" + integrity sha512-UpzcLCXolUWcNu5HtVMHYdXJjArjsF9C0aNnquZYY4uW/Vu0miy5YoWvbV345HauVvcAUnpRuhMMcqTcGOY2+w== + +escape-string-regexp@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-4.0.0.tgz#14ba83a5d373e3d311e5afca29cf5bfad965bf34" + integrity sha512-TtpcNJ3XAzx3Gq8sWRzJaVajRs0uVxA2YAkdb1jm2YkPz4G6egUFAyA3n5vtEIZefPk5Wa4UXbKuS5fKkJWdgA== + +eslint-import-resolver-node@^0.3.9: + version "0.3.9" + resolved "https://registry.yarnpkg.com/eslint-import-resolver-node/-/eslint-import-resolver-node-0.3.9.tgz#d4eaac52b8a2e7c3cd1903eb00f7e053356118ac" + integrity sha512-WFj2isz22JahUv+B788TlO3N6zL3nNJGU8CcZbPZvVEkBPaJdCV4vy5wyghty5ROFbCRnm132v8BScu5/1BQ8g== + dependencies: + debug "^3.2.7" + is-core-module "^2.13.0" + resolve "^1.22.4" + +eslint-module-utils@^2.12.0: + version "2.12.0" + resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.12.0.tgz#fe4cfb948d61f49203d7b08871982b65b9af0b0b" + integrity sha512-wALZ0HFoytlyh/1+4wuZ9FJCD/leWHQzzrxJ8+rebyReSLk7LApMyd3WJaLVoN+D5+WIdJyDK1c6JnE65V4Zyg== + dependencies: + debug "^3.2.7" + +eslint-plugin-css-modules@^2.12.0: + version "2.12.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-css-modules/-/eslint-plugin-css-modules-2.12.0.tgz#c4102c390c7efd68c4d53677c5e763971699322c" + integrity sha512-ruFBdad69ABrbCDCh5mXj7UzNmrvytfzPACjyvZWIAjFZAG8BXpYSbqmE8gU5wF+pIzV3jU2CWhLvfekXT/IgQ== + dependencies: + gonzales-pe "^4.3.0" + lodash "^4.17.2" + +eslint-plugin-import@^2.31.0: + version "2.31.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.31.0.tgz#310ce7e720ca1d9c0bb3f69adfd1c6bdd7d9e0e7" + integrity sha512-ixmkI62Rbc2/w8Vfxyh1jQRTdRTF52VxwRVHl/ykPAmqG+Nb7/kNn+byLP0LxPgI7zWA16Jt82SybJInmMia3A== + dependencies: + "@rtsao/scc" "^1.1.0" + array-includes "^3.1.8" + array.prototype.findlastindex "^1.2.5" + array.prototype.flat "^1.3.2" + array.prototype.flatmap "^1.3.2" + debug "^3.2.7" + doctrine "^2.1.0" + eslint-import-resolver-node "^0.3.9" + eslint-module-utils "^2.12.0" + hasown "^2.0.2" + is-core-module "^2.15.1" + is-glob "^4.0.3" + minimatch "^3.1.2" + object.fromentries "^2.0.8" + object.groupby "^1.0.3" + object.values "^1.2.0" + semver "^6.3.1" + string.prototype.trimend "^1.0.8" + tsconfig-paths "^3.15.0" + +eslint-plugin-jest@^28.11.0: + version "28.11.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-jest/-/eslint-plugin-jest-28.11.0.tgz#2641ecb4411941bbddb3d7cf8a8ff1163fbb510e" + integrity sha512-QAfipLcNCWLVocVbZW8GimKn5p5iiMcgGbRzz8z/P5q7xw+cNEpYqyzFMtIF/ZgF2HLOyy+dYBut+DoYolvqig== + dependencies: + "@typescript-eslint/utils" "^6.0.0 || ^7.0.0 || ^8.0.0" + +eslint-plugin-jsx-a11y@^6.10.2: + version "6.10.2" + resolved "https://registry.yarnpkg.com/eslint-plugin-jsx-a11y/-/eslint-plugin-jsx-a11y-6.10.2.tgz#d2812bb23bf1ab4665f1718ea442e8372e638483" + integrity sha512-scB3nz4WmG75pV8+3eRUQOHZlNSUhFNq37xnpgRkCCELU3XMvXAxLk1eqWWyE22Ki4Q01Fnsw9BA3cJHDPgn2Q== + dependencies: + aria-query "^5.3.2" + array-includes "^3.1.8" + array.prototype.flatmap "^1.3.2" + ast-types-flow "^0.0.8" + axe-core "^4.10.0" + axobject-query "^4.1.0" + damerau-levenshtein "^1.0.8" + emoji-regex "^9.2.2" + hasown "^2.0.2" + jsx-ast-utils "^3.3.5" + language-tags "^1.0.9" + minimatch "^3.1.2" + object.fromentries "^2.0.8" + safe-regex-test "^1.0.3" + string.prototype.includes "^2.0.1" + +eslint-plugin-react-hooks@^5.2.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-react-hooks/-/eslint-plugin-react-hooks-5.2.0.tgz#1be0080901e6ac31ce7971beed3d3ec0a423d9e3" + integrity sha512-+f15FfK64YQwZdJNELETdn5ibXEUQmW1DZL6KXhNnc2heoy/sg9VJJeT7n8TlMWouzWqSWavFkIhHyIbIAEapg== + +eslint-plugin-react-perf@^3.3.3: + version "3.3.3" + resolved "https://registry.yarnpkg.com/eslint-plugin-react-perf/-/eslint-plugin-react-perf-3.3.3.tgz#29f97ab494fff18dc5e6ec34b056d9a21a86ee71" + integrity sha512-EzPdxsRJg5IllCAH9ny/3nK7sv9251tvKmi/d3Ouv5KzI8TB3zNhzScxL9wnh9Hvv8GYC5LEtzTauynfOEYiAw== + +eslint-plugin-react@^7.37.5: + version "7.37.5" + resolved "https://registry.yarnpkg.com/eslint-plugin-react/-/eslint-plugin-react-7.37.5.tgz#2975511472bdda1b272b34d779335c9b0e877065" + integrity sha512-Qteup0SqU15kdocexFNAJMvCJEfa2xUKNV4CC1xsVMrIIqEy3SQ/rqyxCWNzfrd3/ldy6HMlD2e0JDVpDg2qIA== + dependencies: + array-includes "^3.1.8" + array.prototype.findlast "^1.2.5" + array.prototype.flatmap "^1.3.3" + array.prototype.tosorted "^1.1.4" + doctrine "^2.1.0" + es-iterator-helpers "^1.2.1" + estraverse "^5.3.0" + hasown "^2.0.2" + jsx-ast-utils "^2.4.1 || ^3.0.0" + minimatch "^3.1.2" + object.entries "^1.1.9" + object.fromentries "^2.0.8" + object.values "^1.2.1" + prop-types "^15.8.1" + resolve "^2.0.0-next.5" + semver "^6.3.1" + string.prototype.matchall "^4.0.12" + string.prototype.repeat "^1.0.0" + +eslint-plugin-suggest-no-throw@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-suggest-no-throw/-/eslint-plugin-suggest-no-throw-1.0.0.tgz#facc852aa9acd5e0f4d5e0aff71d6a186e3be8e8" + integrity sha512-IoSDNVeH9WRC9T4Nm3HResFthgiPEW8e6YkKzichJ4fZwac0T8qBQ/8qjmv9NeRSEw8q2EbN/THLE/UU38J46g== + +eslint-plugin-testing-library@^7.1.1: + version "7.1.1" + resolved "https://registry.yarnpkg.com/eslint-plugin-testing-library/-/eslint-plugin-testing-library-7.1.1.tgz#df834e821e53fa81c1eb1fad5a0d9ba4c510f9ea" + integrity sha512-nszC833aZPwB6tik1nMkbFqmtgIXTT0sfJEYs0zMBKMlkQ4to2079yUV96SvmLh00ovSBJI4pgcBC1TiIP8mXg== + dependencies: + "@typescript-eslint/scope-manager" "^8.15.0" + "@typescript-eslint/utils" "^8.15.0" + +eslint-scope@^7.2.2: + version "7.2.2" + resolved "https://registry.yarnpkg.com/eslint-scope/-/eslint-scope-7.2.2.tgz#deb4f92563390f32006894af62a22dba1c46423f" + integrity sha512-dOt21O7lTMhDM+X9mB4GX+DZrZtCUJPL/wlcTqxyrx5IvO0IYtILdtrQGQp+8n5S0gwSVmOf9NQrjMOgfQZlIg== + dependencies: + esrecurse "^4.3.0" + estraverse "^5.2.0" + +eslint-visitor-keys@^3.4.1, eslint-visitor-keys@^3.4.3: + version "3.4.3" + resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-3.4.3.tgz#0cd72fe8550e3c2eae156a96a4dddcd1c8ac5800" + integrity sha512-wpc+LXeiyiisxPlEkUzU6svyS1frIO3Mgxj1fdy7Pm8Ygzguax2N3Fa/D/ag1WqbOprdI+uY6wMUl8/a2G+iag== + +eslint-visitor-keys@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/eslint-visitor-keys/-/eslint-visitor-keys-4.2.0.tgz#687bacb2af884fcdda8a6e7d65c606f46a14cd45" + integrity sha512-UyLnSehNt62FFhSwjZlHmeokpRK59rcz29j+F1/aDgbkbRTk7wIc9XzdoasMUbRNKDM0qQt/+BJ4BrpFeABemw== + +eslint@^8.0.1: + version "8.57.1" + resolved "https://registry.yarnpkg.com/eslint/-/eslint-8.57.1.tgz#7df109654aba7e3bbe5c8eae533c5e461d3c6ca9" + integrity sha512-ypowyDxpVSYpkXr9WPv2PAZCtNip1Mv5KTW0SCurXv/9iOpcrH9PaqUElksqEB6pChqHGDRCFTyrZlGhnLNGiA== + dependencies: + "@eslint-community/eslint-utils" "^4.2.0" + "@eslint-community/regexpp" "^4.6.1" + "@eslint/eslintrc" "^2.1.4" + "@eslint/js" "8.57.1" + "@humanwhocodes/config-array" "^0.13.0" + "@humanwhocodes/module-importer" "^1.0.1" + "@nodelib/fs.walk" "^1.2.8" + "@ungap/structured-clone" "^1.2.0" + ajv "^6.12.4" + chalk "^4.0.0" + cross-spawn "^7.0.2" + debug "^4.3.2" + doctrine "^3.0.0" + escape-string-regexp "^4.0.0" + eslint-scope "^7.2.2" + eslint-visitor-keys "^3.4.3" + espree "^9.6.1" + esquery "^1.4.2" + esutils "^2.0.2" + fast-deep-equal "^3.1.3" + file-entry-cache "^6.0.1" + find-up "^5.0.0" + glob-parent "^6.0.2" + globals "^13.19.0" + graphemer "^1.4.0" + ignore "^5.2.0" + imurmurhash "^0.1.4" + is-glob "^4.0.0" + is-path-inside "^3.0.3" + js-yaml "^4.1.0" + json-stable-stringify-without-jsonify "^1.0.1" + levn "^0.4.1" + lodash.merge "^4.6.2" + minimatch "^3.1.2" + natural-compare "^1.4.0" + optionator "^0.9.3" + strip-ansi "^6.0.1" + text-table "^0.2.0" + +espree@^9.6.0, espree@^9.6.1: + version "9.6.1" + resolved "https://registry.yarnpkg.com/espree/-/espree-9.6.1.tgz#a2a17b8e434690a5432f2f8018ce71d331a48c6f" + integrity sha512-oruZaFkjorTpF32kDSI5/75ViwGeZginGGy2NoOSg3Q9bnwlnmDm4HLnkl0RE3n+njDXR037aY1+x58Z/zFdwQ== + dependencies: + acorn "^8.9.0" + acorn-jsx "^5.3.2" + eslint-visitor-keys "^3.4.1" + +esprima@~4.0.0: + version "4.0.1" + resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71" + integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== + +esquery@^1.4.2: + version "1.6.0" + resolved "https://registry.yarnpkg.com/esquery/-/esquery-1.6.0.tgz#91419234f804d852a82dceec3e16cdc22cf9dae7" + integrity sha512-ca9pw9fomFcKPvFLXhBKUK90ZvGibiGOvRJNbjljY7s7uq/5YO4BOzcYtJqExdx99rF6aAcnRxHmcUHcz6sQsg== + dependencies: + estraverse "^5.1.0" + +esrecurse@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/esrecurse/-/esrecurse-4.3.0.tgz#7ad7964d679abb28bee72cec63758b1c5d2c9921" + integrity sha512-KmfKL3b6G+RXvP8N1vr3Tq1kL/oCFgn2NYXEtqP8/L3pKapUA4G8cFVaoF3SU323CD4XypR/ffioHmkti6/Tag== + dependencies: + estraverse "^5.2.0" + +estraverse@^5.1.0, estraverse@^5.2.0, estraverse@^5.3.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/estraverse/-/estraverse-5.3.0.tgz#2eea5290702f26ab8fe5370370ff86c965d21123" + integrity sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA== + +estree-walker@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/estree-walker/-/estree-walker-2.0.2.tgz#52f010178c2a4c117a7757cfe942adb7d2da4cac" + integrity sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w== + +estree-walker@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/estree-walker/-/estree-walker-3.0.3.tgz#67c3e549ec402a487b4fc193d1953a524752340d" + integrity sha512-7RUKfXgSMMkzt6ZuXmqapOurLGPPfgj6l9uRZ7lRGolvk0y2yocc35LdcxKC5PQZdn2DMqioAQ2NoWcrTKmm6g== + dependencies: + "@types/estree" "^1.0.0" + +esutils@^2.0.2: + version "2.0.3" + resolved "https://registry.yarnpkg.com/esutils/-/esutils-2.0.3.tgz#74d2eb4de0b8da1293711910d50775b9b710ef64" + integrity sha512-kVscqXk4OCp68SZ0dkgEKVi6/8ij300KBWTJq32P/dYeWTSwK41WyTxalN1eRmA5Z9UU/LX9D7FWSmV9SAYx6g== + +etag@~1.8.1: + version "1.8.1" + resolved "https://registry.yarnpkg.com/etag/-/etag-1.8.1.tgz#41ae2eeb65efa62268aebfea83ac7d79299b0887" + integrity sha512-aIL5Fx7mawVa300al2BnEE4iNvo1qETxLrPI/o05L7z6go7fCw1J6EQmbK4FmJ2AS7kgVF/KEZWufBfdClMcPg== + +eventemitter3@^4.0.0: + version "4.0.7" + resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-4.0.7.tgz#2de9b68f6528d5644ef5c59526a1b4a07306169f" + integrity sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw== + +eventemitter3@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/eventemitter3/-/eventemitter3-5.0.1.tgz#53f5ffd0a492ac800721bb42c66b841de96423c4" + integrity sha512-GWkBvjiSZK87ELrYOSESUYeVIc9mvLLf/nXalMOS5dYrgZq9o5OVkbZAVM06CVxYsCwH9BDZFPlQTlPA1j4ahA== + +execa@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/execa/-/execa-1.0.0.tgz#c6236a5bb4df6d6f15e88e7f017798216749ddd8" + integrity sha512-adbxcyWV46qiHyvSp50TKt05tB4tK3HcmF7/nxfAdhnox83seTDbwnaqKO4sXRy7roHAIFqJP/Rw/AuEbX61LA== + dependencies: + cross-spawn "^6.0.0" + get-stream "^4.0.0" + is-stream "^1.1.0" + npm-run-path "^2.0.0" + p-finally "^1.0.0" + signal-exit "^3.0.0" + strip-eof "^1.0.0" + +execa@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/execa/-/execa-8.0.1.tgz#51f6a5943b580f963c3ca9c6321796db8cc39b8c" + integrity sha512-VyhnebXciFV2DESc+p6B+y0LjSm0krU4OgJN44qFAhBY0TJ+1V61tYD2+wHusZ6F9n5K+vl8k0sTy7PEfV4qpg== + dependencies: + cross-spawn "^7.0.3" + get-stream "^8.0.1" + human-signals "^5.0.0" + is-stream "^3.0.0" + merge-stream "^2.0.0" + npm-run-path "^5.1.0" + onetime "^6.0.0" + signal-exit "^4.1.0" + strip-final-newline "^3.0.0" + +exenv@^1.2.0: + version "1.2.2" + resolved "https://registry.yarnpkg.com/exenv/-/exenv-1.2.2.tgz#2ae78e85d9894158670b03d47bec1f03bd91bb9d" + integrity sha512-Z+ktTxTwv9ILfgKCk32OX3n/doe+OcLTRtqK9pcL+JsP3J1/VW8Uvl4ZjLlKqeW4rzK4oesDOGMEMRIZqtP4Iw== + +expand-template@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/expand-template/-/expand-template-2.0.3.tgz#6e14b3fcee0f3a6340ecb57d2e8918692052a47c" + integrity sha512-XYfuKMvj4O35f/pOXLObndIRvyQ+/+6AhODh+OKWj9S9498pHHn/IMszH+gt0fBCRWMNfk1ZSp5x3AifmnI2vg== + +expect-type@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/expect-type/-/expect-type-1.2.1.tgz#af76d8b357cf5fa76c41c09dafb79c549e75f71f" + integrity sha512-/kP8CAwxzLVEeFrMm4kMmy4CCDlpipyA7MYLVrdJIkV0fYF0UaigQHRsxHiuY/GEea+bh4KSv3TIlgr+2UL6bw== + +expect@^29.0.0: + version "29.7.0" + resolved "https://registry.yarnpkg.com/expect/-/expect-29.7.0.tgz#578874590dcb3214514084c08115d8aee61e11bc" + integrity sha512-2Zks0hf1VLFYI1kbh0I5jP3KHHyCHpkfyHBzsSXRFgl/Bg9mWYfMW8oD+PdMPlEwy5HNsR9JutYy6pMeOh61nw== + dependencies: + "@jest/expect-utils" "^29.7.0" + jest-get-type "^29.6.3" + jest-matcher-utils "^29.7.0" + jest-message-util "^29.7.0" + jest-util "^29.7.0" + +exponential-backoff@^3.1.1: + version "3.1.2" + resolved "https://registry.yarnpkg.com/exponential-backoff/-/exponential-backoff-3.1.2.tgz#a8f26adb96bf78e8cd8ad1037928d5e5c0679d91" + integrity sha512-8QxYTVXUkuy7fIIoitQkPwGonB8F3Zj8eEO8Sqg9Zv/bkI7RJAzowee4gr81Hak/dUTpA2Z7VfQgoijjPNlUZA== + +express-ws@^5.0.2: + version "5.0.2" + resolved "https://registry.yarnpkg.com/express-ws/-/express-ws-5.0.2.tgz#5b02d41b937d05199c6c266d7cc931c823bda8eb" + integrity sha512-0uvmuk61O9HXgLhGl3QhNSEtRsQevtmbL94/eILaliEADZBHZOQUAiHFrGPrgsjikohyrmSG5g+sCfASTt0lkQ== + dependencies: + ws "^7.4.6" + +express@^4.17.1: + version "4.21.2" + resolved "https://registry.yarnpkg.com/express/-/express-4.21.2.tgz#cf250e48362174ead6cea4a566abef0162c1ec32" + integrity sha512-28HqgMZAmih1Czt9ny7qr6ek2qddF4FclbMzwhCREB6OFfH+rXAnuNCwo1/wFvrtbgsQDb4kSbX9de9lFbrXnA== + dependencies: + accepts "~1.3.8" + array-flatten "1.1.1" + body-parser "1.20.3" + content-disposition "0.5.4" + content-type "~1.0.4" + cookie "0.7.1" + cookie-signature "1.0.6" + debug "2.6.9" + depd "2.0.0" + encodeurl "~2.0.0" + escape-html "~1.0.3" + etag "~1.8.1" + finalhandler "1.3.1" + fresh "0.5.2" + http-errors "2.0.0" + merge-descriptors "1.0.3" + methods "~1.1.2" + on-finished "2.4.1" + parseurl "~1.3.3" + path-to-regexp "0.1.12" + proxy-addr "~2.0.7" + qs "6.13.0" + range-parser "~1.2.1" + safe-buffer "5.2.1" + send "0.19.0" + serve-static "1.16.2" + setprototypeof "1.2.0" + statuses "2.0.1" + type-is "~1.6.18" + utils-merge "1.0.1" + vary "~1.1.2" + +extract-zip@^2.0.0, extract-zip@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/extract-zip/-/extract-zip-2.0.1.tgz#663dca56fe46df890d5f131ef4a06d22bb8ba13a" + integrity sha512-GDhU9ntwuKyGXdZBUgTIe+vXnWj0fppUEtMDL0+idd5Sta8TGpHssn/eusA9mrPr9qNDym6SxAYZjNvCn/9RBg== + dependencies: + debug "^4.1.1" + get-stream "^5.1.0" + yauzl "^2.10.0" + optionalDependencies: + "@types/yauzl" "^2.9.1" + +extsprintf@^1.2.0: + version "1.4.1" + resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.4.1.tgz#8d172c064867f235c0c84a596806d279bf4bcc07" + integrity sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA== + +fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" + integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== + +fast-glob@^3.2.7, fast-glob@^3.3.2: + version "3.3.3" + resolved "https://registry.yarnpkg.com/fast-glob/-/fast-glob-3.3.3.tgz#d06d585ce8dba90a16b0505c543c3ccfb3aeb818" + integrity sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg== + dependencies: + "@nodelib/fs.stat" "^2.0.2" + "@nodelib/fs.walk" "^1.2.3" + glob-parent "^5.1.2" + merge2 "^1.3.0" + micromatch "^4.0.8" + +fast-json-stable-stringify@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633" + integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== + +fast-levenshtein@^2.0.6: + version "2.0.6" + resolved "https://registry.yarnpkg.com/fast-levenshtein/-/fast-levenshtein-2.0.6.tgz#3d8a5c66883a16a30ca8643e851f19baa7797917" + integrity sha512-DCXu6Ifhqcks7TZKY3Hxp3y6qphY5SJZmrWMDrKcERSOXWQdMhU9Ig/PYrzyw/ul9jOIyh0N4M0tbC5hodg8dw== + +fast-safe-stringify@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz#c406a83b6e70d9e35ce3b30a81141df30aeba884" + integrity sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA== + +fastq@^1.6.0: + version "1.19.1" + resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.19.1.tgz#d50eaba803c8846a883c16492821ebcd2cda55f5" + integrity sha512-GwLTyxkCXjXbxqIhTsMI2Nui8huMPtnxg7krajPJAjnEG/iiOS7i+zCtWGZR9G0NBKbXKh6X9m9UIsYX/N6vvQ== + dependencies: + reusify "^1.0.4" + +fbemitter@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/fbemitter/-/fbemitter-3.0.0.tgz#00b2a1af5411254aab416cd75f9e6289bee4bff3" + integrity sha512-KWKaceCwKQU0+HPoop6gn4eOHk50bBv/VxjJtGMfwmJt3D29JpN4H4eisCtIPA+a8GVBam+ldMMpMjJUvpDyHw== + dependencies: + fbjs "^3.0.0" + +fbjs-css-vars@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/fbjs-css-vars/-/fbjs-css-vars-1.0.2.tgz#216551136ae02fe255932c3ec8775f18e2c078b8" + integrity sha512-b2XGFAFdWZWg0phtAWLHCk836A1Xann+I+Dgd3Gk64MHKZO44FfoD1KxyvbSh0qZsIoXQGGlVztIY+oitJPpRQ== + +fbjs@^3.0.0, fbjs@^3.0.1: + version "3.0.5" + resolved "https://registry.yarnpkg.com/fbjs/-/fbjs-3.0.5.tgz#aa0edb7d5caa6340011790bd9249dbef8a81128d" + integrity sha512-ztsSx77JBtkuMrEypfhgc3cI0+0h+svqeie7xHbh1k/IKdcydnvadp/mUaGgjAOXQmQSxsqgaRhS3q9fy+1kxg== + dependencies: + cross-fetch "^3.1.5" + fbjs-css-vars "^1.0.0" + loose-envify "^1.0.0" + object-assign "^4.1.0" + promise "^7.1.1" + setimmediate "^1.0.5" + ua-parser-js "^1.0.35" + +fd-slicer@~1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/fd-slicer/-/fd-slicer-1.1.0.tgz#25c7c89cb1f9077f8891bbe61d8f390eae256f1e" + integrity sha512-cE1qsB/VwyQozZ+q1dGxR8LBYNZeofhEdUNGSMbQD3Gw2lAzX9Zb3uIU6Ebc/Fmyjo9AWWfnn0AUCHqtevs/8g== + dependencies: + pend "~1.2.0" + +fdir@^6.4.4: + version "6.4.4" + resolved "https://registry.yarnpkg.com/fdir/-/fdir-6.4.4.tgz#1cfcf86f875a883e19a8fab53622cfe992e8d2f9" + integrity sha512-1NZP+GK4GfuAv3PqKvxQRDMjdSRZjnkq7KfhlNrCNNlZ0ygQFpebfrnfnq/W7fpUnAv9aGWmY1zKx7FYL3gwhg== + +fetch-blob@^3.1.2, fetch-blob@^3.1.4: + version "3.2.0" + resolved "https://registry.yarnpkg.com/fetch-blob/-/fetch-blob-3.2.0.tgz#f09b8d4bbd45adc6f0c20b7e787e793e309dcce9" + integrity sha512-7yAQpD2UMJzLi1Dqv7qFYnPbaPx7ZfFK6PiIxQ4PfkGPyNyl2Ugx+a/umUonmKqjhM4DnfbMvdX6otXq83soQQ== + dependencies: + node-domexception "^1.0.0" + web-streams-polyfill "^3.0.3" + +fflate@~0.8.2: + version "0.8.2" + resolved "https://registry.yarnpkg.com/fflate/-/fflate-0.8.2.tgz#fc8631f5347812ad6028bbe4a2308b2792aa1dea" + integrity sha512-cPJU47OaAoCbg0pBvzsgpTPhmhqI5eJjh/JIu8tPj5q+T7iLvW/JAYUqmE7KOB4R1ZyEhzBaIQpQpardBF5z8A== + +file-entry-cache@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/file-entry-cache/-/file-entry-cache-6.0.1.tgz#211b2dd9659cb0394b073e7323ac3c933d522027" + integrity sha512-7Gps/XWymbLk2QLYK4NzpMOrYjMhdIxXuIvy2QBsLE6ljuodKvdkWs/cpyJJ3CVIVpH0Oi1Hvg1ovbMzLdFBBg== + dependencies: + flat-cache "^3.0.4" + +filelist@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/filelist/-/filelist-1.0.4.tgz#f78978a1e944775ff9e62e744424f215e58352b5" + integrity sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q== + dependencies: + minimatch "^5.0.1" + +filename-reserved-regex@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/filename-reserved-regex/-/filename-reserved-regex-2.0.0.tgz#abf73dfab735d045440abfea2d91f389ebbfa229" + integrity sha512-lc1bnsSr4L4Bdif8Xb/qrtokGbq5zlsms/CYH8PP+WtCkGNF65DPiQY8vG3SakEdRn8Dlnm+gW/qWKKjS5sZzQ== + +filenamify@^4.1.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/filenamify/-/filenamify-4.3.0.tgz#62391cb58f02b09971c9d4f9d63b3cf9aba03106" + integrity sha512-hcFKyUG57yWGAzu1CMt/dPzYZuv+jAJUT85bL8mrXvNe6hWj6yEHEc4EdcgiA6Z3oi1/9wXJdZPXF2dZNgwgOg== + dependencies: + filename-reserved-regex "^2.0.0" + strip-outer "^1.0.1" + trim-repeated "^1.0.0" + +fill-range@^7.1.1: + version "7.1.1" + resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.1.1.tgz#44265d3cac07e3ea7dc247516380643754a05292" + integrity sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg== + dependencies: + to-regex-range "^5.0.1" + +finalhandler@1.3.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/finalhandler/-/finalhandler-1.3.1.tgz#0c575f1d1d324ddd1da35ad7ece3df7d19088019" + integrity sha512-6BN9trH7bp3qvnrRyzsBz+g3lZxTNZTbVO2EV1CS0WIcDbawYVdYvGflME/9QP0h0pYlCDBCTjYa9nZzMDpyxQ== + dependencies: + debug "2.6.9" + encodeurl "~2.0.0" + escape-html "~1.0.3" + on-finished "2.4.1" + parseurl "~1.3.3" + statuses "2.0.1" + unpipe "~1.0.0" + +find-up@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-2.1.0.tgz#45d1b7e506c717ddd482775a2b77920a3c0c57a7" + integrity sha512-NWzkk0jSJtTt08+FBFMvXoeZnOJD+jTtsRmBYbAIzJdX6l7dLgR7CTubCM5/eDdPUBvLCeVasP1brfVR/9/EZQ== + dependencies: + locate-path "^2.0.0" + +find-up@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/find-up/-/find-up-5.0.0.tgz#4c92819ecb7083561e4f4a240a86be5198f536fc" + integrity sha512-78/PXT1wlLLDgTzDs7sjq9hzz0vXD+zn+7wypEe4fXQxCmdmqfGsEPQxmiCSQI3ajFV91bVSsvNtrJRiW6nGng== + dependencies: + locate-path "^6.0.0" + path-exists "^4.0.0" + +flat-cache@^3.0.4: + version "3.2.0" + resolved "https://registry.yarnpkg.com/flat-cache/-/flat-cache-3.2.0.tgz#2c0c2d5040c99b1632771a9d105725c0115363ee" + integrity sha512-CYcENa+FtcUKLmhhqyctpclsq7QF38pKjZHsGNiSQF5r4FtoKDWabFDl3hzaEQMvT1LHEysw5twgLvpYYb4vbw== + dependencies: + flatted "^3.2.9" + keyv "^4.5.3" + rimraf "^3.0.2" + +flat@^5.0.2: + version "5.0.2" + resolved "https://registry.yarnpkg.com/flat/-/flat-5.0.2.tgz#8ca6fe332069ffa9d324c327198c598259ceb241" + integrity sha512-b6suED+5/3rTpUBdG1gupIl8MPFCAMA0QXwmljLhvCUKcUvdE4gWky9zpuGCcXHOsz4J9wPGNWq6OKpmIzz3hQ== + +flatted@^3.2.9: + version "3.3.3" + resolved "https://registry.yarnpkg.com/flatted/-/flatted-3.3.3.tgz#67c8fad95454a7c7abebf74bb78ee74a44023358" + integrity sha512-GX+ysw4PBCz0PzosHDepZGANEuFCMLrnRTiEy9McGjmkCQYwRq4A/X786G/fjM/+OjsWSU1ZrY5qyARZmO/uwg== + +flora-colossus@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/flora-colossus/-/flora-colossus-2.0.0.tgz#af1e85db0a8256ef05f3fb531c1235236c97220a" + integrity sha512-dz4HxH6pOvbUzZpZ/yXhafjbR2I8cenK5xL0KtBFb7U2ADsR+OwXifnxZjij/pZWF775uSCMzWVd+jDik2H2IA== + dependencies: + debug "^4.3.4" + fs-extra "^10.1.0" + +flux@^4.0.1: + version "4.0.4" + resolved "https://registry.yarnpkg.com/flux/-/flux-4.0.4.tgz#9661182ea81d161ee1a6a6af10d20485ef2ac572" + integrity sha512-NCj3XlayA2UsapRpM7va6wU1+9rE5FIL7qoMcmxWHRzbp0yujihMBm9BBHZ1MDIk5h5o2Bl6eGiCe8rYELAmYw== + dependencies: + fbemitter "^3.0.0" + fbjs "^3.0.1" + +follow-redirects@^1.0.0: + version "1.15.9" + resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.9.tgz#a604fa10e443bf98ca94228d9eebcc2e8a2c8ee1" + integrity sha512-gew4GsXizNgdoRyqmyfMHyAmXsZDk6mHkSxZFCzW9gwlbtOW44CDtYavM+y+72qD/Vq2l550kMF52DT8fOLJqQ== + +for-each@^0.3.3, for-each@^0.3.5: + version "0.3.5" + resolved "https://registry.yarnpkg.com/for-each/-/for-each-0.3.5.tgz#d650688027826920feeb0af747ee7b9421a41d47" + integrity sha512-dKx12eRCVIzqCxFGplyFKJMPvLEWgmNtUrpTiJIR5u97zEhRG8ySrtboPHZXx7daLxQVrl643cTzbab2tkQjxg== + dependencies: + is-callable "^1.2.7" + +foreground-child@^3.1.0: + version "3.3.1" + resolved "https://registry.yarnpkg.com/foreground-child/-/foreground-child-3.3.1.tgz#32e8e9ed1b68a3497befb9ac2b6adf92a638576f" + integrity sha512-gIXjKqtFuWEgzFRJA9WCQeSJLZDjgJUOMCMzxtvFq/37KojM1BFGufqsCy0r4qSQmYLsZYMeyRqzIWOMup03sw== + dependencies: + cross-spawn "^7.0.6" + signal-exit "^4.0.1" + +form-data@^4.0.0: + version "4.0.2" + resolved "https://registry.yarnpkg.com/form-data/-/form-data-4.0.2.tgz#35cabbdd30c3ce73deb2c42d3c8d3ed9ca51794c" + integrity sha512-hGfm/slu0ZabnNt4oaRZ6uREyfCj6P4fT/n6A1rGV+Z0VdGXjfOhVUpkn6qVQONHGIFwmveGXyDs75+nr6FM8w== + dependencies: + asynckit "^0.4.0" + combined-stream "^1.0.8" + es-set-tostringtag "^2.1.0" + mime-types "^2.1.12" + +formdata-polyfill@^4.0.10: + version "4.0.10" + resolved "https://registry.yarnpkg.com/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz#24807c31c9d402e002ab3d8c720144ceb8848423" + integrity sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g== + dependencies: + fetch-blob "^3.1.2" + +forwarded@0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811" + integrity sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow== + +fraction.js@^4.3.7: + version "4.3.7" + resolved "https://registry.yarnpkg.com/fraction.js/-/fraction.js-4.3.7.tgz#06ca0085157e42fda7f9e726e79fefc4068840f7" + integrity sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew== + +fresh@0.5.2: + version "0.5.2" + resolved "https://registry.yarnpkg.com/fresh/-/fresh-0.5.2.tgz#3d8cadd90d976569fa835ab1f8e4b23a105605a7" + integrity sha512-zJ2mQYM18rEFOudeV4GShTGIQ7RbzA7ozbU9I/XBpm7kqgMywgmylMwXHxZJmkVoYkna9d2pVXVXPdYTP9ej8Q== + +fs-constants@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/fs-constants/-/fs-constants-1.0.0.tgz#6be0de9be998ce16af8afc24497b9ee9b7ccd9ad" + integrity sha512-y6OAwoSIf7FyjMIv94u+b5rdheZEjzR63GTyZJm5qh4Bi+2YgwLCcI/fPFZkL5PSixOt6ZNKm+w+Hfp/Bciwow== + +fs-extra@^10.0.0, fs-extra@^10.1.0: + version "10.1.0" + resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-10.1.0.tgz#02873cfbc4084dde127eaa5f9905eef2325d1abf" + integrity sha512-oRXApq54ETRj4eMiFzGnHWGy+zo5raudjuxN0b8H7s/RU2oW0Wvsx9O0ACRN/kRq9E8Vu/ReskGB5o3ji+FzHQ== + dependencies: + graceful-fs "^4.2.0" + jsonfile "^6.0.1" + universalify "^2.0.0" + +fs-extra@^11.1.0, fs-extra@^11.1.1: + version "11.3.0" + resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-11.3.0.tgz#0daced136bbaf65a555a326719af931adc7a314d" + integrity sha512-Z4XaCL6dUDHfP/jT25jJKMmtxvuwbkrD1vNSMFlo9lNLY2c5FHYSQgHPRZUjAB26TpDEoW9HCOgplrdbaPV/ew== + dependencies: + graceful-fs "^4.2.0" + jsonfile "^6.0.1" + universalify "^2.0.0" + +fs-extra@^8.1.0: + version "8.1.0" + resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-8.1.0.tgz#49d43c45a88cd9677668cb7be1b46efdb8d2e1c0" + integrity sha512-yhlQgA6mnOJUKOsRUFsgJdQCvkKhcz8tlZG5HBQfReYZy46OwLcY+Zia0mtdHsOo9y/hP+CxMN0TU9QxoOtG4g== + dependencies: + graceful-fs "^4.2.0" + jsonfile "^4.0.0" + universalify "^0.1.0" + +fs-extra@^9.0.0, fs-extra@^9.0.1: + version "9.1.0" + resolved "https://registry.yarnpkg.com/fs-extra/-/fs-extra-9.1.0.tgz#5954460c764a8da2094ba3554bf839e6b9a7c86d" + integrity sha512-hcg3ZmepS30/7BSFqRvoo3DOMQu7IjqxO5nCDt+zM9XWjb33Wg7ziNT+Qvqbuc3+gWpzO02JubVyk2G4Zvo1OQ== + dependencies: + at-least-node "^1.0.0" + graceful-fs "^4.2.0" + jsonfile "^6.0.1" + universalify "^2.0.0" + +fs-minipass@^2.0.0, fs-minipass@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/fs-minipass/-/fs-minipass-2.1.0.tgz#7f5036fdbf12c63c169190cbe4199c852271f9fb" + integrity sha512-V/JgOLFCS+R6Vcq0slCuaeWEdNC3ouDlJMNIsacH2VtALiu9mV4LPrHc5cDl8k5aw6J8jwgWWpiTo5RYhmIzvg== + dependencies: + minipass "^3.0.0" + +fs.realpath@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/fs.realpath/-/fs.realpath-1.0.0.tgz#1504ad2523158caa40db4a2787cb01411994ea4f" + integrity sha512-OO0pH2lK6a0hZnAdau5ItzHPI6pUlvI7jMVnxUQRtw4owF2wk8lOSabtGDCTP4Ggrg2MbGnWO9X8K1t4+fGMDw== + +fsevents@2.3.2: + version "2.3.2" + resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a" + integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA== + +fsevents@~2.3.2, fsevents@~2.3.3: + version "2.3.3" + resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.3.tgz#cac6407785d03675a2a5e1a5305c697b347d90d6" + integrity sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw== + +function-bind@^1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/function-bind/-/function-bind-1.1.2.tgz#2c02d864d97f3ea6c8830c464cbd11ab6eab7a1c" + integrity sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA== + +function.prototype.name@^1.1.6, function.prototype.name@^1.1.8: + version "1.1.8" + resolved "https://registry.yarnpkg.com/function.prototype.name/-/function.prototype.name-1.1.8.tgz#e68e1df7b259a5c949eeef95cdbde53edffabb78" + integrity sha512-e5iwyodOHhbMr/yNrc7fDYG4qlbIvI5gajyzPnb5TCwyhjApznQh1BMFou9b30SevY43gCJKXycoCBjMbsuW0Q== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.3" + define-properties "^1.2.1" + functions-have-names "^1.2.3" + hasown "^2.0.2" + is-callable "^1.2.7" + +functions-have-names@^1.2.3: + version "1.2.3" + resolved "https://registry.yarnpkg.com/functions-have-names/-/functions-have-names-1.2.3.tgz#0404fe4ee2ba2f607f0e0ec3c80bae994133b834" + integrity sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ== + +fuse.js@^7.1.0: + version "7.1.0" + resolved "https://registry.yarnpkg.com/fuse.js/-/fuse.js-7.1.0.tgz#306228b4befeee11e05b027087c2744158527d09" + integrity sha512-trLf4SzuuUxfusZADLINj+dE8clK1frKdmqiJNb1Es75fmI5oY6X2mxLVUciLLjxqw/xr72Dhy+lER6dGd02FQ== + +galactus@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/galactus/-/galactus-1.0.0.tgz#c2615182afa0c6d0859b92e56ae36d052827db7e" + integrity sha512-R1fam6D4CyKQGNlvJne4dkNF+PvUUl7TAJInvTGa9fti9qAv95quQz29GXapA4d8Ec266mJJxFVh82M4GIIGDQ== + dependencies: + debug "^4.3.4" + flora-colossus "^2.0.0" + fs-extra "^10.1.0" + +gensync@^1.0.0-beta.2: + version "1.0.0-beta.2" + resolved "https://registry.yarnpkg.com/gensync/-/gensync-1.0.0-beta.2.tgz#32a6ee76c3d7f52d46b2b1ae5d93fea8580a25e0" + integrity sha512-3hN7NaskYvMDLQY55gnW3NQ+mesEAepTqlg+VEbj7zzqEMBVNhzcGYYeqFo/TlYz6eQiFcp1HcsCZO+nGgS8zg== + +get-caller-file@^2.0.5: + version "2.0.5" + resolved "https://registry.yarnpkg.com/get-caller-file/-/get-caller-file-2.0.5.tgz#4f94412a82db32f36e3b0b9741f8a97feb031f7e" + integrity sha512-DyFP3BM/3YHTQOCUL/w0OZHR0lpKeGrxotcHWcqNEdnltqFwXVfhEBQ94eIo34AfQpo0rGki4cyIiftY06h2Fg== + +get-east-asian-width@^1.0.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/get-east-asian-width/-/get-east-asian-width-1.3.0.tgz#21b4071ee58ed04ee0db653371b55b4299875389" + integrity sha512-vpeMIQKxczTD/0s2CdEWHcb0eeJe6TFjxb+J5xgX7hScxqrGuyjmv4c1D4A/gelKfyox0gJJwIHF+fLjeaM8kQ== + +get-func-name@^2.0.1, get-func-name@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/get-func-name/-/get-func-name-2.0.2.tgz#0d7cf20cd13fda808669ffa88f4ffc7a3943fc41" + integrity sha512-8vXOvuE167CtIc3OyItco7N/dpRtBbYOsPsXCz7X/PMnlGjYjSGuZJgM1Y7mmew7BKf9BqvLX2tnOVy1BBUsxQ== + +get-intrinsic@^1.2.4, get-intrinsic@^1.2.5, get-intrinsic@^1.2.6, get-intrinsic@^1.2.7, get-intrinsic@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/get-intrinsic/-/get-intrinsic-1.3.0.tgz#743f0e3b6964a93a5491ed1bffaae054d7f98d01" + integrity sha512-9fSjSaos/fRIVIp+xSJlE6lfwhES7LNtKaCBIamHsjr2na1BiABJPo0mOjjz8GJDURarmCPGqaiVg5mfjb98CQ== + dependencies: + call-bind-apply-helpers "^1.0.2" + es-define-property "^1.0.1" + es-errors "^1.3.0" + es-object-atoms "^1.1.1" + function-bind "^1.1.2" + get-proto "^1.0.1" + gopd "^1.2.0" + has-symbols "^1.1.0" + hasown "^2.0.2" + math-intrinsics "^1.1.0" + +get-package-info@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/get-package-info/-/get-package-info-1.0.0.tgz#6432796563e28113cd9474dbbd00052985a4999c" + integrity sha512-SCbprXGAPdIhKAXiG+Mk6yeoFH61JlYunqdFQFHDtLjJlDjFf6x07dsS8acO+xWt52jpdVo49AlVDnUVK1sDNw== + dependencies: + bluebird "^3.1.1" + debug "^2.2.0" + lodash.get "^4.0.0" + read-pkg-up "^2.0.0" + +get-proto@^1.0.0, get-proto@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/get-proto/-/get-proto-1.0.1.tgz#150b3f2743869ef3e851ec0c49d15b1d14d00ee1" + integrity sha512-sTSfBjoXBp89JvIKIefqw7U2CCebsc74kiY6awiGogKtoSGbgjYE/G/+l9sF3MWFPNc9IcoOC4ODfKHfxFmp0g== + dependencies: + dunder-proto "^1.0.1" + es-object-atoms "^1.0.0" + +get-stream@^4.0.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-4.1.0.tgz#c1b255575f3dc21d59bfc79cd3d2b46b1c3a54b5" + integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w== + dependencies: + pump "^3.0.0" + +get-stream@^5.1.0: + version "5.2.0" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3" + integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA== + dependencies: + pump "^3.0.0" + +get-stream@^8.0.1: + version "8.0.1" + resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-8.0.1.tgz#def9dfd71742cd7754a7761ed43749a27d02eca2" + integrity sha512-VaUJspBffn/LMCJVoMvSAdmscJyS1auj5Zulnn5UoYcY531UWmdwhRWkcGKnGU93m5HSXP9LP2usOryrBtQowA== + +get-symbol-description@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/get-symbol-description/-/get-symbol-description-1.1.0.tgz#7bdd54e0befe8ffc9f3b4e203220d9f1e881b6ee" + integrity sha512-w9UMqWwJxHNOvoNzSJ2oPF5wvYcvP7jUvYzhp67yEhTi17ZDBBC1z9pTdGuzjD+EFIqLSYRweZjqfiPzQ06Ebg== + dependencies: + call-bound "^1.0.3" + es-errors "^1.3.0" + get-intrinsic "^1.2.6" + +get-them-args@1.3.2: + version "1.3.2" + resolved "https://registry.yarnpkg.com/get-them-args/-/get-them-args-1.3.2.tgz#74a20ba8a4abece5ae199ad03f2bcc68fdfc9ba5" + integrity sha512-LRn8Jlk+DwZE4GTlDbT3Hikd1wSHgLMme/+7ddlqKd7ldwR6LjJgTVWzBnR01wnYGe4KgrXjg287RaI22UHmAw== + +github-from-package@0.0.0: + version "0.0.0" + resolved "https://registry.yarnpkg.com/github-from-package/-/github-from-package-0.0.0.tgz#97fb5d96bfde8973313f20e8288ef9a167fa64ce" + integrity sha512-SyHy3T1v2NUXn29OsWdxmK6RwHD+vkj3v8en8AOBZ1wBQ/hCAQ5bAQTD02kW4W9tUp/3Qh6J8r9EvntiyCmOOw== + +glob-parent@^5.1.2, glob-parent@~5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4" + integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== + dependencies: + is-glob "^4.0.1" + +glob-parent@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-6.0.2.tgz#6d237d99083950c79290f24c7642a3de9a28f9e3" + integrity sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A== + dependencies: + is-glob "^4.0.3" + +glob@^10.3.10, glob@^10.3.12, glob@^10.3.4, glob@^10.4.5: + version "10.4.5" + resolved "https://registry.yarnpkg.com/glob/-/glob-10.4.5.tgz#f4d9f0b90ffdbab09c9d77f5f29b4262517b0956" + integrity sha512-7Bv8RF0k6xjo7d4A/PxYLbUCfb6c+Vpd2/mB2yRDlew7Jb5hEXiCD9ibfO7wpk8i4sevK6DFny9h7EYbM3/sHg== + dependencies: + foreground-child "^3.1.0" + jackspeak "^3.1.2" + minimatch "^9.0.4" + minipass "^7.1.2" + package-json-from-dist "^1.0.0" + path-scurry "^1.11.1" + +glob@^11.0.0, glob@^11.0.1: + version "11.0.2" + resolved "https://registry.yarnpkg.com/glob/-/glob-11.0.2.tgz#3261e3897bbc603030b041fd77ba636022d51ce0" + integrity sha512-YT7U7Vye+t5fZ/QMkBFrTJ7ZQxInIUjwyAjVj84CYXqgBdv30MFUPGnBR6sQaVq6Is15wYJUsnzTuWaGRBhBAQ== + dependencies: + foreground-child "^3.1.0" + jackspeak "^4.0.1" + minimatch "^10.0.0" + minipass "^7.1.2" + package-json-from-dist "^1.0.0" + path-scurry "^2.0.0" + +glob@^7.1.3, glob@^7.1.6: + version "7.2.3" + resolved "https://registry.yarnpkg.com/glob/-/glob-7.2.3.tgz#b8df0fb802bbfa8e89bd1d938b4e16578ed44f2b" + integrity sha512-nFR0zLpU2YCaRxwoCJvL6UvCH2JFyFVIvwTLsIf21AuHlMskA1hhTdk+LlYJtOlYt9v6dvszD2BGRqBL+iQK9Q== + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^3.1.1" + once "^1.3.0" + path-is-absolute "^1.0.0" + +glob@^8.0.1, glob@^8.1.0: + version "8.1.0" + resolved "https://registry.yarnpkg.com/glob/-/glob-8.1.0.tgz#d388f656593ef708ee3e34640fdfb99a9fd1c33e" + integrity sha512-r8hpEjiQEYlF2QU0df3dS+nxxSIreXQS1qRhMJM0Q5NDdR386C7jb7Hwwod8Fgiuex+k0GFjgft18yvxm5XoCQ== + dependencies: + fs.realpath "^1.0.0" + inflight "^1.0.4" + inherits "2" + minimatch "^5.0.1" + once "^1.3.0" + +global-agent@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/global-agent/-/global-agent-3.0.0.tgz#ae7cd31bd3583b93c5a16437a1afe27cc33a1ab6" + integrity sha512-PT6XReJ+D07JvGoxQMkT6qji/jVNfX/h364XHZOWeRzy64sSFr+xJ5OX7LI3b4MPQzdL4H8Y8M0xzPpsVMwA8Q== + dependencies: + boolean "^3.0.1" + es6-error "^4.1.1" + matcher "^3.0.0" + roarr "^2.15.3" + semver "^7.3.2" + serialize-error "^7.0.1" + +global-dirs@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/global-dirs/-/global-dirs-3.0.1.tgz#0c488971f066baceda21447aecb1a8b911d22485" + integrity sha512-NBcGGFbBA9s1VzD41QXDG+3++t9Mn5t1FpLdhESY6oKY4gYTFpX4wO3sqGUa0Srjtbfj3szX0RnemmrVRUdULA== + dependencies: + ini "2.0.0" + +globals@^11.1.0: + version "11.12.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-11.12.0.tgz#ab8795338868a0babd8525758018c2a7eb95c42e" + integrity sha512-WOBp/EEGUiIsJSp7wcv/y6MO+lV9UoncWqxuFfm8eBwzWNgyfBd6Gz+IeKQ9jCmyhoH99g15M3T+QaVHFjizVA== + +globals@^13.19.0: + version "13.24.0" + resolved "https://registry.yarnpkg.com/globals/-/globals-13.24.0.tgz#8432a19d78ce0c1e833949c36adb345400bb1171" + integrity sha512-AhO5QUcj8llrbG09iWhPU2B204J1xnPeL8kQmVorSsy+Sjj1sk8gIyh6cUocGmH4L0UuhAJy+hJMRA4mgA4mFQ== + dependencies: + type-fest "^0.20.2" + +globalthis@^1.0.1, globalthis@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/globalthis/-/globalthis-1.0.4.tgz#7430ed3a975d97bfb59bcce41f5cabbafa651236" + integrity sha512-DpLKbNU4WylpxJykQujfCcwYWiV/Jhm50Goo0wrVILAv5jOr9d+H+UR3PhSCD2rCCEIg0uc+G+muBTwD54JhDQ== + dependencies: + define-properties "^1.2.1" + gopd "^1.0.1" + +globrex@^0.1.2: + version "0.1.2" + resolved "https://registry.yarnpkg.com/globrex/-/globrex-0.1.2.tgz#dd5d9ec826232730cd6793a5e33a9302985e6098" + integrity sha512-uHJgbwAMwNFf5mLst7IWLNg14x1CkeqglJb/K3doi4dw6q2IvAAmM/Y81kevy83wP+Sst+nutFTYOGg3d1lsxg== + +gonzales-pe@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/gonzales-pe/-/gonzales-pe-4.3.0.tgz#fe9dec5f3c557eead09ff868c65826be54d067b3" + integrity sha512-otgSPpUmdWJ43VXyiNgEYE4luzHCL2pz4wQ0OnDluC6Eg4Ko3Vexy/SrSynglw/eR+OhkzmqFCZa/OFa/RgAOQ== + dependencies: + minimist "^1.2.5" + +goober@^2.1.16: + version "2.1.16" + resolved "https://registry.yarnpkg.com/goober/-/goober-2.1.16.tgz#7d548eb9b83ff0988d102be71f271ca8f9c82a95" + integrity sha512-erjk19y1U33+XAMe1VTvIONHYoSqE4iS7BYUZfHaqeohLmnC0FdxEh7rQU+6MZ4OajItzjZFSRtVANrQwNq6/g== + +gopd@^1.0.1, gopd@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/gopd/-/gopd-1.2.0.tgz#89f56b8217bdbc8802bd299df6d7f1081d7e51a1" + integrity sha512-ZUKRh6/kUFoAiTAtTYPZJ3hw9wNxx+BIBOijnlG9PnrJsCcSjs1wyyD6vJpaYtgnzDrKYRSqf3OO6Rfa93xsRg== + +got@^11.7.0, got@^11.8.5: + version "11.8.6" + resolved "https://registry.yarnpkg.com/got/-/got-11.8.6.tgz#276e827ead8772eddbcfc97170590b841823233a" + integrity sha512-6tfZ91bOr7bOXnK7PRDCGBLa1H4U080YHNaAQ2KsMGlLEzRbk44nsZF2E1IeRc3vtJHPVbKCYgdFbaGO2ljd8g== + dependencies: + "@sindresorhus/is" "^4.0.0" + "@szmarczak/http-timer" "^4.0.5" + "@types/cacheable-request" "^6.0.1" + "@types/responselike" "^1.0.0" + cacheable-lookup "^5.0.3" + cacheable-request "^7.0.2" + decompress-response "^6.0.0" + http2-wrapper "^1.0.0-beta.5.2" + lowercase-keys "^2.0.0" + p-cancelable "^2.0.0" + responselike "^2.0.0" + +graceful-fs@^4.1.2, graceful-fs@^4.1.6, graceful-fs@^4.2.0, graceful-fs@^4.2.6, graceful-fs@^4.2.9: + version "4.2.11" + resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.11.tgz#4183e4e8bf08bb6e05bbb2f7d2e0c8f712ca40e3" + integrity sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ== + +graphemer@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/graphemer/-/graphemer-1.4.0.tgz#fb2f1d55e0e3a1849aeffc90c4fa0dd53a0e66c6" + integrity sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag== + +happy-dom@^16.3.0: + version "16.8.1" + resolved "https://registry.yarnpkg.com/happy-dom/-/happy-dom-16.8.1.tgz#43d7e998fd36aa6062acbdfa88262ef0bc0a105e" + integrity sha512-n0QrmT9lD81rbpKsyhnlz3DgnMZlaOkJPpgi746doA+HvaMC79bdWkwjrNnGJRvDrWTI8iOcJiVTJ5CdT/AZRw== + dependencies: + webidl-conversions "^7.0.0" + whatwg-mimetype "^3.0.0" + +has-bigints@^1.0.2: + version "1.1.0" + resolved "https://registry.yarnpkg.com/has-bigints/-/has-bigints-1.1.0.tgz#28607e965ac967e03cd2a2c70a2636a1edad49fe" + integrity sha512-R3pbpkcIqv2Pm3dUwgjclDRVmWpTJW2DcMzcIhEXEx1oh/CEMObMm3KLmRJOdvhM7o4uQBnwr8pzRK2sJWIqfg== + +has-flag@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd" + integrity sha512-sKJf1+ceQBr4SMkvQnBDNDtf4TXpVhVGateu0t918bl30FnbE2m4vNLX+VWe/dpjlb+HugGYzW7uQXH98HPEYw== + +has-flag@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" + integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== + +has-property-descriptors@^1.0.0, has-property-descriptors@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-property-descriptors/-/has-property-descriptors-1.0.2.tgz#963ed7d071dc7bf5f084c5bfbe0d1b6222586854" + integrity sha512-55JNKuIW+vq4Ke1BjOTjM2YctQIvCT7GFzHwmfZPGo5wnrgkid0YQtnAleFSqumZm4az3n2BS+erby5ipJdgrg== + dependencies: + es-define-property "^1.0.0" + +has-proto@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/has-proto/-/has-proto-1.2.0.tgz#5de5a6eabd95fdffd9818b43055e8065e39fe9d5" + integrity sha512-KIL7eQPfHQRC8+XluaIw7BHUwwqL19bQn4hzNgdr+1wXoU0KKj6rufu47lhY7KbJR2C6T6+PfyN0Ea7wkSS+qQ== + dependencies: + dunder-proto "^1.0.0" + +has-symbols@^1.0.3, has-symbols@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/has-symbols/-/has-symbols-1.1.0.tgz#fc9c6a783a084951d0b971fe1018de813707a338" + integrity sha512-1cDNdwJ2Jaohmb3sg4OmKaMBwuC48sYni5HUw2DvsC8LjGTLK9h+eb1X6RyuOHe4hT0ULCW68iomhjUoKUqlPQ== + +has-tostringtag@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/has-tostringtag/-/has-tostringtag-1.0.2.tgz#2cdc42d40bef2e5b4eeab7c01a73c54ce7ab5abc" + integrity sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw== + dependencies: + has-symbols "^1.0.3" + +hasown@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/hasown/-/hasown-2.0.2.tgz#003eaf91be7adc372e84ec59dc37252cedb80003" + integrity sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ== + dependencies: + function-bind "^1.1.2" + +he@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/he/-/he-1.2.0.tgz#84ae65fa7eafb165fddb61566ae14baf05664f0f" + integrity sha512-F/1DnUGPopORZi0ni+CvrCgHQ5FyEAHRLSApuYWMmrbSwoN2Mn/7k+Gl38gJnR7yyDZk6WLXwiGod1JOWNDKGw== + +hosted-git-info@^2.1.4: + version "2.8.9" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.8.9.tgz#dffc0bf9a21c02209090f2aa69429e1414daf3f9" + integrity sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw== + +hosted-git-info@^4.0.2, hosted-git-info@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz#827b82867e9ff1c8d0c4d9d53880397d2c86d224" + integrity sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA== + dependencies: + lru-cache "^6.0.0" + +html-encoding-sniffer@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/html-encoding-sniffer/-/html-encoding-sniffer-3.0.0.tgz#2cb1a8cf0db52414776e5b2a7a04d5dd98158de9" + integrity sha512-oWv4T4yJ52iKrufjnyZPkrN0CH3QnrUqdB6In1g5Fe1mia8GmF36gnfNySxoZtxD5+NmYw1EElVXiBk93UeskA== + dependencies: + whatwg-encoding "^2.0.0" + +html2canvas-pro@^1.5.8: + version "1.5.8" + resolved "https://registry.yarnpkg.com/html2canvas-pro/-/html2canvas-pro-1.5.8.tgz#d40ba9732c455943fc269f0606895ac00bb0c864" + integrity sha512-bVGAU7IvhBwBlRAmX6QhekX8lsaxmYoF6zIwf/HNlHscjx+KN8jw/U4PQRYqeEVm9+m13hcS1l5ChJB9/e29Lw== + dependencies: + css-line-break "^2.1.0" + text-segmentation "^1.0.3" + +htmlparser2@^9.1.0: + version "9.1.0" + resolved "https://registry.yarnpkg.com/htmlparser2/-/htmlparser2-9.1.0.tgz#cdb498d8a75a51f739b61d3f718136c369bc8c23" + integrity sha512-5zfg6mHUoaer/97TxnGpxmbR7zJtPwIYFMZ/H5ucTlPZhKvtum05yiPK3Mgai3a0DyVxv7qYqoweaEd2nrYQzQ== + dependencies: + domelementtype "^2.3.0" + domhandler "^5.0.3" + domutils "^3.1.0" + entities "^4.5.0" + +http-cache-semantics@^4.0.0, http-cache-semantics@^4.1.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.1.tgz#abe02fcb2985460bf0323be664436ec3476a6d5a" + integrity sha512-er295DKPVsV82j5kw1Gjt+ADA/XYHsajl82cGNQG2eyoPkvgUhX+nDIyelzhIWbbsXP39EHcI6l5tYs2FYqYXQ== + +http-errors@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/http-errors/-/http-errors-2.0.0.tgz#b7774a1486ef73cf7667ac9ae0858c012c57b9d3" + integrity sha512-FtwrG/euBzaEjYeRqOgly7G0qviiXoJWnvEH2Z1plBdXgbyjv34pHTSb9zoeHMyDy33+DWy5Wt9Wo+TURtOYSQ== + dependencies: + depd "2.0.0" + inherits "2.0.4" + setprototypeof "1.2.0" + statuses "2.0.1" + toidentifier "1.0.1" + +http-proxy-agent@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/http-proxy-agent/-/http-proxy-agent-5.0.0.tgz#5129800203520d434f142bc78ff3c170800f2b43" + integrity sha512-n2hY8YdoRE1i7r6M0w9DIw5GgZN0G25P8zLCRQ8rjXtTU3vsNFBI/vWK/UIeE6g5MUUz6avwAPXmL6Fy9D/90w== + dependencies: + "@tootallnate/once" "2" + agent-base "6" + debug "4" + +http-proxy-agent@^7.0.0, http-proxy-agent@^7.0.2: + version "7.0.2" + resolved "https://registry.yarnpkg.com/http-proxy-agent/-/http-proxy-agent-7.0.2.tgz#9a8b1f246866c028509486585f62b8f2c18c270e" + integrity sha512-T1gkAiYYDWYx3V5Bmyu7HcfcvL7mUrTWiM6yOfa3PIphViJ/gFPbvidQ+veqSOHci/PxBcDabeUNCzpOODJZig== + dependencies: + agent-base "^7.1.0" + debug "^4.3.4" + +http-proxy@^1.18.1: + version "1.18.1" + resolved "https://registry.yarnpkg.com/http-proxy/-/http-proxy-1.18.1.tgz#401541f0534884bbf95260334e72f88ee3976549" + integrity sha512-7mz/721AbnJwIVbnaSv1Cz3Am0ZLT/UBwkC92VlxhXv/k/BBQfM2fXElQNC27BVGr0uwUpplYPQM9LnaBMR5NQ== + dependencies: + eventemitter3 "^4.0.0" + follow-redirects "^1.0.0" + requires-port "^1.0.0" + +http-server@^14.1.1: + version "14.1.1" + resolved "https://registry.yarnpkg.com/http-server/-/http-server-14.1.1.tgz#d60fbb37d7c2fdff0f0fbff0d0ee6670bd285e2e" + integrity sha512-+cbxadF40UXd9T01zUHgA+rlo2Bg1Srer4+B4NwIHdaGxAGGv59nYRnGGDJ9LBk7alpS0US+J+bLLdQOOkJq4A== + dependencies: + basic-auth "^2.0.1" + chalk "^4.1.2" + corser "^2.0.1" + he "^1.2.0" + html-encoding-sniffer "^3.0.0" + http-proxy "^1.18.1" + mime "^1.6.0" + minimist "^1.2.6" + opener "^1.5.1" + portfinder "^1.0.28" + secure-compare "3.0.1" + union "~0.5.0" + url-join "^4.0.1" + +http2-wrapper@^1.0.0-beta.5.2: + version "1.0.3" + resolved "https://registry.yarnpkg.com/http2-wrapper/-/http2-wrapper-1.0.3.tgz#b8f55e0c1f25d4ebd08b3b0c2c079f9590800b3d" + integrity sha512-V+23sDMr12Wnz7iTcDeJr3O6AIxlnvT/bmaAAAP/Xda35C90p9599p0F1eHR/N1KILWSoWVAiOMFjBBXaXSMxg== + dependencies: + quick-lru "^5.1.1" + resolve-alpn "^1.0.0" + +https-proxy-agent@^5.0.0: + version "5.0.1" + resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz#c59ef224a04fe8b754f3db0063a25ea30d0005d6" + integrity sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA== + dependencies: + agent-base "6" + debug "4" + +https-proxy-agent@^7.0.0, https-proxy-agent@^7.0.5: + version "7.0.6" + resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-7.0.6.tgz#da8dfeac7da130b05c2ba4b59c9b6cd66611a6b9" + integrity sha512-vK9P5/iUfdl95AI+JVyUuIcVtd4ofvtrOr3HNtM2yxC9bnMbEdp3x01OhQNnjb8IJYi38VlTE3mBXwcfvywuSw== + dependencies: + agent-base "^7.1.2" + debug "4" + +human-signals@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/human-signals/-/human-signals-5.0.0.tgz#42665a284f9ae0dade3ba41ebc37eb4b852f3a28" + integrity sha512-AXcZb6vzzrFAUE61HnN4mpLqd/cSIwNQjtNWR0euPm6y0iqx3G4gOXaIDdtdDwZmhwe82LA6+zinmW4UBWVePQ== + +humanize-ms@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/humanize-ms/-/humanize-ms-1.2.1.tgz#c46e3159a293f6b896da29316d8b6fe8bb79bbed" + integrity sha512-Fl70vYtsAFb/C06PTS9dZBo7ihau+Tu/DNCk/OyHhea07S+aeMWpFFkUaXRa8fI+ScZbEI8dfSxwY7gxZ9SAVQ== + dependencies: + ms "^2.0.0" + +husky@^9.1.7: + version "9.1.7" + resolved "https://registry.yarnpkg.com/husky/-/husky-9.1.7.tgz#d46a38035d101b46a70456a850ff4201344c0b2d" + integrity sha512-5gs5ytaNjBrh5Ow3zrvdUUY+0VxIuWVL4i9irt6friV+BqdCfmV11CQTWMiBYWHbXhco+J1kHfTOUkePhCDvMA== + +iconv-corefoundation@^1.1.7: + version "1.1.7" + resolved "https://registry.yarnpkg.com/iconv-corefoundation/-/iconv-corefoundation-1.1.7.tgz#31065e6ab2c9272154c8b0821151e2c88f1b002a" + integrity sha512-T10qvkw0zz4wnm560lOEg0PovVqUXuOFhhHAkixw8/sycy7TJt7v/RrkEKEQnAw2viPSJu6iAkErxnzR0g8PpQ== + dependencies: + cli-truncate "^2.1.0" + node-addon-api "^1.6.3" + +iconv-lite@0.4.24: + version "0.4.24" + resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b" + integrity sha512-v3MXnZAcvnywkTUEZomIActle7RXXeedOR31wwl7VlyoXO4Qi9arvSenNQWne1TcRwhCL1HwLI21bEqdpj8/rA== + dependencies: + safer-buffer ">= 2.1.2 < 3" + +iconv-lite@0.6.3, iconv-lite@^0.6.2, iconv-lite@^0.6.3: + version "0.6.3" + resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.6.3.tgz#a52f80bf38da1952eb5c681790719871a1a72501" + integrity sha512-4fCk79wshMdzMp2rH06qWrJE4iolqLhCUH+OiuIgU++RB0+94NlDL81atO7GX55uUKueo0txHNtvEyI6D7WdMw== + dependencies: + safer-buffer ">= 2.1.2 < 3.0.0" + +ieee754@^1.1.13: + version "1.2.1" + resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" + integrity sha512-dcyqhDvX1C46lXZcVqCpK+FtMRQVdIMN6/Df5js2zouUsqG7I6sFxitIC+7KYK29KdXOLHdu9zL4sFnoVQnqaA== + +ignore@^5.2.0, ignore@^5.3.1: + version "5.3.2" + resolved "https://registry.yarnpkg.com/ignore/-/ignore-5.3.2.tgz#3cd40e729f3643fd87cb04e50bf0eb722bc596f5" + integrity sha512-hsBTNUqQTDwkWtcdYI2i06Y/nUBEsNEDJKjWdigLvegy8kDuJAS8uRlpkkcQpyEXL0Z/pjDy5HBmMjRCJ2gq+g== + +immediate@~3.0.5: + version "3.0.6" + resolved "https://registry.yarnpkg.com/immediate/-/immediate-3.0.6.tgz#9db1dbd0faf8de6fbe0f5dd5e56bb606280de69b" + integrity sha512-XXOFtyqDjNDAQxVfYxuF7g9Il/IbWmmlQg2MYKOH8ExIT1qg6xc4zyS3HaEEATgs1btfzxq15ciUiY7gjSXRGQ== + +import-fresh@^3.2.1: + version "3.3.1" + resolved "https://registry.yarnpkg.com/import-fresh/-/import-fresh-3.3.1.tgz#9cecb56503c0ada1f2741dbbd6546e4b13b57ccf" + integrity sha512-TR3KfrTZTYLPB6jUjfx6MF9WcWrHL9su5TObK4ZkYgBdWKPOFoSoQIdEuTuR82pmtxH2spWG9h6etwfr1pLBqQ== + dependencies: + parent-module "^1.0.0" + resolve-from "^4.0.0" + +imurmurhash@^0.1.4: + version "0.1.4" + resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" + integrity sha512-JmXMZ6wuvDmLiHEml9ykzqO6lwFbof0GG4IkcGaENdCRDDmMVnny7s5HsIgHCbaq0w2MyPhDqkhTUgS2LU2PHA== + +indent-string@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/indent-string/-/indent-string-4.0.0.tgz#624f8f4497d619b2d9768531d58f4122854d7251" + integrity sha512-EdDDZu4A2OyIK7Lr/2zG+w5jmbuk1DVBnEwREQvBzspBJkCEbRa8GxU1lghYcaGJCnRWibjDXlq779X1/y5xwg== + +index-to-position@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/index-to-position/-/index-to-position-1.1.0.tgz#2e50bd54c8040bdd6d9b3d95ec2a8fedf86b4d44" + integrity sha512-XPdx9Dq4t9Qk1mTMbWONJqU7boCoumEH7fRET37HX5+khDUl3J2W6PdALxhILYlIYx2amlwYcRPp28p0tSiojg== + +infer-owner@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/infer-owner/-/infer-owner-1.0.4.tgz#c4cefcaa8e51051c2a40ba2ce8a3d27295af9467" + integrity sha512-IClj+Xz94+d7irH5qRyfJonOdfTzuDaifE6ZPWfx0N0+/ATZCbuTPq2prFl526urkQd90WyUKIh1DfBQ2hMz9A== + +inflight@^1.0.4: + version "1.0.6" + resolved "https://registry.yarnpkg.com/inflight/-/inflight-1.0.6.tgz#49bd6331d7d02d0c09bc910a1075ba8165b56df9" + integrity sha512-k92I/b08q4wvFscXCLvqfsHCrjrF7yiXsQuIVvVE7N82W3+aqpzuUdBbfhWcy/FZR3/4IgflMgKLOsvPDrGCJA== + dependencies: + once "^1.3.0" + wrappy "1" + +inherits@2, inherits@2.0.4, inherits@^2.0.3, inherits@^2.0.4, inherits@~2.0.3: + version "2.0.4" + resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" + integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== + +ini@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/ini/-/ini-2.0.0.tgz#e5fd556ecdd5726be978fa1001862eacb0a94bc5" + integrity sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA== + +ini@~1.3.0: + version "1.3.8" + resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c" + integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew== + +internal-slot@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/internal-slot/-/internal-slot-1.1.0.tgz#1eac91762947d2f7056bc838d93e13b2e9604961" + integrity sha512-4gd7VpWNQNB4UKKCFFVcp1AVv+FMOgs9NKzjHKusc8jTMhd5eL1NqQqOpE0KzMds804/yHlglp3uxgluOqAPLw== + dependencies: + es-errors "^1.3.0" + hasown "^2.0.2" + side-channel "^1.1.0" + +interpret@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/interpret/-/interpret-3.1.1.tgz#5be0ceed67ca79c6c4bc5cf0d7ee843dcea110c4" + integrity sha512-6xwYfHbajpoF0xLW+iwLkhwgvLoZDfjYfoFNu8ftMoXINzwuymNLd9u/KmwtdT2GbR+/Cz66otEGEVVUHX9QLQ== + +ip-address@^9.0.5: + version "9.0.5" + resolved "https://registry.yarnpkg.com/ip-address/-/ip-address-9.0.5.tgz#117a960819b08780c3bd1f14ef3c1cc1d3f3ea5a" + integrity sha512-zHtQzGojZXTwZTHQqra+ETKd4Sn3vgi7uBmlPoXVWZqYvuKmtI0l/VZTjqGmJY9x88GGOaZ9+G9ES8hC4T4X8g== + dependencies: + jsbn "1.1.0" + sprintf-js "^1.1.3" + +ipaddr.js@1.9.1: + version "1.9.1" + resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3" + integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g== + +is-array-buffer@^3.0.4, is-array-buffer@^3.0.5: + version "3.0.5" + resolved "https://registry.yarnpkg.com/is-array-buffer/-/is-array-buffer-3.0.5.tgz#65742e1e687bd2cc666253068fd8707fe4d44280" + integrity sha512-DDfANUiiG2wC1qawP66qlTugJeL5HyzMpfr8lLK+jMQirGzNod0B12cFB/9q838Ru27sBwfw78/rdoU7RERz6A== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.3" + get-intrinsic "^1.2.6" + +is-arrayish@^0.2.1: + version "0.2.1" + resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d" + integrity sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg== + +is-async-function@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/is-async-function/-/is-async-function-2.1.1.tgz#3e69018c8e04e73b738793d020bfe884b9fd3523" + integrity sha512-9dgM/cZBnNvjzaMYHVoxxfPj2QXt22Ev7SuuPrs+xav0ukGB0S6d4ydZdEiM48kLx5kDV+QBPrpVnFyefL8kkQ== + dependencies: + async-function "^1.0.0" + call-bound "^1.0.3" + get-proto "^1.0.1" + has-tostringtag "^1.0.2" + safe-regex-test "^1.1.0" + +is-bigint@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/is-bigint/-/is-bigint-1.1.0.tgz#dda7a3445df57a42583db4228682eba7c4170672" + integrity sha512-n4ZT37wG78iz03xPRKJrHTdZbe3IicyucEtdRsV5yglwc3GyUfbAfpSeD0FJ41NbUNSt5wbhqfp1fS+BgnvDFQ== + dependencies: + has-bigints "^1.0.2" + +is-binary-path@~2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09" + integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw== + dependencies: + binary-extensions "^2.0.0" + +is-boolean-object@^1.2.1: + version "1.2.2" + resolved "https://registry.yarnpkg.com/is-boolean-object/-/is-boolean-object-1.2.2.tgz#7067f47709809a393c71ff5bb3e135d8a9215d9e" + integrity sha512-wa56o2/ElJMYqjCjGkXri7it5FbebW5usLw/nPmCMs5DeZ7eziSYZhSmPRn0txqeW4LnAmQQU7FgqLpsEFKM4A== + dependencies: + call-bound "^1.0.3" + has-tostringtag "^1.0.2" + +is-callable@^1.2.7: + version "1.2.7" + resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.7.tgz#3bc2a85ea742d9e36205dcacdd72ca1fdc51b055" + integrity sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA== + +is-ci@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/is-ci/-/is-ci-3.0.1.tgz#db6ecbed1bd659c43dac0f45661e7674103d1867" + integrity sha512-ZYvCgrefwqoQ6yTyYUbQu64HsITZ3NfKX1lzaEYdkTDcfKzzCI/wthRRYKkdjHKFVgNiXKAKm65Zo1pk2as/QQ== + dependencies: + ci-info "^3.2.0" + +is-core-module@^2.13.0, is-core-module@^2.15.1, is-core-module@^2.16.0: + version "2.16.1" + resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.16.1.tgz#2a98801a849f43e2add644fbb6bc6229b19a4ef4" + integrity sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w== + dependencies: + hasown "^2.0.2" + +is-data-view@^1.0.1, is-data-view@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/is-data-view/-/is-data-view-1.0.2.tgz#bae0a41b9688986c2188dda6657e56b8f9e63b8e" + integrity sha512-RKtWF8pGmS87i2D6gqQu/l7EYRlVdfzemCJN/P3UOs//x1QE7mfhvzHIApBTRf7axvT6DMGwSwBXYCT0nfB9xw== + dependencies: + call-bound "^1.0.2" + get-intrinsic "^1.2.6" + is-typed-array "^1.1.13" + +is-date-object@^1.0.5, is-date-object@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/is-date-object/-/is-date-object-1.1.0.tgz#ad85541996fc7aa8b2729701d27b7319f95d82f7" + integrity sha512-PwwhEakHVKTdRNVOw+/Gyh0+MzlCl4R6qKvkhuvLtPMggI1WAHt9sOwZxQLSGpUaDnrdyDsomoRgNnCfKNSXXg== + dependencies: + call-bound "^1.0.2" + has-tostringtag "^1.0.2" + +is-docker@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-docker/-/is-docker-3.0.0.tgz#90093aa3106277d8a77a5910dbae71747e15a200" + integrity sha512-eljcgEDlEns/7AXFosB5K/2nCM4P7FQPkGc/DWLy5rmFEWvZayGrik1d9/QIY5nJ4f9YsVvBkA6kJpHn9rISdQ== + +is-electron@^2.2.0: + version "2.2.2" + resolved "https://registry.yarnpkg.com/is-electron/-/is-electron-2.2.2.tgz#3778902a2044d76de98036f5dc58089ac4d80bb9" + integrity sha512-FO/Rhvz5tuw4MCWkpMzHFKWD2LsfHzIb7i6MdPYZ/KW7AlxawyLkqdy+jPZP1WubqEADE3O4FUENlJHDfQASRg== + +is-extglob@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" + integrity sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ== + +is-finalizationregistry@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/is-finalizationregistry/-/is-finalizationregistry-1.1.1.tgz#eefdcdc6c94ddd0674d9c85887bf93f944a97c90" + integrity sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg== + dependencies: + call-bound "^1.0.3" + +is-fullwidth-code-point@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" + integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== + +is-fullwidth-code-point@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-4.0.0.tgz#fae3167c729e7463f8461ce512b080a49268aa88" + integrity sha512-O4L094N2/dZ7xqVdrXhh9r1KODPJpFms8B5sGdJLPy664AgvXsreZUyCQQNItZRDlYug4xStLjNp/sz3HvBowQ== + +is-generator-function@^1.0.10: + version "1.1.0" + resolved "https://registry.yarnpkg.com/is-generator-function/-/is-generator-function-1.1.0.tgz#bf3eeda931201394f57b5dba2800f91a238309ca" + integrity sha512-nPUB5km40q9e8UfN/Zc24eLlzdSf9OfKByBw9CIdw4H1giPMeA0OIJvbchsCu4npfI2QcMVBsGEBHKZ7wLTWmQ== + dependencies: + call-bound "^1.0.3" + get-proto "^1.0.0" + has-tostringtag "^1.0.2" + safe-regex-test "^1.1.0" + +is-glob@^4.0.0, is-glob@^4.0.1, is-glob@^4.0.3, is-glob@~4.0.1: + version "4.0.3" + resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084" + integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg== + dependencies: + is-extglob "^2.1.1" + +is-inside-container@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-inside-container/-/is-inside-container-1.0.0.tgz#e81fba699662eb31dbdaf26766a61d4814717ea4" + integrity sha512-KIYLCCJghfHZxqjYBE7rEy0OBuTd5xCHS7tHVgvCLkx7StIoaxwNW3hCALgEUjFfeRk+MG/Qxmp/vtETEF3tRA== + dependencies: + is-docker "^3.0.0" + +is-interactive@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/is-interactive/-/is-interactive-1.0.0.tgz#cea6e6ae5c870a7b0a0004070b7b587e0252912e" + integrity sha512-2HvIEKRoqS62guEC+qBjpvRubdX910WCMuJTZ+I9yvqKU2/12eSL549HMwtabb4oupdj2sMP50k+XJfB/8JE6w== + +is-interactive@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/is-interactive/-/is-interactive-2.0.0.tgz#40c57614593826da1100ade6059778d597f16e90" + integrity sha512-qP1vozQRI+BMOPcjFzrjXuQvdak2pHNUMZoeG2eRbiSqyvbEf/wQtEOTOX1guk6E3t36RkaqiSt8A/6YElNxLQ== + +is-lambda@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/is-lambda/-/is-lambda-1.0.1.tgz#3d9877899e6a53efc0160504cde15f82e6f061d5" + integrity sha512-z7CMFGNrENq5iFB9Bqo64Xk6Y9sg+epq1myIcdHaGnbMTYOxvzsEtdYqQUylB7LxfkvgrrjP32T6Ywciio9UIQ== + +is-map@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/is-map/-/is-map-2.0.3.tgz#ede96b7fe1e270b3c4465e3a465658764926d62e" + integrity sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw== + +is-number-object@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/is-number-object/-/is-number-object-1.1.1.tgz#144b21e95a1bc148205dcc2814a9134ec41b2541" + integrity sha512-lZhclumE1G6VYD8VHe35wFaIif+CTy5SJIi5+3y4psDgWu4wPDoBhF8NxUOinEc7pHgiTsT6MaBb92rKhhD+Xw== + dependencies: + call-bound "^1.0.3" + has-tostringtag "^1.0.2" + +is-number@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" + integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== + +is-path-inside@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz#d231362e53a07ff2b0e0ea7fed049161ffd16283" + integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ== + +is-plain-obj@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-plain-obj/-/is-plain-obj-2.1.0.tgz#45e42e37fccf1f40da8e5f76ee21515840c09287" + integrity sha512-YWnfyRwxL/+SsrWYfOpUtz5b3YD+nyfkHvjbcanzk8zgyO4ASD67uVMRt8k5bM4lLMDnXfriRhOpemw+NfT1eA== + +is-regex@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/is-regex/-/is-regex-1.2.1.tgz#76d70a3ed10ef9be48eb577887d74205bf0cad22" + integrity sha512-MjYsKHO5O7mCsmRGxWcLWheFqN9DJ/2TmngvjKXihe6efViPqc274+Fx/4fYj/r03+ESvBdTXK0V6tA3rgez1g== + dependencies: + call-bound "^1.0.2" + gopd "^1.2.0" + has-tostringtag "^1.0.2" + hasown "^2.0.2" + +is-set@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/is-set/-/is-set-2.0.3.tgz#8ab209ea424608141372ded6e0cb200ef1d9d01d" + integrity sha512-iPAjerrse27/ygGLxw+EBR9agv9Y6uLeYVJMu+QNCoouJ1/1ri0mGrcWpfCqFZuzzx3WjtwxG098X+n4OuRkPg== + +is-shared-array-buffer@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/is-shared-array-buffer/-/is-shared-array-buffer-1.0.4.tgz#9b67844bd9b7f246ba0708c3a93e34269c774f6f" + integrity sha512-ISWac8drv4ZGfwKl5slpHG9OwPNty4jOWPRIhBpxOoD+hqITiwuipOQ2bNthAzwA3B4fIjO4Nln74N0S9byq8A== + dependencies: + call-bound "^1.0.3" + +is-stream@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-1.1.0.tgz#12d4a3dd4e68e0b79ceb8dbc84173ae80d91ca44" + integrity sha512-uQPm8kcs47jx38atAcWTVxyltQYoPT68y9aWYdV6yWXSyW8mzSat0TL6CiWdZeCdF3KrAvpVtnHbTv4RN+rqdQ== + +is-stream@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/is-stream/-/is-stream-3.0.0.tgz#e6bfd7aa6bef69f4f472ce9bb681e3e57b4319ac" + integrity sha512-LnQR4bZ9IADDRSkvpqMGvt/tEJWclzklNgSw48V5EAaAeDd6qGvN8ei6k5p0tvxSR171VmGyHuTiAOfxAbr8kA== + +is-string@^1.0.7, is-string@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/is-string/-/is-string-1.1.1.tgz#92ea3f3d5c5b6e039ca8677e5ac8d07ea773cbb9" + integrity sha512-BtEeSsoaQjlSPBemMQIrY1MY0uM6vnS1g5fmufYOtnxLGUZM2178PKbhsk7Ffv58IX+ZtcvoGwccYsh0PglkAA== + dependencies: + call-bound "^1.0.3" + has-tostringtag "^1.0.2" + +is-symbol@^1.0.4, is-symbol@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/is-symbol/-/is-symbol-1.1.1.tgz#f47761279f532e2b05a7024a7506dbbedacd0634" + integrity sha512-9gGx6GTtCQM73BgmHQXfDmLtfjjTUDSyoxTCbp5WtoixAhfgsDirWIcVQ/IHpvI5Vgd5i/J5F7B9cN/WlVbC/w== + dependencies: + call-bound "^1.0.2" + has-symbols "^1.1.0" + safe-regex-test "^1.1.0" + +is-typed-array@^1.1.13, is-typed-array@^1.1.14, is-typed-array@^1.1.15: + version "1.1.15" + resolved "https://registry.yarnpkg.com/is-typed-array/-/is-typed-array-1.1.15.tgz#4bfb4a45b61cee83a5a46fba778e4e8d59c0ce0b" + integrity sha512-p3EcsicXjit7SaskXHs1hA91QxgTw46Fv6EFKKGS5DRFLD8yKnohjF3hxoju94b/OcMZoQukzpPpBE9uLVKzgQ== + dependencies: + which-typed-array "^1.1.16" + +is-unicode-supported@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz#3f26c76a809593b52bfa2ecb5710ed2779b522a7" + integrity sha512-knxG2q4UC3u8stRGyAVJCOdxFmv5DZiRcdlIaAQXAbSfJya+OhopNotLQrstBhququ4ZpuKbDc/8S6mgXgPFPw== + +is-unicode-supported@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-1.3.0.tgz#d824984b616c292a2e198207d4a609983842f714" + integrity sha512-43r2mRvz+8JRIKnWJ+3j8JtjRKZ6GmjzfaE/qiBJnikNnYv/6bagRJ1kUhNk8R5EX/GkobD+r+sfxCPJsiKBLQ== + +is-unicode-supported@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/is-unicode-supported/-/is-unicode-supported-2.1.0.tgz#09f0ab0de6d3744d48d265ebb98f65d11f2a9b3a" + integrity sha512-mE00Gnza5EEB3Ds0HfMyllZzbBrmLOX3vfWoj9A9PEnTfratQ/BcaJOuMhnkhjXvb2+FkY3VuHqtAGpTPmglFQ== + +is-weakmap@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/is-weakmap/-/is-weakmap-2.0.2.tgz#bf72615d649dfe5f699079c54b83e47d1ae19cfd" + integrity sha512-K5pXYOm9wqY1RgjpL3YTkF39tni1XajUIkawTLUo9EZEVUFga5gSQJF8nNS7ZwJQ02y+1YCNYcMh+HIf1ZqE+w== + +is-weakref@^1.0.2, is-weakref@^1.1.0: + version "1.1.1" + resolved "https://registry.yarnpkg.com/is-weakref/-/is-weakref-1.1.1.tgz#eea430182be8d64174bd96bffbc46f21bf3f9293" + integrity sha512-6i9mGWSlqzNMEqpCp93KwRS1uUOodk2OJ6b+sq7ZPDSy2WuI5NFIxp/254TytR8ftefexkWn5xNiHUNpPOfSew== + dependencies: + call-bound "^1.0.3" + +is-weakset@^2.0.3: + version "2.0.4" + resolved "https://registry.yarnpkg.com/is-weakset/-/is-weakset-2.0.4.tgz#c9f5deb0bc1906c6d6f1027f284ddf459249daca" + integrity sha512-mfcwb6IzQyOKTs84CQMrOwW4gQcaTOAWJ0zzJCl2WSPDrWk/OzDaImWFH3djXhb24g4eudZfLRozAvPGw4d9hQ== + dependencies: + call-bound "^1.0.3" + get-intrinsic "^1.2.6" + +is-wsl@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/is-wsl/-/is-wsl-3.1.0.tgz#e1c657e39c10090afcbedec61720f6b924c3cbd2" + integrity sha512-UcVfVfaK4Sc4m7X3dUSoHoozQGBEFeDC+zVo06t98xe8CzHSZZBekNXH+tu0NalHolcJ/QAGqS46Hef7QXBIMw== + dependencies: + is-inside-container "^1.0.0" + +isarray@^2.0.5: + version "2.0.5" + resolved "https://registry.yarnpkg.com/isarray/-/isarray-2.0.5.tgz#8af1e4c1221244cc62459faf38940d4e644a5723" + integrity sha512-xHjhDr3cNBK0BzdUJSPXZntQUx/mwMS5Rw4A7lPJ90XGAO6ISP/ePDNuo0vhqOZU+UD5JoodwCAAoZQd3FeAKw== + +isarray@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/isarray/-/isarray-1.0.0.tgz#bb935d48582cba168c06834957a54a3e07124f11" + integrity sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ== + +isbinaryfile@^4.0.8: + version "4.0.10" + resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.10.tgz#0c5b5e30c2557a2f06febd37b7322946aaee42b3" + integrity sha512-iHrqe5shvBUcFbmZq9zOQHBoeOhZJu6RQGrDpBgenUm/Am+F3JM2MgQj+rK3Z601fzrL5gLZWtAPH2OBaSVcyw== + +isbinaryfile@^5.0.0: + version "5.0.4" + resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-5.0.4.tgz#2a2edefa76cafa66613fe4c1ea52f7f031017bdf" + integrity sha512-YKBKVkKhty7s8rxddb40oOkuP0NbaeXrQvLin6QMHL7Ypiy2RW9LwOVrVgZRyOrhQlayMd9t+D8yDy8MKFTSDQ== + +isexe@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" + integrity sha512-RHxMLp9lnKHGHRng9QFhRCMbYAcVpn69smSGcq3f36xjgVVWThj4qqLbTLlq7Ssj8B+fIQ1EuCEGI2lKsyQeIw== + +isomorphic-fetch@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/isomorphic-fetch/-/isomorphic-fetch-3.0.0.tgz#0267b005049046d2421207215d45d6a262b8b8b4" + integrity sha512-qvUtwJ3j6qwsF3jLxkZ72qCgjMysPzDfeV240JHiGZsANBYd+EEuu35v7dfrJ9Up0Ak07D7GGSkGhCHTqg/5wA== + dependencies: + node-fetch "^2.6.1" + whatwg-fetch "^3.4.1" + +iterator.prototype@^1.1.4: + version "1.1.5" + resolved "https://registry.yarnpkg.com/iterator.prototype/-/iterator.prototype-1.1.5.tgz#12c959a29de32de0aa3bbbb801f4d777066dae39" + integrity sha512-H0dkQoCa3b2VEeKQBOxFph+JAbcrQdE7KC0UkqwpLmv2EC4P41QXP+rqo9wYodACiG5/WM5s9oDApTU8utwj9g== + dependencies: + define-data-property "^1.1.4" + es-object-atoms "^1.0.0" + get-intrinsic "^1.2.6" + get-proto "^1.0.0" + has-symbols "^1.1.0" + set-function-name "^2.0.2" + +jackspeak@^3.1.2: + version "3.4.3" + resolved "https://registry.yarnpkg.com/jackspeak/-/jackspeak-3.4.3.tgz#8833a9d89ab4acde6188942bd1c53b6390ed5a8a" + integrity sha512-OGlZQpz2yfahA/Rd1Y8Cd9SIEsqvXkLVoSw/cgwhnhFMDbsQFeZYoJJ7bIZBS9BcamUW96asq/npPWugM+RQBw== + dependencies: + "@isaacs/cliui" "^8.0.2" + optionalDependencies: + "@pkgjs/parseargs" "^0.11.0" + +jackspeak@^4.0.1: + version "4.1.0" + resolved "https://registry.yarnpkg.com/jackspeak/-/jackspeak-4.1.0.tgz#c489c079f2b636dc4cbe9b0312a13ff1282e561b" + integrity sha512-9DDdhb5j6cpeitCbvLO7n7J4IxnbM6hoF6O1g4HQ5TfhvvKN8ywDM7668ZhMHRqVmxqhps/F6syWK2KcPxYlkw== + dependencies: + "@isaacs/cliui" "^8.0.2" + +jake@^10.8.5: + version "10.9.2" + resolved "https://registry.yarnpkg.com/jake/-/jake-10.9.2.tgz#6ae487e6a69afec3a5e167628996b59f35ae2b7f" + integrity sha512-2P4SQ0HrLQ+fw6llpLnOaGAvN2Zu6778SJMrCUwns4fOoG9ayrTiZk3VV8sCPkVZF8ab0zksVpS8FDY5pRCNBA== + dependencies: + async "^3.2.3" + chalk "^4.0.2" + filelist "^1.0.4" + minimatch "^3.1.2" + +jest-diff@^29.7.0: + version "29.7.0" + resolved "https://registry.yarnpkg.com/jest-diff/-/jest-diff-29.7.0.tgz#017934a66ebb7ecf6f205e84699be10afd70458a" + integrity sha512-LMIgiIrhigmPrs03JHpxUh2yISK3vLFPkAodPeo0+BuF7wA2FoQbkEg1u8gBYBThncu7e1oEDUfIXVuTqLRUjw== + dependencies: + chalk "^4.0.0" + diff-sequences "^29.6.3" + jest-get-type "^29.6.3" + pretty-format "^29.7.0" + +jest-get-type@^29.6.3: + version "29.6.3" + resolved "https://registry.yarnpkg.com/jest-get-type/-/jest-get-type-29.6.3.tgz#36f499fdcea197c1045a127319c0481723908fd1" + integrity sha512-zrteXnqYxfQh7l5FHyL38jL39di8H8rHoecLH3JNxH3BwOrBsNeabdap5e0I23lD4HHI8W5VFBZqG4Eaq5LNcw== + +jest-matcher-utils@^29.7.0: + version "29.7.0" + resolved "https://registry.yarnpkg.com/jest-matcher-utils/-/jest-matcher-utils-29.7.0.tgz#ae8fec79ff249fd592ce80e3ee474e83a6c44f12" + integrity sha512-sBkD+Xi9DtcChsI3L3u0+N0opgPYnCRPtGcQYrgXmR+hmt/fYfWAL0xRXYU8eWOdfuLgBe0YCW3AFtnRLagq/g== + dependencies: + chalk "^4.0.0" + jest-diff "^29.7.0" + jest-get-type "^29.6.3" + pretty-format "^29.7.0" + +jest-message-util@^29.7.0: + version "29.7.0" + resolved "https://registry.yarnpkg.com/jest-message-util/-/jest-message-util-29.7.0.tgz#8bc392e204e95dfe7564abbe72a404e28e51f7f3" + integrity sha512-GBEV4GRADeP+qtB2+6u61stea8mGcOT4mCtrYISZwfu9/ISHFJ/5zOMXYbpBE9RsS5+Gb63DW4FgmnKJ79Kf6w== + dependencies: + "@babel/code-frame" "^7.12.13" + "@jest/types" "^29.6.3" + "@types/stack-utils" "^2.0.0" + chalk "^4.0.0" + graceful-fs "^4.2.9" + micromatch "^4.0.4" + pretty-format "^29.7.0" + slash "^3.0.0" + stack-utils "^2.0.3" + +jest-util@^29.7.0: + version "29.7.0" + resolved "https://registry.yarnpkg.com/jest-util/-/jest-util-29.7.0.tgz#23c2b62bfb22be82b44de98055802ff3710fc0bc" + integrity sha512-z6EbKajIpqGKU56y5KBUgy1dt1ihhQJgWzUlZHArA/+X2ad7Cb5iF+AK1EWVL/Bo7Rz9uurpqw6SiBCefUbCGA== + dependencies: + "@jest/types" "^29.6.3" + "@types/node" "*" + chalk "^4.0.0" + ci-info "^3.2.0" + graceful-fs "^4.2.9" + picomatch "^2.2.3" + +jiti@^1.21.6: + version "1.21.7" + resolved "https://registry.yarnpkg.com/jiti/-/jiti-1.21.7.tgz#9dd81043424a3d28458b193d965f0d18a2300ba9" + integrity sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A== + +jose@^4.15.9: + version "4.15.9" + resolved "https://registry.yarnpkg.com/jose/-/jose-4.15.9.tgz#9b68eda29e9a0614c042fa29387196c7dd800100" + integrity sha512-1vUQX+IdDMVPj4k8kOxgUqlcK518yluMuGZwqlr44FS1ppZB/5GWh4rZG89erpOBOJjU/OBsnCVFfapsRz6nEA== + +js-levenshtein@^1.1.6: + version "1.1.6" + resolved "https://registry.yarnpkg.com/js-levenshtein/-/js-levenshtein-1.1.6.tgz#c6cee58eb3550372df8deb85fad5ce66ce01d59d" + integrity sha512-X2BB11YZtrRqY4EnQcLX5Rh373zbK4alC1FW7D7MBhL2gtcC17cTnr6DmfHZeS0s2rTHjUTMMHfG7gO8SSdw+g== + +"js-tokens@^3.0.0 || ^4.0.0", js-tokens@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-4.0.0.tgz#19203fb59991df98e3a287050d4647cdeaf32499" + integrity sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ== + +js-tokens@^9.0.1: + version "9.0.1" + resolved "https://registry.yarnpkg.com/js-tokens/-/js-tokens-9.0.1.tgz#2ec43964658435296f6761b34e10671c2d9527f4" + integrity sha512-mxa9E9ITFOt0ban3j6L5MpjwegGz6lBQmM1IJkWeBZGcMxto50+eWdjC/52xDbS2vy0k7vIMK0Fe2wfL9OQSpQ== + +js-yaml@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-4.1.0.tgz#c1fb65f8f5017901cdd2c951864ba18458a10602" + integrity sha512-wpxZs9NoxZaJESJGIZTyDEaYpl0FKSA+FB9aJiyemKhMwkxQg63h4T1KJgUGHpTqPDNRcmmYLugrRjJlBtWvRA== + dependencies: + argparse "^2.0.1" + +jsbn@1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/jsbn/-/jsbn-1.1.0.tgz#b01307cb29b618a1ed26ec79e911f803c4da0040" + integrity sha512-4bYVV3aAMtDTTu4+xsDYa6sy9GyJ69/amsu9sYF2zqjiEoZA5xJi3BrfX3uY+/IekIu7MwdObdbDWpoZdBv3/A== + +jsesc@^3.0.2: + version "3.1.0" + resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-3.1.0.tgz#74d335a234f67ed19907fdadfac7ccf9d409825d" + integrity sha512-/sM3dO2FOzXjKQhJuo0Q173wf2KOo8t4I8vHy6lF9poUp7bKT0/NHE8fPX23PwfhnykfqnC2xRxOnVw5XuGIaA== + +jsesc@~3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/jsesc/-/jsesc-3.0.2.tgz#bb8b09a6597ba426425f2e4a07245c3d00b9343e" + integrity sha512-xKqzzWXDttJuOcawBt4KnKHHIf5oQ/Cxax+0PWFG+DFDgHNAdi+TXECADI+RYiFUMmx8792xsMbbgXj4CwnP4g== + +json-buffer@3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.1.tgz#9338802a30d3b6605fbe0613e094008ca8c05a13" + integrity sha512-4bV5BfR2mqfQTJm+V5tPPdf+ZpuhiIvTuAB5g8kcrXOZpTT/QwwVRWBywX1ozr6lEuPdbHxwaJlm9G6mI2sfSQ== + +json-rpc-2.0@^1.6.0, json-rpc-2.0@^1.7.0: + version "1.7.0" + resolved "https://registry.yarnpkg.com/json-rpc-2.0/-/json-rpc-2.0-1.7.0.tgz#840deb0bc168463e12bceb462f7fe225e793fc17" + integrity sha512-asnLgC1qD5ytP+fvBP8uL0rvj+l8P6iYICbzZ8dVxCpESffVjzA7KkYkbKCIbavs7cllwH1ZUaNtJwphdeRqpg== + +json-schema-traverse@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" + integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== + +json-schema-traverse@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz#ae7bcb3656ab77a73ba5c49bf654f38e6b6860e2" + integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug== + +json-stable-stringify-without-jsonify@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/json-stable-stringify-without-jsonify/-/json-stable-stringify-without-jsonify-1.0.1.tgz#9db7b59496ad3f3cfef30a75142d2d930ad72651" + integrity sha512-Bdboy+l7tA3OGW6FjyFHWkP5LuByj1Tk33Ljyq0axyzdk9//JSi2u3fP1QSmd1KNwq6VOKYGlAu87CisVir6Pw== + +json-stringify-safe@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/json-stringify-safe/-/json-stringify-safe-5.0.1.tgz#1296a2d58fd45f19a0f6ce01d65701e2c735b6eb" + integrity sha512-ZClg6AaYvamvYEE82d3Iyd3vSSIjQ+odgjaTzRuO3s7toCdFKczob2i0zCh7JE8kWn17yvAWhUVxvqGwUalsRA== + +json5@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/json5/-/json5-1.0.2.tgz#63d98d60f21b313b77c4d6da18bfa69d80e1d593" + integrity sha512-g1MWMLBiz8FKi1e4w0UyVL3w+iJceWAFBAaBnnGKOpNa5f8TLktkbre1+s6oICydWAm+HRUGTmI+//xv2hvXYA== + dependencies: + minimist "^1.2.0" + +json5@^2.2.3: + version "2.2.3" + resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.3.tgz#78cd6f1a19bdc12b73db5ad0c61efd66c1e29283" + integrity sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg== + +jsonc-parser@^3.2.0: + version "3.3.1" + resolved "https://registry.yarnpkg.com/jsonc-parser/-/jsonc-parser-3.3.1.tgz#f2a524b4f7fd11e3d791e559977ad60b98b798b4" + integrity sha512-HUgH65KyejrUFPvHFPbqOY0rsFip3Bo5wb4ngvdi1EpCYWUQDC5V+Y7mZws+DLkr4M//zQJoanu1SP+87Dv1oQ== + +jsonfile@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-4.0.0.tgz#8771aae0799b64076b76640fca058f9c10e33ecb" + integrity sha512-m6F1R3z8jjlf2imQHS2Qez5sjKWQzbuuhuJ/FKYFRZvPE3PuHcSMVZzfsLhGVOkfd20obL5SWEBew5ShlquNxg== + optionalDependencies: + graceful-fs "^4.1.6" + +jsonfile@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/jsonfile/-/jsonfile-6.1.0.tgz#bc55b2634793c679ec6403094eb13698a6ec0aae" + integrity sha512-5dgndWOriYSm5cnYaJNhalLNDKOqFwyDB/rr1E9ZsGciGvKPs8R2xYGCacuf3z6K1YKDz182fd+fY3cn3pMqXQ== + dependencies: + universalify "^2.0.0" + optionalDependencies: + graceful-fs "^4.1.6" + +jsonwebtoken@^9.0.0: + version "9.0.2" + resolved "https://registry.yarnpkg.com/jsonwebtoken/-/jsonwebtoken-9.0.2.tgz#65ff91f4abef1784697d40952bb1998c504caaf3" + integrity sha512-PRp66vJ865SSqOlgqS8hujT5U4AOgMfhrwYIuIhfKaoSCZcirrmASQr8CX7cUg+RMih+hgznrjp99o+W4pJLHQ== + dependencies: + jws "^3.2.2" + lodash.includes "^4.3.0" + lodash.isboolean "^3.0.3" + lodash.isinteger "^4.0.4" + lodash.isnumber "^3.0.3" + lodash.isplainobject "^4.0.6" + lodash.isstring "^4.0.1" + lodash.once "^4.0.0" + ms "^2.1.1" + semver "^7.5.4" + +"jsx-ast-utils@^2.4.1 || ^3.0.0", jsx-ast-utils@^3.3.5: + version "3.3.5" + resolved "https://registry.yarnpkg.com/jsx-ast-utils/-/jsx-ast-utils-3.3.5.tgz#4766bd05a8e2a11af222becd19e15575e52a853a" + integrity sha512-ZZow9HBI5O6EPgSJLUb8n2NKgmVWTwCvHGwFuJlMjvLFqlGG6pjirPhtdsseaLZjSibD8eegzmYpUZwoIlj2cQ== + dependencies: + array-includes "^3.1.6" + array.prototype.flat "^1.3.1" + object.assign "^4.1.4" + object.values "^1.1.6" + +jszip@^3.10.1: + version "3.10.1" + resolved "https://registry.yarnpkg.com/jszip/-/jszip-3.10.1.tgz#34aee70eb18ea1faec2f589208a157d1feb091c2" + integrity sha512-xXDvecyTpGLrqFrvkrUSoxxfJI5AH7U8zxxtVclpsUtMCq4JQ290LY8AW5c7Ggnr/Y/oK+bQMbqK2qmtk3pN4g== + dependencies: + lie "~3.3.0" + pako "~1.0.2" + readable-stream "~2.3.6" + setimmediate "^1.0.5" + +junk@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/junk/-/junk-3.1.0.tgz#31499098d902b7e98c5d9b9c80f43457a88abfa1" + integrity sha512-pBxcB3LFc8QVgdggvZWyeys+hnrNWg4OcZIU/1X59k5jQdLBlCsYGRQaz234SqoRLTCgMH00fY0xRJH+F9METQ== + +jwa@^1.4.1: + version "1.4.1" + resolved "https://registry.yarnpkg.com/jwa/-/jwa-1.4.1.tgz#743c32985cb9e98655530d53641b66c8645b039a" + integrity sha512-qiLX/xhEEFKUAJ6FiBMbes3w9ATzyk5W7Hvzpa/SLYdxNtng+gcurvrI7TbACjIXlsJyr05/S1oUhZrc63evQA== + dependencies: + buffer-equal-constant-time "1.0.1" + ecdsa-sig-formatter "1.0.11" + safe-buffer "^5.0.1" + +jws@^3.2.2: + version "3.2.2" + resolved "https://registry.yarnpkg.com/jws/-/jws-3.2.2.tgz#001099f3639468c9414000e99995fa52fb478304" + integrity sha512-YHlZCB6lMTllWDtSPHz/ZXTsi8S00usEV6v1tjq8tOUZzw7DpSDWVXjXDre6ed1w/pd495ODpHZYSdkRTsa0HA== + dependencies: + jwa "^1.4.1" + safe-buffer "^5.0.1" + +keytar@^7.7.0: + version "7.9.0" + resolved "https://registry.yarnpkg.com/keytar/-/keytar-7.9.0.tgz#4c6225708f51b50cbf77c5aae81721964c2918cb" + integrity sha512-VPD8mtVtm5JNtA2AErl6Chp06JBfy7diFQ7TQQhdpWOl6MrCRB+eRbvAZUsbGQS9kiMq0coJsy0W0vHpDCkWsQ== + dependencies: + node-addon-api "^4.3.0" + prebuild-install "^7.0.1" + +keyv@^4.0.0, keyv@^4.5.3: + version "4.5.4" + resolved "https://registry.yarnpkg.com/keyv/-/keyv-4.5.4.tgz#a879a99e29452f942439f2a405e3af8b31d4de93" + integrity sha512-oxVHkHR/EJf2CNXnWxRLW6mg7JyCCUcG0DtEGmL2ctUo1PNTin1PUil+r/+4r5MpVgC/fn1kjsx7mjSujKqIpw== + dependencies: + json-buffer "3.0.1" + +kill-port@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/kill-port/-/kill-port-2.0.1.tgz#e5e18e2706b13d54320938be42cb7d40609b15cf" + integrity sha512-e0SVOV5jFo0mx8r7bS29maVWp17qGqLBZ5ricNSajON6//kmb7qqqNnml4twNE8Dtj97UQD+gNFOaipS/q1zzQ== + dependencies: + get-them-args "1.3.2" + shell-exec "1.0.2" + +language-subtag-registry@^0.3.20: + version "0.3.23" + resolved "https://registry.yarnpkg.com/language-subtag-registry/-/language-subtag-registry-0.3.23.tgz#23529e04d9e3b74679d70142df3fd2eb6ec572e7" + integrity sha512-0K65Lea881pHotoGEa5gDlMxt3pctLi2RplBb7Ezh4rRdLEOtgi7n4EwK9lamnUCkKBqaeKRVebTq6BAxSkpXQ== + +language-tags@^1.0.9: + version "1.0.9" + resolved "https://registry.yarnpkg.com/language-tags/-/language-tags-1.0.9.tgz#1ffdcd0ec0fafb4b1be7f8b11f306ad0f9c08777" + integrity sha512-MbjN408fEndfiQXbFQ1vnd+1NoLDsnQW41410oQBXiyXDMYH5z505juWa4KUE1LqxRC7DgOgZDbKLxHIwm27hA== + dependencies: + language-subtag-registry "^0.3.20" + +lazy-val@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/lazy-val/-/lazy-val-1.0.5.tgz#6cf3b9f5bc31cee7ee3e369c0832b7583dcd923d" + integrity sha512-0/BnGCCfyUMkBpeDgWihanIAF9JmZhHBgUhEqzvf+adhNGLoP6TaiI5oF8oyb3I45P+PcnrqihSf01M0l0G5+Q== + +leven@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/leven/-/leven-3.1.0.tgz#77891de834064cccba82ae7842bb6b14a13ed7f2" + integrity sha512-qsda+H8jTaUaN/x5vzW2rzc+8Rw4TAQ/4KjB46IwK5VH+IlVeeeje/EoZRpiXvIqjFgK84QffqPztGI3VBLG1A== + +levn@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/levn/-/levn-0.4.1.tgz#ae4562c007473b932a6200d403268dd2fffc6ade" + integrity sha512-+bT2uH4E5LGE7h/n3evcS/sQlJXCpIp6ym8OWJ5eV6+67Dsql/LaaT7qJBAt2rzfoa/5QBGBhxDix1dMt2kQKQ== + dependencies: + prelude-ls "^1.2.1" + type-check "~0.4.0" + +lie@~3.3.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/lie/-/lie-3.3.0.tgz#dcf82dee545f46074daf200c7c1c5a08e0f40f6a" + integrity sha512-UaiMJzeWRlEujzAuw5LokY1L5ecNQYZKfmyZ9L7wDHb/p5etKaxXhohBcrw0EYby+G/NA52vRSN4N39dxHAIwQ== + dependencies: + immediate "~3.0.5" + +lilconfig@^3.0.0, lilconfig@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/lilconfig/-/lilconfig-3.1.3.tgz#a1bcfd6257f9585bf5ae14ceeebb7b559025e4c4" + integrity sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw== + +lines-and-columns@^1.1.6: + version "1.2.4" + resolved "https://registry.yarnpkg.com/lines-and-columns/-/lines-and-columns-1.2.4.tgz#eca284f75d2965079309dc0ad9255abb2ebc1632" + integrity sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg== + +linkify-it@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/linkify-it/-/linkify-it-5.0.0.tgz#9ef238bfa6dc70bd8e7f9572b52d369af569b421" + integrity sha512-5aHCbzQRADcdP+ATqnDuhhJ/MRIqDkZX5pyjFHRRysS8vZ5AbqGEoFIb6pYHPZ+L/OC2Lc+xT8uHVVR5CAK/wQ== + dependencies: + uc.micro "^2.0.0" + +listr2@^7.0.2: + version "7.0.2" + resolved "https://registry.yarnpkg.com/listr2/-/listr2-7.0.2.tgz#3aa3e1549dfaf3c57ab5eeaba754da3b87f33063" + integrity sha512-rJysbR9GKIalhTbVL2tYbF2hVyDnrf7pFUZBwjPaMIdadYHmeT+EVi/Bu3qd7ETQPahTotg2WRCatXwRBW554g== + dependencies: + cli-truncate "^3.1.0" + colorette "^2.0.20" + eventemitter3 "^5.0.1" + log-update "^5.0.1" + rfdc "^1.3.0" + wrap-ansi "^8.1.0" + +load-json-file@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/load-json-file/-/load-json-file-2.0.0.tgz#7947e42149af80d696cbf797bcaabcfe1fe29ca8" + integrity sha512-3p6ZOGNbiX4CdvEd1VcE6yi78UrGNpjHO33noGwHCnT/o2fyllJDepsm8+mFFv/DvtwFHht5HIHSyOy5a+ChVQ== + dependencies: + graceful-fs "^4.1.2" + parse-json "^2.2.0" + pify "^2.0.0" + strip-bom "^3.0.0" + +local-pkg@^0.5.0: + version "0.5.1" + resolved "https://registry.yarnpkg.com/local-pkg/-/local-pkg-0.5.1.tgz#69658638d2a95287534d4c2fff757980100dbb6d" + integrity sha512-9rrA30MRRP3gBD3HTGnC6cDFpaE1kVDWxWgqWJUN0RvDNAo+Nz/9GxB+nHOH0ifbVFy0hSA1V6vFDvnx54lTEQ== + dependencies: + mlly "^1.7.3" + pkg-types "^1.2.1" + +locate-path@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-2.0.0.tgz#2b568b265eec944c6d9c0de9c3dbbbca0354cd8e" + integrity sha512-NCI2kiDkyR7VeEKm27Kda/iQHyKJe1Bu0FlTbYp3CqJu+9IFe9bLyAjMxf5ZDDbEg+iMPzB5zYyUTSm8wVTKmA== + dependencies: + p-locate "^2.0.0" + path-exists "^3.0.0" + +locate-path@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/locate-path/-/locate-path-6.0.0.tgz#55321eb309febbc59c4801d931a72452a681d286" + integrity sha512-iPZK6eYjbxRu3uB4/WZ3EsEIMJFMqAoopl3R+zuq0UjcAm/MO6KCweDgPfP3elTztoKP3KtnVHxTn2NHBSDVUw== + dependencies: + p-locate "^5.0.0" + +lodash.curry@^4.0.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/lodash.curry/-/lodash.curry-4.1.1.tgz#248e36072ede906501d75966200a86dab8b23170" + integrity sha512-/u14pXGviLaweY5JI0IUzgzF2J6Ne8INyzAZjImcryjgkZ+ebruBxy2/JaOOkTqScddcYtakjhSaeemV8lR0tA== + +lodash.debounce@^4.0.8: + version "4.0.8" + resolved "https://registry.yarnpkg.com/lodash.debounce/-/lodash.debounce-4.0.8.tgz#82d79bff30a67c4005ffd5e2515300ad9ca4d7af" + integrity sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow== + +lodash.escaperegexp@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/lodash.escaperegexp/-/lodash.escaperegexp-4.1.2.tgz#64762c48618082518ac3df4ccf5d5886dae20347" + integrity sha512-TM9YBvyC84ZxE3rgfefxUWiQKLilstD6k7PTGt6wfbtXF8ixIJLOL3VYyV/z+ZiPLsVxAsKAFVwWlWeb2Y8Yyw== + +lodash.flow@^3.3.0: + version "3.5.0" + resolved "https://registry.yarnpkg.com/lodash.flow/-/lodash.flow-3.5.0.tgz#87bf40292b8cf83e4e8ce1a3ae4209e20071675a" + integrity sha512-ff3BX/tSioo+XojX4MOsOMhJw0nZoUEF011LX8g8d3gvjVbxd89cCio4BCXronjxcTUIJUoqKEUA+n4CqvvRPw== + +lodash.get@^4.0.0: + version "4.4.2" + resolved "https://registry.yarnpkg.com/lodash.get/-/lodash.get-4.4.2.tgz#2d177f652fa31e939b4438d5341499dfa3825e99" + integrity sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ== + +lodash.includes@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/lodash.includes/-/lodash.includes-4.3.0.tgz#60bb98a87cb923c68ca1e51325483314849f553f" + integrity sha512-W3Bx6mdkRTGtlJISOvVD/lbqjTlPPUDTMnlXZFnVwi9NKJ6tiAk6LVdlhZMm17VZisqhKcgzpO5Wz91PCt5b0w== + +lodash.isboolean@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/lodash.isboolean/-/lodash.isboolean-3.0.3.tgz#6c2e171db2a257cd96802fd43b01b20d5f5870f6" + integrity sha512-Bz5mupy2SVbPHURB98VAcw+aHh4vRV5IPNhILUCsOzRmsTmSQ17jIuqopAentWoehktxGd9e/hbIXq980/1QJg== + +lodash.isequal@^4.5.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/lodash.isequal/-/lodash.isequal-4.5.0.tgz#415c4478f2bcc30120c22ce10ed3226f7d3e18e0" + integrity sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ== + +lodash.isinteger@^4.0.4: + version "4.0.4" + resolved "https://registry.yarnpkg.com/lodash.isinteger/-/lodash.isinteger-4.0.4.tgz#619c0af3d03f8b04c31f5882840b77b11cd68343" + integrity sha512-DBwtEWN2caHQ9/imiNeEA5ys1JoRtRfY3d7V9wkqtbycnAmTvRRmbHKDV4a0EYc678/dia0jrte4tjYwVBaZUA== + +lodash.isnumber@^3.0.3: + version "3.0.3" + resolved "https://registry.yarnpkg.com/lodash.isnumber/-/lodash.isnumber-3.0.3.tgz#3ce76810c5928d03352301ac287317f11c0b1ffc" + integrity sha512-QYqzpfwO3/CWf3XP+Z+tkQsfaLL/EnUlXWVkIk5FUPc4sBdTehEqZONuyRt2P67PXAk+NXmTBcc97zw9t1FQrw== + +lodash.isplainobject@^4.0.6: + version "4.0.6" + resolved "https://registry.yarnpkg.com/lodash.isplainobject/-/lodash.isplainobject-4.0.6.tgz#7c526a52d89b45c45cc690b88163be0497f550cb" + integrity sha512-oSXzaWypCMHkPC3NvBEaPHf0KsA5mvPrOPgQWDsbg8n7orZ290M0BmC/jgRZ4vcJ6DTAhjrsSYgdsW/F+MFOBA== + +lodash.isstring@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/lodash.isstring/-/lodash.isstring-4.0.1.tgz#d527dfb5456eca7cc9bb95d5daeaf88ba54a5451" + integrity sha512-0wJxfxH1wgO3GrbuP+dTTk7op+6L41QCXbGINEmD+ny/G/eCqGzxyCsh7159S+mgDDcoarnBw6PC1PS5+wUGgw== + +lodash.merge@^4.6.2: + version "4.6.2" + resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a" + integrity sha512-0KpjqXRVvrYyCsX1swR/XTK0va6VQkQM6MNo7PqW77ByjAhoARA8EfrP1N4+KlKj8YS0ZUCtRT/YUuhyYDujIQ== + +lodash.once@^4.0.0: + version "4.1.1" + resolved "https://registry.yarnpkg.com/lodash.once/-/lodash.once-4.1.1.tgz#0dd3971213c7c56df880977d504c88fb471a97ac" + integrity sha512-Sb487aTOCr9drQVL8pIxOzVhafOjZN9UU54hiN8PU3uAiSV7lx1yYNpbNmex2PK6dSJoNTSJUUswT651yww3Mg== + +lodash@^4.17.15, lodash@^4.17.2, lodash@^4.17.20: + version "4.17.21" + resolved "https://registry.yarnpkg.com/lodash/-/lodash-4.17.21.tgz#679591c564c3bffaae8454cf0b3df370c3d6911c" + integrity sha512-v2kDEe57lecTulaDIuNTPy3Ry4gLGJ6Z1O3vE1krgXZNrsQ+LFTGHVxVjcXPs17LhbZVGedAJv8XZ1tvj5FvSg== + +log-symbols@^4.0.0, log-symbols@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-4.1.0.tgz#3fbdbb95b4683ac9fc785111e792e558d4abd503" + integrity sha512-8XPvpAA8uyhfteu8pIvQxpJZ7SYYdpUivZpGy6sFsBuKRY/7rQGavedeB8aK+Zkyq6upMFVL/9AW6vOYzfRyLg== + dependencies: + chalk "^4.1.0" + is-unicode-supported "^0.1.0" + +log-symbols@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/log-symbols/-/log-symbols-6.0.0.tgz#bb95e5f05322651cac30c0feb6404f9f2a8a9439" + integrity sha512-i24m8rpwhmPIS4zscNzK6MSEhk0DUWa/8iYQWxhffV8jkI4Phvs3F+quL5xvS0gdQR0FyTCMMH33Y78dDTzzIw== + dependencies: + chalk "^5.3.0" + is-unicode-supported "^1.3.0" + +log-update@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/log-update/-/log-update-5.0.1.tgz#9e928bf70cb183c1f0c9e91d9e6b7115d597ce09" + integrity sha512-5UtUDQ/6edw4ofyljDNcOVJQ4c7OjDro4h3y8e1GQL5iYElYclVHJ3zeWchylvMaKnDbDilC8irOVyexnA/Slw== + dependencies: + ansi-escapes "^5.0.0" + cli-cursor "^4.0.0" + slice-ansi "^5.0.0" + strip-ansi "^7.0.1" + wrap-ansi "^8.0.1" + +loose-envify@^1.0.0, loose-envify@^1.1.0, loose-envify@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/loose-envify/-/loose-envify-1.4.0.tgz#71ee51fa7be4caec1a63839f7e682d8132d30caf" + integrity sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q== + dependencies: + js-tokens "^3.0.0 || ^4.0.0" + +loupe@^2.3.6, loupe@^2.3.7: + version "2.3.7" + resolved "https://registry.yarnpkg.com/loupe/-/loupe-2.3.7.tgz#6e69b7d4db7d3ab436328013d37d1c8c3540c697" + integrity sha512-zSMINGVYkdpYSOBmLi0D1Uo7JU9nVdQKrHxC8eYlV+9YKK9WePqAlL7lSlorG/U2Fw1w0hTBmaa/jrQ3UbPHtA== + dependencies: + get-func-name "^2.0.1" + +loupe@^3.1.0, loupe@^3.1.3: + version "3.1.3" + resolved "https://registry.yarnpkg.com/loupe/-/loupe-3.1.3.tgz#042a8f7986d77f3d0f98ef7990a2b2fef18b0fd2" + integrity sha512-kkIp7XSkP78ZxJEsSxW3712C6teJVoeHHwgo9zJ380de7IYyJ2ISlxojcH2pC5OFLewESmnRi/+XCDIEEVyoug== + +lowercase-keys@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz#2603e78b7b4b0006cbca2fbcc8a3202558ac9479" + integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA== + +lru-cache@^10.2.0: + version "10.4.3" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-10.4.3.tgz#410fc8a17b70e598013df257c2446b7f3383f119" + integrity sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ== + +lru-cache@^11.0.0: + version "11.1.0" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-11.1.0.tgz#afafb060607108132dbc1cf8ae661afb69486117" + integrity sha512-QIXZUBJUx+2zHUdQujWejBkcD9+cs94tLn0+YL8UrCh+D5sCXZ4c7LaEH48pNwRY3MLDgqUFyhlCyjJPf1WP0A== + +lru-cache@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-5.1.1.tgz#1da27e6710271947695daf6848e847f01d84b920" + integrity sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w== + dependencies: + yallist "^3.0.2" + +lru-cache@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" + integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== + dependencies: + yallist "^4.0.0" + +lru-cache@^7.7.1: + version "7.18.3" + resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-7.18.3.tgz#f793896e0fd0e954a59dfdd82f0773808df6aa89" + integrity sha512-jumlc0BIUrS3qJGgIkWZsyfAM7NCWiBcCDhnd+3NNM5KbBmLTgHVfWBcg6W+rLUsIpzpERPsvwUP7CckAQSOoA== + +lz-string@^1.5.0: + version "1.5.0" + resolved "https://registry.yarnpkg.com/lz-string/-/lz-string-1.5.0.tgz#c1ab50f77887b712621201ba9fd4e3a6ed099941" + integrity sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ== + +magic-string@^0.30.17, magic-string@^0.30.5: + version "0.30.17" + resolved "https://registry.yarnpkg.com/magic-string/-/magic-string-0.30.17.tgz#450a449673d2460e5bbcfba9a61916a1714c7453" + integrity sha512-sNPKHvyjVf7gyjwS4xGTaW/mCnF8wnjtifKBEhxfZ7E/S8tQ0rssrwGNn6q8JH/ohItJfSQp9mBtQYuTlH5QnA== + dependencies: + "@jridgewell/sourcemap-codec" "^1.5.0" + +make-dir@^1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-1.3.0.tgz#79c1033b80515bd6d24ec9933e860ca75ee27f0c" + integrity sha512-2w31R7SJtieJJnQtGc7RVL2StM2vGYVfqUOvUDxH6bC6aJTxPxTF0GnIgCyu7tjockiUWAYQRbxa7vKn34s5sQ== + dependencies: + pify "^3.0.0" + +make-error@^1.1.1: + version "1.3.6" + resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2" + integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw== + +make-fetch-happen@^10.2.1: + version "10.2.1" + resolved "https://registry.yarnpkg.com/make-fetch-happen/-/make-fetch-happen-10.2.1.tgz#f5e3835c5e9817b617f2770870d9492d28678164" + integrity sha512-NgOPbRiaQM10DYXvN3/hhGVI2M5MtITFryzBGxHM5p4wnFxsVCbxkrBrDsk+EZ5OB4jEOT7AjDxtdF+KVEFT7w== + dependencies: + agentkeepalive "^4.2.1" + cacache "^16.1.0" + http-cache-semantics "^4.1.0" + http-proxy-agent "^5.0.0" + https-proxy-agent "^5.0.0" + is-lambda "^1.0.1" + lru-cache "^7.7.1" + minipass "^3.1.6" + minipass-collect "^1.0.2" + minipass-fetch "^2.0.3" + minipass-flush "^1.0.5" + minipass-pipeline "^1.2.4" + negotiator "^0.6.3" + promise-retry "^2.0.1" + socks-proxy-agent "^7.0.0" + ssri "^9.0.0" + +map-age-cleaner@^0.1.1: + version "0.1.3" + resolved "https://registry.yarnpkg.com/map-age-cleaner/-/map-age-cleaner-0.1.3.tgz#7d583a7306434c055fe474b0f45078e6e1b4b92a" + integrity sha512-bJzx6nMoP6PDLPBFmg7+xRKeFZvFboMrGlxmNj9ClvX53KrmvM5bXFXEWjbz4cz1AFn+jWJ9z/DJSz7hrs0w3w== + dependencies: + p-defer "^1.0.0" + +markdown-it@^14.1.0: + version "14.1.0" + resolved "https://registry.yarnpkg.com/markdown-it/-/markdown-it-14.1.0.tgz#3c3c5992883c633db4714ccb4d7b5935d98b7d45" + integrity sha512-a54IwgWPaeBCAAsv13YgmALOF1elABB08FxO9i+r4VFk5Vl4pKokRPeX8u5TCgSsPi6ec1otfLjdOpVcgbpshg== + dependencies: + argparse "^2.0.1" + entities "^4.4.0" + linkify-it "^5.0.0" + mdurl "^2.0.0" + punycode.js "^2.3.1" + uc.micro "^2.1.0" + +matcher@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/matcher/-/matcher-3.0.0.tgz#bd9060f4c5b70aa8041ccc6f80368760994f30ca" + integrity sha512-OkeDaAZ/bQCxeFAozM55PKcKU0yJMPGifLwV4Qgjitu+5MoAfSQN4lsLJeXZ1b8w0x+/Emda6MZgXS1jvsapng== + dependencies: + escape-string-regexp "^4.0.0" + +math-intrinsics@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/math-intrinsics/-/math-intrinsics-1.1.0.tgz#a0dd74be81e2aa5c2f27e65ce283605ee4e2b7f9" + integrity sha512-/IXtbwEk5HTPyEwyKX6hGkYXxM9nbj64B+ilVJnC/R6B0pH5G4V3b0pVbL7DBj4tkhBAppbQUlf6F6Xl9LHu1g== + +mdurl@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/mdurl/-/mdurl-2.0.0.tgz#80676ec0433025dd3e17ee983d0fe8de5a2237e0" + integrity sha512-Lf+9+2r+Tdp5wXDXC4PcIBjTDtq4UKjCPMQhKIuzpJNW0b96kVqSwW0bT7FhRSfmAiFYgP+SCRvdrDozfh0U5w== + +media-typer@0.3.0: + version "0.3.0" + resolved "https://registry.yarnpkg.com/media-typer/-/media-typer-0.3.0.tgz#8710d7af0aa626f8fffa1ce00168545263255748" + integrity sha512-dq+qelQ9akHpcOl/gUVRTxVIOkAJ1wR3QAvb4RsVjS8oVoFjDGTc679wJYmUmknUF5HwMLOgb5O+a3KxfWapPQ== + +mem@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/mem/-/mem-4.3.0.tgz#461af497bc4ae09608cdb2e60eefb69bff744178" + integrity sha512-qX2bG48pTqYRVmDB37rn/6PT7LcR8T7oAX3bf99u1Tt1nzxYfxkgqDwUwolPlXweM0XzBOBFzSx4kfp7KP1s/w== + dependencies: + map-age-cleaner "^0.1.1" + mimic-fn "^2.0.0" + p-is-promise "^2.0.0" + +merge-descriptors@1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/merge-descriptors/-/merge-descriptors-1.0.3.tgz#d80319a65f3c7935351e5cfdac8f9318504dbed5" + integrity sha512-gaNvAS7TZ897/rVaZ0nMtAyxNyi/pdbjbAwUpFQpN70GqnVfOiXpeUUMKRBmzXaSQ8DdTX4/0ms62r2K+hE6mQ== + +merge-stream@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/merge-stream/-/merge-stream-2.0.0.tgz#52823629a14dd00c9770fb6ad47dc6310f2c1f60" + integrity sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w== + +merge2@^1.3.0: + version "1.4.1" + resolved "https://registry.yarnpkg.com/merge2/-/merge2-1.4.1.tgz#4368892f885e907455a6fd7dc55c0c9d404990ae" + integrity sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg== + +meshoptimizer@~0.18.1: + version "0.18.1" + resolved "https://registry.yarnpkg.com/meshoptimizer/-/meshoptimizer-0.18.1.tgz#cdb90907f30a7b5b1190facd3b7ee6b7087797d8" + integrity sha512-ZhoIoL7TNV4s5B6+rx5mC//fw8/POGyNxS/DZyCJeiZ12ScLfVwRE/GfsxwiTkMYYD5DmK2/JXnEVXqL4rF+Sw== + +methods@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/methods/-/methods-1.1.2.tgz#5529a4d67654134edcc5266656835b0f851afcee" + integrity sha512-iclAHeNqNm68zFtnZ0e+1L2yUIdvzNoauKU4WBA3VvH/vPFieF7qfRlwUZU+DA9P9bPXIS90ulxoUoCH23sV2w== + +micromatch@^4.0.4, micromatch@^4.0.8: + version "4.0.8" + resolved "https://registry.yarnpkg.com/micromatch/-/micromatch-4.0.8.tgz#d66fa18f3a47076789320b9b1af32bd86d9fa202" + integrity sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA== + dependencies: + braces "^3.0.3" + picomatch "^2.3.1" + +mime-db@1.52.0: + version "1.52.0" + resolved "https://registry.yarnpkg.com/mime-db/-/mime-db-1.52.0.tgz#bbabcdc02859f4987301c856e3387ce5ec43bf70" + integrity sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg== + +mime-types@^2.1.12, mime-types@~2.1.24, mime-types@~2.1.34: + version "2.1.35" + resolved "https://registry.yarnpkg.com/mime-types/-/mime-types-2.1.35.tgz#381a871b62a734450660ae3deee44813f70d959a" + integrity sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw== + dependencies: + mime-db "1.52.0" + +mime@1.6.0, mime@^1.3.4, mime@^1.6.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1" + integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg== + +mime@^2.5.2: + version "2.6.0" + resolved "https://registry.yarnpkg.com/mime/-/mime-2.6.0.tgz#a2a682a95cd4d0cb1d6257e28f83da7e35800367" + integrity sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg== + +mimic-fn@^2.0.0, mimic-fn@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" + integrity sha512-OqbOk5oEQeAZ8WXWydlu9HJjz9WVdEIvamMCcXmuqUYjTknH/sqsWvhQ3vgwKFRR1HpjvNBKQ37nbJgYzGqGcg== + +mimic-fn@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-4.0.0.tgz#60a90550d5cb0b239cca65d893b1a53b29871ecc" + integrity sha512-vqiC06CuhBTUdZH+RYl8sFrL096vA45Ok5ISO6sE/Mr1jRbGH4Csnhi8f3wKVl7x8mO4Au7Ir9D3Oyv1VYMFJw== + +mimic-function@^5.0.0: + version "5.0.1" + resolved "https://registry.yarnpkg.com/mimic-function/-/mimic-function-5.0.1.tgz#acbe2b3349f99b9deaca7fb70e48b83e94e67076" + integrity sha512-VP79XUPxV2CigYP3jWwAUFSku2aKqBH7uTAapFWCBqutsbmDo96KY5o8uh6U+/YSIn5OxJnXp73beVkpqMIGhA== + +mimic-response@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-1.0.1.tgz#4923538878eef42063cb8a3e3b0798781487ab1b" + integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ== + +mimic-response@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-3.1.0.tgz#2d1d59af9c1b129815accc2c46a022a5ce1fa3c9" + integrity sha512-z0yWI+4FDrrweS8Zmt4Ej5HdJmky15+L2e6Wgn3+iK5fWzb6T3fhNFq2+MeTRb064c6Wr4N/wv0DzQTjNzHNGQ== + +min-indent@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/min-indent/-/min-indent-1.0.1.tgz#a63f681673b30571fbe8bc25686ae746eefa9869" + integrity sha512-I9jwMn07Sy/IwOj3zVkVik2JTvgpaykDZEigL6Rx6N9LbMywwUSMtxET+7lVoDLLd3O3IXwJwvuuns8UB/HeAg== + +minimatch@^10.0.0: + version "10.0.1" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-10.0.1.tgz#ce0521856b453c86e25f2c4c0d03e6ff7ddc440b" + integrity sha512-ethXTt3SGGR+95gudmqJ1eNhRO7eGEGIgYA9vnPatK4/etz2MEVDno5GMCibdMTuBMyElzIlgxMna3K94XDIDQ== + dependencies: + brace-expansion "^2.0.1" + +minimatch@^3.0.3, minimatch@^3.0.4, minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" + integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== + dependencies: + brace-expansion "^1.1.7" + +minimatch@^5.0.1, minimatch@^5.1.0, minimatch@^5.1.6: + version "5.1.6" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.1.6.tgz#1cfcb8cf5522ea69952cd2af95ae09477f122a96" + integrity sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g== + dependencies: + brace-expansion "^2.0.1" + +minimatch@^9.0.3, minimatch@^9.0.4: + version "9.0.5" + resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-9.0.5.tgz#d74f9dd6b57d83d8e98cfb82133b03978bc929e5" + integrity sha512-G6T0ZX48xgozx7587koeX9Ys2NYy6Gmv//P89sEte9V9whIapMNF4idKxnW2QtCcLiTWlb/wfCabAtAFWhhBow== + dependencies: + brace-expansion "^2.0.1" + +minimist@^1.2.0, minimist@^1.2.3, minimist@^1.2.5, minimist@^1.2.6, minimist@^1.2.8: + version "1.2.8" + resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.8.tgz#c1a464e7693302e082a075cee0c057741ac4772c" + integrity sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA== + +minipass-collect@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/minipass-collect/-/minipass-collect-1.0.2.tgz#22b813bf745dc6edba2576b940022ad6edc8c617" + integrity sha512-6T6lH0H8OG9kITm/Jm6tdooIbogG9e0tLgpY6mphXSm/A9u8Nq1ryBG+Qspiub9LjWlBPsPS3tWQ/Botq4FdxA== + dependencies: + minipass "^3.0.0" + +minipass-fetch@^2.0.3: + version "2.1.2" + resolved "https://registry.yarnpkg.com/minipass-fetch/-/minipass-fetch-2.1.2.tgz#95560b50c472d81a3bc76f20ede80eaed76d8add" + integrity sha512-LT49Zi2/WMROHYoqGgdlQIZh8mLPZmOrN2NdJjMXxYe4nkN6FUyuPuOAOedNJDrx0IRGg9+4guZewtp8hE6TxA== + dependencies: + minipass "^3.1.6" + minipass-sized "^1.0.3" + minizlib "^2.1.2" + optionalDependencies: + encoding "^0.1.13" + +minipass-flush@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/minipass-flush/-/minipass-flush-1.0.5.tgz#82e7135d7e89a50ffe64610a787953c4c4cbb373" + integrity sha512-JmQSYYpPUqX5Jyn1mXaRwOda1uQ8HP5KAT/oDSLCzt1BYRhQU0/hDtsB1ufZfEEzMZ9aAVmsBw8+FWsIXlClWw== + dependencies: + minipass "^3.0.0" + +minipass-pipeline@^1.2.4: + version "1.2.4" + resolved "https://registry.yarnpkg.com/minipass-pipeline/-/minipass-pipeline-1.2.4.tgz#68472f79711c084657c067c5c6ad93cddea8214c" + integrity sha512-xuIq7cIOt09RPRJ19gdi4b+RiNvDFYe5JH+ggNvBqGqpQXcru3PcRmOZuHBKWK1Txf9+cQ+HMVN4d6z46LZP7A== + dependencies: + minipass "^3.0.0" + +minipass-sized@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/minipass-sized/-/minipass-sized-1.0.3.tgz#70ee5a7c5052070afacfbc22977ea79def353b70" + integrity sha512-MbkQQ2CTiBMlA2Dm/5cY+9SWFEN8pzzOXi6rlM5Xxq0Yqbda5ZQy9sU75a673FE9ZK0Zsbr6Y5iP6u9nktfg2g== + dependencies: + minipass "^3.0.0" + +minipass@^3.0.0, minipass@^3.1.1, minipass@^3.1.6: + version "3.3.6" + resolved "https://registry.yarnpkg.com/minipass/-/minipass-3.3.6.tgz#7bba384db3a1520d18c9c0e5251c3444e95dd94a" + integrity sha512-DxiNidxSEK+tHG6zOIklvNOwm3hvCrbUrdtzY74U6HKTJxvIDfOUL5W5P2Ghd3DTkhhKPYGqeNUIh5qcM4YBfw== + dependencies: + yallist "^4.0.0" + +minipass@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/minipass/-/minipass-5.0.0.tgz#3e9788ffb90b694a5d0ec94479a45b5d8738133d" + integrity sha512-3FnjYuehv9k6ovOEbyOswadCDPX1piCfhV8ncmYtHOjuPwylVWsghTLo7rabjC3Rx5xD4HDx8Wm1xnMF7S5qFQ== + +"minipass@^5.0.0 || ^6.0.2 || ^7.0.0", minipass@^7.1.2: + version "7.1.2" + resolved "https://registry.yarnpkg.com/minipass/-/minipass-7.1.2.tgz#93a9626ce5e5e66bd4db86849e7515e92340a707" + integrity sha512-qOOzS1cBTWYF4BH8fVePDBOO9iptMnGUEZwNc/cMWnTV2nVLZ7VoNWEPHkYczZA0pdoA7dl6e7FL659nX9S2aw== + +minizlib@^2.1.1, minizlib@^2.1.2: + version "2.1.2" + resolved "https://registry.yarnpkg.com/minizlib/-/minizlib-2.1.2.tgz#e90d3466ba209b932451508a11ce3d3632145931" + integrity sha512-bAxsR8BVfj60DWXHE3u30oHzfl4G7khkSuPW+qvpd7jFRHm7dLxOjUk1EHACJ/hxLY8phGJ0YhYHZo7jil7Qdg== + dependencies: + minipass "^3.0.0" + yallist "^4.0.0" + +mkdirp-classic@^0.5.2, mkdirp-classic@^0.5.3: + version "0.5.3" + resolved "https://registry.yarnpkg.com/mkdirp-classic/-/mkdirp-classic-0.5.3.tgz#fa10c9115cc6d8865be221ba47ee9bed78601113" + integrity sha512-gKLcREMhtuZRwRAfqP3RFW+TK4JqApVBtOIftVgjuABpAtpxhPGaDcfvbhNvD0B8iD1oUr/txX35NjcaY6Ns/A== + +mkdirp@^1.0.3, mkdirp@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/mkdirp/-/mkdirp-1.0.4.tgz#3eb5ed62622756d79a5f0e2a221dfebad75c2f7e" + integrity sha512-vVqVZQyf3WLx2Shd0qJ9xuvqgAyKPLAiqITEtqW0oIUjzo3PePDd6fW9iFz30ef7Ysp/oiWqbhszeGWW2T6Gzw== + +mlly@^1.7.3, mlly@^1.7.4: + version "1.7.4" + resolved "https://registry.yarnpkg.com/mlly/-/mlly-1.7.4.tgz#3d7295ea2358ec7a271eaa5d000a0f84febe100f" + integrity sha512-qmdSIPC4bDJXgZTCR7XosJiNKySV7O215tsPtDN9iEO/7q/76b/ijtgRu/+epFXSJhijtTCCGp3DWS549P3xKw== + dependencies: + acorn "^8.14.0" + pathe "^2.0.1" + pkg-types "^1.3.0" + ufo "^1.5.4" + +mocha@^11.1.0: + version "11.1.0" + resolved "https://registry.yarnpkg.com/mocha/-/mocha-11.1.0.tgz#20d7c6ac4d6d6bcb60a8aa47971fca74c65c3c66" + integrity sha512-8uJR5RTC2NgpY3GrYcgpZrsEd9zKbPDpob1RezyR2upGHRQtHWofmzTMzTMSV6dru3tj5Ukt0+Vnq1qhFEEwAg== + dependencies: + ansi-colors "^4.1.3" + browser-stdout "^1.3.1" + chokidar "^3.5.3" + debug "^4.3.5" + diff "^5.2.0" + escape-string-regexp "^4.0.0" + find-up "^5.0.0" + glob "^10.4.5" + he "^1.2.0" + js-yaml "^4.1.0" + log-symbols "^4.1.0" + minimatch "^5.1.6" + ms "^2.1.3" + serialize-javascript "^6.0.2" + strip-json-comments "^3.1.1" + supports-color "^8.1.1" + workerpool "^6.5.1" + yargs "^17.7.2" + yargs-parser "^21.1.1" + yargs-unparser "^2.0.0" + +ms@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.0.0.tgz#5608aeadfc00be6c2901df5f9861788de0d597c8" + integrity sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A== + +ms@2.1.3, ms@^2.0.0, ms@^2.1.1, ms@^2.1.3: + version "2.1.3" + resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" + integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== + +multicast-dns@^7.2.5: + version "7.2.5" + resolved "https://registry.yarnpkg.com/multicast-dns/-/multicast-dns-7.2.5.tgz#77eb46057f4d7adbd16d9290fa7299f6fa64cced" + integrity sha512-2eznPJP8z2BFLX50tf0LuODrpINqP1RVIm/CObbTcBRITQgmC/TjcREF1NeTBzIcR5XO/ukWo+YHOjBbFwIupg== + dependencies: + dns-packet "^5.2.2" + thunky "^1.0.2" + +mute-stream@~0.0.4: + version "0.0.8" + resolved "https://registry.yarnpkg.com/mute-stream/-/mute-stream-0.0.8.tgz#1630c42b2251ff81e2a283de96a5497ea92e5e0d" + integrity sha512-nnbWWOkoWyUsTjKrhgD0dcz22mdkSnpYqbEjIm2nhwhuxlSkpywJmBo8h0ZqJdkp73mb90SssHkN4rsRaBAfAA== + +mz@^2.7.0: + version "2.7.0" + resolved "https://registry.yarnpkg.com/mz/-/mz-2.7.0.tgz#95008057a56cafadc2bc63dde7f9ff6955948e32" + integrity sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q== + dependencies: + any-promise "^1.0.0" + object-assign "^4.0.1" + thenify-all "^1.0.0" + +nanoid@^3.3.8: + version "3.3.11" + resolved "https://registry.yarnpkg.com/nanoid/-/nanoid-3.3.11.tgz#4f4f112cefbe303202f2199838128936266d185b" + integrity sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w== + +napi-build-utils@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/napi-build-utils/-/napi-build-utils-2.0.0.tgz#13c22c0187fcfccce1461844136372a47ddc027e" + integrity sha512-GEbrYkbfF7MoNaoh2iGG84Mnf/WZfB0GdGEsM8wz7Expx/LlWf5U8t9nvJKXSp3qr5IsEbK04cBGhol/KwOsWA== + +natural-compare@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" + integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== + +negotiator@0.6.3: + version "0.6.3" + resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.3.tgz#58e323a72fedc0d6f9cd4d31fe49f51479590ccd" + integrity sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg== + +negotiator@^0.6.3: + version "0.6.4" + resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.4.tgz#777948e2452651c570b712dd01c23e262713fff7" + integrity sha512-myRT3DiWPHqho5PrJaIRyaMv2kgYf0mUVgBNOYMuCH5Ki1yEiQaf/ZJuQ62nvpc44wL5WDbTX7yGJi1Neevw8w== + +nice-try@^1.0.4: + version "1.0.5" + resolved "https://registry.yarnpkg.com/nice-try/-/nice-try-1.0.5.tgz#a3378a7696ce7d223e88fc9b764bd7ef1089e366" + integrity sha512-1nh45deeb5olNY7eX82BkPO7SSxR5SSYJiPTrTdFUVYwAl8CKMA5N9PjTYkHiRjisVcxcQ1HXdLhx2qxxJzLNQ== + +node-abi@^3.3.0, node-abi@^3.45.0: + version "3.74.0" + resolved "https://registry.yarnpkg.com/node-abi/-/node-abi-3.74.0.tgz#5bfb4424264eaeb91432d2adb9da23c63a301ed0" + integrity sha512-c5XK0MjkGBrQPGYG24GBADZud0NCbznxNx0ZkS+ebUTrmV1qTDxPxSL8zEAPURXSbLRWVexxmP4986BziahL5w== + dependencies: + semver "^7.3.5" + +node-addon-api@^1.6.3: + version "1.7.2" + resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-1.7.2.tgz#3df30b95720b53c24e59948b49532b662444f54d" + integrity sha512-ibPK3iA+vaY1eEjESkQkM0BbCqFOaZMiXRTtdB0u7b4djtY6JnsjvPdUHVMg6xQt3B8fpTTWHI9A+ADjM9frzg== + +node-addon-api@^4.3.0: + version "4.3.0" + resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-4.3.0.tgz#52a1a0b475193e0928e98e0426a0d1254782b77f" + integrity sha512-73sE9+3UaLYYFmDsFZnqCInzPyh3MqIwZO9cw58yIqAZhONrrabrYyYe3TuIqtIiOuTXVhsGau8hcrhhwSsDIQ== + +node-api-version@^0.2.0: + version "0.2.1" + resolved "https://registry.yarnpkg.com/node-api-version/-/node-api-version-0.2.1.tgz#19bad54f6d65628cbee4e607a325e4488ace2de9" + integrity sha512-2xP/IGGMmmSQpI1+O/k72jF/ykvZ89JeuKX3TLJAYPDVLUalrshrLHkeVcCCZqG/eEa635cr8IBYzgnDvM2O8Q== + dependencies: + semver "^7.3.5" + +node-domexception@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/node-domexception/-/node-domexception-1.0.0.tgz#6888db46a1f71c0b76b3f7555016b63fe64766e5" + integrity sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ== + +node-fetch@^2.6.1, node-fetch@^2.6.7, node-fetch@^2.7.0: + version "2.7.0" + resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-2.7.0.tgz#d0f0fa6e3e2dc1d27efcd8ad99d550bda94d187d" + integrity sha512-c4FRfUm/dbcWZ7U+1Wq0AwCyFL+3nt2bEw05wfxSz+DWpWsitgmSgYmy2dQdWyKC1694ELPqMs/YzUSNozLt8A== + dependencies: + whatwg-url "^5.0.0" + +node-fetch@^3.3.2: + version "3.3.2" + resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-3.3.2.tgz#d1e889bacdf733b4ff3b2b243eb7a12866a0b78b" + integrity sha512-dRB78srN/l6gqWulah9SrxeYnxeddIG30+GOqK/9OlLVyLg3HPnr6SqOWTWOXKRwC2eGYCkZ59NNuSgvSrpgOA== + dependencies: + data-uri-to-buffer "^4.0.0" + fetch-blob "^3.1.4" + formdata-polyfill "^4.0.10" + +node-forge@^1.2.1: + version "1.3.1" + resolved "https://registry.yarnpkg.com/node-forge/-/node-forge-1.3.1.tgz#be8da2af243b2417d5f646a770663a92b7e9ded3" + integrity sha512-dPEtOeMvF9VMcYV/1Wb8CPoVAXtp6MKMlcbAt4ddqmGqUJ6fQZFXkNZNkNlfevtNkGtaSoXf/vNNNSvgrdXwtA== + +node-releases@^2.0.19: + version "2.0.19" + resolved "https://registry.yarnpkg.com/node-releases/-/node-releases-2.0.19.tgz#9e445a52950951ec4d177d843af370b411caf314" + integrity sha512-xxOWJsBKtzAq7DY0J+DTzuz58K8e7sJbdgwkbMWQe8UYB6ekmsQ45q0M/tJDsGaZmbC+l7n57UV8Hl5tHxO9uw== + +nopt@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/nopt/-/nopt-6.0.0.tgz#245801d8ebf409c6df22ab9d95b65e1309cdb16d" + integrity sha512-ZwLpbTgdhuZUnZzjd7nb1ZV+4DoiC6/sfiVKok72ym/4Tlf+DFdlHYmT2JPmcNNWV6Pi3SDf1kT+A4r9RTuT9g== + dependencies: + abbrev "^1.0.0" + +normalize-package-data@^2.3.2: + version "2.5.0" + resolved "https://registry.yarnpkg.com/normalize-package-data/-/normalize-package-data-2.5.0.tgz#e66db1838b200c1dfc233225d12cb36520e234a8" + integrity sha512-/5CMN3T0R4XTj4DcGaexo+roZSdSFW/0AOOTROrjxzCG1wrWXEsGbRKevjlIL+ZDE4sZlJr5ED4YW0yqmkK+eA== + dependencies: + hosted-git-info "^2.1.4" + resolve "^1.10.0" + semver "2 || 3 || 4 || 5" + validate-npm-package-license "^3.0.1" + +normalize-path@^3.0.0, normalize-path@~3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65" + integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== + +normalize-range@^0.1.2: + version "0.1.2" + resolved "https://registry.yarnpkg.com/normalize-range/-/normalize-range-0.1.2.tgz#2d10c06bdfd312ea9777695a4d28439456b75942" + integrity sha512-bdok/XvKII3nUpklnV6P2hxtMNrCboOjAcyBuQnWEhO665FwrSNRxU+AqpsyvO6LgGYPspN+lu5CLtw4jPRKNA== + +normalize-url@^6.0.1: + version "6.1.0" + resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-6.1.0.tgz#40d0885b535deffe3f3147bec877d05fe4c5668a" + integrity sha512-DlL+XwOy3NxAQ8xuC0okPgK46iuVNAK01YN7RueYBqqFeGsBjV9XmCAzAdgt+667bCl5kPh9EqKKDwnaPG1I7A== + +npm-run-path@^2.0.0: + version "2.0.2" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-2.0.2.tgz#35a9232dfa35d7067b4cb2ddf2357b1871536c5f" + integrity sha512-lJxZYlT4DW/bRUtFh1MQIWqmLwQfAxnqWG4HhEdjMlkrJYnJn0Jrr2u3mgxqaWsdiBc76TYkTG/mhrnYTuzfHw== + dependencies: + path-key "^2.0.0" + +npm-run-path@^5.1.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/npm-run-path/-/npm-run-path-5.3.0.tgz#e23353d0ebb9317f174e93417e4a4d82d0249e9f" + integrity sha512-ppwTtiJZq0O/ai0z7yfudtBpWIoxM8yE6nHi1X47eFR2EWORqfbu6CnPlNsjeN683eT0qG6H/Pyf9fCcvjnnnQ== + dependencies: + path-key "^4.0.0" + +nth-check@^2.0.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/nth-check/-/nth-check-2.1.1.tgz#c9eab428effce36cd6b92c924bdb000ef1f1ed1d" + integrity sha512-lqjrjmaOoAnWfMmBPL+XNnynZh2+swxiX3WUE0s4yEHI6m+AwrK2UZOimIRl3X/4QctVqS8AiZjFqyOGrMXb/w== + dependencies: + boolbase "^1.0.0" + +object-assign@^4.0.1, object-assign@^4.1.0, object-assign@^4.1.1: + version "4.1.1" + resolved "https://registry.yarnpkg.com/object-assign/-/object-assign-4.1.1.tgz#2109adc7965887cfc05cbbd442cac8bfbb360863" + integrity sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg== + +object-hash@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/object-hash/-/object-hash-2.2.0.tgz#5ad518581eefc443bd763472b8ff2e9c2c0d54a5" + integrity sha512-gScRMn0bS5fH+IuwyIFgnh9zBdo4DV+6GhygmWM9HyNJSgS0hScp1f5vjtm7oIIOiT9trXrShAkLFSc2IqKNgw== + +object-hash@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/object-hash/-/object-hash-3.0.0.tgz#73f97f753e7baffc0e2cc9d6e079079744ac82e9" + integrity sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw== + +object-inspect@^1.13.3: + version "1.13.4" + resolved "https://registry.yarnpkg.com/object-inspect/-/object-inspect-1.13.4.tgz#8375265e21bc20d0fa582c22e1b13485d6e00213" + integrity sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew== + +object-keys@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e" + integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA== + +object.assign@^4.1.4, object.assign@^4.1.7: + version "4.1.7" + resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.7.tgz#8c14ca1a424c6a561b0bb2a22f66f5049a945d3d" + integrity sha512-nK28WOo+QIjBkDduTINE4JkF/UJJKyf2EJxvJKfblDpyg0Q+pkOHNTL0Qwy6NP6FhE/EnzV73BxxqcJaXY9anw== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.3" + define-properties "^1.2.1" + es-object-atoms "^1.0.0" + has-symbols "^1.1.0" + object-keys "^1.1.1" + +object.entries@^1.1.9: + version "1.1.9" + resolved "https://registry.yarnpkg.com/object.entries/-/object.entries-1.1.9.tgz#e4770a6a1444afb61bd39f984018b5bede25f8b3" + integrity sha512-8u/hfXFRBD1O0hPUjioLhoWFHRmt6tKA4/vZPyckBr18l1KE9uHrFaFaUi8MDRTpi4uak2goyPTSNJLXX2k2Hw== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.4" + define-properties "^1.2.1" + es-object-atoms "^1.1.1" + +object.fromentries@^2.0.8: + version "2.0.8" + resolved "https://registry.yarnpkg.com/object.fromentries/-/object.fromentries-2.0.8.tgz#f7195d8a9b97bd95cbc1999ea939ecd1a2b00c65" + integrity sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ== + dependencies: + call-bind "^1.0.7" + define-properties "^1.2.1" + es-abstract "^1.23.2" + es-object-atoms "^1.0.0" + +object.groupby@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/object.groupby/-/object.groupby-1.0.3.tgz#9b125c36238129f6f7b61954a1e7176148d5002e" + integrity sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ== + dependencies: + call-bind "^1.0.7" + define-properties "^1.2.1" + es-abstract "^1.23.2" + +object.values@^1.1.6, object.values@^1.2.0, object.values@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.2.1.tgz#deed520a50809ff7f75a7cfd4bc64c7a038c6216" + integrity sha512-gXah6aZrcUxjWg2zR2MwouP2eHlCBzdV4pygudehaKXSGW4v2AsRQUK+lwwXhii6KFZcunEnmSUoYp5CXibxtA== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.3" + define-properties "^1.2.1" + es-object-atoms "^1.0.0" + +oidc-token-hash@^5.0.3: + version "5.1.0" + resolved "https://registry.yarnpkg.com/oidc-token-hash/-/oidc-token-hash-5.1.0.tgz#74bda0c35dd9f71ea9ce0db72ce8dabf5f90ef79" + integrity sha512-y0W+X7Ppo7oZX6eovsRkuzcSM40Bicg2JEJkDJ4irIt1wsYAP5MLSNv+QAogO8xivMffw/9OvV3um1pxXgt1uA== + +on-finished@2.4.1: + version "2.4.1" + resolved "https://registry.yarnpkg.com/on-finished/-/on-finished-2.4.1.tgz#58c8c44116e54845ad57f14ab10b03533184ac3f" + integrity sha512-oVlzkg3ENAhCk2zdv7IJwd/QUD4z2RxRwpkcGY8psCVcCYZNq4wYnVWALHM+brtuJjePWiYF/ClmuDr8Ch5+kg== + dependencies: + ee-first "1.1.1" + +once@^1.3.0, once@^1.3.1, once@^1.4.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" + integrity sha512-lNaJgI+2Q5URQBkccEKHTQOPaXdUxnZZElQTZY0MFUAuaEqe1E+Nyvgdz/aIyNi6Z9MzO5dv1H8n58/GELp3+w== + dependencies: + wrappy "1" + +onetime@^5.1.0: + version "5.1.2" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-5.1.2.tgz#d0e96ebb56b07476df1dd9c4806e5237985ca45e" + integrity sha512-kbpaSSGJTWdAY5KPVeMOKXSrPtr8C8C7wodJbcsd51jRnmD+GZu8Y0VoU6Dm5Z4vWr0Ig/1NKuWRKf7j5aaYSg== + dependencies: + mimic-fn "^2.1.0" + +onetime@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-6.0.0.tgz#7c24c18ed1fd2e9bca4bd26806a33613c77d34b4" + integrity sha512-1FlR+gjXK7X+AsAHso35MnyN5KqGwJRi/31ft6x0M194ht7S+rWAvd7PHss9xSKMzE0asv1pyIHaJYq+BbacAQ== + dependencies: + mimic-fn "^4.0.0" + +onetime@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/onetime/-/onetime-7.0.0.tgz#9f16c92d8c9ef5120e3acd9dd9957cceecc1ab60" + integrity sha512-VXJjc87FScF88uafS3JllDgvAm+c/Slfz06lorj2uAY34rlUu0Nt+v8wreiImcrgAjjIHp1rXpTDlLOGw29WwQ== + dependencies: + mimic-function "^5.0.0" + +open@^10.1.0: + version "10.1.1" + resolved "https://registry.yarnpkg.com/open/-/open-10.1.1.tgz#5fd814699e47ae3e1a09962d39f4f4441cae6c22" + integrity sha512-zy1wx4+P3PfhXSEPJNtZmJXfhkkIaxU1VauWIrDZw1O7uJRDRJtKr9n3Ic4NgbA16KyOxOXO2ng9gYwCdXuSXA== + dependencies: + default-browser "^5.2.1" + define-lazy-prop "^3.0.0" + is-inside-container "^1.0.0" + is-wsl "^3.1.0" + +openapi-types@^12.0.0: + version "12.1.3" + resolved "https://registry.yarnpkg.com/openapi-types/-/openapi-types-12.1.3.tgz#471995eb26c4b97b7bd356aacf7b91b73e777dd3" + integrity sha512-N4YtSYJqghVu4iek2ZUvcN/0aqH1kRDuNqzcycDxhOUpg7GdvLa2F3DgS6yBNhInhv2r/6I0Flkn7CqL8+nIcw== + +openapi-typescript@^7.6.1: + version "7.6.1" + resolved "https://registry.yarnpkg.com/openapi-typescript/-/openapi-typescript-7.6.1.tgz#e39d1e21ebf43f91712703f7063118246d099d19" + integrity sha512-F7RXEeo/heF3O9lOXo2bNjCOtfp7u+D6W3a3VNEH2xE6v+fxLtn5nq0uvUcA1F5aT+CMhNeC5Uqtg5tlXFX/ag== + dependencies: + "@redocly/openapi-core" "^1.28.0" + ansi-colors "^4.1.3" + change-case "^5.4.4" + parse-json "^8.1.0" + supports-color "^9.4.0" + yargs-parser "^21.1.1" + +opener@^1.5.1: + version "1.5.2" + resolved "https://registry.yarnpkg.com/opener/-/opener-1.5.2.tgz#5d37e1f35077b9dcac4301372271afdeb2a13598" + integrity sha512-ur5UIdyw5Y7yEj9wLzhqXiy6GZ3Mwx0yGI+5sMn2r0N0v3cKJvUmFH5yPP+WXh9e0xfyzyJX95D8l088DNFj7A== + +openid-client@^5.6.5: + version "5.7.1" + resolved "https://registry.yarnpkg.com/openid-client/-/openid-client-5.7.1.tgz#34cace862a3e6472ed7d0a8616ef73b7fb85a9c3" + integrity sha512-jDBPgSVfTnkIh71Hg9pRvtJc6wTwqjRkN88+gCFtYWrlP4Yx2Dsrow8uPi3qLr/aeymPF3o2+dS+wOpglK04ew== + dependencies: + jose "^4.15.9" + lru-cache "^6.0.0" + object-hash "^2.2.0" + oidc-token-hash "^5.0.3" + +optionator@^0.9.3: + version "0.9.4" + resolved "https://registry.yarnpkg.com/optionator/-/optionator-0.9.4.tgz#7ea1c1a5d91d764fb282139c88fe11e182a3a734" + integrity sha512-6IpQ7mKUxRcZNLIObR0hz7lxsapSSIYNZJwXPGeF0mTVqGKFIXj1DQcMoT22S3ROcLyY/rz0PWaWZ9ayWmad9g== + dependencies: + deep-is "^0.1.3" + fast-levenshtein "^2.0.6" + levn "^0.4.1" + prelude-ls "^1.2.1" + type-check "^0.4.0" + word-wrap "^1.2.5" + +ora@^5.1.0, ora@^5.4.1: + version "5.4.1" + resolved "https://registry.yarnpkg.com/ora/-/ora-5.4.1.tgz#1b2678426af4ac4a509008e5e4ac9e9959db9e18" + integrity sha512-5b6Y85tPxZZ7QytO+BQzysW31HJku27cRIlkbAXaNx+BdcVi+LlRFmVXzeF6a7JCwJpyw5c4b+YSVImQIrBpuQ== + dependencies: + bl "^4.1.0" + chalk "^4.1.0" + cli-cursor "^3.1.0" + cli-spinners "^2.5.0" + is-interactive "^1.0.0" + is-unicode-supported "^0.1.0" + log-symbols "^4.1.0" + strip-ansi "^6.0.0" + wcwidth "^1.0.1" + +ora@^8.1.0: + version "8.2.0" + resolved "https://registry.yarnpkg.com/ora/-/ora-8.2.0.tgz#8fbbb7151afe33b540dd153f171ffa8bd38e9861" + integrity sha512-weP+BZ8MVNnlCm8c0Qdc1WSWq4Qn7I+9CJGm7Qali6g44e/PUzbjNqJX5NJ9ljlNMosfJvg1fKEGILklK9cwnw== + dependencies: + chalk "^5.3.0" + cli-cursor "^5.0.0" + cli-spinners "^2.9.2" + is-interactive "^2.0.0" + is-unicode-supported "^2.0.0" + log-symbols "^6.0.0" + stdin-discarder "^0.2.2" + string-width "^7.2.0" + strip-ansi "^7.1.0" + +own-keys@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/own-keys/-/own-keys-1.0.1.tgz#e4006910a2bf913585289676eebd6f390cf51358" + integrity sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg== + dependencies: + get-intrinsic "^1.2.6" + object-keys "^1.1.1" + safe-push-apply "^1.0.0" + +p-cancelable@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-2.1.1.tgz#aab7fbd416582fa32a3db49859c122487c5ed2cf" + integrity sha512-BZOr3nRQHOntUjTrH8+Lh54smKHoHyur8We1V8DSMVrl5A2malOOwuJRnKRDjSnkoeBh4at6BwEnb5I7Jl31wg== + +p-defer@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/p-defer/-/p-defer-1.0.0.tgz#9f6eb182f6c9aa8cd743004a7d4f96b196b0fb0c" + integrity sha512-wB3wfAxZpk2AzOfUMJNL+d36xothRSyj8EXOa4f6GMqYDN9BJaaSISbsk+wS9abmnebVw95C2Kb5t85UmpCxuw== + +p-finally@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/p-finally/-/p-finally-1.0.0.tgz#3fbcfb15b899a44123b34b6dcc18b724336a2cae" + integrity sha512-LICb2p9CB7FS+0eR1oqWnHhp0FljGLZCWBE9aix0Uye9W8LTQPwMTYVGWQWIw9RdQiDg4+epXQODwIYJtSJaow== + +p-is-promise@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/p-is-promise/-/p-is-promise-2.1.0.tgz#918cebaea248a62cf7ffab8e3bca8c5f882fc42e" + integrity sha512-Y3W0wlRPK8ZMRbNq97l4M5otioeA5lm1z7bkNkxCka8HSPjR0xRWmpCmc9utiaLP9Jb1eD8BgeIxTW4AIF45Pg== + +p-limit@^1.1.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-1.3.0.tgz#b86bd5f0c25690911c7590fcbfc2010d54b3ccb8" + integrity sha512-vvcXsLAJ9Dr5rQOPk7toZQZJApBl2K4J6dANSsEuh6QI41JYcsS/qhTGa9ErIUUgK3WNQoJYvylxvjqmiqEA9Q== + dependencies: + p-try "^1.0.0" + +p-limit@^3.0.2, "p-limit@^3.1.0 ": + version "3.1.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-3.1.0.tgz#e1daccbe78d0d1388ca18c64fea38e3e57e3706b" + integrity sha512-TYOanM3wGwNGsZN2cVTYPArw454xnXj5qmWF1bEoAc4+cU/ol7GVh7odevjp1FNHduHc3KZMcFduxU5Xc6uJRQ== + dependencies: + yocto-queue "^0.1.0" + +p-limit@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/p-limit/-/p-limit-5.0.0.tgz#6946d5b7140b649b7a33a027d89b4c625b3a5985" + integrity sha512-/Eaoq+QyLSiXQ4lyYV23f14mZRQcXnxfHrN0vCai+ak9G0pp9iEQukIIZq5NccEvwRB8PUnZT0KsOoDCINS1qQ== + dependencies: + yocto-queue "^1.0.0" + +p-locate@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-2.0.0.tgz#20a0103b222a70c8fd39cc2e580680f3dde5ec43" + integrity sha512-nQja7m7gSKuewoVRen45CtVfODR3crN3goVQ0DDZ9N3yHxgpkuBhZqsaiotSQRrADUrne346peY7kT3TSACykg== + dependencies: + p-limit "^1.1.0" + +p-locate@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/p-locate/-/p-locate-5.0.0.tgz#83c8315c6785005e3bd021839411c9e110e6d834" + integrity sha512-LaNjtRWUBY++zB5nE/NwcaoMylSPk+S+ZHNB1TzdbMJMny6dynpAGt7X/tl/QYq3TIeE6nxHppbo2LGymrG5Pw== + dependencies: + p-limit "^3.0.2" + +p-map@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/p-map/-/p-map-4.0.0.tgz#bb2f95a5eda2ec168ec9274e06a747c3e2904d2b" + integrity sha512-/bjOqmgETBYB5BoEeGVea8dmvHb2m9GLy1E9W43yeyfP6QQCZGFNa+XRceJEuDB6zqr+gKpIAmlLebMpykw/MQ== + dependencies: + aggregate-error "^3.0.0" + +p-try@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/p-try/-/p-try-1.0.0.tgz#cbc79cdbaf8fd4228e13f621f2b1a237c1b207b3" + integrity sha512-U1etNYuMJoIz3ZXSrrySFjsXQTWOx2/jdi86L+2pRvph/qMKL6sbcCYdH23fqsbm8TH2Gn0OybpT4eSFlCVHww== + +package-json-from-dist@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/package-json-from-dist/-/package-json-from-dist-1.0.1.tgz#4f1471a010827a86f94cfd9b0727e36d267de505" + integrity sha512-UEZIS3/by4OC8vL3P2dTXRETpebLI2NiI5vIrjaD/5UtrkFX/tNbwjTSRAGC/+7CAo2pIcBaRgWmcBBHcsaCIw== + +pako@~1.0.2: + version "1.0.11" + resolved "https://registry.yarnpkg.com/pako/-/pako-1.0.11.tgz#6c9599d340d54dfd3946380252a35705a6b992bf" + integrity sha512-4hLB8Py4zZce5s4yd9XzopqwVv/yGNhV1Bl8NTmCq1763HeK2+EwVTv+leGeL13Dnh2wfbqowVPXCIO0z4taYw== + +parent-module@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/parent-module/-/parent-module-1.0.1.tgz#691d2709e78c79fae3a156622452d00762caaaa2" + integrity sha512-GQ2EWRpQV8/o+Aw8YqtfZZPfNRWZYkbidE9k5rpl/hC3vtHHBfGm2Ifi6qWV+coDGkrUKZAxE3Lot5kcsRlh+g== + dependencies: + callsites "^3.0.0" + +parse-author@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/parse-author/-/parse-author-2.0.0.tgz#d3460bf1ddd0dfaeed42da754242e65fb684a81f" + integrity sha512-yx5DfvkN8JsHL2xk2Os9oTia467qnvRgey4ahSm2X8epehBLx/gWLcy5KI+Y36ful5DzGbCS6RazqZGgy1gHNw== + dependencies: + author-regex "^1.0.0" + +parse-color@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/parse-color/-/parse-color-1.0.0.tgz#7b748b95a83f03f16a94f535e52d7f3d94658619" + integrity sha512-fuDHYgFHJGbpGMgw9skY/bj3HL/Jrn4l/5rSspy00DoT4RyLnDcRvPxdZ+r6OFwIsgAuhDh4I09tAId4mI12bw== + dependencies: + color-convert "~0.5.0" + +parse-json@^2.2.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-2.2.0.tgz#f480f40434ef80741f8469099f8dea18f55a4dc9" + integrity sha512-QR/GGaKCkhwk1ePQNYDRKYZ3mwU9ypsKhB0XyFnLQdomyEqk3e8wpW3V5Jp88zbxK4n5ST1nqo+g9juTpownhQ== + dependencies: + error-ex "^1.2.0" + +parse-json@^8.1.0: + version "8.3.0" + resolved "https://registry.yarnpkg.com/parse-json/-/parse-json-8.3.0.tgz#88a195a2157025139a2317a4f2f9252b61304ed5" + integrity sha512-ybiGyvspI+fAoRQbIPRddCcSTV9/LsJbf0e/S85VLowVGzRmokfneg2kwVW/KU5rOXrPSbF1qAKPMgNTqqROQQ== + dependencies: + "@babel/code-frame" "^7.26.2" + index-to-position "^1.1.0" + type-fest "^4.39.1" + +parse-semver@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/parse-semver/-/parse-semver-1.1.1.tgz#9a4afd6df063dc4826f93fba4a99cf223f666cb8" + integrity sha512-Eg1OuNntBMH0ojvEKSrvDSnwLmvVuUOSdylH/pSCPNMIspLlweJyIWXCE+k/5hm3cj/EBUYwmWkjhBALNP4LXQ== + dependencies: + semver "^5.1.0" + +parse5-htmlparser2-tree-adapter@^7.0.0: + version "7.1.0" + resolved "https://registry.yarnpkg.com/parse5-htmlparser2-tree-adapter/-/parse5-htmlparser2-tree-adapter-7.1.0.tgz#b5a806548ed893a43e24ccb42fbb78069311e81b" + integrity sha512-ruw5xyKs6lrpo9x9rCZqZZnIUntICjQAd0Wsmp396Ul9lN/h+ifgVV1x1gZHi8euej6wTfpqX8j+BFQxF0NS/g== + dependencies: + domhandler "^5.0.3" + parse5 "^7.0.0" + +parse5-parser-stream@^7.1.2: + version "7.1.2" + resolved "https://registry.yarnpkg.com/parse5-parser-stream/-/parse5-parser-stream-7.1.2.tgz#d7c20eadc37968d272e2c02660fff92dd27e60e1" + integrity sha512-JyeQc9iwFLn5TbvvqACIF/VXG6abODeB3Fwmv/TGdLk2LfbWkaySGY72at4+Ty7EkPZj854u4CrICqNk2qIbow== + dependencies: + parse5 "^7.0.0" + +parse5@^7.0.0, parse5@^7.1.2: + version "7.3.0" + resolved "https://registry.yarnpkg.com/parse5/-/parse5-7.3.0.tgz#d7e224fa72399c7a175099f45fc2ad024b05ec05" + integrity sha512-IInvU7fabl34qmi9gY8XOVxhYyMyuH2xUNpb2q8/Y+7552KlejkRvqvD19nMoUW/uQGGbqNpA6Tufu5FL5BZgw== + dependencies: + entities "^6.0.0" + +parseurl@~1.3.3: + version "1.3.3" + resolved "https://registry.yarnpkg.com/parseurl/-/parseurl-1.3.3.tgz#9da19e7bee8d12dff0513ed5b76957793bc2e8d4" + integrity sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ== + +path-exists@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-3.0.0.tgz#ce0ebeaa5f78cb18925ea7d810d7b59b010fd515" + integrity sha512-bpC7GYwiDYQ4wYLe+FA8lhRjhQCMcQGuSgGGqDkg/QerRWw9CmGRT0iSOVRSZJ29NMLZgIzqaljJ63oaL4NIJQ== + +path-exists@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-exists/-/path-exists-4.0.0.tgz#513bdbe2d3b95d7762e8c1137efa195c6c61b5b3" + integrity sha512-ak9Qy5Q7jYb2Wwcey5Fpvg2KoAc/ZIhLSLOSBmRmygPsGwkVVt0fZa0qrtMz+m6tJTAHfZQ8FnmB4MG4LWy7/w== + +path-is-absolute@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/path-is-absolute/-/path-is-absolute-1.0.1.tgz#174b9268735534ffbc7ace6bf53a5a9e1b5c5f5f" + integrity sha512-AVbw3UJ2e9bq64vSaS9Am0fje1Pa8pbGqTTsmXfaIiMpnr5DlDhfJOuLj9Sf95ZPVDAUerDfEk88MPmPe7UCQg== + +path-key@^2.0.0, path-key@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-2.0.1.tgz#411cadb574c5a140d3a4b1910d40d80cc9f40b40" + integrity sha512-fEHGKCSmUSDPv4uoj8AlD+joPlq3peND+HRYyxFz4KPw4z926S/b8rIuFs2FYJg3BwsxJf6A9/3eIdLaYC+9Dw== + +path-key@^3.1.0: + version "3.1.1" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-3.1.1.tgz#581f6ade658cbba65a0d3380de7753295054f375" + integrity sha512-ojmeN0qd+y0jszEtoY48r0Peq5dwMEkIlCOu6Q5f41lfkswXuKtYrhgoTpLnyIcHm24Uhqx+5Tqm2InSwLhE6Q== + +path-key@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/path-key/-/path-key-4.0.0.tgz#295588dc3aee64154f877adb9d780b81c554bf18" + integrity sha512-haREypq7xkM7ErfgIyA0z+Bj4AGKlMSdlQE2jvJo6huWD1EdkKYV+G/T4nq0YEF2vgTT8kqMFKo1uHn950r4SQ== + +path-parse@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/path-parse/-/path-parse-1.0.7.tgz#fbc114b60ca42b30d9daf5858e4bd68bbedb6735" + integrity sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw== + +path-scurry@^1.11.1: + version "1.11.1" + resolved "https://registry.yarnpkg.com/path-scurry/-/path-scurry-1.11.1.tgz#7960a668888594a0720b12a911d1a742ab9f11d2" + integrity sha512-Xa4Nw17FS9ApQFJ9umLiJS4orGjm7ZzwUrwamcGQuHSzDyth9boKDaycYdDcZDuqYATXw4HFXgaqWTctW/v1HA== + dependencies: + lru-cache "^10.2.0" + minipass "^5.0.0 || ^6.0.2 || ^7.0.0" + +path-scurry@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/path-scurry/-/path-scurry-2.0.0.tgz#9f052289f23ad8bf9397a2a0425e7b8615c58580" + integrity sha512-ypGJsmGtdXUOeM5u93TyeIEfEhM6s+ljAhrk5vAvSx8uyY/02OvrZnA0YNGUrPXfpJMgI1ODd3nwz8Npx4O4cg== + dependencies: + lru-cache "^11.0.0" + minipass "^7.1.2" + +path-to-regexp@0.1.12: + version "0.1.12" + resolved "https://registry.yarnpkg.com/path-to-regexp/-/path-to-regexp-0.1.12.tgz#d5e1a12e478a976d432ef3c58d534b9923164bb7" + integrity sha512-RA1GjUVMnvYFxuqovrEqZoxxW5NUZqbwKtYz/Tt7nXerk0LbLblQmrsgdeOxV5SFHf0UDggjS/bSeOZwt1pmEQ== + +path-type@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/path-type/-/path-type-2.0.0.tgz#f012ccb8415b7096fc2daa1054c3d72389594c73" + integrity sha512-dUnb5dXUf+kzhC/W/F4e5/SkluXIFf5VUHolW1Eg1irn1hGWjPGdsRcvYJ1nD6lhk8Ir7VM0bHJKsYTx8Jx9OQ== + dependencies: + pify "^2.0.0" + +pathe@^1.1.1: + version "1.1.2" + resolved "https://registry.yarnpkg.com/pathe/-/pathe-1.1.2.tgz#6c4cb47a945692e48a1ddd6e4094d170516437ec" + integrity sha512-whLdWMYL2TwI08hn8/ZqAbrVemu0LNaNNJZX73O6qaIdCTfXutsLhMkjdENX0qhsQ9uIimo4/aQOmXkoon2nDQ== + +pathe@^2.0.1, pathe@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/pathe/-/pathe-2.0.3.tgz#3ecbec55421685b70a9da872b2cff3e1cbed1716" + integrity sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w== + +pathval@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/pathval/-/pathval-1.1.1.tgz#8534e77a77ce7ac5a2512ea21e0fdb8fcf6c3d8d" + integrity sha512-Dp6zGqpTdETdR63lehJYPeIOqpiNBNtc7BpWSLrOje7UaIsE5aY92r/AunQA7rsXvet3lrJ3JnZX29UPTKXyKQ== + +pathval@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/pathval/-/pathval-2.0.0.tgz#7e2550b422601d4f6b8e26f1301bc8f15a741a25" + integrity sha512-vE7JKRyES09KiunauX7nd2Q9/L7lhok4smP9RZTDeD4MVs72Dp2qNFVz39Nz5a0FVEW0BJR6C0DYrq6unoziZA== + +pe-library@^0.4.1: + version "0.4.1" + resolved "https://registry.yarnpkg.com/pe-library/-/pe-library-0.4.1.tgz#e269be0340dcb13aa6949d743da7d658c3e2fbea" + integrity sha512-eRWB5LBz7PpDu4PUlwT0PhnQfTQJlDDdPa35urV4Osrm0t0AqQFGn+UIkU3klZvwJ8KPO3VbBFsXquA6p6kqZw== + +pe-library@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/pe-library/-/pe-library-1.0.1.tgz#02735430885a622576a53cd8827658b7d2fada0e" + integrity sha512-nh39Mo1eGWmZS7y+mK/dQIqg7S1lp38DpRxkyoHf0ZcUs/HDc+yyTjuOtTvSMZHmfSLuSQaX945u05Y2Q6UWZg== + +pend@~1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50" + integrity sha512-F3asv42UuXchdzt+xXqfW1OGlVBe+mxa2mqI0pg5yAHZPvFmY3Y6drSf/GQ1A86WgWEN9Kzh/WrgKa6iGcHXLg== + +picocolors@^1.0.0, picocolors@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/picocolors/-/picocolors-1.1.1.tgz#3d321af3eab939b083c8f929a1d12cda81c26b6b" + integrity sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA== + +picomatch@^2.0.4, picomatch@^2.2.1, picomatch@^2.2.3, picomatch@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" + integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== + +picomatch@^4.0.2: + version "4.0.2" + resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-4.0.2.tgz#77c742931e8f3b8820946c76cd0c1f13730d1dab" + integrity sha512-M7BAV6Rlcy5u+m6oPhAPFgJTzAioX/6B0DxyvDlo9l8+T3nLKbrczg2WLUyzd45L8RqfUMyGPzekbMvX2Ldkwg== + +pify@^2.0.0, pify@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/pify/-/pify-2.3.0.tgz#ed141a6ac043a849ea588498e7dca8b15330e90c" + integrity sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog== + +pify@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/pify/-/pify-3.0.0.tgz#e5a4acd2c101fdf3d9a4d07f0dbc4db49dd28176" + integrity sha512-C3FsVNH1udSEX48gGX1xfvwTWfsYWj5U+8/uK15BGzIGrKoUpghX8hWZwa/OFnakBiiVNmBvemTJR5mcy7iPcg== + +pirates@^4.0.1: + version "4.0.7" + resolved "https://registry.yarnpkg.com/pirates/-/pirates-4.0.7.tgz#643b4a18c4257c8a65104b73f3049ce9a0a15e22" + integrity sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA== + +pixelmatch@^5.3.0: + version "5.3.0" + resolved "https://registry.yarnpkg.com/pixelmatch/-/pixelmatch-5.3.0.tgz#5e5321a7abedfb7962d60dbf345deda87cb9560a" + integrity sha512-o8mkY4E/+LNUf6LzX96ht6k6CEDi65k9G2rjMtBe9Oo+VPKSvl+0GKHuH/AlG+GA5LPG/i5hrekkxUc3s2HU+Q== + dependencies: + pngjs "^6.0.0" + +pkg-types@^1.2.1, pkg-types@^1.3.0: + version "1.3.1" + resolved "https://registry.yarnpkg.com/pkg-types/-/pkg-types-1.3.1.tgz#bd7cc70881192777eef5326c19deb46e890917df" + integrity sha512-/Jm5M4RvtBFVkKWRu2BLUTNP8/M2a+UwuAX+ae4770q1qVGtfjG+WTCupoZixokjmHiry8uI+dlY8KXYV5HVVQ== + dependencies: + confbox "^0.1.8" + mlly "^1.7.4" + pathe "^2.0.1" + +playwright-core@1.52.0: + version "1.52.0" + resolved "https://registry.yarnpkg.com/playwright-core/-/playwright-core-1.52.0.tgz#238f1f0c3edd4ebba0434ce3f4401900319a3dca" + integrity sha512-l2osTgLXSMeuLZOML9qYODUQoPPnUsKsb5/P6LJ2e6uPKXUdPK5WYhN4z03G+YNbWmGDY4YENauNu4ZKczreHg== + +playwright@1.52.0: + version "1.52.0" + resolved "https://registry.yarnpkg.com/playwright/-/playwright-1.52.0.tgz#26cb9a63346651e1c54c8805acfd85683173d4bd" + integrity sha512-JAwMNMBlxJ2oD1kce4KPtMkDeKGHQstdpFPcPH3maElAXon/QZeTvtsfXmTMRyO9TslfoYOXkSsvao2nE1ilTw== + dependencies: + playwright-core "1.52.0" + optionalDependencies: + fsevents "2.3.2" + +plist@3.1.0, plist@^3.0.0, plist@^3.0.4, plist@^3.0.5, plist@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/plist/-/plist-3.1.0.tgz#797a516a93e62f5bde55e0b9cc9c967f860893c9" + integrity sha512-uysumyrvkUX0rX/dEVqt8gC3sTBzd4zoWfLeS29nb53imdaXVvLINYXTI2GNqzaMuvacNx4uJQ8+b3zXR0pkgQ== + dependencies: + "@xmldom/xmldom" "^0.8.8" + base64-js "^1.5.1" + xmlbuilder "^15.1.1" + +pluralize@^8.0.0: + version "8.0.0" + resolved "https://registry.yarnpkg.com/pluralize/-/pluralize-8.0.0.tgz#1a6fa16a38d12a1901e0320fa017051c539ce3b1" + integrity sha512-Nc3IT5yHzflTfbjgqWcCPpo7DaKy4FnpB0l/zCAW0Tc7jxAiuqSxHasntB3D7887LSrA93kDJ9IXovxJYxyLCA== + +pngjs@^6.0.0: + version "6.0.0" + resolved "https://registry.yarnpkg.com/pngjs/-/pngjs-6.0.0.tgz#ca9e5d2aa48db0228a52c419c3308e87720da821" + integrity sha512-TRzzuFRRmEoSW/p1KVAmiOgPco2Irlah+bGFCeNfJXxxYGwSw7YwAOAcd7X28K/m5bjBWKsC29KyoMfHbypayg== + +pngjs@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/pngjs/-/pngjs-7.0.0.tgz#a8b7446020ebbc6ac739db6c5415a65d17090e26" + integrity sha512-LKWqWJRhstyYo9pGvgor/ivk2w94eSjE3RGVuzLGlr3NmD8bf7RcYGze1mNdEHRP6TRP6rMuDHk5t44hnTRyow== + +portfinder@^1.0.28: + version "1.0.36" + resolved "https://registry.yarnpkg.com/portfinder/-/portfinder-1.0.36.tgz#4eef523c15e972417a9ee496c3e9c95b8f649d52" + integrity sha512-gMKUzCoP+feA7t45moaSx7UniU7PgGN3hA8acAB+3Qn7/js0/lJ07fYZlxt9riE9S3myyxDCyAFzSrLlta0c9g== + dependencies: + async "^3.2.6" + debug "^4.3.6" + +possible-typed-array-names@^1.0.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/possible-typed-array-names/-/possible-typed-array-names-1.1.0.tgz#93e3582bc0e5426586d9d07b79ee40fc841de4ae" + integrity sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg== + +postcss-import@^15.1.0: + version "15.1.0" + resolved "https://registry.yarnpkg.com/postcss-import/-/postcss-import-15.1.0.tgz#41c64ed8cc0e23735a9698b3249ffdbf704adc70" + integrity sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew== + dependencies: + postcss-value-parser "^4.0.0" + read-cache "^1.0.0" + resolve "^1.1.7" + +postcss-js@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/postcss-js/-/postcss-js-4.0.1.tgz#61598186f3703bab052f1c4f7d805f3991bee9d2" + integrity sha512-dDLF8pEO191hJMtlHFPRa8xsizHaM82MLfNkUHdUtVEV3tgTp5oj+8qbEqYM57SLfc74KSbw//4SeJma2LRVIw== + dependencies: + camelcase-css "^2.0.1" + +postcss-load-config@^4.0.2: + version "4.0.2" + resolved "https://registry.yarnpkg.com/postcss-load-config/-/postcss-load-config-4.0.2.tgz#7159dcf626118d33e299f485d6afe4aff7c4a3e3" + integrity sha512-bSVhyJGL00wMVoPUzAVAnbEoWyqRxkjv64tUl427SKnPrENtq6hJwUojroMz2VB+Q1edmi4IfrAPpami5VVgMQ== + dependencies: + lilconfig "^3.0.0" + yaml "^2.3.4" + +postcss-nested@^6.2.0: + version "6.2.0" + resolved "https://registry.yarnpkg.com/postcss-nested/-/postcss-nested-6.2.0.tgz#4c2d22ab5f20b9cb61e2c5c5915950784d068131" + integrity sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ== + dependencies: + postcss-selector-parser "^6.1.1" + +postcss-selector-parser@^6.1.1, postcss-selector-parser@^6.1.2: + version "6.1.2" + resolved "https://registry.yarnpkg.com/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz#27ecb41fb0e3b6ba7a1ec84fff347f734c7929de" + integrity sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg== + dependencies: + cssesc "^3.0.0" + util-deprecate "^1.0.2" + +postcss-value-parser@^4.0.0, postcss-value-parser@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz#723c09920836ba6d3e5af019f92bc0971c02e514" + integrity sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ== + +postcss@^8.4.43, postcss@^8.4.47, postcss@^8.5.3: + version "8.5.3" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.5.3.tgz#1463b6f1c7fb16fe258736cba29a2de35237eafb" + integrity sha512-dle9A3yYxlBSrt8Fu+IpjGT8SY8hN0mlaA6GY8t0P5PjIOZemULz/E2Bnm/2dcUOena75OTNkHI76uZBNUUq3A== + dependencies: + nanoid "^3.3.8" + picocolors "^1.1.1" + source-map-js "^1.2.1" + +postinstall-postinstall@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/postinstall-postinstall/-/postinstall-postinstall-2.1.0.tgz#4f7f77441ef539d1512c40bd04c71b06a4704ca3" + integrity sha512-7hQX6ZlZXIoRiWNrbMQaLzUUfH+sSx39u8EJ9HYuDc1kLo9IXKWjM5RSquZN1ad5GnH8CGFM78fsAAQi3OKEEQ== + +postject@^1.0.0-alpha.6: + version "1.0.0-alpha.6" + resolved "https://registry.yarnpkg.com/postject/-/postject-1.0.0-alpha.6.tgz#9d022332272e2cfce8dea4cfce1ee6dd1b2ee135" + integrity sha512-b9Eb8h2eVqNE8edvKdwqkrY6O7kAwmI8kcnBv1NScolYJbo59XUF0noFq+lxbC1yN20bmC0WBEbDC5H/7ASb0A== + dependencies: + commander "^9.4.0" + +prebuild-install@^7.0.1: + version "7.1.3" + resolved "https://registry.yarnpkg.com/prebuild-install/-/prebuild-install-7.1.3.tgz#d630abad2b147443f20a212917beae68b8092eec" + integrity sha512-8Mf2cbV7x1cXPUILADGI3wuhfqWvtiLA1iclTDbFRZkgRQS0NqsPZphna9V+HyTEadheuPmjaJMsbzKQFOzLug== + dependencies: + detect-libc "^2.0.0" + expand-template "^2.0.3" + github-from-package "0.0.0" + minimist "^1.2.3" + mkdirp-classic "^0.5.3" + napi-build-utils "^2.0.0" + node-abi "^3.3.0" + pump "^3.0.0" + rc "^1.2.7" + simple-get "^4.0.0" + tar-fs "^2.0.0" + tunnel-agent "^0.6.0" + +prelude-ls@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/prelude-ls/-/prelude-ls-1.2.1.tgz#debc6489d7a6e6b0e7611888cec880337d316396" + integrity sha512-vkcDPrRZo1QZLbn5RLGPpg/WmIQ65qoWWhcGKf/b5eplkkarX0m9z8ppCat4mlOqUsWpyNuYgO3VRyrYHSzX5g== + +prettier@^2.8.8: + version "2.8.8" + resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.8.8.tgz#e8c5d7e98a4305ffe3de2e1fc4aca1a71c28b1da" + integrity sha512-tdN8qQGvNjw4CHbY+XXk0JgCXn9QiF21a55rBe5LJAU+kDyC4WQn4+awm2Xfk2lQMk5fKup9XgzTZtGkjBdP9Q== + +pretty-format@^27.0.2: + version "27.5.1" + resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-27.5.1.tgz#2181879fdea51a7a5851fb39d920faa63f01d88e" + integrity sha512-Qb1gy5OrP5+zDf2Bvnzdl3jsTf1qXVMazbvCoKhtKqVs4/YK4ozX4gKQJJVyNe+cajNPn0KoC0MC3FUmaHWEmQ== + dependencies: + ansi-regex "^5.0.1" + ansi-styles "^5.0.0" + react-is "^17.0.1" + +pretty-format@^29.0.0, pretty-format@^29.7.0: + version "29.7.0" + resolved "https://registry.yarnpkg.com/pretty-format/-/pretty-format-29.7.0.tgz#ca42c758310f365bfa71a0bda0a807160b776812" + integrity sha512-Pdlw/oPxN+aXdmM9R00JVC9WVFoCLTKJvDVLgmJ+qAffBMxsV85l/Lu7sNx4zSzPyoL2euImuEwHhOXdEgNFZQ== + dependencies: + "@jest/schemas" "^29.6.3" + ansi-styles "^5.0.0" + react-is "^18.0.0" + +proc-log@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/proc-log/-/proc-log-2.0.1.tgz#8f3f69a1f608de27878f91f5c688b225391cb685" + integrity sha512-Kcmo2FhfDTXdcbfDH76N7uBYHINxc/8GW7UAVuVP9I+Va3uHSerrnKV6dLooga/gh7GlgzuCCr/eoldnL1muGw== + +process-nextick-args@~2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/process-nextick-args/-/process-nextick-args-2.0.1.tgz#7820d9b16120cc55ca9ae7792680ae7dba6d7fe2" + integrity sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag== + +progress@^2.0.3: + version "2.0.3" + resolved "https://registry.yarnpkg.com/progress/-/progress-2.0.3.tgz#7e8cf8d8f5b8f239c1bc68beb4eb78567d572ef8" + integrity sha512-7PiHtLll5LdnKIMw100I+8xJXR5gW2QwWYkT6iJva0bXitZKa/XMrSbdmg3r2Xnaidz9Qumd0VPaMrZlF9V9sA== + +promise-inflight@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/promise-inflight/-/promise-inflight-1.0.1.tgz#98472870bf228132fcbdd868129bad12c3c029e3" + integrity sha512-6zWPyEOFaQBJYcGMHBKTKJ3u6TBsnMFOIZSa6ce1e/ZrrsOlnHRHbabMjLiBYKp+n44X9eUI6VUPaukCXHuG4g== + +promise-retry@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/promise-retry/-/promise-retry-2.0.1.tgz#ff747a13620ab57ba688f5fc67855410c370da22" + integrity sha512-y+WKFlBR8BGXnsNlIHFGPZmyDf3DFMoLhaflAnyZgV6rG6xu+JwesTo2Q9R6XwYmtmwAFCkAk3e35jEdoeh/3g== + dependencies: + err-code "^2.0.2" + retry "^0.12.0" + +promise@^7.1.1: + version "7.3.1" + resolved "https://registry.yarnpkg.com/promise/-/promise-7.3.1.tgz#064b72602b18f90f29192b8b1bc418ffd1ebd3bf" + integrity sha512-nolQXZ/4L+bP/UGlkfaIujX9BKxGwmQ9OT4mOt5yvy8iK1h3wqTEJCijzGANTCCl9nWjY41juyAn2K3Q1hLLTg== + dependencies: + asap "~2.0.3" + +prop-types@^15.7.2, prop-types@^15.8.1: + version "15.8.1" + resolved "https://registry.yarnpkg.com/prop-types/-/prop-types-15.8.1.tgz#67d87bf1a694f48435cf332c24af10214a3140b5" + integrity sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg== + dependencies: + loose-envify "^1.4.0" + object-assign "^4.1.1" + react-is "^16.13.1" + +proxy-addr@~2.0.7: + version "2.0.7" + resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.7.tgz#f19fe69ceab311eeb94b42e70e8c2070f9ba1025" + integrity sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg== + dependencies: + forwarded "0.2.0" + ipaddr.js "1.9.1" + +pump@^3.0.0: + version "3.0.2" + resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.2.tgz#836f3edd6bc2ee599256c924ffe0d88573ddcbf8" + integrity sha512-tUPXtzlGM8FE3P0ZL6DVs/3P58k9nk8/jZeQCurTJylQA8qFYzHFfhBJkuqyE0FifOsQ0uKWekiZ5g8wtr28cw== + dependencies: + end-of-stream "^1.1.0" + once "^1.3.1" + +punycode.js@^2.3.1: + version "2.3.1" + resolved "https://registry.yarnpkg.com/punycode.js/-/punycode.js-2.3.1.tgz#6b53e56ad75588234e79f4affa90972c7dd8cdb7" + integrity sha512-uxFIHU0YlHYhDQtV4R9J6a52SLx28BCjT+4ieh7IGbgwVJWO+km431c4yRlREUAsAmt/uMjQUyQHNEPf0M39CA== + +punycode@^2.1.0: + version "2.3.1" + resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.3.1.tgz#027422e2faec0b25e1549c3e1bd8309b9133b6e5" + integrity sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg== + +pure-color@^1.2.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/pure-color/-/pure-color-1.3.0.tgz#1fe064fb0ac851f0de61320a8bf796836422f33e" + integrity sha512-QFADYnsVoBMw1srW7OVKEYjG+MbIa49s54w1MA1EDY6r2r/sTcKKYqRX1f4GYvnXP7eN/Pe9HFcX+hwzmrXRHA== + +qs@6.13.0: + version "6.13.0" + resolved "https://registry.yarnpkg.com/qs/-/qs-6.13.0.tgz#6ca3bd58439f7e245655798997787b0d88a51906" + integrity sha512-+38qI9SOr8tfZ4QmJNplMUxqjbe7LKvvZgWdExBOmd+egZTtjLB67Gu0HRX3u/XOq7UU2Nx6nsjvS16Z9uwfpg== + dependencies: + side-channel "^1.0.6" + +qs@^6.4.0, qs@^6.9.1: + version "6.14.0" + resolved "https://registry.yarnpkg.com/qs/-/qs-6.14.0.tgz#c63fa40680d2c5c941412a0e899c89af60c0a930" + integrity sha512-YWWTjgABSKcvs/nWBi9PycY/JiPJqOD4JA6o9Sej2AtvSGarXxKC3OQSk4pAarbdQlKAh5D4FCQkJNkW+GAn3w== + dependencies: + side-channel "^1.1.0" + +queue-microtask@^1.2.2: + version "1.2.3" + resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" + integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== + +quick-lru@^5.1.1: + version "5.1.1" + resolved "https://registry.yarnpkg.com/quick-lru/-/quick-lru-5.1.1.tgz#366493e6b3e42a3a6885e2e99d18f80fb7a8c932" + integrity sha512-WuyALRjWPDGtt/wzJiadO5AXY+8hZ80hVpe6MyivgraREW751X3SbhRvG3eLKOYN+8VEvqLcf3wdnt44Z4S4SA== + +randombytes@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/randombytes/-/randombytes-2.1.0.tgz#df6f84372f0270dc65cdf6291349ab7a473d4f2a" + integrity sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ== + dependencies: + safe-buffer "^5.1.0" + +range-parser@~1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/range-parser/-/range-parser-1.2.1.tgz#3cf37023d199e1c24d1a55b84800c2f3e6468031" + integrity sha512-Hrgsx+orqoygnmhFbKaHE6c296J+HTAQXoxEF6gNupROmmGJRoyzfG3ccAveqCBrwr/2yxQ5BVd/GTl5agOwSg== + +raw-body@2.5.2: + version "2.5.2" + resolved "https://registry.yarnpkg.com/raw-body/-/raw-body-2.5.2.tgz#99febd83b90e08975087e8f1f9419a149366b68a" + integrity sha512-8zGqypfENjCIqGhgXToC8aB2r7YrBX+AQAfIPs/Mlk+BtPTztOvTS01NRW/3Eh60J+a48lt8qsCzirQ6loCVfA== + dependencies: + bytes "3.1.2" + http-errors "2.0.0" + iconv-lite "0.4.24" + unpipe "1.0.0" + +rc@^1.2.7: + version "1.2.8" + resolved "https://registry.yarnpkg.com/rc/-/rc-1.2.8.tgz#cd924bf5200a075b83c188cd6b9e211b7fc0d3ed" + integrity sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw== + dependencies: + deep-extend "^0.6.0" + ini "~1.3.0" + minimist "^1.2.0" + strip-json-comments "~2.0.1" + +re-resizable@^6.11.2: + version "6.11.2" + resolved "https://registry.yarnpkg.com/re-resizable/-/re-resizable-6.11.2.tgz#2e8f7119ca3881d5b5aea0ffa014a80e5c1252b3" + integrity sha512-2xI2P3OHs5qw7K0Ud1aLILK6MQxW50TcO+DetD9eIV58j84TqYeHoZcL9H4GXFXXIh7afhH8mv5iUCXII7OW7A== + +react-base16-styling@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/react-base16-styling/-/react-base16-styling-0.6.0.tgz#ef2156d66cf4139695c8a167886cb69ea660792c" + integrity sha512-yvh/7CArceR/jNATXOKDlvTnPKPmGZz7zsenQ3jUwLzHkNUR0CvY3yGYJbWJ/nnxsL8Sgmt5cO3/SILVuPO6TQ== + dependencies: + base16 "^1.0.0" + lodash.curry "^4.0.1" + lodash.flow "^3.3.0" + pure-color "^1.2.0" + +react-dom@^18.2.0: + version "18.3.1" + resolved "https://registry.yarnpkg.com/react-dom/-/react-dom-18.3.1.tgz#c2265d79511b57d479b3dd3fdfa51536494c5cb4" + integrity sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw== + dependencies: + loose-envify "^1.1.0" + scheduler "^0.23.2" + +react-hot-toast@^2.5.2: + version "2.5.2" + resolved "https://registry.yarnpkg.com/react-hot-toast/-/react-hot-toast-2.5.2.tgz#b55328966a26add56513e2dc1682e2cb4753c244" + integrity sha512-Tun3BbCxzmXXM7C+NI4qiv6lT0uwGh4oAfeJyNOjYUejTsm35mK9iCaYLGv8cBz9L5YxZLx/2ii7zsIwPtPUdw== + dependencies: + csstype "^3.1.3" + goober "^2.1.16" + +react-hotkeys-hook@^4.6.1: + version "4.6.2" + resolved "https://registry.yarnpkg.com/react-hotkeys-hook/-/react-hotkeys-hook-4.6.2.tgz#26dd20f59d23204814f223d5c5f3979a3fe83c88" + integrity sha512-FmP+ZriY3EG59Ug/lxNfrObCnW9xQShgk7Nb83+CkpfkcCpfS95ydv+E9JuXA5cp8KtskU7LGlIARpkc92X22Q== + +react-is@^16.13.1: + version "16.13.1" + resolved "https://registry.yarnpkg.com/react-is/-/react-is-16.13.1.tgz#789729a4dc36de2999dc156dd6c1d9c18cea56a4" + integrity sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ== + +react-is@^17.0.1: + version "17.0.2" + resolved "https://registry.yarnpkg.com/react-is/-/react-is-17.0.2.tgz#e691d4a8e9c789365655539ab372762b0efb54f0" + integrity sha512-w2GsyukL62IJnlaff/nRegPQR94C/XXamvMWmSHRJ4y7Ts/4ocGRmTHvOs8PSE6pB3dWOrD/nueuU5sduBsQ4w== + +react-is@^18.0.0: + version "18.3.1" + resolved "https://registry.yarnpkg.com/react-is/-/react-is-18.3.1.tgz#e83557dc12eae63a99e003a46388b1dcbb44db7e" + integrity sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg== + +react-json-view@^1.21.3: + version "1.21.3" + resolved "https://registry.yarnpkg.com/react-json-view/-/react-json-view-1.21.3.tgz#f184209ee8f1bf374fb0c41b0813cff54549c475" + integrity sha512-13p8IREj9/x/Ye4WI/JpjhoIwuzEgUAtgJZNBJckfzJt1qyh24BdTm6UQNGnyTq9dapQdrqvquZTo3dz1X6Cjw== + dependencies: + flux "^4.0.1" + react-base16-styling "^0.6.0" + react-lifecycles-compat "^3.0.4" + react-textarea-autosize "^8.3.2" + +react-lifecycles-compat@^3.0.0, react-lifecycles-compat@^3.0.4: + version "3.0.4" + resolved "https://registry.yarnpkg.com/react-lifecycles-compat/-/react-lifecycles-compat-3.0.4.tgz#4f1a273afdfc8f3488a8c516bfda78f872352362" + integrity sha512-fBASbA6LnOU9dOU2eW7aQ8xmYBSXUIWr+UmF9b1efZBazGNO+rcXT/icdKnYm2pTwcRylVUYwW7H1PHfLekVzA== + +react-modal-promise@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/react-modal-promise/-/react-modal-promise-1.0.2.tgz#122620b7f19eec73683affadfa77c543d88edc40" + integrity sha512-dqT618ROhG8qh1+O6EZkia5ELw3zaZWGpMX2YfEH4bgwYENPuFonqKw1W70LFx3K/SCZvVBcD6UYEI12yzYXzg== + +react-modal@^3.16.3: + version "3.16.3" + resolved "https://registry.yarnpkg.com/react-modal/-/react-modal-3.16.3.tgz#c412d41915782e3c261253435d01468e2439b11b" + integrity sha512-yCYRJB5YkeQDQlTt17WGAgFJ7jr2QYcWa1SHqZ3PluDmnKJ/7+tVU+E6uKyZ0nODaeEj+xCpK4LcSnKXLMC0Nw== + dependencies: + exenv "^1.2.0" + prop-types "^15.7.2" + react-lifecycles-compat "^3.0.0" + warning "^4.0.3" + +react-refresh@^0.17.0: + version "0.17.0" + resolved "https://registry.yarnpkg.com/react-refresh/-/react-refresh-0.17.0.tgz#b7e579c3657f23d04eccbe4ad2e58a8ed51e7e53" + integrity sha512-z6F7K9bV85EfseRCp2bzrpyQ0Gkw1uLoCel9XBVWPg/TjRj94SkJzUTGfOa4bs7iJvBWtQG0Wq7wnI0syw3EBQ== + +react-router-dom@^6.28.0: + version "6.30.0" + resolved "https://registry.yarnpkg.com/react-router-dom/-/react-router-dom-6.30.0.tgz#a64774104508bff56b1affc2796daa3f7e76b7df" + integrity sha512-x30B78HV5tFk8ex0ITwzC9TTZMua4jGyA9IUlH1JLQYQTFyxr/ZxwOJq7evg1JX1qGVUcvhsmQSKdPncQrjTgA== + dependencies: + "@remix-run/router" "1.23.0" + react-router "6.30.0" + +react-router@6.30.0: + version "6.30.0" + resolved "https://registry.yarnpkg.com/react-router/-/react-router-6.30.0.tgz#9789d775e63bc0df60f39ced77c8c41f1e01ff90" + integrity sha512-D3X8FyH9nBcTSHGdEKurK7r8OYE1kKFn3d/CF+CoxbSHkxU7o37+Uh7eAHRXr6k2tSExXYO++07PeXJtA/dEhQ== + dependencies: + "@remix-run/router" "1.23.0" + +react-textarea-autosize@^8.3.2: + version "8.5.9" + resolved "https://registry.yarnpkg.com/react-textarea-autosize/-/react-textarea-autosize-8.5.9.tgz#ab8627b09aa04d8a2f45d5b5cd94c84d1d4a8893" + integrity sha512-U1DGlIQN5AwgjTyOEnI1oCcMuEr1pv1qOtklB2l4nyMGbHzWrI0eFsYK0zos2YWqAolJyG0IWJaqWmWj5ETh0A== + dependencies: + "@babel/runtime" "^7.20.13" + use-composed-ref "^1.3.0" + use-latest "^1.2.1" + +react@^18.3.1: + version "18.3.1" + resolved "https://registry.yarnpkg.com/react/-/react-18.3.1.tgz#49ab892009c53933625bd16b2533fc754cab2891" + integrity sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ== + dependencies: + loose-envify "^1.1.0" + +read-binary-file-arch@^1.0.6: + version "1.0.6" + resolved "https://registry.yarnpkg.com/read-binary-file-arch/-/read-binary-file-arch-1.0.6.tgz#959c4637daa932280a9b911b1a6766a7e44288fc" + integrity sha512-BNg9EN3DD3GsDXX7Aa8O4p92sryjkmzYYgmgTAc6CA4uGLEDzFfxOxugu21akOxpcXHiEgsYkC6nPsQvLLLmEg== + dependencies: + debug "^4.3.4" + +read-cache@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/read-cache/-/read-cache-1.0.0.tgz#e664ef31161166c9751cdbe8dbcf86b5fb58f774" + integrity sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA== + dependencies: + pify "^2.3.0" + +read-pkg-up@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-2.0.0.tgz#6b72a8048984e0c41e79510fd5e9fa99b3b549be" + integrity sha512-1orxQfbWGUiTn9XsPlChs6rLie/AV9jwZTGmu2NZw/CUDJQchXJFYE0Fq5j7+n558T1JhDWLdhyd1Zj+wLY//w== + dependencies: + find-up "^2.0.0" + read-pkg "^2.0.0" + +read-pkg@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/read-pkg/-/read-pkg-2.0.0.tgz#8ef1c0623c6a6db0dc6713c4bfac46332b2368f8" + integrity sha512-eFIBOPW7FGjzBuk3hdXEuNSiTZS/xEMlH49HxMyzb0hyPfu4EhVjT2DH32K1hSSmVq4sebAWnZuuY5auISUTGA== + dependencies: + load-json-file "^2.0.0" + normalize-package-data "^2.3.2" + path-type "^2.0.0" + +read@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/read/-/read-1.0.7.tgz#b3da19bd052431a97671d44a42634adf710b40c4" + integrity sha512-rSOKNYUmaxy0om1BNjMN4ezNT6VKK+2xF4GBhc81mkH7L60i6dp8qPYrkndNLT3QPphoII3maL9PVC9XmhHwVQ== + dependencies: + mute-stream "~0.0.4" + +readable-stream@^3.1.1, readable-stream@^3.4.0: + version "3.6.2" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.2.tgz#56a9b36ea965c00c5a93ef31eb111a0f11056967" + integrity sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA== + dependencies: + inherits "^2.0.3" + string_decoder "^1.1.1" + util-deprecate "^1.0.1" + +readable-stream@~2.3.6: + version "2.3.8" + resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-2.3.8.tgz#91125e8042bba1b9887f49345f6277027ce8be9b" + integrity sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA== + dependencies: + core-util-is "~1.0.0" + inherits "~2.0.3" + isarray "~1.0.0" + process-nextick-args "~2.0.0" + safe-buffer "~5.1.1" + string_decoder "~1.1.1" + util-deprecate "~1.0.1" + +readdirp@^4.0.1: + version "4.1.2" + resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-4.1.2.tgz#eb85801435fbf2a7ee58f19e0921b068fc69948d" + integrity sha512-GDhwkLfywWL2s6vEjyhri+eXmfH6j1L7JE27WhqLeYzoh/A3DBaYGEj2H/HFZCn/kMfim73FXxEJTw06WtxQwg== + +readdirp@~3.6.0: + version "3.6.0" + resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.6.0.tgz#74a370bd857116e245b29cc97340cd431a02a6c7" + integrity sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA== + dependencies: + picomatch "^2.2.1" + +recast@^0.23.1: + version "0.23.11" + resolved "https://registry.yarnpkg.com/recast/-/recast-0.23.11.tgz#8885570bb28cf773ba1dc600da7f502f7883f73f" + integrity sha512-YTUo+Flmw4ZXiWfQKGcwwc11KnoRAYgzAE2E7mXKCjSviTKShtxBsN6YUUBB2gtaBzKzeKunxhUwNHQuRryhWA== + dependencies: + ast-types "^0.16.1" + esprima "~4.0.0" + source-map "~0.6.1" + tiny-invariant "^1.3.3" + tslib "^2.0.1" + +rechoir@^0.8.0: + version "0.8.0" + resolved "https://registry.yarnpkg.com/rechoir/-/rechoir-0.8.0.tgz#49f866e0d32146142da3ad8f0eff352b3215ff22" + integrity sha512-/vxpCXddiX8NGfGO/mTafwjq4aFa/71pvamip0++IQk3zG8cbCj0fifNPrjjF1XMXUne91jL9OoxmdykoEtifQ== + dependencies: + resolve "^1.20.0" + +redent@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/redent/-/redent-3.0.0.tgz#e557b7998316bb53c9f1f56fa626352c6963059f" + integrity sha512-6tDA8g98We0zd0GvVeMT9arEOnTw9qM03L9cJXaCjrip1OO764RDBLBfrB4cwzNGDj5OA5ioymC9GkizgWJDUg== + dependencies: + indent-string "^4.0.0" + strip-indent "^3.0.0" + +reflect.getprototypeof@^1.0.6, reflect.getprototypeof@^1.0.9: + version "1.0.10" + resolved "https://registry.yarnpkg.com/reflect.getprototypeof/-/reflect.getprototypeof-1.0.10.tgz#c629219e78a3316d8b604c765ef68996964e7bf9" + integrity sha512-00o4I+DVrefhv+nX0ulyi3biSHCPDe+yLv5o/p6d/UVlirijB8E16FtfwSAi4g3tcqrQ4lRAqQSoFEZJehYEcw== + dependencies: + call-bind "^1.0.8" + define-properties "^1.2.1" + es-abstract "^1.23.9" + es-errors "^1.3.0" + es-object-atoms "^1.0.0" + get-intrinsic "^1.2.7" + get-proto "^1.0.1" + which-builtin-type "^1.2.1" + +regenerate-unicode-properties@^10.2.0: + version "10.2.0" + resolved "https://registry.yarnpkg.com/regenerate-unicode-properties/-/regenerate-unicode-properties-10.2.0.tgz#626e39df8c372338ea9b8028d1f99dc3fd9c3db0" + integrity sha512-DqHn3DwbmmPVzeKj9woBadqmXxLvQoQIwu7nopMc72ztvxVmVk2SBhSnx67zuye5TP+lJsb/TBQsjLKhnDf3MA== + dependencies: + regenerate "^1.4.2" + +regenerate@^1.4.2: + version "1.4.2" + resolved "https://registry.yarnpkg.com/regenerate/-/regenerate-1.4.2.tgz#b9346d8827e8f5a32f7ba29637d398b69014848a" + integrity sha512-zrceR/XhGYU/d/opr2EKO7aRHUeiBI8qjtfHqADTwZd6Szfy16la6kqD0MIUs5z5hx6AaKa+PixpPrR289+I0A== + +regenerator-runtime@^0.14.0: + version "0.14.1" + resolved "https://registry.yarnpkg.com/regenerator-runtime/-/regenerator-runtime-0.14.1.tgz#356ade10263f685dda125100cd862c1db895327f" + integrity sha512-dYnhHh0nJoMfnkZs6GmmhFknAGRrLznOu5nc9ML+EJxGvrx6H7teuevqVqCuPcPK//3eDrrjQhehXVx9cnkGdw== + +regenerator-transform@^0.15.2: + version "0.15.2" + resolved "https://registry.yarnpkg.com/regenerator-transform/-/regenerator-transform-0.15.2.tgz#5bbae58b522098ebdf09bca2f83838929001c7a4" + integrity sha512-hfMp2BoF0qOk3uc5V20ALGDS2ddjQaLrdl7xrGXvAIow7qeWRM2VA2HuCHkUKk9slq3VwEwLNK3DFBqDfPGYtg== + dependencies: + "@babel/runtime" "^7.8.4" + +regexp.prototype.flags@^1.5.3: + version "1.5.4" + resolved "https://registry.yarnpkg.com/regexp.prototype.flags/-/regexp.prototype.flags-1.5.4.tgz#1ad6c62d44a259007e55b3970e00f746efbcaa19" + integrity sha512-dYqgNSZbDwkaJ2ceRd9ojCGjBq+mOm9LmtXnAnEGyHhN/5R7iDW2TRw3h+o/jCFxus3P2LfWIIiwowAjANm7IA== + dependencies: + call-bind "^1.0.8" + define-properties "^1.2.1" + es-errors "^1.3.0" + get-proto "^1.0.1" + gopd "^1.2.0" + set-function-name "^2.0.2" + +regexpu-core@^6.2.0: + version "6.2.0" + resolved "https://registry.yarnpkg.com/regexpu-core/-/regexpu-core-6.2.0.tgz#0e5190d79e542bf294955dccabae04d3c7d53826" + integrity sha512-H66BPQMrv+V16t8xtmq+UC0CBpiTBA60V8ibS1QVReIp8T1z8hwFxqcGzm9K6lgsN7sB5edVH8a+ze6Fqm4weA== + dependencies: + regenerate "^1.4.2" + regenerate-unicode-properties "^10.2.0" + regjsgen "^0.8.0" + regjsparser "^0.12.0" + unicode-match-property-ecmascript "^2.0.0" + unicode-match-property-value-ecmascript "^2.1.0" + +regjsgen@^0.8.0: + version "0.8.0" + resolved "https://registry.yarnpkg.com/regjsgen/-/regjsgen-0.8.0.tgz#df23ff26e0c5b300a6470cad160a9d090c3a37ab" + integrity sha512-RvwtGe3d7LvWiDQXeQw8p5asZUmfU1G/l6WbUXeHta7Y2PEIvBTwH6E2EfmYUK8pxcxEdEmaomqyp0vZZ7C+3Q== + +regjsparser@^0.12.0: + version "0.12.0" + resolved "https://registry.yarnpkg.com/regjsparser/-/regjsparser-0.12.0.tgz#0e846df6c6530586429377de56e0475583b088dc" + integrity sha512-cnE+y8bz4NhMjISKbgeVJtqNbtf5QpjZP+Bslo+UqkIt9QPnX9q095eiRRASJG1/tz6dlNr6Z5NsBiWYokp6EQ== + dependencies: + jsesc "~3.0.2" + +require-directory@^2.1.1: + version "2.1.1" + resolved "https://registry.yarnpkg.com/require-directory/-/require-directory-2.1.1.tgz#8c64ad5fd30dab1c976e2344ffe7f792a6a6df42" + integrity sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q== + +require-from-string@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-2.0.2.tgz#89a7fdd938261267318eafe14f9c32e598c36909" + integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw== + +requires-port@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/requires-port/-/requires-port-1.0.0.tgz#925d2601d39ac485e091cf0da5c6e694dc3dcaff" + integrity sha512-KigOCHcocU3XODJxsu8i/j8T9tzT4adHiecwORRQ0ZZFcp7ahwXuRU1m+yuO90C5ZUyGeGfocHDI14M3L3yDAQ== + +resedit@^1.7.0: + version "1.7.2" + resolved "https://registry.yarnpkg.com/resedit/-/resedit-1.7.2.tgz#b1041170b99811710c13f949c7d225871de4cc78" + integrity sha512-vHjcY2MlAITJhC0eRD/Vv8Vlgmu9Sd3LX9zZvtGzU5ZImdTN3+d6e/4mnTyV8vEbyf1sgNIrWxhWlrys52OkEA== + dependencies: + pe-library "^0.4.1" + +resedit@^2.0.0: + version "2.0.3" + resolved "https://registry.yarnpkg.com/resedit/-/resedit-2.0.3.tgz#5145a9faabca44b917d5636dbe8e67ec7f62c6f2" + integrity sha512-oTeemxwoMuxxTYxXUwjkrOPfngTQehlv0/HoYFNkB4uzsP1Un1A9nI8JQKGOFkxpqkC7qkMs0lUsGrvUlbLNUA== + dependencies: + pe-library "^1.0.1" + +resolve-alpn@^1.0.0: + version "1.2.1" + resolved "https://registry.yarnpkg.com/resolve-alpn/-/resolve-alpn-1.2.1.tgz#b7adbdac3546aaaec20b45e7d8265927072726f9" + integrity sha512-0a1F4l73/ZFZOakJnQ3FvkJ2+gSTQWz/r2KE5OdDY0TxPm5h4GkqkWWfM47T7HsbnOtcJVEF4epCVy6u7Q3K+g== + +resolve-from@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/resolve-from/-/resolve-from-4.0.0.tgz#4abcd852ad32dd7baabfe9b40e00a36db5f392e6" + integrity sha512-pb/MYmXstAkysRFx8piNI1tGFNQIFA3vkE3Gq4EuA1dF6gHp/+vgZqsCGJapvy8N3Q+4o7FwvquPJcnZ7RYy4g== + +resolve@^1.1.6, resolve@^1.1.7, resolve@^1.10.0, resolve@^1.14.2, resolve@^1.20.0, resolve@^1.22.1, resolve@^1.22.4, resolve@^1.22.8: + version "1.22.10" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-1.22.10.tgz#b663e83ffb09bbf2386944736baae803029b8b39" + integrity sha512-NPRy+/ncIMeDlTAsuqwKIiferiawhefFJtkNSW0qZJEqMEb+qBt/77B/jGeeek+F0uOeN05CDa6HXbbIgtVX4w== + dependencies: + is-core-module "^2.16.0" + path-parse "^1.0.7" + supports-preserve-symlinks-flag "^1.0.0" + +resolve@^2.0.0-next.5: + version "2.0.0-next.5" + resolved "https://registry.yarnpkg.com/resolve/-/resolve-2.0.0-next.5.tgz#6b0ec3107e671e52b68cd068ef327173b90dc03c" + integrity sha512-U7WjGVG9sH8tvjW5SmGbQuui75FiyjAX72HX15DwBBwF9dNiQZRQAg9nnPhYy+TUnE0+VcrttuvNI8oSxZcocA== + dependencies: + is-core-module "^2.13.0" + path-parse "^1.0.7" + supports-preserve-symlinks-flag "^1.0.0" + +responselike@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/responselike/-/responselike-2.0.1.tgz#9a0bc8fdc252f3fb1cca68b016591059ba1422bc" + integrity sha512-4gl03wn3hj1HP3yzgdI7d3lCkF95F21Pz4BPGvKHinyQzALR5CapwC8yIi0Rh58DEMQ/SguC03wFj2k0M/mHhw== + dependencies: + lowercase-keys "^2.0.0" + +restore-cursor@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-3.1.0.tgz#39f67c54b3a7a58cea5236d95cf0034239631f7e" + integrity sha512-l+sSefzHpj5qimhFSE5a8nufZYAM3sBSVMAPtYkmC+4EH2anSGaEMXSD0izRQbu9nfyQ9y5JrVmp7E8oZrUjvA== + dependencies: + onetime "^5.1.0" + signal-exit "^3.0.2" + +restore-cursor@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-4.0.0.tgz#519560a4318975096def6e609d44100edaa4ccb9" + integrity sha512-I9fPXU9geO9bHOt9pHHOhOkYerIMsmVaWB0rA2AI9ERh/+x/i7MV5HKBNrg+ljO5eoPVgCcnFuRjJ9uH6I/3eg== + dependencies: + onetime "^5.1.0" + signal-exit "^3.0.2" + +restore-cursor@^5.0.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/restore-cursor/-/restore-cursor-5.1.0.tgz#0766d95699efacb14150993f55baf0953ea1ebe7" + integrity sha512-oMA2dcrw6u0YfxJQXm342bFKX/E4sG9rbTzO9ptUcR/e8A33cHuvStiYOwH7fszkZlZ1z/ta9AAoPk2F4qIOHA== + dependencies: + onetime "^7.0.0" + signal-exit "^4.1.0" + +retry@^0.12.0: + version "0.12.0" + resolved "https://registry.yarnpkg.com/retry/-/retry-0.12.0.tgz#1b42a6266a21f07421d1b0b54b7dc167b01c013b" + integrity sha512-9LkiTwjUh6rT555DtE9rTX+BKByPfrMzEAtnlEtdEwr3Nkffwiihqe2bWADg+OQRjt9gl6ICdmB/ZFDCGAtSow== + +reusify@^1.0.4: + version "1.1.0" + resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.1.0.tgz#0fe13b9522e1473f51b558ee796e08f11f9b489f" + integrity sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw== + +rfdc@^1.3.0: + version "1.4.1" + resolved "https://registry.yarnpkg.com/rfdc/-/rfdc-1.4.1.tgz#778f76c4fb731d93414e8f925fbecf64cce7f6ca" + integrity sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA== + +rimraf@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/rimraf/-/rimraf-3.0.2.tgz#f1a5402ba6220ad52cc1282bac1ae3aa49fd061a" + integrity sha512-JZkJMZkAGFFPP2YqXZXPbMlMBgsxzE8ILs4lMIX/2o0L9UBw9O/Y3o6wFw/i9YLapcUJWwqbi3kdxIPdC62TIA== + dependencies: + glob "^7.1.3" + +roarr@^2.15.3: + version "2.15.4" + resolved "https://registry.yarnpkg.com/roarr/-/roarr-2.15.4.tgz#f5fe795b7b838ccfe35dc608e0282b9eba2e7afd" + integrity sha512-CHhPh+UNHD2GTXNYhPWLnU8ONHdI+5DI+4EYIAOaiD63rHeYlZvyh8P+in5999TTSFgUYuKUAjzRI4mdh/p+2A== + dependencies: + boolean "^3.0.1" + detect-node "^2.0.4" + globalthis "^1.0.1" + json-stringify-safe "^5.0.1" + semver-compare "^1.0.0" + sprintf-js "^1.1.2" + +rollup-plugin-dts@^6.1.1: + version "6.2.1" + resolved "https://registry.yarnpkg.com/rollup-plugin-dts/-/rollup-plugin-dts-6.2.1.tgz#120a40734f740115da44931d7915a370fe420701" + integrity sha512-sR3CxYUl7i2CHa0O7bA45mCrgADyAQ0tVtGSqi3yvH28M+eg1+g5d7kQ9hLvEz5dorK3XVsH5L2jwHLQf72DzA== + dependencies: + magic-string "^0.30.17" + optionalDependencies: + "@babel/code-frame" "^7.26.2" + +rollup@^4.20.0, rollup@^4.34.9, rollup@^4.40.0: + version "4.40.0" + resolved "https://registry.yarnpkg.com/rollup/-/rollup-4.40.0.tgz#13742a615f423ccba457554f006873d5a4de1920" + integrity sha512-Noe455xmA96nnqH5piFtLobsGbCij7Tu+tb3c1vYjNbTkfzGqXqQXG3wJaYXkRZuQ0vEYN4bhwg7QnIrqB5B+w== + dependencies: + "@types/estree" "1.0.7" + optionalDependencies: + "@rollup/rollup-android-arm-eabi" "4.40.0" + "@rollup/rollup-android-arm64" "4.40.0" + "@rollup/rollup-darwin-arm64" "4.40.0" + "@rollup/rollup-darwin-x64" "4.40.0" + "@rollup/rollup-freebsd-arm64" "4.40.0" + "@rollup/rollup-freebsd-x64" "4.40.0" + "@rollup/rollup-linux-arm-gnueabihf" "4.40.0" + "@rollup/rollup-linux-arm-musleabihf" "4.40.0" + "@rollup/rollup-linux-arm64-gnu" "4.40.0" + "@rollup/rollup-linux-arm64-musl" "4.40.0" + "@rollup/rollup-linux-loongarch64-gnu" "4.40.0" + "@rollup/rollup-linux-powerpc64le-gnu" "4.40.0" + "@rollup/rollup-linux-riscv64-gnu" "4.40.0" + "@rollup/rollup-linux-riscv64-musl" "4.40.0" + "@rollup/rollup-linux-s390x-gnu" "4.40.0" + "@rollup/rollup-linux-x64-gnu" "4.40.0" + "@rollup/rollup-linux-x64-musl" "4.40.0" + "@rollup/rollup-win32-arm64-msvc" "4.40.0" + "@rollup/rollup-win32-ia32-msvc" "4.40.0" + "@rollup/rollup-win32-x64-msvc" "4.40.0" + fsevents "~2.3.2" + +run-applescript@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/run-applescript/-/run-applescript-7.0.0.tgz#e5a553c2bffd620e169d276c1cd8f1b64778fbeb" + integrity sha512-9by4Ij99JUr/MCFBUkDKLWK3G9HVXmabKz9U5MlIAIuvuzkiOicRYs8XJLxX+xahD+mLiiCYDqF9dKAgtzKP1A== + +run-parallel@^1.1.9: + version "1.2.0" + resolved "https://registry.yarnpkg.com/run-parallel/-/run-parallel-1.2.0.tgz#66d1368da7bdf921eb9d95bd1a9229e7f21a43ee" + integrity sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA== + dependencies: + queue-microtask "^1.2.2" + +safe-array-concat@^1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/safe-array-concat/-/safe-array-concat-1.1.3.tgz#c9e54ec4f603b0bbb8e7e5007a5ee7aecd1538c3" + integrity sha512-AURm5f0jYEOydBj7VQlVvDrjeFgthDdEF5H1dP+6mNpoXOMo1quQqJ4wvJDyRZ9+pO3kGWoOdmV08cSv2aJV6Q== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.2" + get-intrinsic "^1.2.6" + has-symbols "^1.1.0" + isarray "^2.0.5" + +safe-buffer@5.1.2, safe-buffer@~5.1.0, safe-buffer@~5.1.1: + version "5.1.2" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.1.2.tgz#991ec69d296e0313747d59bdfd2b745c35f8828d" + integrity sha512-Gd2UZBJDkXlY7GbJxfsE8/nvKkUEU1G38c1siN6QP6a9PT9MmHB8GnpscSmMJSoF8LOIrt8ud/wPtojys4G6+g== + +safe-buffer@5.2.1, safe-buffer@^5.0.1, safe-buffer@^5.1.0, safe-buffer@~5.2.0: + version "5.2.1" + resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" + integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== + +safe-push-apply@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/safe-push-apply/-/safe-push-apply-1.0.0.tgz#01850e981c1602d398c85081f360e4e6d03d27f5" + integrity sha512-iKE9w/Z7xCzUMIZqdBsp6pEQvwuEebH4vdpjcDWnyzaI6yl6O9FHvVpmGelvEHNsoY6wGblkxR6Zty/h00WiSA== + dependencies: + es-errors "^1.3.0" + isarray "^2.0.5" + +safe-regex-test@^1.0.3, safe-regex-test@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/safe-regex-test/-/safe-regex-test-1.1.0.tgz#7f87dfb67a3150782eaaf18583ff5d1711ac10c1" + integrity sha512-x/+Cz4YrimQxQccJf5mKEbIa1NzeCRNI5Ecl/ekmlYaampdNLPalVyIcCZNNH3MvmqBugV5TMYZXv0ljslUlaw== + dependencies: + call-bound "^1.0.2" + es-errors "^1.3.0" + is-regex "^1.2.1" + +"safer-buffer@>= 2.1.2 < 3", "safer-buffer@>= 2.1.2 < 3.0.0": + version "2.1.2" + resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" + integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== + +sanitize-filename@^1.6.3: + version "1.6.3" + resolved "https://registry.yarnpkg.com/sanitize-filename/-/sanitize-filename-1.6.3.tgz#755ebd752045931977e30b2025d340d7c9090378" + integrity sha512-y/52Mcy7aw3gRm7IrcGDFx/bCk4AhRh2eI9luHOQM86nZsqwiRkkq2GekHXBBD+SmPidc8i2PqtYZl+pWJ8Oeg== + dependencies: + truncate-utf8-bytes "^1.0.0" + +sax@>=0.6.0, sax@^1.2.4: + version "1.4.1" + resolved "https://registry.yarnpkg.com/sax/-/sax-1.4.1.tgz#44cc8988377f126304d3b3fc1010c733b929ef0f" + integrity sha512-+aWOz7yVScEGoKNd4PA10LZ8sk0A/z5+nXQG5giUO5rprX9jgYsTdov9qCchZiPIZezbZH+jRut8nPodFAX4Jg== + +scheduler@^0.23.2: + version "0.23.2" + resolved "https://registry.yarnpkg.com/scheduler/-/scheduler-0.23.2.tgz#414ba64a3b282892e944cf2108ecc078d115cdc3" + integrity sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ== + dependencies: + loose-envify "^1.1.0" + +secure-compare@3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/secure-compare/-/secure-compare-3.0.1.tgz#f1a0329b308b221fae37b9974f3d578d0ca999e3" + integrity sha512-AckIIV90rPDcBcglUwXPF3kg0P0qmPsPXAj6BBEENQE1p5yA1xfmDJzfi1Tappj37Pv2mVbKpL3Z1T+Nn7k1Qw== + +semver-compare@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/semver-compare/-/semver-compare-1.0.0.tgz#0dee216a1c941ab37e9efb1788f6afc5ff5537fc" + integrity sha512-YM3/ITh2MJ5MtzaM429anh+x2jiLVjqILF4m4oyQB18W7Ggea7BfqdH/wGMK7dDiMghv/6WG7znWMwUDzJiXow== + +"semver@2 || 3 || 4 || 5", semver@^5.1.0, semver@^5.5.0: + version "5.7.2" + resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.2.tgz#48d55db737c3287cd4835e17fa13feace1c41ef8" + integrity sha512-cBznnQ9KjJqU67B52RMC65CMarK2600WFnbkcaiwWq3xy/5haFJlshgnpjovMVJ+Hff49d8GEn0b87C5pDQ10g== + +semver@^6.2.0, semver@^6.3.1: + version "6.3.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" + integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== + +semver@^7.1.3, semver@^7.2.1, semver@^7.3.2, semver@^7.3.5, semver@^7.3.7, semver@^7.3.8, semver@^7.5.2, semver@^7.5.3, semver@^7.5.4, semver@^7.6.0, semver@^7.6.2, semver@^7.6.3: + version "7.7.1" + resolved "https://registry.yarnpkg.com/semver/-/semver-7.7.1.tgz#abd5098d82b18c6c81f6074ff2647fd3e7220c9f" + integrity sha512-hlq8tAfn0m/61p4BVRcPzIGr6LKiMwo4VM6dGi6pt4qcRkmNzTcWq6eCEjEh+qXjkMDvPlOFFSGwQjoEa6gyMA== + +send@0.19.0: + version "0.19.0" + resolved "https://registry.yarnpkg.com/send/-/send-0.19.0.tgz#bbc5a388c8ea6c048967049dbeac0e4a3f09d7f8" + integrity sha512-dW41u5VfLXu8SJh5bwRmyYUbAoSB3c9uQh6L8h/KtsFREPWpbX1lrljJo186Jc4nmci/sGUZ9a0a0J2zgfq2hw== + dependencies: + debug "2.6.9" + depd "2.0.0" + destroy "1.2.0" + encodeurl "~1.0.2" + escape-html "~1.0.3" + etag "~1.8.1" + fresh "0.5.2" + http-errors "2.0.0" + mime "1.6.0" + ms "2.1.3" + on-finished "2.4.1" + range-parser "~1.2.1" + statuses "2.0.1" + +serialize-error@^7.0.1: + version "7.0.1" + resolved "https://registry.yarnpkg.com/serialize-error/-/serialize-error-7.0.1.tgz#f1360b0447f61ffb483ec4157c737fab7d778e18" + integrity sha512-8I8TjW5KMOKsZQTvoxjuSIa7foAwPWGOts+6o7sgjz41/qMD9VQHEDxi6PBvK2l0MXUmqZyNpUK+T2tQaaElvw== + dependencies: + type-fest "^0.13.1" + +serialize-javascript@^6.0.2: + version "6.0.2" + resolved "https://registry.yarnpkg.com/serialize-javascript/-/serialize-javascript-6.0.2.tgz#defa1e055c83bf6d59ea805d8da862254eb6a6c2" + integrity sha512-Saa1xPByTTq2gdeFZYLLo+RFE35NHZkAbqZeWNd3BpzppeVisAqpDjcp8dyf6uIvEqJRd46jemmyA4iFIeVk8g== + dependencies: + randombytes "^2.1.0" + +serve-static@1.16.2: + version "1.16.2" + resolved "https://registry.yarnpkg.com/serve-static/-/serve-static-1.16.2.tgz#b6a5343da47f6bdd2673848bf45754941e803296" + integrity sha512-VqpjJZKadQB/PEbEwvFdO43Ax5dFBZ2UECszz8bQ7pi7wt//PWe1P6MN7eCnjsatYtBT6EuiClbjSWP2WrIoTw== + dependencies: + encodeurl "~2.0.0" + escape-html "~1.0.3" + parseurl "~1.3.3" + send "0.19.0" + +set-function-length@^1.2.2: + version "1.2.2" + resolved "https://registry.yarnpkg.com/set-function-length/-/set-function-length-1.2.2.tgz#aac72314198eaed975cf77b2c3b6b880695e5449" + integrity sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg== + dependencies: + define-data-property "^1.1.4" + es-errors "^1.3.0" + function-bind "^1.1.2" + get-intrinsic "^1.2.4" + gopd "^1.0.1" + has-property-descriptors "^1.0.2" + +set-function-name@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/set-function-name/-/set-function-name-2.0.2.tgz#16a705c5a0dc2f5e638ca96d8a8cd4e1c2b90985" + integrity sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ== + dependencies: + define-data-property "^1.1.4" + es-errors "^1.3.0" + functions-have-names "^1.2.3" + has-property-descriptors "^1.0.2" + +set-proto@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/set-proto/-/set-proto-1.0.0.tgz#0760dbcff30b2d7e801fd6e19983e56da337565e" + integrity sha512-RJRdvCo6IAnPdsvP/7m6bsQqNnn1FCBX5ZNtFL98MmFF/4xAIJTIg1YbHW5DC2W5SKZanrC6i4HsJqlajw/dZw== + dependencies: + dunder-proto "^1.0.1" + es-errors "^1.3.0" + es-object-atoms "^1.0.0" + +setimmediate@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/setimmediate/-/setimmediate-1.0.5.tgz#290cbb232e306942d7d7ea9b83732ab7856f8285" + integrity sha512-MATJdZp8sLqDl/68LfQmbP8zKPLQNV6BIZoIgrscFDQ+RsvK/BxeDQOgyxKKoh0y/8h3BqVFnCqQ/gd+reiIXA== + +setprototypeof@1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/setprototypeof/-/setprototypeof-1.2.0.tgz#66c9a24a73f9fc28cbe66b09fed3d33dcaf1b424" + integrity sha512-E5LDX7Wrp85Kil5bhZv46j8jOeboKq5JMmYM3gVGdGH8xFpPWXUMsNrlODCrkoxMEeNi/XZIwuRvY4XNwYMJpw== + +shebang-command@^1.2.0: + version "1.2.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-1.2.0.tgz#44aac65b695b03398968c39f363fee5deafdf1ea" + integrity sha512-EV3L1+UQWGor21OmnvojK36mhg+TyIKDh3iFBKBohr5xeXIhNBcx8oWdgkTEEQ+BEFFYdLRuqMfd5L84N1V5Vg== + dependencies: + shebang-regex "^1.0.0" + +shebang-command@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/shebang-command/-/shebang-command-2.0.0.tgz#ccd0af4f8835fbdc265b82461aaf0c36663f34ea" + integrity sha512-kHxr2zZpYtdmrN1qDjrrX/Z1rR1kG8Dx+gkpK1G4eXmvXswmcE1hTWBWYUzlraYw1/yZp6YuDY77YtvbN0dmDA== + dependencies: + shebang-regex "^3.0.0" + +shebang-regex@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-1.0.0.tgz#da42f49740c0b42db2ca9728571cb190c98efea3" + integrity sha512-wpoSFAxys6b2a2wHZ1XpDSgD7N9iVjg29Ph9uV/uaP9Ex/KXlkTZTeddxDPSYQpgvzKLGJke2UU0AzoGCjNIvQ== + +shebang-regex@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/shebang-regex/-/shebang-regex-3.0.0.tgz#ae16f1644d873ecad843b0307b143362d4c42172" + integrity sha512-7++dFhtcx3353uBaq8DDR4NuxBetBzC7ZQOhmTQInHEd6bSrXdiEyzCvG07Z44UYdLShWUyXt5M/yhz8ekcb1A== + +shell-exec@1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/shell-exec/-/shell-exec-1.0.2.tgz#2e9361b0fde1d73f476c4b6671fa17785f696756" + integrity sha512-jyVd+kU2X+mWKMmGhx4fpWbPsjvD53k9ivqetutVW/BQ+WIZoDoP4d8vUMGezV6saZsiNoW2f9GIhg9Dondohg== + +side-channel-list@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/side-channel-list/-/side-channel-list-1.0.0.tgz#10cb5984263115d3b7a0e336591e290a830af8ad" + integrity sha512-FCLHtRD/gnpCiCHEiJLOwdmFP+wzCmDEkc9y7NsYxeF4u7Btsn1ZuwgwJGxImImHicJArLP4R0yX4c2KCrMrTA== + dependencies: + es-errors "^1.3.0" + object-inspect "^1.13.3" + +side-channel-map@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/side-channel-map/-/side-channel-map-1.0.1.tgz#d6bb6b37902c6fef5174e5f533fab4c732a26f42" + integrity sha512-VCjCNfgMsby3tTdo02nbjtM/ewra6jPHmpThenkTYh8pG9ucZ/1P8So4u4FGBek/BjpOVsDCMoLA/iuBKIFXRA== + dependencies: + call-bound "^1.0.2" + es-errors "^1.3.0" + get-intrinsic "^1.2.5" + object-inspect "^1.13.3" + +side-channel-weakmap@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/side-channel-weakmap/-/side-channel-weakmap-1.0.2.tgz#11dda19d5368e40ce9ec2bdc1fb0ecbc0790ecea" + integrity sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A== + dependencies: + call-bound "^1.0.2" + es-errors "^1.3.0" + get-intrinsic "^1.2.5" + object-inspect "^1.13.3" + side-channel-map "^1.0.1" + +side-channel@^1.0.6, side-channel@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/side-channel/-/side-channel-1.1.0.tgz#c3fcff9c4da932784873335ec9765fa94ff66bc9" + integrity sha512-ZX99e6tRweoUXqR+VBrslhda51Nh5MTQwou5tnUDgbtyM0dBgmhEDtWGP/xbKn6hqfPRHujUNwz5fy/wbbhnpw== + dependencies: + es-errors "^1.3.0" + object-inspect "^1.13.3" + side-channel-list "^1.0.0" + side-channel-map "^1.0.1" + side-channel-weakmap "^1.0.2" + +siginfo@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/siginfo/-/siginfo-2.0.0.tgz#32e76c70b79724e3bb567cb9d543eb858ccfaf30" + integrity sha512-ybx0WO1/8bSBLEWXZvEd7gMW3Sn3JFlW3TvX1nREbDLRNQNaeNN8WK0meBwPdAaOI7TtRRRJn/Es1zhrrCHu7g== + +signal-exit@^3.0.0, signal-exit@^3.0.2: + version "3.0.7" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9" + integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ== + +signal-exit@^4.0.1, signal-exit@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-4.1.0.tgz#952188c1cbd546070e2dd20d0f41c0ae0530cb04" + integrity sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw== + +simple-concat@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/simple-concat/-/simple-concat-1.0.1.tgz#f46976082ba35c2263f1c8ab5edfe26c41c9552f" + integrity sha512-cSFtAPtRhljv69IK0hTVZQ+OfE9nePi/rtJmw5UjHeVyVroEqJXP1sFztKUy1qU+xvz3u/sfYJLa947b7nAN2Q== + +simple-get@^4.0.0: + version "4.0.1" + resolved "https://registry.yarnpkg.com/simple-get/-/simple-get-4.0.1.tgz#4a39db549287c979d352112fa03fd99fd6bc3543" + integrity sha512-brv7p5WgH0jmQJr1ZDDfKDOSeWWg+OVypG99A/5vYGPqJ6pxiaHLy8nxtFjBA7oMa01ebA9gfh1uMCFqOuXxvA== + dependencies: + decompress-response "^6.0.0" + once "^1.3.1" + simple-concat "^1.0.0" + +simple-update-notifier@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/simple-update-notifier/-/simple-update-notifier-2.0.0.tgz#d70b92bdab7d6d90dfd73931195a30b6e3d7cebb" + integrity sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w== + dependencies: + semver "^7.5.3" + +sketch-helpers@^0.0.4: + version "0.0.4" + resolved "https://registry.yarnpkg.com/sketch-helpers/-/sketch-helpers-0.0.4.tgz#c6e4257451cd65483ab99ff7d3b10da04e98374d" + integrity sha512-xSt+Ku4VFDk4fBW3kRj+raZ49fFSJ32q1ph05GKQvZ9mIUI+W2/3iJJSBfBWwIdxlNiMx6RoUe2O+5vwtkPT3A== + +slash@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" + integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== + +slice-ansi@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-3.0.0.tgz#31ddc10930a1b7e0b67b08c96c2f49b77a789787" + integrity sha512-pSyv7bSTC7ig9Dcgbw9AuRNUb5k5V6oDudjZoMBSr13qpLBG7tB+zgCkARjq7xIUgdz5P1Qe8u+rSGdouOOIyQ== + dependencies: + ansi-styles "^4.0.0" + astral-regex "^2.0.0" + is-fullwidth-code-point "^3.0.0" + +slice-ansi@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-5.0.0.tgz#b73063c57aa96f9cd881654b15294d95d285c42a" + integrity sha512-FC+lgizVPfie0kkhqUScwRu1O/lF6NOgJmlCgK+/LYxDCTk8sGelYaHDhFcDN+Sn3Cv+3VSa4Byeo+IMCzpMgQ== + dependencies: + ansi-styles "^6.0.0" + is-fullwidth-code-point "^4.0.0" + +smart-buffer@^4.0.2, smart-buffer@^4.2.0: + version "4.2.0" + resolved "https://registry.yarnpkg.com/smart-buffer/-/smart-buffer-4.2.0.tgz#6e1d71fa4f18c05f7d0ff216dd16a481d0e8d9ae" + integrity sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg== + +socks-proxy-agent@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/socks-proxy-agent/-/socks-proxy-agent-7.0.0.tgz#dc069ecf34436621acb41e3efa66ca1b5fed15b6" + integrity sha512-Fgl0YPZ902wEsAyiQ+idGd1A7rSFx/ayC1CQVMw5P+EQx2V0SgpGtf6OKFhVjPflPUl9YMmEOnmfjCdMUsygww== + dependencies: + agent-base "^6.0.2" + debug "^4.3.3" + socks "^2.6.2" + +socks@^2.6.2: + version "2.8.4" + resolved "https://registry.yarnpkg.com/socks/-/socks-2.8.4.tgz#07109755cdd4da03269bda4725baa061ab56d5cc" + integrity sha512-D3YaD0aRxR3mEcqnidIs7ReYJFVzWdd6fXJYUM8ixcQcJRGTka/b3saV0KflYhyVJXKhb947GndU35SxYNResQ== + dependencies: + ip-address "^9.0.5" + smart-buffer "^4.2.0" + +source-map-js@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.2.1.tgz#1ce5650fddd87abc099eda37dcff024c2667ae46" + integrity sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA== + +source-map-support@^0.5.13, source-map-support@^0.5.19: + version "0.5.21" + resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.21.tgz#04fe7c7f9e1ed2d662233c28cb2b35b9f63f6e4f" + integrity sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w== + dependencies: + buffer-from "^1.0.0" + source-map "^0.6.0" + +source-map@^0.6.0, source-map@~0.6.1: + version "0.6.1" + resolved "https://registry.yarnpkg.com/source-map/-/source-map-0.6.1.tgz#74722af32e9614e9c287a8d0bbde48b5e2f1a263" + integrity sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g== + +spdx-correct@^3.0.0: + version "3.2.0" + resolved "https://registry.yarnpkg.com/spdx-correct/-/spdx-correct-3.2.0.tgz#4f5ab0668f0059e34f9c00dce331784a12de4e9c" + integrity sha512-kN9dJbvnySHULIluDHy32WHRUu3Og7B9sbY7tsFLctQkIqnMh3hErYgdMjTYuqmcXX+lK5T1lnUt3G7zNswmZA== + dependencies: + spdx-expression-parse "^3.0.0" + spdx-license-ids "^3.0.0" + +spdx-exceptions@^2.1.0: + version "2.5.0" + resolved "https://registry.yarnpkg.com/spdx-exceptions/-/spdx-exceptions-2.5.0.tgz#5d607d27fc806f66d7b64a766650fa890f04ed66" + integrity sha512-PiU42r+xO4UbUS1buo3LPJkjlO7430Xn5SVAhdpzzsPHsjbYVflnnFdATgabnLude+Cqu25p6N+g2lw/PFsa4w== + +spdx-expression-parse@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/spdx-expression-parse/-/spdx-expression-parse-3.0.1.tgz#cf70f50482eefdc98e3ce0a6833e4a53ceeba679" + integrity sha512-cbqHunsQWnJNE6KhVSMsMeH5H/L9EpymbzqTQ3uLwNCLZ1Q481oWaofqH7nO6V07xlXwY6PhQdQ2IedWx/ZK4Q== + dependencies: + spdx-exceptions "^2.1.0" + spdx-license-ids "^3.0.0" + +spdx-license-ids@^3.0.0: + version "3.0.21" + resolved "https://registry.yarnpkg.com/spdx-license-ids/-/spdx-license-ids-3.0.21.tgz#6d6e980c9df2b6fc905343a3b2d702a6239536c3" + integrity sha512-Bvg/8F5XephndSK3JffaRqdT+gyhfqIPwDHpX80tJrF8QQRYMo8sNMeaZ2Dp5+jhwKnUmIOyFFQfHRkjJm5nXg== + +split@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/split/-/split-1.0.1.tgz#605bd9be303aa59fb35f9229fbea0ddec9ea07d9" + integrity sha512-mTyOoPbrivtXnwnIxZRFYRrPNtEFKlpB2fvjSnCQUiAA6qAZzqwna5envK4uk6OIeP17CsdF3rSBGYVBsU0Tkg== + dependencies: + through "2" + +sprintf-js@^1.1.2, sprintf-js@^1.1.3: + version "1.1.3" + resolved "https://registry.yarnpkg.com/sprintf-js/-/sprintf-js-1.1.3.tgz#4914b903a2f8b685d17fdf78a70e917e872e444a" + integrity sha512-Oo+0REFV59/rz3gfJNKQiBlwfHaSESl1pcGyABQsnnIfWOFt6JNj5gCog2U6MLZ//IGYD+nA8nI+mTShREReaA== + +ssri@^9.0.0: + version "9.0.1" + resolved "https://registry.yarnpkg.com/ssri/-/ssri-9.0.1.tgz#544d4c357a8d7b71a19700074b6883fcb4eae057" + integrity sha512-o57Wcn66jMQvfHG1FlYbWeZWW/dHZhJXjpIcTfXldXEk5nz5lStPo3mK0OJQfGR3RbZUlbISexbljkJzuEj/8Q== + dependencies: + minipass "^3.1.1" + +stack-utils@^2.0.3: + version "2.0.6" + resolved "https://registry.yarnpkg.com/stack-utils/-/stack-utils-2.0.6.tgz#aaf0748169c02fc33c8232abccf933f54a1cc34f" + integrity sha512-XlkWvfIm6RmsWtNJx+uqtKLS8eqFbxUg0ZzLXqY0caEy9l7hruX8IpiDnjsLavoBgqCCR71TqWO8MaXYheJ3RQ== + dependencies: + escape-string-regexp "^2.0.0" + +stackback@0.0.2: + version "0.0.2" + resolved "https://registry.yarnpkg.com/stackback/-/stackback-0.0.2.tgz#1ac8a0d9483848d1695e418b6d031a3c3ce68e3b" + integrity sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw== + +stat-mode@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/stat-mode/-/stat-mode-1.0.0.tgz#68b55cb61ea639ff57136f36b216a291800d1465" + integrity sha512-jH9EhtKIjuXZ2cWxmXS8ZP80XyC3iasQxMDV8jzhNJpfDb7VbQLVW4Wvsxz9QZvzV+G4YoSfBUVKDOyxLzi/sg== + +statuses@2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63" + integrity sha512-RwNA9Z/7PrK06rYLIzFMlaF+l73iwpzsqRIFgbMLbTcLD6cOao82TaWefPXQvB2fOC4AjuYSEndS7N/mTCbkdQ== + +std-env@^3.5.0, std-env@^3.9.0: + version "3.9.0" + resolved "https://registry.yarnpkg.com/std-env/-/std-env-3.9.0.tgz#1a6f7243b339dca4c9fd55e1c7504c77ef23e8f1" + integrity sha512-UGvjygr6F6tpH7o2qyqR6QYpwraIjKSdtzyBdyytFOHmPZY917kwdwLG0RbOjWOnKmnm3PeHjaoLLMie7kPLQw== + +stdin-discarder@^0.2.2: + version "0.2.2" + resolved "https://registry.yarnpkg.com/stdin-discarder/-/stdin-discarder-0.2.2.tgz#390037f44c4ae1a1ae535c5fe38dc3aba8d997be" + integrity sha512-UhDfHmA92YAlNnCfhmq0VeNL5bDbiZGg7sZ2IvPsXubGkiNa9EC+tUTsjBRsYUAz87btI6/1wf4XoVvQ3uRnmQ== + +"string-width-cjs@npm:string-width@^4.2.0": + version "4.2.3" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" + integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== + dependencies: + emoji-regex "^8.0.0" + is-fullwidth-code-point "^3.0.0" + strip-ansi "^6.0.1" + +string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: + version "4.2.3" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" + integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== + dependencies: + emoji-regex "^8.0.0" + is-fullwidth-code-point "^3.0.0" + strip-ansi "^6.0.1" + +string-width@^5.0.0, string-width@^5.0.1, string-width@^5.1.2: + version "5.1.2" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-5.1.2.tgz#14f8daec6d81e7221d2a357e668cab73bdbca794" + integrity sha512-HnLOCR3vjcY8beoNLtcjZ5/nxn2afmME6lhrDrebokqMap+XbeW8n9TXpPDOqdGK5qcI3oT0GKTW6wC7EMiVqA== + dependencies: + eastasianwidth "^0.2.0" + emoji-regex "^9.2.2" + strip-ansi "^7.0.1" + +string-width@^7.2.0: + version "7.2.0" + resolved "https://registry.yarnpkg.com/string-width/-/string-width-7.2.0.tgz#b5bb8e2165ce275d4d43476dd2700ad9091db6dc" + integrity sha512-tsaTIkKW9b4N+AEj+SVA+WhJzV7/zMhcSu78mLKWSk7cXMOSHsBKFWUs0fWwq8QyK3MgJBQRX6Gbi4kYbdvGkQ== + dependencies: + emoji-regex "^10.3.0" + get-east-asian-width "^1.0.0" + strip-ansi "^7.1.0" + +string.prototype.includes@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/string.prototype.includes/-/string.prototype.includes-2.0.1.tgz#eceef21283640761a81dbe16d6c7171a4edf7d92" + integrity sha512-o7+c9bW6zpAdJHTtujeePODAhkuicdAryFsfVKwA+wGw89wJ4GTY484WTucM9hLtDEOpOvI+aHnzqnC5lHp4Rg== + dependencies: + call-bind "^1.0.7" + define-properties "^1.2.1" + es-abstract "^1.23.3" + +string.prototype.matchall@^4.0.12: + version "4.0.12" + resolved "https://registry.yarnpkg.com/string.prototype.matchall/-/string.prototype.matchall-4.0.12.tgz#6c88740e49ad4956b1332a911e949583a275d4c0" + integrity sha512-6CC9uyBL+/48dYizRf7H7VAYCMCNTBeM78x/VTUe9bFEaxBepPJDa1Ow99LqI/1yF7kuy7Q3cQsYMrcjGUcskA== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.3" + define-properties "^1.2.1" + es-abstract "^1.23.6" + es-errors "^1.3.0" + es-object-atoms "^1.0.0" + get-intrinsic "^1.2.6" + gopd "^1.2.0" + has-symbols "^1.1.0" + internal-slot "^1.1.0" + regexp.prototype.flags "^1.5.3" + set-function-name "^2.0.2" + side-channel "^1.1.0" + +string.prototype.repeat@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/string.prototype.repeat/-/string.prototype.repeat-1.0.0.tgz#e90872ee0308b29435aa26275f6e1b762daee01a" + integrity sha512-0u/TldDbKD8bFCQ/4f5+mNRrXwZ8hg2w7ZR8wa16e8z9XpePWl3eGEcUD0OXpEH/VJH/2G3gjUtR3ZOiBe2S/w== + dependencies: + define-properties "^1.1.3" + es-abstract "^1.17.5" + +string.prototype.trim@^1.2.10: + version "1.2.10" + resolved "https://registry.yarnpkg.com/string.prototype.trim/-/string.prototype.trim-1.2.10.tgz#40b2dd5ee94c959b4dcfb1d65ce72e90da480c81" + integrity sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.2" + define-data-property "^1.1.4" + define-properties "^1.2.1" + es-abstract "^1.23.5" + es-object-atoms "^1.0.0" + has-property-descriptors "^1.0.2" + +string.prototype.trimend@^1.0.8, string.prototype.trimend@^1.0.9: + version "1.0.9" + resolved "https://registry.yarnpkg.com/string.prototype.trimend/-/string.prototype.trimend-1.0.9.tgz#62e2731272cd285041b36596054e9f66569b6942" + integrity sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ== + dependencies: + call-bind "^1.0.8" + call-bound "^1.0.2" + define-properties "^1.2.1" + es-object-atoms "^1.0.0" + +string.prototype.trimstart@^1.0.8: + version "1.0.8" + resolved "https://registry.yarnpkg.com/string.prototype.trimstart/-/string.prototype.trimstart-1.0.8.tgz#7ee834dda8c7c17eff3118472bb35bfedaa34dde" + integrity sha512-UXSH262CSZY1tfu3G3Secr6uGLCFVPMhIqHjlgCUtCCcgihYc/xKs9djMTMUOb2j1mVSeU8EU6NWc/iQKU6Gfg== + dependencies: + call-bind "^1.0.7" + define-properties "^1.2.1" + es-object-atoms "^1.0.0" + +string_decoder@^1.1.1: + version "1.3.0" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" + integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== + dependencies: + safe-buffer "~5.2.0" + +string_decoder@~1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.1.1.tgz#9cf1611ba62685d7030ae9e4ba34149c3af03fc8" + integrity sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg== + dependencies: + safe-buffer "~5.1.0" + +"strip-ansi-cjs@npm:strip-ansi@^6.0.1": + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + +strip-ansi@^6.0.0, strip-ansi@^6.0.1: + version "6.0.1" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" + integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== + dependencies: + ansi-regex "^5.0.1" + +strip-ansi@^7.0.1, strip-ansi@^7.1.0: + version "7.1.0" + resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-7.1.0.tgz#d5b6568ca689d8561370b0707685d22434faff45" + integrity sha512-iq6eVVI64nQQTRYq2KtEg2d2uU7LElhTJwsH4YzIHZshxlgZms/wIc4VoDQTlG/IvVIrBKG06CrZnp0qv7hkcQ== + dependencies: + ansi-regex "^6.0.1" + +strip-bom@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-bom/-/strip-bom-3.0.0.tgz#2334c18e9c759f7bdd56fdef7e9ae3d588e68ed3" + integrity sha512-vavAMRXOgBVNF6nyEEmL3DBK19iRpDcoIwW+swQ+CbGiu7lju6t+JklA1MHweoWtadgt4ISVUsXLyDq34ddcwA== + +strip-eof@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/strip-eof/-/strip-eof-1.0.0.tgz#bb43ff5598a6eb05d89b59fcd129c983313606bf" + integrity sha512-7FCwGGmx8mD5xQd3RPUvnSpUXHM3BWuzjtpD4TXsfcZ9EL4azvVVUscFYwD9nx8Kh+uCBC00XBtAykoMHwTh8Q== + +strip-final-newline@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-final-newline/-/strip-final-newline-3.0.0.tgz#52894c313fbff318835280aed60ff71ebf12b8fd" + integrity sha512-dOESqjYr96iWYylGObzd39EuNTa5VJxyvVAEm5Jnh7KGo75V43Hk1odPQkNDyXNmUR6k+gEiDVXnjB8HJ3crXw== + +strip-indent@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/strip-indent/-/strip-indent-3.0.0.tgz#c32e1cee940b6b3432c771bc2c54bcce73cd3001" + integrity sha512-laJTa3Jb+VQpaC6DseHhF7dXVqHTfJPCRDaEbid/drOhgitgYku/letMUqOXFoWV0zIIUbjpdH2t+tYj4bQMRQ== + dependencies: + min-indent "^1.0.0" + +strip-json-comments@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006" + integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig== + +strip-json-comments@~2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a" + integrity sha512-4gB8na07fecVVkOI6Rs4e7T6NOTki5EmL7TUduTs6bu3EdnSycntVJ4re8kgZA+wx9IueI2Y11bfbgwtzuE0KQ== + +strip-literal@^2.0.0: + version "2.1.1" + resolved "https://registry.yarnpkg.com/strip-literal/-/strip-literal-2.1.1.tgz#26906e65f606d49f748454a08084e94190c2e5ad" + integrity sha512-631UJ6O00eNGfMiWG78ck80dfBab8X6IVFB51jZK5Icd7XAs60Z5y7QdSd/wGIklnWvRbUNloVzhOKKmutxQ6Q== + dependencies: + js-tokens "^9.0.1" + +strip-outer@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/strip-outer/-/strip-outer-1.0.1.tgz#b2fd2abf6604b9d1e6013057195df836b8a9d631" + integrity sha512-k55yxKHwaXnpYGsOzg4Vl8+tDrWylxDEpknGjhTiZB8dFRU5rTo9CAzeycivxV3s+zlTKwrs6WxMxR95n26kwg== + dependencies: + escape-string-regexp "^1.0.2" + +style-mod@^4.0.0, style-mod@^4.1.0: + version "4.1.2" + resolved "https://registry.yarnpkg.com/style-mod/-/style-mod-4.1.2.tgz#ca238a1ad4786520f7515a8539d5a63691d7bf67" + integrity sha512-wnD1HyVqpJUI2+eKZ+eo1UwghftP6yuFheBqqe+bWCotBjC2K1YnteJILRMs3SM4V/0dLEW1SC27MWP5y+mwmw== + +sucrase@^3.35.0: + version "3.35.0" + resolved "https://registry.yarnpkg.com/sucrase/-/sucrase-3.35.0.tgz#57f17a3d7e19b36d8995f06679d121be914ae263" + integrity sha512-8EbVDiu9iN/nESwxeSxDKe0dunta1GOlHufmSSXxMD2z2/tMZpDMpvXQGsc+ajGo8y2uYUmixaSRUc/QPoQ0GA== + dependencies: + "@jridgewell/gen-mapping" "^0.3.2" + commander "^4.0.0" + glob "^10.3.10" + lines-and-columns "^1.1.6" + mz "^2.7.0" + pirates "^4.0.1" + ts-interface-checker "^0.1.9" + +sudo-prompt@^9.1.1: + version "9.2.1" + resolved "https://registry.yarnpkg.com/sudo-prompt/-/sudo-prompt-9.2.1.tgz#77efb84309c9ca489527a4e749f287e6bdd52afd" + integrity sha512-Mu7R0g4ig9TUuGSxJavny5Rv0egCEtpZRNMrZaYS1vxkiIxGiGUwoezU3LazIQ+KE04hTrTfNPgxU5gzi7F5Pw== + +sumchecker@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/sumchecker/-/sumchecker-3.0.1.tgz#6377e996795abb0b6d348e9b3e1dfb24345a8e42" + integrity sha512-MvjXzkz/BOfyVDkG0oFOtBxHX2u3gKbMHIF/dXblZsgD3BWOFLmHovIpZY7BykJdAjcqRCBi1WYBNdEC9yI7vg== + dependencies: + debug "^4.1.0" + +supports-color@^5.3.0: + version "5.5.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f" + integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow== + dependencies: + has-flag "^3.0.0" + +supports-color@^7.1.0: + version "7.2.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" + integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== + dependencies: + has-flag "^4.0.0" + +supports-color@^8.1.1: + version "8.1.1" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c" + integrity sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q== + dependencies: + has-flag "^4.0.0" + +supports-color@^9.4.0: + version "9.4.0" + resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-9.4.0.tgz#17bfcf686288f531db3dea3215510621ccb55954" + integrity sha512-VL+lNrEoIXww1coLPOmiEmK/0sGigko5COxI09KzHc2VJXJsQ37UaQ+8quuxjDeA7+KnLGTWRyOXSLLR2Wb4jw== + +supports-preserve-symlinks-flag@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09" + integrity sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w== + +tailwindcss@^3.4.17: + version "3.4.17" + resolved "https://registry.yarnpkg.com/tailwindcss/-/tailwindcss-3.4.17.tgz#ae8406c0f96696a631c790768ff319d46d5e5a63" + integrity sha512-w33E2aCvSDP0tW9RZuNXadXlkHXqFzSkQew/aIa2i/Sj8fThxwovwlXHSPXTbAHwEIhBFXAedUhP2tueAKP8Og== + dependencies: + "@alloc/quick-lru" "^5.2.0" + arg "^5.0.2" + chokidar "^3.6.0" + didyoumean "^1.2.2" + dlv "^1.1.3" + fast-glob "^3.3.2" + glob-parent "^6.0.2" + is-glob "^4.0.3" + jiti "^1.21.6" + lilconfig "^3.1.3" + micromatch "^4.0.8" + normalize-path "^3.0.0" + object-hash "^3.0.0" + picocolors "^1.1.1" + postcss "^8.4.47" + postcss-import "^15.1.0" + postcss-js "^4.0.1" + postcss-load-config "^4.0.2" + postcss-nested "^6.2.0" + postcss-selector-parser "^6.1.2" + resolve "^1.22.8" + sucrase "^3.35.0" + +tar-fs@^2.0.0: + version "2.1.2" + resolved "https://registry.yarnpkg.com/tar-fs/-/tar-fs-2.1.2.tgz#425f154f3404cb16cb8ff6e671d45ab2ed9596c5" + integrity sha512-EsaAXwxmx8UB7FRKqeozqEPop69DXcmYwTQwXvyAPF352HJsPdkVhvTaDPYqfNgruveJIJy3TA2l+2zj8LJIJA== + dependencies: + chownr "^1.1.1" + mkdirp-classic "^0.5.2" + pump "^3.0.0" + tar-stream "^2.1.4" + +tar-stream@^2.1.4: + version "2.2.0" + resolved "https://registry.yarnpkg.com/tar-stream/-/tar-stream-2.2.0.tgz#acad84c284136b060dc3faa64474aa9aebd77287" + integrity sha512-ujeqbceABgwMZxEJnk2HDY2DlnUZ+9oEcb1KzTVfYHio0UE6dG71n60d8D2I4qNvleWrrXpmjpt7vZeF1LnMZQ== + dependencies: + bl "^4.0.3" + end-of-stream "^1.4.1" + fs-constants "^1.0.0" + inherits "^2.0.3" + readable-stream "^3.1.1" + +tar@^6.0.5, tar@^6.1.11, tar@^6.1.12, tar@^6.2.1: + version "6.2.1" + resolved "https://registry.yarnpkg.com/tar/-/tar-6.2.1.tgz#717549c541bc3c2af15751bea94b1dd068d4b03a" + integrity sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A== + dependencies: + chownr "^2.0.0" + fs-minipass "^2.0.0" + minipass "^5.0.0" + minizlib "^2.1.1" + mkdirp "^1.0.3" + yallist "^4.0.0" + +temp-file@^3.4.0: + version "3.4.0" + resolved "https://registry.yarnpkg.com/temp-file/-/temp-file-3.4.0.tgz#766ea28911c683996c248ef1a20eea04d51652c7" + integrity sha512-C5tjlC/HCtVUOi3KWVokd4vHVViOmGjtLwIh4MuzPo/nMYTV/p1urt3RnMz2IWXDdKEGJH3k5+KPxtqRsUYGtg== + dependencies: + async-exit-hook "^2.0.1" + fs-extra "^10.0.0" + +text-segmentation@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/text-segmentation/-/text-segmentation-1.0.3.tgz#52a388159efffe746b24a63ba311b6ac9f2d7943" + integrity sha512-iOiPUo/BGnZ6+54OsWxZidGCsdU8YbE4PSpdPinp7DeMtUJNJBoJ/ouUSTJjHkh1KntHaltHl/gDs2FC4i5+Nw== + dependencies: + utrie "^1.0.2" + +text-table@^0.2.0: + version "0.2.0" + resolved "https://registry.yarnpkg.com/text-table/-/text-table-0.2.0.tgz#7f5ee823ae805207c00af2df4a84ec3fcfa570b4" + integrity sha512-N+8UisAXDGk8PFXP4HAzVR9nbfmVJ3zYLAWiTIoqC5v5isinhr+r5uaO8+7r3BMfuNIufIsA7RdpVgacC2cSpw== + +thenify-all@^1.0.0: + version "1.6.0" + resolved "https://registry.yarnpkg.com/thenify-all/-/thenify-all-1.6.0.tgz#1a1918d402d8fc3f98fbf234db0bcc8cc10e9726" + integrity sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA== + dependencies: + thenify ">= 3.1.0 < 4" + +"thenify@>= 3.1.0 < 4": + version "3.3.1" + resolved "https://registry.yarnpkg.com/thenify/-/thenify-3.3.1.tgz#8932e686a4066038a016dd9e2ca46add9838a95f" + integrity sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw== + dependencies: + any-promise "^1.0.0" + +three@^0.175.0: + version "0.175.0" + resolved "https://registry.yarnpkg.com/three/-/three-0.175.0.tgz#67b357b0b1ee8ef0445b9a768f59363ab1fa7921" + integrity sha512-nNE3pnTHxXN/Phw768u0Grr7W4+rumGg/H6PgeseNJojkJtmeHJfZWi41Gp2mpXl1pg1pf1zjwR4McM1jTqkpg== + +through@2: + version "2.3.8" + resolved "https://registry.yarnpkg.com/through/-/through-2.3.8.tgz#0dd4c9ffaabc357960b1b724115d7e0e86a2e1f5" + integrity sha512-w89qg7PI8wAdvX60bMDP+bFoD5Dvhm9oLheFp5O4a2QF0cSBGsBX4qZmadPMvVqlLJBBci+WqGGOAPvcDeNSVg== + +thunky@^1.0.2: + version "1.1.0" + resolved "https://registry.yarnpkg.com/thunky/-/thunky-1.1.0.tgz#5abaf714a9405db0504732bbccd2cedd9ef9537d" + integrity sha512-eHY7nBftgThBqOyHGVN+l8gF0BucP09fMo0oO/Lb0w1OF80dJv+lDVpXG60WMQvkcxAkNybKsrEIE3ZtKGmPrA== + +tiny-async-pool@1.3.0: + version "1.3.0" + resolved "https://registry.yarnpkg.com/tiny-async-pool/-/tiny-async-pool-1.3.0.tgz#c013e1b369095e7005db5595f95e646cca6ef8a5" + integrity sha512-01EAw5EDrcVrdgyCLgoSPvqznC0sVxDSVeiOz09FUpjh71G79VCqneOr+xvt7T1r76CF6ZZfPjHorN2+d+3mqA== + dependencies: + semver "^5.5.0" + +tiny-invariant@^1.3.3: + version "1.3.3" + resolved "https://registry.yarnpkg.com/tiny-invariant/-/tiny-invariant-1.3.3.tgz#46680b7a873a0d5d10005995eb90a70d74d60127" + integrity sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg== + +tiny-typed-emitter@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/tiny-typed-emitter/-/tiny-typed-emitter-2.1.0.tgz#b3b027fdd389ff81a152c8e847ee2f5be9fad7b5" + integrity sha512-qVtvMxeXbVej0cQWKqVSSAHmKZEHAvxdF8HEUBFWts8h+xEo5m/lEiPakuyZ3BnCBjOD8i24kzNOiOLLgsSxhA== + +tinybench@^2.5.1, tinybench@^2.9.0: + version "2.9.0" + resolved "https://registry.yarnpkg.com/tinybench/-/tinybench-2.9.0.tgz#103c9f8ba6d7237a47ab6dd1dcff77251863426b" + integrity sha512-0+DUvqWMValLmha6lr4kD8iAMK1HzV0/aKnCtWb9v9641TnP/MFb7Pc2bxoxQjTXAErryXVgUOfv2YqNllqGeg== + +tinyexec@^0.3.2: + version "0.3.2" + resolved "https://registry.yarnpkg.com/tinyexec/-/tinyexec-0.3.2.tgz#941794e657a85e496577995c6eef66f53f42b3d2" + integrity sha512-KQQR9yN7R5+OSwaK0XQoj22pwHoTlgYqmUscPYoknOoWCWfj/5/ABTMRi69FrKU5ffPVh5QcFikpWJI/P1ocHA== + +tinyglobby@^0.2.13: + version "0.2.13" + resolved "https://registry.yarnpkg.com/tinyglobby/-/tinyglobby-0.2.13.tgz#a0e46515ce6cbcd65331537e57484af5a7b2ff7e" + integrity sha512-mEwzpUgrLySlveBwEVDMKk5B57bhLPYovRfPAXD5gA/98Opn0rCDj3GtLwFvCvH5RK9uPCExUROW5NjDwvqkxw== + dependencies: + fdir "^6.4.4" + picomatch "^4.0.2" + +tinypool@^0.8.3: + version "0.8.4" + resolved "https://registry.yarnpkg.com/tinypool/-/tinypool-0.8.4.tgz#e217fe1270d941b39e98c625dcecebb1408c9aa8" + integrity sha512-i11VH5gS6IFeLY3gMBQ00/MmLncVP7JLXOw1vlgkytLmJK7QnEr7NXf0LBdxfmNPAeyetukOk0bOYrJrFGjYJQ== + +tinypool@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/tinypool/-/tinypool-1.0.2.tgz#706193cc532f4c100f66aa00b01c42173d9051b2" + integrity sha512-al6n+QEANGFOMf/dmUMsuS5/r9B06uwlyNjZZql/zv8J7ybHCgoihBNORZCY2mzUuAnomQa2JdhyHKzZxPCrFA== + +tinyrainbow@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/tinyrainbow/-/tinyrainbow-2.0.0.tgz#9509b2162436315e80e3eee0fcce4474d2444294" + integrity sha512-op4nsTR47R6p0vMUUoYl/a+ljLFVtlfaXkLQmqfLR1qHma1h/ysYk4hEXZ880bf2CYgTskvTa/e196Vd5dDQXw== + +tinyspy@^2.2.0: + version "2.2.1" + resolved "https://registry.yarnpkg.com/tinyspy/-/tinyspy-2.2.1.tgz#117b2342f1f38a0dbdcc73a50a454883adf861d1" + integrity sha512-KYad6Vy5VDWV4GH3fjpseMQ/XU2BhIYP7Vzd0LG44qRWm/Yt2WCOTicFdvmgo6gWaqooMQCawTtILVQJupKu7A== + +tinyspy@^3.0.2: + version "3.0.2" + resolved "https://registry.yarnpkg.com/tinyspy/-/tinyspy-3.0.2.tgz#86dd3cf3d737b15adcf17d7887c84a75201df20a" + integrity sha512-n1cw8k1k0x4pgA2+9XrOkFydTerNcJ1zWCO5Nn9scWHTD+5tp8dghT2x1uduQePZTZgd3Tupf+x9BxJjeJi77Q== + +tmp-promise@^3.0.2: + version "3.0.3" + resolved "https://registry.yarnpkg.com/tmp-promise/-/tmp-promise-3.0.3.tgz#60a1a1cc98c988674fcbfd23b6e3367bdeac4ce7" + integrity sha512-RwM7MoPojPxsOBYnyd2hy0bxtIlVrihNs9pj5SUvY8Zz1sQcQG2tG1hSr8PDxfgEB8RNKDhqbIlroIarSNDNsQ== + dependencies: + tmp "^0.2.0" + +tmp@^0.2.0, tmp@^0.2.3: + version "0.2.3" + resolved "https://registry.yarnpkg.com/tmp/-/tmp-0.2.3.tgz#eb783cc22bc1e8bebd0671476d46ea4eb32a79ae" + integrity sha512-nZD7m9iCPC5g0pYmcaxogYKggSfLsdxl8of3Q/oIbqCqLLIO9IAF0GWjX1z9NZRHPiXv8Wex4yDCaZsgEw0Y8w== + +to-regex-range@^5.0.1: + version "5.0.1" + resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" + integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== + dependencies: + is-number "^7.0.0" + +toidentifier@1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/toidentifier/-/toidentifier-1.0.1.tgz#3be34321a88a820ed1bd80dfaa33e479fbb8dd35" + integrity sha512-o5sSPKEkg/DIQNmH43V0/uerLrpzVedkUh8tGNvaeXpfpuwjKenlSox/2O/BTlZUtEe+JG7s5YhEz608PlAHRA== + +tr46@~0.0.3: + version "0.0.3" + resolved "https://registry.yarnpkg.com/tr46/-/tr46-0.0.3.tgz#8184fd347dac9cdc185992f3a6622e14b9d9ab6a" + integrity sha512-N3WMsuqV66lT30CrXNbEjx4GEwlow3v6rr4mCcv6prnfwhS01rkgyFdjPNBYd9br7LpXV1+Emh01fHnq2Gdgrw== + +trim-repeated@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/trim-repeated/-/trim-repeated-1.0.0.tgz#e3646a2ea4e891312bf7eace6cfb05380bc01c21" + integrity sha512-pkonvlKk8/ZuR0D5tLW8ljt5I8kmxp2XKymhepUeOdCEfKpZaktSArkLHZt76OB1ZvO9bssUsDty4SWhLvZpLg== + dependencies: + escape-string-regexp "^1.0.2" + +truncate-utf8-bytes@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/truncate-utf8-bytes/-/truncate-utf8-bytes-1.0.2.tgz#405923909592d56f78a5818434b0b78489ca5f2b" + integrity sha512-95Pu1QXQvruGEhv62XCMO3Mm90GscOCClvrIUwCM0PYOXK3kaF3l3sIHxx71ThJfcbM2O5Au6SO3AWCSEfW4mQ== + dependencies: + utf8-byte-length "^1.0.1" + +ts-api-utils@^2.0.1: + version "2.1.0" + resolved "https://registry.yarnpkg.com/ts-api-utils/-/ts-api-utils-2.1.0.tgz#595f7094e46eed364c13fd23e75f9513d29baf91" + integrity sha512-CUgTZL1irw8u29bzrOD/nH85jqyc74D6SshFgujOIA7osm2Rz7dYH77agkx7H4FBNxDq7Cjf+IjaX/8zwFW+ZQ== + +ts-interface-checker@^0.1.9: + version "0.1.13" + resolved "https://registry.yarnpkg.com/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz#784fd3d679722bc103b1b4b8030bcddb5db2a699" + integrity sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA== + +ts-node@^10.0.0, ts-node@^10.9.1, ts-node@^10.9.2: + version "10.9.2" + resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-10.9.2.tgz#70f021c9e185bccdca820e26dc413805c101c71f" + integrity sha512-f0FFpIdcHgn8zcPSbf1dRevwt047YMnaiJM3u2w2RewrB+fob/zePZcrOyQoLMMO7aBIddLcQIEK5dYjkLnGrQ== + dependencies: + "@cspotcode/source-map-support" "^0.8.0" + "@tsconfig/node10" "^1.0.7" + "@tsconfig/node12" "^1.0.7" + "@tsconfig/node14" "^1.0.0" + "@tsconfig/node16" "^1.0.2" + acorn "^8.4.1" + acorn-walk "^8.1.1" + arg "^4.1.0" + create-require "^1.1.0" + diff "^4.0.1" + make-error "^1.1.1" + v8-compile-cache-lib "^3.0.1" + yn "3.1.1" + +tsconfck@^3.0.3: + version "3.1.5" + resolved "https://registry.yarnpkg.com/tsconfck/-/tsconfck-3.1.5.tgz#2f07f9be6576825e7a77470a5304ce06c7746e61" + integrity sha512-CLDfGgUp7XPswWnezWwsCRxNmgQjhYq3VXHM0/XIRxhVrKw0M1if9agzryh1QS3nxjCROvV+xWxoJO1YctzzWg== + +tsconfig-paths@^3.15.0: + version "3.15.0" + resolved "https://registry.yarnpkg.com/tsconfig-paths/-/tsconfig-paths-3.15.0.tgz#5299ec605e55b1abb23ec939ef15edaf483070d4" + integrity sha512-2Ac2RgzDe/cn48GvOe3M+o82pEFewD3UPbyoUHHdKasHwJKjds4fLXWf/Ux5kATBKN20oaFGu+jbElp1pos0mg== + dependencies: + "@types/json5" "^0.0.29" + json5 "^1.0.2" + minimist "^1.2.6" + strip-bom "^3.0.0" + +tslib@^2.0.1, tslib@^2.2.0, tslib@^2.3.0, tslib@^2.6.2, tslib@~2.8: + version "2.8.1" + resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.8.1.tgz#612efe4ed235d567e8aba5f2a5fab70280ade83f" + integrity sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w== + +tunnel-agent@^0.6.0: + version "0.6.0" + resolved "https://registry.yarnpkg.com/tunnel-agent/-/tunnel-agent-0.6.0.tgz#27a5dea06b36b04a0a9966774b290868f0fc40fd" + integrity sha512-McnNiV1l8RYeY8tBgEpuodCC1mLUdbSN+CYBL7kJsJNInOP8UjDDEwdk6Mw60vdLLrr5NHKZhMAOSrR2NZuQ+w== + dependencies: + safe-buffer "^5.0.1" + +tunnel@0.0.6: + version "0.0.6" + resolved "https://registry.yarnpkg.com/tunnel/-/tunnel-0.0.6.tgz#72f1314b34a5b192db012324df2cc587ca47f92c" + integrity sha512-1h/Lnq9yajKY2PEbBadPXj3VxsDDu844OnaAo52UVmIzIvwwtBPIuNvkjuzBlTWpfJyUbG3ez0KSBibQkj4ojg== + +type-check@^0.4.0, type-check@~0.4.0: + version "0.4.0" + resolved "https://registry.yarnpkg.com/type-check/-/type-check-0.4.0.tgz#07b8203bfa7056c0657050e3ccd2c37730bab8f1" + integrity sha512-XleUoc9uwGXqjWwXaUTZAmzMcFZ5858QA2vvx1Ur5xIcixXIP+8LnFDgRplU30us6teqdlskFfu+ae4K79Ooew== + dependencies: + prelude-ls "^1.2.1" + +type-detect@^4.0.0, type-detect@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/type-detect/-/type-detect-4.1.0.tgz#deb2453e8f08dcae7ae98c626b13dddb0155906c" + integrity sha512-Acylog8/luQ8L7il+geoSxhEkazvkslg7PSNKOX59mbB9cOveP5aq9h74Y7YU8yDpJwetzQQrfIwtf4Wp4LKcw== + +type-fest@^0.13.1: + version "0.13.1" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.13.1.tgz#0172cb5bce80b0bd542ea348db50c7e21834d934" + integrity sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg== + +type-fest@^0.20.2: + version "0.20.2" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.20.2.tgz#1bf207f4b28f91583666cb5fbd327887301cd5f4" + integrity sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ== + +type-fest@^1.0.2: + version "1.4.0" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-1.4.0.tgz#e9fb813fe3bf1744ec359d55d1affefa76f14be1" + integrity sha512-yGSza74xk0UG8k+pLh5oeoYirvIiWo5t0/o3zHHAO2tRDiZcxWP7fywNlXhqb6/r6sWvwi+RsyQMWhVLe4BVuA== + +type-fest@^4.39.1: + version "4.40.0" + resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-4.40.0.tgz#62bc09caccb99a75e1ad6b9b4653e8805e5e1eee" + integrity sha512-ABHZ2/tS2JkvH1PEjxFDTUWC8dB5OsIGZP4IFLhR293GqT5Y5qB1WwL2kMPYhQW9DVgVD8Hd7I8gjwPIf5GFkw== + +type-is@~1.6.18: + version "1.6.18" + resolved "https://registry.yarnpkg.com/type-is/-/type-is-1.6.18.tgz#4e552cd05df09467dcbc4ef739de89f2cf37c131" + integrity sha512-TkRKr9sUTxEH8MdfuCSP7VizJyzRNMjj2J2do2Jr3Kym598JVdEksuzPQCnlFPW4ky9Q+iA+ma9BGm06XQBy8g== + dependencies: + media-typer "0.3.0" + mime-types "~2.1.24" + +typed-array-buffer@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/typed-array-buffer/-/typed-array-buffer-1.0.3.tgz#a72395450a4869ec033fd549371b47af3a2ee536" + integrity sha512-nAYYwfY3qnzX30IkA6AQZjVbtK6duGontcQm1WSG1MD94YLqK0515GNApXkoxKOWMusVssAHWLh9SeaoefYFGw== + dependencies: + call-bound "^1.0.3" + es-errors "^1.3.0" + is-typed-array "^1.1.14" + +typed-array-byte-length@^1.0.3: + version "1.0.3" + resolved "https://registry.yarnpkg.com/typed-array-byte-length/-/typed-array-byte-length-1.0.3.tgz#8407a04f7d78684f3d252aa1a143d2b77b4160ce" + integrity sha512-BaXgOuIxz8n8pIq3e7Atg/7s+DpiYrxn4vdot3w9KbnBhcRQq6o3xemQdIfynqSeXeDrF32x+WvfzmOjPiY9lg== + dependencies: + call-bind "^1.0.8" + for-each "^0.3.3" + gopd "^1.2.0" + has-proto "^1.2.0" + is-typed-array "^1.1.14" + +typed-array-byte-offset@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/typed-array-byte-offset/-/typed-array-byte-offset-1.0.4.tgz#ae3698b8ec91a8ab945016108aef00d5bff12355" + integrity sha512-bTlAFB/FBYMcuX81gbL4OcpH5PmlFHqlCCpAl8AlEzMz5k53oNDvN8p1PNOWLEmI2x4orp3raOFB51tv9X+MFQ== + dependencies: + available-typed-arrays "^1.0.7" + call-bind "^1.0.8" + for-each "^0.3.3" + gopd "^1.2.0" + has-proto "^1.2.0" + is-typed-array "^1.1.15" + reflect.getprototypeof "^1.0.9" + +typed-array-length@^1.0.7: + version "1.0.7" + resolved "https://registry.yarnpkg.com/typed-array-length/-/typed-array-length-1.0.7.tgz#ee4deff984b64be1e118b0de8c9c877d5ce73d3d" + integrity sha512-3KS2b+kL7fsuk/eJZ7EQdnEmQoaho/r6KUef7hxvltNA5DR8NAUM+8wJMbJyZ4G9/7i3v5zPBIMN5aybAh2/Jg== + dependencies: + call-bind "^1.0.7" + for-each "^0.3.3" + gopd "^1.0.1" + is-typed-array "^1.1.13" + possible-typed-array-names "^1.0.0" + reflect.getprototypeof "^1.0.6" + +typed-rest-client@^1.8.4: + version "1.8.11" + resolved "https://registry.yarnpkg.com/typed-rest-client/-/typed-rest-client-1.8.11.tgz#6906f02e3c91e8d851579f255abf0fd60800a04d" + integrity sha512-5UvfMpd1oelmUPRbbaVnq+rHP7ng2cE4qoQkQeAqxRL6PklkxsM0g32/HL0yfvruK6ojQ5x8EE+HF4YV6DtuCA== + dependencies: + qs "^6.9.1" + tunnel "0.0.6" + underscore "^1.12.1" + +typescript-eslint@^8.30.1: + version "8.31.0" + resolved "https://registry.yarnpkg.com/typescript-eslint/-/typescript-eslint-8.31.0.tgz#955e2e81011afbc11db3abd5f1d073a4b12e1270" + integrity sha512-u+93F0sB0An8WEAPtwxVhFby573E8ckdjwUUQUj9QA4v8JAvgtoDdIyYR3XFwFHq2W1KJ1AurwJCO+w+Y1ixyQ== + dependencies: + "@typescript-eslint/eslint-plugin" "8.31.0" + "@typescript-eslint/parser" "8.31.0" + "@typescript-eslint/utils" "8.31.0" + +typescript@^5.2.2, typescript@^5.4.3, typescript@^5.7.2, typescript@^5.8.3: + version "5.8.3" + resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.8.3.tgz#92f8a3e5e3cf497356f4178c34cd65a7f5e8440e" + integrity sha512-p1diW6TqL9L07nNxvRMM7hMMw4c5XOo/1ibL4aAIGmSAt9slTE1Xgw5KWuof2uTOvCg9BY7ZRi+GaF+7sfgPeQ== + +ua-parser-js@^1.0.35, ua-parser-js@^1.0.37: + version "1.0.40" + resolved "https://registry.yarnpkg.com/ua-parser-js/-/ua-parser-js-1.0.40.tgz#ac6aff4fd8ea3e794a6aa743ec9c2fc29e75b675" + integrity sha512-z6PJ8Lml+v3ichVojCiB8toQJBuwR42ySM4ezjXIqXK3M0HczmKQ3LF4rhU55PfD99KEEXQG6yb7iOMyvYuHew== + +uc.micro@^2.0.0, uc.micro@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/uc.micro/-/uc.micro-2.1.0.tgz#f8d3f7d0ec4c3dea35a7e3c8efa4cb8b45c9e7ee" + integrity sha512-ARDJmphmdvUk6Glw7y9DQ2bFkKBHwQHLi2lsaH6PPmz/Ka9sFOBsBluozhDltWmnv9u/cF6Rt87znRTPV+yp/A== + +ufo@^1.5.4: + version "1.6.1" + resolved "https://registry.yarnpkg.com/ufo/-/ufo-1.6.1.tgz#ac2db1d54614d1b22c1d603e3aef44a85d8f146b" + integrity sha512-9a4/uxlTWJ4+a5i0ooc1rU7C7YOw3wT+UGqdeNNHWnOF9qcMBgLRS+4IYUqbczewFx4mLEig6gawh7X6mFlEkA== + +unbox-primitive@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/unbox-primitive/-/unbox-primitive-1.1.0.tgz#8d9d2c9edeea8460c7f35033a88867944934d1e2" + integrity sha512-nWJ91DjeOkej/TA8pXQ3myruKpKEYgqvpw9lz4OPHj/NWFNluYrjbz9j01CJ8yKQd2g4jFoOkINCTW2I5LEEyw== + dependencies: + call-bound "^1.0.3" + has-bigints "^1.0.2" + has-symbols "^1.1.0" + which-boxed-primitive "^1.1.1" + +underscore@^1.12.1: + version "1.13.7" + resolved "https://registry.yarnpkg.com/underscore/-/underscore-1.13.7.tgz#970e33963af9a7dda228f17ebe8399e5fbe63a10" + integrity sha512-GMXzWtsc57XAtguZgaQViUOzs0KTkk8ojr3/xAxXLITqf/3EMwxC0inyETfDFjH/Krbhuep0HNbbjI9i/q3F3g== + +undici-types@~6.19.2: + version "6.19.8" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.19.8.tgz#35111c9d1437ab83a7cdc0abae2f26d88eda0a02" + integrity sha512-ve2KP6f/JnbPBFyobGHuerC9g1FYGn/F8n1LWTwNxCEzd6IfqTwUQcNXgEtmmQ6DlRrC1hrSrBnCZPokRrDHjw== + +undici-types@~6.21.0: + version "6.21.0" + resolved "https://registry.yarnpkg.com/undici-types/-/undici-types-6.21.0.tgz#691d00af3909be93a7faa13be61b3a5b50ef12cb" + integrity sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ== + +undici@^6.19.5: + version "6.21.2" + resolved "https://registry.yarnpkg.com/undici/-/undici-6.21.2.tgz#49c5884e8f9039c65a89ee9018ef3c8e2f1f4928" + integrity sha512-uROZWze0R0itiAKVPsYhFov9LxrPMHLMEQFszeI2gCN6bnIIZ8twzBCJcN2LJrBBLfrP0t1FW0g+JmKVl8Vk1g== + +unicode-canonical-property-names-ecmascript@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/unicode-canonical-property-names-ecmascript/-/unicode-canonical-property-names-ecmascript-2.0.1.tgz#cb3173fe47ca743e228216e4a3ddc4c84d628cc2" + integrity sha512-dA8WbNeb2a6oQzAQ55YlT5vQAWGV9WXOsi3SskE3bcCdM0P4SDd+24zS/OCacdRq5BkdsRj9q3Pg6YyQoxIGqg== + +unicode-match-property-ecmascript@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/unicode-match-property-ecmascript/-/unicode-match-property-ecmascript-2.0.0.tgz#54fd16e0ecb167cf04cf1f756bdcc92eba7976c3" + integrity sha512-5kaZCrbp5mmbz5ulBkDkbY0SsPOjKqVS35VpL9ulMPfSl0J0Xsm+9Evphv9CoIZFwre7aJoa94AY6seMKGVN5Q== + dependencies: + unicode-canonical-property-names-ecmascript "^2.0.0" + unicode-property-aliases-ecmascript "^2.0.0" + +unicode-match-property-value-ecmascript@^2.1.0: + version "2.2.0" + resolved "https://registry.yarnpkg.com/unicode-match-property-value-ecmascript/-/unicode-match-property-value-ecmascript-2.2.0.tgz#a0401aee72714598f739b68b104e4fe3a0cb3c71" + integrity sha512-4IehN3V/+kkr5YeSSDDQG8QLqO26XpL2XP3GQtqwlT/QYSECAwFztxVHjlbh0+gjJ3XmNLS0zDsbgs9jWKExLg== + +unicode-property-aliases-ecmascript@^2.0.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/unicode-property-aliases-ecmascript/-/unicode-property-aliases-ecmascript-2.1.0.tgz#43d41e3be698bd493ef911077c9b131f827e8ccd" + integrity sha512-6t3foTQI9qne+OZoVQB/8x8rk2k1eVy1gRXhV3oFQ5T6R1dqQ1xtin3XqSlx3+ATBkliTaR/hHyJBm+LVPNM8w== + +union@~0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/union/-/union-0.5.0.tgz#b2c11be84f60538537b846edb9ba266ba0090075" + integrity sha512-N6uOhuW6zO95P3Mel2I2zMsbsanvvtgn6jVqJv4vbVcz/JN0OkL9suomjQGmWtxJQXOCqUJvquc1sMeNz/IwlA== + dependencies: + qs "^6.4.0" + +unique-filename@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/unique-filename/-/unique-filename-2.0.1.tgz#e785f8675a9a7589e0ac77e0b5c34d2eaeac6da2" + integrity sha512-ODWHtkkdx3IAR+veKxFV+VBkUMcN+FaqzUUd7IZzt+0zhDZFPFxhlqwPF3YQvMHx1TD0tdgYl+kuPnJ8E6ql7A== + dependencies: + unique-slug "^3.0.0" + +unique-slug@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/unique-slug/-/unique-slug-3.0.0.tgz#6d347cf57c8a7a7a6044aabd0e2d74e4d76dc7c9" + integrity sha512-8EyMynh679x/0gqE9fT9oilG+qEt+ibFyqjuVTsZn1+CMxH+XLlpvr2UZx4nVcCwTpx81nICr2JQFkM+HPLq4w== + dependencies: + imurmurhash "^0.1.4" + +universalify@^0.1.0: + version "0.1.2" + resolved "https://registry.yarnpkg.com/universalify/-/universalify-0.1.2.tgz#b646f69be3942dabcecc9d6639c80dc105efaa66" + integrity sha512-rBJeI5CXAlmy1pV+617WB9J63U6XcazHHF2f2dbJix4XzpUF0RS3Zbj0FGIOCAva5P/d/GBOYaACQ1w+0azUkg== + +universalify@^2.0.0: + version "2.0.1" + resolved "https://registry.yarnpkg.com/universalify/-/universalify-2.0.1.tgz#168efc2180964e6386d061e094df61afe239b18d" + integrity sha512-gptHNQghINnc/vTGIk0SOFGFNXw7JVrlRUtConJRlvaw6DuX0wO5Jeko9sWrMBhh+PsYAZ7oXAiOnf/UKogyiw== + +unpipe@1.0.0, unpipe@~1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/unpipe/-/unpipe-1.0.0.tgz#b2bf4ee8514aae6165b4817829d21b2ef49904ec" + integrity sha512-pjy2bYhSsufwWlKwPc+l3cN7+wuJlK6uz0YdJEOlQDbl6jo/YlPi4mb8agUkVC8BF7V8NuzeyPNqRksA3hztKQ== + +update-browserslist-db@^1.1.1: + version "1.1.3" + resolved "https://registry.yarnpkg.com/update-browserslist-db/-/update-browserslist-db-1.1.3.tgz#348377dd245216f9e7060ff50b15a1b740b75420" + integrity sha512-UxhIZQ+QInVdunkDAaiazvvT/+fXL5Osr0JZlJulepYu6Jd7qJtDZjlur0emRlT71EN3ScPoE7gvsuIKKNavKw== + dependencies: + escalade "^3.2.0" + picocolors "^1.1.1" + +uri-js-replace@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/uri-js-replace/-/uri-js-replace-1.0.1.tgz#c285bb352b701c9dfdaeffc4da5be77f936c9048" + integrity sha512-W+C9NWNLFOoBI2QWDp4UT9pv65r2w5Cx+3sTYFvtMdDBxkKt1syCqsUdSFAChbEe1uK5TfS04wt/nGwmaeIQ0g== + +uri-js@^4.2.2: + version "4.4.1" + resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" + integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== + dependencies: + punycode "^2.1.0" + +url-join@^4.0.1: + version "4.0.1" + resolved "https://registry.yarnpkg.com/url-join/-/url-join-4.0.1.tgz#b642e21a2646808ffa178c4c5fda39844e12cde7" + integrity sha512-jk1+QP6ZJqyOiuEI9AEWQfju/nB2Pw466kbA0LEZljHwKeMgd9WrAEgEGxjPDD2+TNbbb37rTyhEfrCXfuKXnA== + +use-composed-ref@^1.3.0: + version "1.4.0" + resolved "https://registry.yarnpkg.com/use-composed-ref/-/use-composed-ref-1.4.0.tgz#09e023bf798d005286ad85cd20674bdf5770653b" + integrity sha512-djviaxuOOh7wkj0paeO1Q/4wMZ8Zrnag5H6yBvzN7AKKe8beOaED9SF5/ByLqsku8NP4zQqsvM2u3ew/tJK8/w== + +use-isomorphic-layout-effect@^1.1.1, use-isomorphic-layout-effect@^1.1.2: + version "1.2.0" + resolved "https://registry.yarnpkg.com/use-isomorphic-layout-effect/-/use-isomorphic-layout-effect-1.2.0.tgz#afb292eb284c39219e8cb8d3d62d71999361a21d" + integrity sha512-q6ayo8DWoPZT0VdG4u3D3uxcgONP3Mevx2i2b0434cwWBoL+aelL1DzkXI6w3PhTZzUeR2kaVlZn70iCiseP6w== + +use-latest@^1.2.1: + version "1.3.0" + resolved "https://registry.yarnpkg.com/use-latest/-/use-latest-1.3.0.tgz#549b9b0d4c1761862072f0899c6f096eb379137a" + integrity sha512-mhg3xdm9NaM8q+gLT8KryJPnRFOz1/5XPBhmDEVZK1webPzDjrPk7f/mbpeLqTgB9msytYWANxgALOCJKnLvcQ== + dependencies: + use-isomorphic-layout-effect "^1.1.1" + +use-sync-external-store@^1.2.0: + version "1.5.0" + resolved "https://registry.yarnpkg.com/use-sync-external-store/-/use-sync-external-store-1.5.0.tgz#55122e2a3edd2a6c106174c27485e0fd59bcfca0" + integrity sha512-Rb46I4cGGVBmjamjphe8L/UnvJD+uPPtTkNvX5mZgqdbavhI4EbgIWJiIHXJ8bc/i9EQGPRh4DwEURJ552Do0A== + +username@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/username/-/username-5.1.0.tgz#a7f9325adce2d0166448cdd55d4985b1360f2508" + integrity sha512-PCKbdWw85JsYMvmCv5GH3kXmM66rCd9m1hBEDutPNv94b/pqCMT4NtcKyeWYvLFiE8b+ha1Jdl8XAaUdPn5QTg== + dependencies: + execa "^1.0.0" + mem "^4.3.0" + +utf8-byte-length@^1.0.1: + version "1.0.5" + resolved "https://registry.yarnpkg.com/utf8-byte-length/-/utf8-byte-length-1.0.5.tgz#f9f63910d15536ee2b2d5dd4665389715eac5c1e" + integrity sha512-Xn0w3MtiQ6zoz2vFyUVruaCL53O/DwUvkEeOvj+uulMm0BkUGYWmBYVyElqZaSLhY6ZD0ulfU3aBra2aVT4xfA== + +util-deprecate@^1.0.1, util-deprecate@^1.0.2, util-deprecate@~1.0.1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" + integrity sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw== + +utils-merge@1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/utils-merge/-/utils-merge-1.0.1.tgz#9f95710f50a267947b2ccc124741c1028427e713" + integrity sha512-pMZTvIkT1d+TFGvDOqodOclx0QWkkgi6Tdoa8gC8ffGAAqz9pzPTZWAybbsHHoED/ztMtkv/VoYTYyShUn81hA== + +utrie@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/utrie/-/utrie-1.0.2.tgz#d42fe44de9bc0119c25de7f564a6ed1b2c87a645" + integrity sha512-1MLa5ouZiOmQzUbjbu9VmjLzn1QLXBhwpUa7kdLUQK+KQ5KA9I1vk5U4YHe/X2Ch7PYnJfWuWT+VbuxbGwljhw== + dependencies: + base64-arraybuffer "^1.0.2" + +uuid@^10.0.0: + version "10.0.0" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-10.0.0.tgz#5a95aa454e6e002725c79055fd42aaba30ca6294" + integrity sha512-8XkAphELsDnEGrDxUOHB3RGvXz6TeuYSGEZBOjtTtPm2lwhGBjLgOzLHB63IUWfBpNucQjND6d3AOudO+H3RWQ== + +uuid@^11.1.0: + version "11.1.0" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-11.1.0.tgz#9549028be1753bb934fc96e2bca09bb4105ae912" + integrity sha512-0/A9rDy9P7cJ+8w1c9WD9V//9Wj15Ce2MPz8Ri6032usz+NfePxx5AcN3bN+r6ZL6jEo066/yNYB3tn4pQEx+A== + +uuid@^8.3.0: + version "8.3.2" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-8.3.2.tgz#80d5b5ced271bb9af6c445f21a1a04c606cefbe2" + integrity sha512-+NYs2QeMWy+GWFOEm9xnn6HCDp0l7QBD7ml8zLUmJ+93Q5NF0NocErnwkTkXVFNiX3/fpC6afS8Dhb/gz7R7eg== + +v8-compile-cache-lib@^3.0.1: + version "3.0.1" + resolved "https://registry.yarnpkg.com/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz#6336e8d71965cb3d35a1bbb7868445a7c05264bf" + integrity sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg== + +validate-npm-package-license@^3.0.1: + version "3.0.4" + resolved "https://registry.yarnpkg.com/validate-npm-package-license/-/validate-npm-package-license-3.0.4.tgz#fc91f6b9c7ba15c857f4cb2c5defeec39d4f410a" + integrity sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew== + dependencies: + spdx-correct "^3.0.0" + spdx-expression-parse "^3.0.0" + +vary@~1.1.2: + version "1.1.2" + resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" + integrity sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg== + +verror@^1.10.0: + version "1.10.1" + resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.1.tgz#4bf09eeccf4563b109ed4b3d458380c972b0cdeb" + integrity sha512-veufcmxri4e3XSrT0xwfUR7kguIkaxBeosDg00yDWhk49wdwkSUrvvsm7nc75e1PUyvIeZj6nS8VQRYz2/S4Xg== + dependencies: + assert-plus "^1.0.0" + core-util-is "1.0.2" + extsprintf "^1.2.0" + +vite-node@1.6.1: + version "1.6.1" + resolved "https://registry.yarnpkg.com/vite-node/-/vite-node-1.6.1.tgz#fff3ef309296ea03ceaa6ca4bb660922f5416c57" + integrity sha512-YAXkfvGtuTzwWbDSACdJSg4A4DZiAqckWe90Zapc/sEX3XvHcw1NdurM/6od8J207tSDqNbSsgdCacBgvJKFuA== + dependencies: + cac "^6.7.14" + debug "^4.3.4" + pathe "^1.1.1" + picocolors "^1.0.0" + vite "^5.0.0" + +vite-node@3.1.2: + version "3.1.2" + resolved "https://registry.yarnpkg.com/vite-node/-/vite-node-3.1.2.tgz#b17869a12307f5260b20ba4b58cf493afee70aa7" + integrity sha512-/8iMryv46J3aK13iUXsei5G/A3CUlW4665THCPS+K8xAaqrVWiGB4RfXMQXCLjpK9P2eK//BczrVkn5JLAk6DA== + dependencies: + cac "^6.7.14" + debug "^4.4.0" + es-module-lexer "^1.6.0" + pathe "^2.0.3" + vite "^5.0.0 || ^6.0.0" + +vite-plugin-package-version@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/vite-plugin-package-version/-/vite-plugin-package-version-1.1.0.tgz#7d8088955aa21e4ec93353c98992b3f58c4bf13c" + integrity sha512-TPoFZXNanzcaKCIrC3e2L/TVRkkRLB6l4RPN/S7KbG7rWfyLcCEGsnXvxn6qR7fyZwXalnnSN/I9d6pSFjHpEA== + +vite-plugin-top-level-await@^1.5.0: + version "1.5.0" + resolved "https://registry.yarnpkg.com/vite-plugin-top-level-await/-/vite-plugin-top-level-await-1.5.0.tgz#e3f76302921152bf29d1658f169d168f8937e78b" + integrity sha512-r/DtuvHrSqUVk23XpG2cl8gjt1aATMG5cjExXL1BUTcSNab6CzkcPua9BPEc9fuTP5UpwClCxUe3+dNGL0yrgQ== + dependencies: + "@rollup/plugin-virtual" "^3.0.2" + "@swc/core" "^1.10.16" + uuid "^10.0.0" + +vite-tsconfig-paths@^4.3.2: + version "4.3.2" + resolved "https://registry.yarnpkg.com/vite-tsconfig-paths/-/vite-tsconfig-paths-4.3.2.tgz#321f02e4b736a90ff62f9086467faf4e2da857a9" + integrity sha512-0Vd/a6po6Q+86rPlntHye7F31zA2URZMbH8M3saAZ/xR9QoGN/L21bxEGfXdWmFdNkqPpRdxFT7nmNe12e9/uA== + dependencies: + debug "^4.1.1" + globrex "^0.1.2" + tsconfck "^3.0.3" + +vite-tsconfig-paths@^5.1.4: + version "5.1.4" + resolved "https://registry.yarnpkg.com/vite-tsconfig-paths/-/vite-tsconfig-paths-5.1.4.tgz#d9a71106a7ff2c1c840c6f1708042f76a9212ed4" + integrity sha512-cYj0LRuLV2c2sMqhqhGpaO3LretdtMn/BVX4cPLanIZuwwrkVl+lK84E/miEXkCHWXuq65rhNN4rXsBcOB3S4w== + dependencies: + debug "^4.1.1" + globrex "^0.1.2" + tsconfck "^3.0.3" + +vite@^5.0.0, vite@^5.4.18: + version "5.4.18" + resolved "https://registry.yarnpkg.com/vite/-/vite-5.4.18.tgz#b5af357f9d5ebb2e0c085779b7a37a77f09168a4" + integrity sha512-1oDcnEp3lVyHCuQ2YFelM4Alm2o91xNoMncRm1U7S+JdYfYOvbiGZ3/CxGttrOu2M/KcGz7cRC2DoNUA6urmMA== + dependencies: + esbuild "^0.21.3" + postcss "^8.4.43" + rollup "^4.20.0" + optionalDependencies: + fsevents "~2.3.3" + +"vite@^5.0.0 || ^6.0.0": + version "6.3.3" + resolved "https://registry.yarnpkg.com/vite/-/vite-6.3.3.tgz#497392c3f2243194e4dbf09ea83e9a3dddf49b88" + integrity sha512-5nXH+QsELbFKhsEfWLkHrvgRpTdGJzqOZ+utSdmPTvwHmvU6ITTm3xx+mRusihkcI8GeC7lCDyn3kDtiki9scw== + dependencies: + esbuild "^0.25.0" + fdir "^6.4.4" + picomatch "^4.0.2" + postcss "^8.5.3" + rollup "^4.34.9" + tinyglobby "^0.2.13" + optionalDependencies: + fsevents "~2.3.3" + +vitest-webgl-canvas-mock@^1.1.0: + version "1.1.0" + resolved "https://registry.yarnpkg.com/vitest-webgl-canvas-mock/-/vitest-webgl-canvas-mock-1.1.0.tgz#17d79f8b2601e7cd77903c59ed744d4d86324e25" + integrity sha512-F/5+XvBs7cSZPe41IGQTbSjNimB4NntPnRqv4eWb42voFKQINH8y2xZkibNUxYJCGIuDFsYp1lDQgTvWLahSzA== + dependencies: + cssfontparser "^1.2.1" + parse-color "^1.0.0" + +vitest@^1.6.1: + version "1.6.1" + resolved "https://registry.yarnpkg.com/vitest/-/vitest-1.6.1.tgz#b4a3097adf8f79ac18bc2e2e0024c534a7a78d2f" + integrity sha512-Ljb1cnSJSivGN0LqXd/zmDbWEM0RNNg2t1QW/XUhYl/qPqyu7CsqeWtqQXHVaJsecLPuDoak2oJcZN2QoRIOag== + dependencies: + "@vitest/expect" "1.6.1" + "@vitest/runner" "1.6.1" + "@vitest/snapshot" "1.6.1" + "@vitest/spy" "1.6.1" + "@vitest/utils" "1.6.1" + acorn-walk "^8.3.2" + chai "^4.3.10" + debug "^4.3.4" + execa "^8.0.1" + local-pkg "^0.5.0" + magic-string "^0.30.5" + pathe "^1.1.1" + picocolors "^1.0.0" + std-env "^3.5.0" + strip-literal "^2.0.0" + tinybench "^2.5.1" + tinypool "^0.8.3" + vite "^5.0.0" + vite-node "1.6.1" + why-is-node-running "^2.2.2" + +vitest@^3.1.1: + version "3.1.2" + resolved "https://registry.yarnpkg.com/vitest/-/vitest-3.1.2.tgz#63afc16b6da3bea6e39f5387d80719e70634ba66" + integrity sha512-WaxpJe092ID1C0mr+LH9MmNrhfzi8I65EX/NRU/Ld016KqQNRgxSOlGNP1hHN+a/F8L15Mh8klwaF77zR3GeDQ== + dependencies: + "@vitest/expect" "3.1.2" + "@vitest/mocker" "3.1.2" + "@vitest/pretty-format" "^3.1.2" + "@vitest/runner" "3.1.2" + "@vitest/snapshot" "3.1.2" + "@vitest/spy" "3.1.2" + "@vitest/utils" "3.1.2" + chai "^5.2.0" + debug "^4.4.0" + expect-type "^1.2.1" + magic-string "^0.30.17" + pathe "^2.0.3" + std-env "^3.9.0" + tinybench "^2.9.0" + tinyexec "^0.3.2" + tinyglobby "^0.2.13" + tinypool "^1.0.2" + tinyrainbow "^2.0.0" + vite "^5.0.0 || ^6.0.0" + vite-node "3.1.2" + why-is-node-running "^2.3.0" + +vscode-jsonrpc@8.2.0: + version "8.2.0" + resolved "https://registry.yarnpkg.com/vscode-jsonrpc/-/vscode-jsonrpc-8.2.0.tgz#f43dfa35fb51e763d17cd94dcca0c9458f35abf9" + integrity sha512-C+r0eKJUIfiDIfwJhria30+TYWPtuHJXHtI7J0YlOmKAo7ogxP20T0zxB7HZQIFhIyvoBPwWskjxrvAtfjyZfA== + +vscode-jsonrpc@^8.2.1: + version "8.2.1" + resolved "https://registry.yarnpkg.com/vscode-jsonrpc/-/vscode-jsonrpc-8.2.1.tgz#a322cc0f1d97f794ffd9c4cd2a898a0bde097f34" + integrity sha512-kdjOSJ2lLIn7r1rtrMbbNCHjyMPfRnowdKjBQ+mGq6NAW5QY2bEZC/khaC5OR8svbbjvLEaIXkOq45e2X9BIbQ== + +vscode-languageclient@^9.0.1: + version "9.0.1" + resolved "https://registry.yarnpkg.com/vscode-languageclient/-/vscode-languageclient-9.0.1.tgz#cdfe20267726c8d4db839dc1e9d1816e1296e854" + integrity sha512-JZiimVdvimEuHh5olxhxkht09m3JzUGwggb5eRUkzzJhZ2KjCN0nh55VfiED9oez9DyF8/fz1g1iBV3h+0Z2EA== + dependencies: + minimatch "^5.1.0" + semver "^7.3.7" + vscode-languageserver-protocol "3.17.5" + +vscode-languageserver-protocol@3.17.5, vscode-languageserver-protocol@^3.17.5: + version "3.17.5" + resolved "https://registry.yarnpkg.com/vscode-languageserver-protocol/-/vscode-languageserver-protocol-3.17.5.tgz#864a8b8f390835572f4e13bd9f8313d0e3ac4bea" + integrity sha512-mb1bvRJN8SVznADSGWM9u/b07H7Ecg0I3OgXDuLdn307rl/J3A9YD6/eYOssqhecL27hK1IPZAsaqh00i/Jljg== + dependencies: + vscode-jsonrpc "8.2.0" + vscode-languageserver-types "3.17.5" + +vscode-languageserver-types@3.17.5: + version "3.17.5" + resolved "https://registry.yarnpkg.com/vscode-languageserver-types/-/vscode-languageserver-types-3.17.5.tgz#3273676f0cf2eab40b3f44d085acbb7f08a39d8a" + integrity sha512-Ld1VelNuX9pdF39h2Hgaeb5hEZM2Z3jUrrMgWQAu82jMtZp7p3vJT3BzToKtZI7NgQssZje5o0zryOrhQvzQAg== + +vscode-uri@^3.1.0: + version "3.1.0" + resolved "https://registry.yarnpkg.com/vscode-uri/-/vscode-uri-3.1.0.tgz#dd09ec5a66a38b5c3fffc774015713496d14e09c" + integrity sha512-/BpdSx+yCQGnCvecbyXdxHDkuk55/G3xwnC0GqY4gmQ3j+A+g8kzzgB4Nk/SINjqn6+waqw3EgbVF2QKExkRxQ== + +w3c-keyname@^2.2.4: + version "2.2.8" + resolved "https://registry.yarnpkg.com/w3c-keyname/-/w3c-keyname-2.2.8.tgz#7b17c8c6883d4e8b86ac8aba79d39e880f8869c5" + integrity sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ== + +warning@^4.0.3: + version "4.0.3" + resolved "https://registry.yarnpkg.com/warning/-/warning-4.0.3.tgz#16e9e077eb8a86d6af7d64aa1e05fd85b4678ca3" + integrity sha512-rpJyN222KWIvHJ/F53XSZv0Zl/accqHR8et1kpaMTD/fLCRxtV8iX8czMzY7sVZupTI3zcUTg8eycS2kNF9l6w== + dependencies: + loose-envify "^1.0.0" + +wcwidth@^1.0.1: + version "1.0.1" + resolved "https://registry.yarnpkg.com/wcwidth/-/wcwidth-1.0.1.tgz#f0b0dcf915bc5ff1528afadb2c0e17b532da2fe8" + integrity sha512-XHPEwS0q6TaxcvG85+8EYkbiCux2XtWG2mkc47Ng2A77BQu9+DqIOJldST4HgPkuea7dvKSj5VgX3P1d4rW8Tg== + dependencies: + defaults "^1.0.3" + +web-streams-polyfill@^3.0.3: + version "3.3.3" + resolved "https://registry.yarnpkg.com/web-streams-polyfill/-/web-streams-polyfill-3.3.3.tgz#2073b91a2fdb1fbfbd401e7de0ac9f8214cecb4b" + integrity sha512-d2JWLCivmZYTSIoge9MsgFCZrt571BikcWGYkjC1khllbTeDlGqZ2D8vD8E/lJa8WGWbb7Plm8/XJYV7IJHZZw== + +web-vitals@^3.5.2: + version "3.5.2" + resolved "https://registry.yarnpkg.com/web-vitals/-/web-vitals-3.5.2.tgz#5bb58461bbc173c3f00c2ddff8bfe6e680999ca9" + integrity sha512-c0rhqNcHXRkY/ogGDJQxZ9Im9D19hDihbzSQJrsioex+KnFgmMzBiy57Z1EjkhX/+OjyBpclDCzz2ITtjokFmg== + +webidl-conversions@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-3.0.1.tgz#24534275e2a7bc6be7bc86611cc16ae0a5654871" + integrity sha512-2JAn3z8AR6rjK8Sm8orRC0h/bcl/DqL7tRPdGZ4I1CjdF+EaMLmYxBHyXuKL849eucPFhvBoxMsflfOb8kxaeQ== + +webidl-conversions@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/webidl-conversions/-/webidl-conversions-7.0.0.tgz#256b4e1882be7debbf01d05f0aa2039778ea080a" + integrity sha512-VwddBukDzu71offAQR975unBIGqfKZpM+8ZX6ySk8nYhVoo5CYaZyzt3YBvYtRtO+aoGlqxPg/B87NGVZ/fu6g== + +whatwg-encoding@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/whatwg-encoding/-/whatwg-encoding-2.0.0.tgz#e7635f597fd87020858626805a2729fa7698ac53" + integrity sha512-p41ogyeMUrw3jWclHWTQg1k05DSVXPLcVxRTYsXUk+ZooOCZLcoYgPZ/HL/D/N+uQPOtcp1me1WhBEaX02mhWg== + dependencies: + iconv-lite "0.6.3" + +whatwg-encoding@^3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/whatwg-encoding/-/whatwg-encoding-3.1.1.tgz#d0f4ef769905d426e1688f3e34381a99b60b76e5" + integrity sha512-6qN4hJdMwfYBtE3YBTTHhoeuUrDBPZmbQaxWAqSALV/MeEnR5z1xd8UKud2RAkFoPkmB+hli1TZSnyi84xz1vQ== + dependencies: + iconv-lite "0.6.3" + +whatwg-fetch@^3.4.1: + version "3.6.20" + resolved "https://registry.yarnpkg.com/whatwg-fetch/-/whatwg-fetch-3.6.20.tgz#580ce6d791facec91d37c72890995a0b48d31c70" + integrity sha512-EqhiFU6daOA8kpjOWTL0olhVOF3i7OrFzSYiGsEMB8GcXS+RrzauAERX65xMeNWVqxA6HXH2m69Z9LaKKdisfg== + +whatwg-mimetype@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/whatwg-mimetype/-/whatwg-mimetype-3.0.0.tgz#5fa1a7623867ff1af6ca3dc72ad6b8a4208beba7" + integrity sha512-nt+N2dzIutVRxARx1nghPKGv1xHikU7HKdfafKkLNLindmPU/ch3U31NOCGGA/dmPcmb1VlofO0vnKAcsm0o/Q== + +whatwg-mimetype@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/whatwg-mimetype/-/whatwg-mimetype-4.0.0.tgz#bc1bf94a985dc50388d54a9258ac405c3ca2fc0a" + integrity sha512-QaKxh0eNIi2mE9p2vEdzfagOKHCcj1pJ56EEHGQOVxp8r9/iszLUUV7v89x9O1p/T+NlTM5W7jW6+cz4Fq1YVg== + +whatwg-url@^5.0.0: + version "5.0.0" + resolved "https://registry.yarnpkg.com/whatwg-url/-/whatwg-url-5.0.0.tgz#966454e8765462e37644d3626f6742ce8b70965d" + integrity sha512-saE57nupxk6v3HY35+jzBwYa0rKSy0XR8JSxZPwgLr7ys0IBzhGviA1/TUGJLmSVqs8pb9AnvICXEuOHLprYTw== + dependencies: + tr46 "~0.0.3" + webidl-conversions "^3.0.0" + +which-boxed-primitive@^1.1.0, which-boxed-primitive@^1.1.1: + version "1.1.1" + resolved "https://registry.yarnpkg.com/which-boxed-primitive/-/which-boxed-primitive-1.1.1.tgz#d76ec27df7fa165f18d5808374a5fe23c29b176e" + integrity sha512-TbX3mj8n0odCBFVlY8AxkqcHASw3L60jIuF8jFP78az3C2YhmGvqbHBpAjTRH2/xqYunrJ9g1jSyjCjpoWzIAA== + dependencies: + is-bigint "^1.1.0" + is-boolean-object "^1.2.1" + is-number-object "^1.1.1" + is-string "^1.1.1" + is-symbol "^1.1.1" + +which-builtin-type@^1.2.1: + version "1.2.1" + resolved "https://registry.yarnpkg.com/which-builtin-type/-/which-builtin-type-1.2.1.tgz#89183da1b4907ab089a6b02029cc5d8d6574270e" + integrity sha512-6iBczoX+kDQ7a3+YJBnh3T+KZRxM/iYNPXicqk66/Qfm1b93iu+yOImkg0zHbj5LNOcNv1TEADiZ0xa34B4q6Q== + dependencies: + call-bound "^1.0.2" + function.prototype.name "^1.1.6" + has-tostringtag "^1.0.2" + is-async-function "^2.0.0" + is-date-object "^1.1.0" + is-finalizationregistry "^1.1.0" + is-generator-function "^1.0.10" + is-regex "^1.2.1" + is-weakref "^1.0.2" + isarray "^2.0.5" + which-boxed-primitive "^1.1.0" + which-collection "^1.0.2" + which-typed-array "^1.1.16" + +which-collection@^1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/which-collection/-/which-collection-1.0.2.tgz#627ef76243920a107e7ce8e96191debe4b16c2a0" + integrity sha512-K4jVyjnBdgvc86Y6BkaLZEN933SwYOuBFkdmBu9ZfkcAbdVbpITnDmjvZ/aQjRXQrv5EPkTnD1s39GiiqbngCw== + dependencies: + is-map "^2.0.3" + is-set "^2.0.3" + is-weakmap "^2.0.2" + is-weakset "^2.0.3" + +which-typed-array@^1.1.16, which-typed-array@^1.1.18: + version "1.1.19" + resolved "https://registry.yarnpkg.com/which-typed-array/-/which-typed-array-1.1.19.tgz#df03842e870b6b88e117524a4b364b6fc689f956" + integrity sha512-rEvr90Bck4WZt9HHFC4DJMsjvu7x+r6bImz0/BrbWb7A2djJ8hnZMrWnHo9F8ssv0OMErasDhftrfROTyqSDrw== + dependencies: + available-typed-arrays "^1.0.7" + call-bind "^1.0.8" + call-bound "^1.0.4" + for-each "^0.3.5" + get-proto "^1.0.1" + gopd "^1.2.0" + has-tostringtag "^1.0.2" + +which@^1.2.9: + version "1.3.1" + resolved "https://registry.yarnpkg.com/which/-/which-1.3.1.tgz#a45043d54f5805316da8d62f9f50918d3da70b0a" + integrity sha512-HxJdYWq1MTIQbJ3nw0cqssHoTNU267KlrDuGZ1WYlxDStUtKUhOaJmh112/TZmHxxUfuJqPXSOm7tDyas0OSIQ== + dependencies: + isexe "^2.0.0" + +which@^2.0.1, which@^2.0.2: + version "2.0.2" + resolved "https://registry.yarnpkg.com/which/-/which-2.0.2.tgz#7c6a8dd0a636a0327e10b59c9286eee93f3f51b1" + integrity sha512-BLI3Tl1TW3Pvl70l3yq3Y64i+awpwXqsGBYWkkqMtnbXgrMD+yj7rhW0kuEDxzJaYXGjEW5ogapKNMEKNMjibA== + dependencies: + isexe "^2.0.0" + +why-is-node-running@^2.2.2, why-is-node-running@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/why-is-node-running/-/why-is-node-running-2.3.0.tgz#a3f69a97107f494b3cdc3bdddd883a7d65cebf04" + integrity sha512-hUrmaWBdVDcxvYqnyh09zunKzROWjbZTiNy8dBEjkS7ehEDQibXJ7XvlmtbwuTclUiIyN+CyXQD4Vmko8fNm8w== + dependencies: + siginfo "^2.0.0" + stackback "0.0.2" + +win-ca@^3.5.1: + version "3.5.1" + resolved "https://registry.yarnpkg.com/win-ca/-/win-ca-3.5.1.tgz#2ef37ac24b0a1daa2714b4c5ef258c5242429e00" + integrity sha512-RNy9gpBS6cxWHjfbqwBA7odaHyT+YQNhtdpJZwYCFoxB/Dq22oeOZ9YCXMwjhLytKpo7JJMnKdJ/ve7N12zzfQ== + dependencies: + is-electron "^2.2.0" + make-dir "^1.3.0" + node-forge "^1.2.1" + split "^1.0.1" + +word-wrap@^1.2.5: + version "1.2.5" + resolved "https://registry.yarnpkg.com/word-wrap/-/word-wrap-1.2.5.tgz#d2c45c6dd4fbce621a66f136cbe328afd0410b34" + integrity sha512-BN22B5eaMMI9UMtjrGd5g5eCYPpCPDUy0FJXbYsaT5zYxjFOckS53SQDE3pWkVoWpHXVb3BrYcEN4Twa55B5cA== + +workerpool@^6.5.1: + version "6.5.1" + resolved "https://registry.yarnpkg.com/workerpool/-/workerpool-6.5.1.tgz#060f73b39d0caf97c6db64da004cd01b4c099544" + integrity sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA== + +"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0": + version "7.0.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" + integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + +wrap-ansi@^7.0.0: + version "7.0.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" + integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== + dependencies: + ansi-styles "^4.0.0" + string-width "^4.1.0" + strip-ansi "^6.0.0" + +wrap-ansi@^8.0.1, wrap-ansi@^8.1.0: + version "8.1.0" + resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-8.1.0.tgz#56dc22368ee570face1b49819975d9b9a5ead214" + integrity sha512-si7QWI6zUMq56bESFvagtmzMdGOtoxfR+Sez11Mobfc7tm+VkUckk9bW2UeffTGVUbOksxmSw0AA2gs8g71NCQ== + dependencies: + ansi-styles "^6.1.0" + string-width "^5.0.1" + strip-ansi "^7.0.1" + +wrappy@1: + version "1.0.2" + resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" + integrity sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ== + +ws@^7.4.6: + version "7.5.10" + resolved "https://registry.yarnpkg.com/ws/-/ws-7.5.10.tgz#58b5c20dc281633f6c19113f39b349bd8bd558d9" + integrity sha512-+dbF1tHwZpXcbOJdVOkzLDxZP1ailvSxM6ZweXTegylPny803bFhA+vqBYw4s31NSAk4S2Qz+AKXK9a4wkdjcQ== + +ws@^8.18.1: + version "8.18.1" + resolved "https://registry.yarnpkg.com/ws/-/ws-8.18.1.tgz#ea131d3784e1dfdff91adb0a4a116b127515e3cb" + integrity sha512-RKW2aJZMXeMxVpnZ6bck+RswznaxmzdULiBr6KY7XkTnW8uvt0iT9H5DkHUChXrc+uurzwa0rVI16n/Xzjdz1w== + +xml2js@^0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/xml2js/-/xml2js-0.5.0.tgz#d9440631fbb2ed800203fad106f2724f62c493b7" + integrity sha512-drPFnkQJik/O+uPKpqSgr22mpuFHqKdbS835iAQrUC73L2F5WkboIRd63ai/2Yg6I1jzifPFKH2NTK+cfglkIA== + dependencies: + sax ">=0.6.0" + xmlbuilder "~11.0.0" + +xmlbuilder@>=11.0.1, xmlbuilder@^15.1.1: + version "15.1.1" + resolved "https://registry.yarnpkg.com/xmlbuilder/-/xmlbuilder-15.1.1.tgz#9dcdce49eea66d8d10b42cae94a79c3c8d0c2ec5" + integrity sha512-yMqGBqtXyeN1e3TGYvgNgDVZ3j84W4cwkOXQswghol6APgZWaff9lnbvN7MHYJOiXsvGPXtjTYJEiC9J2wv9Eg== + +xmlbuilder@~11.0.0: + version "11.0.1" + resolved "https://registry.yarnpkg.com/xmlbuilder/-/xmlbuilder-11.0.1.tgz#be9bae1c8a046e76b31127726347d0ad7002beb3" + integrity sha512-fDlsI/kFEx7gLvbecc0/ohLG50fugQp8ryHzMTuW9vSa1GJ0XYWKnhsUx7oie3G98+r56aTQIUB4kht42R3JvA== + +"xstate-beta@npm:xstate@beta": + version "5.0.0-beta.54" + resolved "https://registry.yarnpkg.com/xstate/-/xstate-5.0.0-beta.54.tgz#d80f1a9e43ad883a65fc9b399161bd39633bd9bf" + integrity sha512-BTnCPBQ2iTKe4uCnHEe1hNx6VTbXU+5mQGybSQHOjTLiBi4Ryi+tL9T6N1tmqagvM8rfl4XRfvndogfWCWcdpw== + +xstate@^4.33.4: + version "4.38.3" + resolved "https://registry.yarnpkg.com/xstate/-/xstate-4.38.3.tgz#4e15e7ad3aa0ca1eea2010548a5379966d8f1075" + integrity sha512-SH7nAaaPQx57dx6qvfcIgqKRXIh4L0A1iYEqim4s1u7c9VoCgzZc+63FY90AKU4ZzOC2cfJzTnpO4zK7fCUzzw== + +xstate@^5.19.2: + version "5.19.2" + resolved "https://registry.yarnpkg.com/xstate/-/xstate-5.19.2.tgz#db3f1ee614bbb6a49ad3f0c96ddbf98562d456ba" + integrity sha512-B8fL2aP0ogn5aviAXFzI5oZseAMqN00fg/TeDa3ZtatyDcViYLIfuQl4y8qmHCiKZgGEzmnTyNtNQL9oeJE2gw== + +xterm-addon-fit@^0.5.0: + version "0.5.0" + resolved "https://registry.yarnpkg.com/xterm-addon-fit/-/xterm-addon-fit-0.5.0.tgz#2d51b983b786a97dcd6cde805e700c7f913bc596" + integrity sha512-DsS9fqhXHacEmsPxBJZvfj2la30Iz9xk+UKjhQgnYNkrUIN5CYLbw7WEfz117c7+S86S/tpHPfvNxJsF5/G8wQ== + +xterm-addon-search@^0.8.0: + version "0.8.2" + resolved "https://registry.yarnpkg.com/xterm-addon-search/-/xterm-addon-search-0.8.2.tgz#be7aa74d5ff12c901707c6ff674229f214318032" + integrity sha512-I1863mjn8P6uVrqm/X+btalVsqjAKLhnhpbP7SavAOpEkI1jJhbHU2UTp7NjeRtcKTks6UWk/ycgds5snDSejg== + +xterm@^4.9.0: + version "4.19.0" + resolved "https://registry.yarnpkg.com/xterm/-/xterm-4.19.0.tgz#c0f9d09cd61de1d658f43ca75f992197add9ef6d" + integrity sha512-c3Cp4eOVsYY5Q839dR5IejghRPpxciGmLWWaP9g+ppfMeBChMeLa1DCA+pmX/jyDZ+zxFOmlJL/82qVdayVoGQ== + +y18n@^5.0.5: + version "5.0.8" + resolved "https://registry.yarnpkg.com/y18n/-/y18n-5.0.8.tgz#7f4934d0f7ca8c56f95314939ddcd2dd91ce1d55" + integrity sha512-0pfFzegeDWJHJIAmTLRP2DwHjdF5s7jo9tuztdQxAhINCdvS+3nGINqPd00AphqJR/0LhANUS6/+7SCb98YOfA== + +yallist@^3.0.2: + version "3.1.1" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-3.1.1.tgz#dbb7daf9bfd8bac9ab45ebf602b8cbad0d5d08fd" + integrity sha512-a4UGQaWPH59mOXUYnAG2ewncQS4i4F43Tv3JoAM+s2VDAmS9NsK8GpDMLrCHPksFT7h3K6TOoUNn2pb7RoXx4g== + +yallist@^4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" + integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== + +yaml-ast-parser@0.0.43: + version "0.0.43" + resolved "https://registry.yarnpkg.com/yaml-ast-parser/-/yaml-ast-parser-0.0.43.tgz#e8a23e6fb4c38076ab92995c5dca33f3d3d7c9bb" + integrity sha512-2PTINUwsRqSd+s8XxKaJWQlUuEMHJQyEuh2edBbW8KNJz0SJPwUSD2zRWqezFEdN7IzAgeuYHFUCF7o8zRdZ0A== + +yaml@^2.3.4: + version "2.7.1" + resolved "https://registry.yarnpkg.com/yaml/-/yaml-2.7.1.tgz#44a247d1b88523855679ac7fa7cda6ed7e135cf6" + integrity sha512-10ULxpnOCQXxJvBgxsn9ptjq6uviG/htZKk9veJGhlqn3w/DxQ631zFF+nlQXLwmImeS5amR2dl2U8sg6U9jsQ== + +yargs-parser@^21.1.1: + version "21.1.1" + resolved "https://registry.yarnpkg.com/yargs-parser/-/yargs-parser-21.1.1.tgz#9096bceebf990d21bb31fa9516e0ede294a77d35" + integrity sha512-tVpsJW7DdjecAiFpbIB1e3qxIQsE6NoPc5/eTdrbbIC4h0LVsWhnoa3g+m2HclBIujHzsxZ4VJVA+GUuc2/LBw== + +yargs-unparser@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/yargs-unparser/-/yargs-unparser-2.0.0.tgz#f131f9226911ae5d9ad38c432fe809366c2325eb" + integrity sha512-7pRTIA9Qc1caZ0bZ6RYRGbHJthJWuakf+WmHK0rVeLkNrrGhfoabBNdue6kdINI6r4if7ocq9aD/n7xwKOdzOA== + dependencies: + camelcase "^6.0.0" + decamelize "^4.0.0" + flat "^5.0.2" + is-plain-obj "^2.1.0" + +yargs@^17.0.1, yargs@^17.6.2, yargs@^17.7.2: + version "17.7.2" + resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269" + integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w== + dependencies: + cliui "^8.0.1" + escalade "^3.1.1" + get-caller-file "^2.0.5" + require-directory "^2.1.1" + string-width "^4.2.3" + y18n "^5.0.5" + yargs-parser "^21.1.1" + +yauzl@^2.10.0, yauzl@^2.3.1: + version "2.10.0" + resolved "https://registry.yarnpkg.com/yauzl/-/yauzl-2.10.0.tgz#c7eb17c93e112cb1086fa6d8e51fb0667b79a5f9" + integrity sha512-p4a9I6X6nu6IhoGmBqAcbJy1mlC4j27vEPZX9F4L4/vZT3Lyq1VkFHw/V/PUcB9Buo+DG3iHkT0x3Qya58zc3g== + dependencies: + buffer-crc32 "~0.2.3" + fd-slicer "~1.1.0" + +yazl@^2.2.2: + version "2.5.1" + resolved "https://registry.yarnpkg.com/yazl/-/yazl-2.5.1.tgz#a3d65d3dd659a5b0937850e8609f22fffa2b5c35" + integrity sha512-phENi2PLiHnHb6QBVot+dJnaAZ0xosj7p3fWl+znIjBDlnMI2PsZCJZ306BPTFOaHf5qdDEI8x5qFrSOBN5vrw== + dependencies: + buffer-crc32 "~0.2.3" + +yn@3.1.1: + version "3.1.1" + resolved "https://registry.yarnpkg.com/yn/-/yn-3.1.1.tgz#1e87401a09d767c1d5eab26a6e4c185182d2eb50" + integrity sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q== + +yocto-queue@^0.1.0: + version "0.1.0" + resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-0.1.0.tgz#0294eb3dee05028d31ee1a5fa2c556a6aaf10a1b" + integrity sha512-rVksvsnNCdJ/ohGc6xgPwyN8eheCxsiLM8mxuE/t/mOVqJewPuO1miLpTHQiRgTKCLexL4MeAFVagts7HmNZ2Q== + +yocto-queue@^1.0.0: + version "1.2.1" + resolved "https://registry.yarnpkg.com/yocto-queue/-/yocto-queue-1.2.1.tgz#36d7c4739f775b3cbc28e6136e21aa057adec418" + integrity sha512-AyeEbWOu/TAXdxlV9wmGcR0+yh2j3vYPGOECcIj2S7MkrLyC7ne+oye2BKTItt0ii2PHk4cDy+95+LshzbXnGg==