diff --git a/.github/dependabot.yml b/.github/dependabot.yml index 2b885816f7..011e014b22 100644 --- a/.github/dependabot.yml +++ b/.github/dependabot.yml @@ -8,21 +8,21 @@ updates: - package-ecosystem: 'npm' # See documentation for possible values directory: '/' # Location of package manifests schedule: - interval: 'daily' + interval: 'weekly' reviewers: - franknoirot - irev-dev - package-ecosystem: 'github-actions' # See documentation for possible values directory: '/' # Location of package manifests schedule: - interval: 'daily' + interval: 'weekly' reviewers: - adamchalmers - jessfraz - package-ecosystem: 'cargo' # See documentation for possible values directory: '/src/wasm-lib/' # Location of package manifests schedule: - interval: 'daily' + interval: 'weekly' reviewers: - adamchalmers - jessfraz diff --git a/.github/workflows/build-test-publish-apps.yml b/.github/workflows/build-test-publish-apps.yml index cf51db1459..e418dd103a 100644 --- a/.github/workflows/build-test-publish-apps.yml +++ b/.github/workflows/build-test-publish-apps.yml @@ -181,6 +181,7 @@ jobs: - name: Build the app (release) if: ${{ env.BUILD_RELEASE == 'true' }} env: + PUBLISH_FOR_PULL_REQUEST: true APPLE_ID: ${{ secrets.APPLE_ID }} APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} APPLE_APP_SPECIFIC_PASSWORD: ${{ secrets.APPLE_PASSWORD }} @@ -370,7 +371,7 @@ jobs: project_id: ${{ env.GOOGLE_CLOUD_PROJECT_ID }} - name: Upload release files to public bucket - uses: google-github-actions/upload-cloud-storage@v2.2.0 + uses: google-github-actions/upload-cloud-storage@v2.2.1 with: path: out glob: 'Zoo*' @@ -378,7 +379,7 @@ jobs: destination: ${{ env.BUCKET_DIR }} - name: Upload update endpoint to public bucket - uses: google-github-actions/upload-cloud-storage@v2.2.0 + uses: google-github-actions/upload-cloud-storage@v2.2.1 with: path: out glob: 'latest*' @@ -386,7 +387,7 @@ jobs: destination: ${{ env.BUCKET_DIR }} - name: Upload download endpoint to public bucket - uses: google-github-actions/upload-cloud-storage@v2.2.0 + uses: google-github-actions/upload-cloud-storage@v2.2.1 with: path: last_download.json destination: ${{ env.BUCKET_DIR }} diff --git a/.github/workflows/static-analysis.yml b/.github/workflows/static-analysis.yml index 155160e887..8908acd98a 100644 --- a/.github/workflows/static-analysis.yml +++ b/.github/workflows/static-analysis.yml @@ -81,6 +81,31 @@ jobs: - name: Run codespell run: codespell --config .codespellrc # Edit this file to tweak the typo list and other configuration. + yarn-unit-test-kcl-samples: + runs-on: ubuntu-latest + steps: + - uses: actions/checkout@v4 + + - uses: actions/setup-node@v4 + with: + node-version-file: '.nvmrc' + cache: 'yarn' + + - run: yarn install + - run: yarn build:wasm + + - run: yarn simpleserver:bg + if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} + + - name: Install Chromium Browser + if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} + run: yarn playwright install chromium --with-deps + + - name: run unit tests for kcl samples + if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} + run: yarn test:unit:kcl-samples + env: + VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} yarn-unit-test: runs-on: ubuntu-latest diff --git a/docs/kcl/index.md b/docs/kcl/index.md index 75f3c19d4b..0e3a6705c2 100644 --- a/docs/kcl/index.md +++ b/docs/kcl/index.md @@ -84,9 +84,13 @@ layout: manual * [`rem`](kcl/rem) * [`revolve`](kcl/revolve) * [`segAng`](kcl/segAng) +* [`segEnd`](kcl/segEnd) * [`segEndX`](kcl/segEndX) * [`segEndY`](kcl/segEndY) * [`segLen`](kcl/segLen) +* [`segStart`](kcl/segStart) +* [`segStartX`](kcl/segStartX) +* [`segStartY`](kcl/segStartY) * [`shell`](kcl/shell) * [`sin`](kcl/sin) * [`sqrt`](kcl/sqrt) diff --git a/docs/kcl/segEnd.md b/docs/kcl/segEnd.md new file mode 100644 index 0000000000..85ae38b7fc --- /dev/null +++ b/docs/kcl/segEnd.md @@ -0,0 +1,53 @@ +--- +title: "segEnd" +excerpt: "Compute the ending point of the provided line segment." +layout: manual +--- + +Compute the ending point of the provided line segment. + + + +```js +segEnd(tag: TagIdentifier) -> [number] +``` + + +### Arguments + +| Name | Type | Description | Required | +|----------|------|-------------|----------| +| `tag` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | + +### Returns + +`[number]` + + +### Examples + +```js +w = 15 +cube = startSketchAt([0, 0]) + |> line([w, 0], %, $line1) + |> line([0, w], %, $line2) + |> line([-w, 0], %, $line3) + |> line([0, -w], %, $line4) + |> close(%) + |> extrude(5, %) + +fn cylinder = (radius, tag) => { + return startSketchAt([0, 0]) + |> circle({ radius: radius, center: segEnd(tag) }, %) + |> extrude(radius, %) +} + +cylinder(1, line1) +cylinder(2, line2) +cylinder(3, line3) +cylinder(4, line4) +``` + +![Rendered example of segEnd 0]() + + diff --git a/docs/kcl/segStart.md b/docs/kcl/segStart.md new file mode 100644 index 0000000000..dba51ef15c --- /dev/null +++ b/docs/kcl/segStart.md @@ -0,0 +1,56 @@ +--- +title: "segStart" +excerpt: "Compute the starting point of the provided line segment." +layout: manual +--- + +Compute the starting point of the provided line segment. + + + +```js +segStart(tag: TagIdentifier) -> [number] +``` + + +### Arguments + +| Name | Type | Description | Required | +|----------|------|-------------|----------| +| `tag` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | + +### Returns + +`[number]` + + +### Examples + +```js +w = 15 +cube = startSketchAt([0, 0]) + |> line([w, 0], %, $line1) + |> line([0, w], %, $line2) + |> line([-w, 0], %, $line3) + |> line([0, -w], %, $line4) + |> close(%) + |> extrude(5, %) + +fn cylinder = (radius, tag) => { + return startSketchAt([0, 0]) + |> circle({ + radius: radius, + center: segStart(tag) + }, %) + |> extrude(radius, %) +} + +cylinder(1, line1) +cylinder(2, line2) +cylinder(3, line3) +cylinder(4, line4) +``` + +![Rendered example of segStart 0]() + + diff --git a/docs/kcl/segStartX.md b/docs/kcl/segStartX.md new file mode 100644 index 0000000000..fbfea9d334 --- /dev/null +++ b/docs/kcl/segStartX.md @@ -0,0 +1,43 @@ +--- +title: "segStartX" +excerpt: "Compute the starting point of the provided line segment along the 'x' axis." +layout: manual +--- + +Compute the starting point of the provided line segment along the 'x' axis. + + + +```js +segStartX(tag: TagIdentifier) -> number +``` + + +### Arguments + +| Name | Type | Description | Required | +|----------|------|-------------|----------| +| `tag` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | + +### Returns + +`number` + + +### Examples + +```js +exampleSketch = startSketchOn('XZ') + |> startProfileAt([0, 0], %) + |> line([20, 0], %, $thing) + |> line([0, 5], %) + |> line([20 - segStartX(thing), 0], %) + |> line([-20, 10], %) + |> close(%) + +example = extrude(5, exampleSketch) +``` + +![Rendered example of segStartX 0]() + + diff --git a/docs/kcl/segStartY.md b/docs/kcl/segStartY.md new file mode 100644 index 0000000000..33dabd7dd1 --- /dev/null +++ b/docs/kcl/segStartY.md @@ -0,0 +1,44 @@ +--- +title: "segStartY" +excerpt: "Compute the starting point of the provided line segment along the 'y' axis." +layout: manual +--- + +Compute the starting point of the provided line segment along the 'y' axis. + + + +```js +segStartY(tag: TagIdentifier) -> number +``` + + +### Arguments + +| Name | Type | Description | Required | +|----------|------|-------------|----------| +| `tag` | [`TagIdentifier`](/docs/kcl/types#tag-identifier) | | Yes | + +### Returns + +`number` + + +### Examples + +```js +exampleSketch = startSketchOn('XZ') + |> startProfileAt([0, 0], %) + |> line([20, 0], %) + |> line([0, 3], %, $thing) + |> line([-10, 0], %) + |> line([0, 20 - segStartY(thing)], %) + |> line([-10, 0], %) + |> close(%) + +example = extrude(5, exampleSketch) +``` + +![Rendered example of segStartY 0]() + + diff --git a/docs/kcl/std.json b/docs/kcl/std.json index 2a49b6a5fc..9c42301943 100644 --- a/docs/kcl/std.json +++ b/docs/kcl/std.json @@ -744,8 +744,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -1596,8 +1596,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -2499,8 +2499,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -3859,8 +3859,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -4711,8 +4711,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -5614,8 +5614,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -7007,8 +7007,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -8325,8 +8325,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -9691,8 +9691,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -11110,8 +11110,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -12428,8 +12428,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -13794,8 +13794,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -15213,8 +15213,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -16531,8 +16531,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -17897,8 +17897,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -19312,8 +19312,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -20241,8 +20241,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -21559,8 +21559,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -22861,8 +22861,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -24265,8 +24265,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -25583,8 +25583,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -26949,8 +26949,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -28353,8 +28353,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -29671,8 +29671,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -31037,8 +31037,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -32492,8 +32492,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -33810,8 +33810,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -35176,8 +35176,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -37004,8 +37004,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -38322,8 +38322,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -39688,8 +39688,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -41151,8 +41151,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -42107,8 +42107,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -43443,8 +43443,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -44382,8 +44382,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -45732,8 +45732,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Solid": { "description": "An solid is a collection of extrude surfaces.", @@ -47050,8 +47050,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Solid": { "description": "An solid is a collection of extrude surfaces.", @@ -48804,8 +48804,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -50182,8 +50182,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -51500,8 +51500,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -52866,8 +52866,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -54374,8 +54374,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -55409,8 +55409,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -57215,8 +57215,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -58171,8 +58171,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -59507,8 +59507,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -60446,8 +60446,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -62267,8 +62267,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -63165,8 +63165,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -64063,8 +64063,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -64627,8 +64627,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -66002,8 +66002,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -67783,8 +67783,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -69148,8 +69148,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -70514,8 +70514,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -71540,8 +71540,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -72915,8 +72915,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -74397,8 +74397,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 } } }, @@ -75138,8 +75138,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -76527,8 +76527,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -78076,8 +78076,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -79394,8 +79394,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -80760,8 +80760,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -82154,8 +82154,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -83472,8 +83472,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -84838,8 +84838,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -86258,8 +86258,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -87576,8 +87576,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -88558,8 +88558,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -90229,8 +90229,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -92509,8 +92509,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -93973,8 +93971,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -96253,8 +96251,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -97721,8 +97717,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -100001,8 +99997,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -101924,8 +101918,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -102995,8 +102989,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -104303,8 +104297,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -105955,8 +105949,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -107327,8 +107321,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -108387,8 +108381,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -109768,8 +109762,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -111538,8 +111532,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -112910,8 +112904,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -113954,8 +113948,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -115335,8 +115329,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -116711,8 +116705,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -118092,8 +118086,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -119474,8 +119468,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Solid": { "description": "An solid is a collection of extrude surfaces.", @@ -120792,8 +120786,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Solid": { "description": "An solid is a collection of extrude surfaces.", @@ -122546,8 +122540,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -123972,8 +123966,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -125366,8 +125360,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -126755,8 +126749,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -127917,8 +127911,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -130197,8 +130191,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -132052,8 +132044,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -134332,8 +134324,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -135794,8 +135784,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -138074,8 +138064,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -139947,8 +139935,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -142227,8 +142215,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -144082,8 +144068,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -146362,8 +146348,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -147826,8 +147810,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -150106,8 +150090,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -151568,8 +151550,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "TagEngineInfo": { "description": "Engine information for a tag.", @@ -153848,8 +153830,6 @@ "description": "KCL value for an optional parameter which was not given an argument. (remember, parameters are in the function declaration, arguments are in the function call/application).", "type": "object", "required": [ - "end", - "start", "type" ], "properties": { @@ -156150,8 +156130,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -157189,8 +157169,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -158128,8 +158108,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -159893,8 +159873,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -160121,8 +160101,8 @@ ] }, { - "name": "segEndX", - "summary": "Compute the ending point of the provided line segment along the 'x' axis.", + "name": "segEnd", + "summary": "Compute the ending point of the provided line segment.", "description": "", "tags": [], "args": [ @@ -160791,8 +160771,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -161003,24 +160983,29 @@ ], "returnValue": { "name": "", - "type": "number", + "type": "[number]", "schema": { "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", - "title": "double", - "type": "number", - "format": "double" + "title": "Array_size_2_of_double", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 }, "required": true }, "unpublished": false, "deprecated": false, "examples": [ - "exampleSketch = startSketchOn('XZ')\n |> startProfileAt([0, 0], %)\n |> line([20, 0], %, $thing)\n |> line([0, 5], %)\n |> line([segEndX(thing), 0], %)\n |> line([-20, 10], %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" + "w = 15\ncube = startSketchAt([0, 0])\n |> line([w, 0], %, $line1)\n |> line([0, w], %, $line2)\n |> line([-w, 0], %, $line3)\n |> line([0, -w], %, $line4)\n |> close(%)\n |> extrude(5, %)\n\nfn cylinder = (radius, tag) => {\n return startSketchAt([0, 0])\n |> circle({ radius: radius, center: segEnd(tag) }, %)\n |> extrude(radius, %)\n}\n\ncylinder(1, line1)\ncylinder(2, line2)\ncylinder(3, line3)\ncylinder(4, line4)" ] }, { - "name": "segEndY", - "summary": "Compute the ending point of the provided line segment along the 'y' axis.", + "name": "segEndX", + "summary": "Compute the ending point of the provided line segment along the 'x' axis.", "description": "", "tags": [], "args": [ @@ -161689,8 +161674,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -161913,12 +161898,12 @@ "unpublished": false, "deprecated": false, "examples": [ - "exampleSketch = startSketchOn('XZ')\n |> startProfileAt([0, 0], %)\n |> line([20, 0], %)\n |> line([0, 3], %, $thing)\n |> line([-10, 0], %)\n |> line([0, segEndY(thing)], %)\n |> line([-10, 0], %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" + "exampleSketch = startSketchOn('XZ')\n |> startProfileAt([0, 0], %)\n |> line([20, 0], %, $thing)\n |> line([0, 5], %)\n |> line([segEndX(thing), 0], %)\n |> line([-20, 10], %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" ] }, { - "name": "segLen", - "summary": "Compute the length of the provided line segment.", + "name": "segEndY", + "summary": "Compute the ending point of the provided line segment along the 'y' axis.", "description": "", "tags": [], "args": [ @@ -162587,8 +162572,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -162811,102 +162796,46 @@ "unpublished": false, "deprecated": false, "examples": [ - "exampleSketch = startSketchOn(\"XZ\")\n |> startProfileAt([0, 0], %)\n |> angledLine({ angle: 60, length: 10 }, %, $thing)\n |> tangentialArc({ offset: -120, radius: 5 }, %)\n |> angledLine({ angle: -60, length: segLen(thing) }, %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" + "exampleSketch = startSketchOn('XZ')\n |> startProfileAt([0, 0], %)\n |> line([20, 0], %)\n |> line([0, 3], %, $thing)\n |> line([-10, 0], %)\n |> line([0, segEndY(thing)], %)\n |> line([-10, 0], %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" ] }, { - "name": "shell", - "summary": "Remove volume from a 3-dimensional shape such that a wall of the", - "description": "provided thickness remains, taking volume starting at the provided face, leaving it open in that direction.", + "name": "segLen", + "summary": "Compute the length of the provided line segment.", + "description": "", "tags": [], "args": [ { - "name": "data", - "type": "ShellData", + "name": "tag", + "type": "TagIdentifier", "schema": { "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", - "title": "ShellData", - "description": "Data for shells.", + "title": "TagIdentifier", "type": "object", "required": [ - "faces", - "thickness" + "__meta", + "value" ], "properties": { - "thickness": { - "description": "The thickness of the shell.", - "type": "number", - "format": "double" + "value": { + "type": "string" }, - "faces": { - "description": "The faces you want removed.", + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { "type": "array", "items": { - "$ref": "#/components/schemas/FaceTag" + "$ref": "#/components/schemas/Metadata" } } }, "definitions": { - "FaceTag": { - "description": "A tag for a face.", - "anyOf": [ - { - "$ref": "#/components/schemas/StartOrEnd" - }, - { - "description": "A tag for the face.", - "allOf": [ - { - "$ref": "#/components/schemas/TagIdentifier" - } - ] - } - ] - }, - "StartOrEnd": { - "oneOf": [ - { - "description": "The start face as in before you extruded. This could also be known as the bottom face. But we do not call it bottom because it would be the top face if you extruded it in the opposite direction or flipped the camera.", - "type": "string", - "enum": [ - "start" - ] - }, - { - "description": "The end face after you extruded. This could also be known as the top face. But we do not call it top because it would be the bottom face if you extruded it in the opposite direction or flipped the camera.", - "type": "string", - "enum": [ - "end" - ] - } - ] - }, - "TagIdentifier": { - "type": "object", - "required": [ - "__meta", - "value" - ], - "properties": { - "value": { - "type": "string" - }, - "info": { - "allOf": [ - { - "$ref": "#/components/schemas/TagEngineInfo" - } - ], - "nullable": true - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, "TagEngineInfo": { "description": "Engine information for a tag.", "type": "object", @@ -163541,8 +163470,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -163749,168 +163678,62 @@ } }, "required": true + } + ], + "returnValue": { + "name": "", + "type": "number", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "double", + "type": "number", + "format": "double" }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "exampleSketch = startSketchOn(\"XZ\")\n |> startProfileAt([0, 0], %)\n |> angledLine({ angle: 60, length: 10 }, %, $thing)\n |> tangentialArc({ offset: -120, radius: 5 }, %)\n |> angledLine({ angle: -60, length: segLen(thing) }, %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" + ] + }, + { + "name": "segStart", + "summary": "Compute the starting point of the provided line segment.", + "description": "", + "tags": [], + "args": [ { - "name": "solid_set", - "type": "SolidSet", + "name": "tag", + "type": "TagIdentifier", "schema": { "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", - "title": "SolidSet", - "description": "A solid or a group of solids.", - "oneOf": [ - { - "description": "An solid is a collection of extrude surfaces.", - "type": "object", - "required": [ - "__meta", - "height", - "id", - "sketch", - "type", - "value" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "solid" - ] - }, - "id": { - "description": "The id of the solid.", - "type": "string", - "format": "uuid" - }, - "value": { - "description": "The extrude surfaces.", - "type": "array", - "items": { - "$ref": "#/components/schemas/ExtrudeSurface" - } - }, - "sketch": { - "description": "The sketch.", - "allOf": [ - { - "$ref": "#/components/schemas/Sketch" - } - ] - }, - "height": { - "description": "The height of the solid.", - "type": "number", - "format": "double" - }, - "startCapId": { - "description": "The id of the extrusion start cap", - "type": "string", - "format": "uuid", - "nullable": true - }, - "endCapId": { - "description": "The id of the extrusion end cap", - "type": "string", - "format": "uuid", - "nullable": true - }, - "edgeCuts": { - "description": "Chamfers or fillets on this solid.", - "type": "array", - "items": { - "$ref": "#/components/schemas/EdgeCut" - } - }, - "__meta": { - "description": "Metadata.", - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } - }, - { - "type": [ - "object", - "array" - ], - "items": { - "$ref": "#/components/schemas/Solid" - }, - "required": [ - "type" - ], - "properties": { - "type": { - "type": "string", - "enum": [ - "solids" - ] - } - } - } + "title": "TagIdentifier", + "type": "object", + "required": [ + "__meta", + "value" ], - "definitions": { - "FaceTag": { - "description": "A tag for a face.", - "anyOf": [ - { - "$ref": "#/components/schemas/StartOrEnd" - }, - { - "description": "A tag for the face.", - "allOf": [ - { - "$ref": "#/components/schemas/TagIdentifier" - } - ] - } - ] + "properties": { + "value": { + "type": "string" }, - "StartOrEnd": { - "oneOf": [ - { - "description": "The start face as in before you extruded. This could also be known as the bottom face. But we do not call it bottom because it would be the top face if you extruded it in the opposite direction or flipped the camera.", - "type": "string", - "enum": [ - "start" - ] - }, + "info": { + "allOf": [ { - "description": "The end face after you extruded. This could also be known as the top face. But we do not call it top because it would be the bottom face if you extruded it in the opposite direction or flipped the camera.", - "type": "string", - "enum": [ - "end" - ] + "$ref": "#/components/schemas/TagEngineInfo" } - ] - }, - "TagIdentifier": { - "type": "object", - "required": [ - "__meta", - "value" ], - "properties": { - "value": { - "type": "string" - }, - "info": { - "allOf": [ - { - "$ref": "#/components/schemas/TagEngineInfo" - } - ], - "nullable": true - }, - "__meta": { - "type": "array", - "items": { - "$ref": "#/components/schemas/Metadata" - } - } - } + "nullable": true }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + }, + "definitions": { "TagEngineInfo": { "description": "Engine information for a tag.", "type": "object", @@ -164545,8 +164368,3767 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 + }, + "ExtrudeSurface": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + }, + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "fillet" + ] + }, + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + ] + }, + "Metadata": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + } + }, + "required": true + } + ], + "returnValue": { + "name": "", + "type": "[number]", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "Array_size_2_of_double", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "w = 15\ncube = startSketchAt([0, 0])\n |> line([w, 0], %, $line1)\n |> line([0, w], %, $line2)\n |> line([-w, 0], %, $line3)\n |> line([0, -w], %, $line4)\n |> close(%)\n |> extrude(5, %)\n\nfn cylinder = (radius, tag) => {\n return startSketchAt([0, 0])\n |> circle({\n radius: radius,\n center: segStart(tag)\n }, %)\n |> extrude(radius, %)\n}\n\ncylinder(1, line1)\ncylinder(2, line2)\ncylinder(3, line3)\ncylinder(4, line4)" + ] + }, + { + "name": "segStartX", + "summary": "Compute the starting point of the provided line segment along the 'x' axis.", + "description": "", + "tags": [], + "args": [ + { + "name": "tag", + "type": "TagIdentifier", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "TagIdentifier", + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + }, + "definitions": { + "TagEngineInfo": { + "description": "Engine information for a tag.", + "type": "object", + "required": [ + "id", + "sketch" + ], + "properties": { + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" + }, + "sketch": { + "description": "The sketch the tag is on.", + "type": "string", + "format": "uuid" + }, + "path": { + "description": "The path the tag is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Path" + } + ], + "nullable": true + }, + "surface": { + "description": "The surface information for the tag.", + "allOf": [ + { + "$ref": "#/components/schemas/ExtrudeSurface" + } + ], + "nullable": true + } + } + }, + "Path": { + "description": "A path.", + "oneOf": [ + { + "description": "A path that goes to a point.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ToPoint" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment that goes to a point", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArcTo" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArc" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "a complete arc", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Circle" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "the arc's radius", + "type": "number", + "format": "double" + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A path that is horizontal.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type", + "x" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Horizontal" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "An angled line to.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "AngledLineTo" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Base" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A circular arc, not necessarily tangential to the current point.", + "type": "object", + "required": [ + "__geoMeta", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Arc" + ] + }, + "center": { + "description": "Center of the circle that this arc is drawn on.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "Radius of the circle that this arc is drawn on.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + } + ] + }, + "TagDeclarator": { + "type": "object", + "required": [ + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "GeoMeta": { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + "SourceRange": { + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 3, + "minItems": 3 + }, + "ExtrudeSurface": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + }, + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "fillet" + ] + }, + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + ] + }, + "Metadata": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + } + }, + "required": true + } + ], + "returnValue": { + "name": "", + "type": "number", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "double", + "type": "number", + "format": "double" + }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "exampleSketch = startSketchOn('XZ')\n |> startProfileAt([0, 0], %)\n |> line([20, 0], %, $thing)\n |> line([0, 5], %)\n |> line([20 - segStartX(thing), 0], %)\n |> line([-20, 10], %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" + ] + }, + { + "name": "segStartY", + "summary": "Compute the starting point of the provided line segment along the 'y' axis.", + "description": "", + "tags": [], + "args": [ + { + "name": "tag", + "type": "TagIdentifier", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "TagIdentifier", + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + }, + "definitions": { + "TagEngineInfo": { + "description": "Engine information for a tag.", + "type": "object", + "required": [ + "id", + "sketch" + ], + "properties": { + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" + }, + "sketch": { + "description": "The sketch the tag is on.", + "type": "string", + "format": "uuid" + }, + "path": { + "description": "The path the tag is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Path" + } + ], + "nullable": true + }, + "surface": { + "description": "The surface information for the tag.", + "allOf": [ + { + "$ref": "#/components/schemas/ExtrudeSurface" + } + ], + "nullable": true + } + } + }, + "Path": { + "description": "A path.", + "oneOf": [ + { + "description": "A path that goes to a point.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ToPoint" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment that goes to a point", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArcTo" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArc" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "a complete arc", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Circle" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "the arc's radius", + "type": "number", + "format": "double" + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A path that is horizontal.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type", + "x" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Horizontal" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "An angled line to.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "AngledLineTo" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Base" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A circular arc, not necessarily tangential to the current point.", + "type": "object", + "required": [ + "__geoMeta", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Arc" + ] + }, + "center": { + "description": "Center of the circle that this arc is drawn on.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "Radius of the circle that this arc is drawn on.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + } + ] + }, + "TagDeclarator": { + "type": "object", + "required": [ + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "GeoMeta": { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + "SourceRange": { + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 3, + "minItems": 3 + }, + "ExtrudeSurface": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + }, + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "fillet" + ] + }, + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + ] + }, + "Metadata": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + } + }, + "required": true + } + ], + "returnValue": { + "name": "", + "type": "number", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "double", + "type": "number", + "format": "double" + }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "exampleSketch = startSketchOn('XZ')\n |> startProfileAt([0, 0], %)\n |> line([20, 0], %)\n |> line([0, 3], %, $thing)\n |> line([-10, 0], %)\n |> line([0, 20 - segStartY(thing)], %)\n |> line([-10, 0], %)\n |> close(%)\n\nexample = extrude(5, exampleSketch)" + ] + }, + { + "name": "shell", + "summary": "Remove volume from a 3-dimensional shape such that a wall of the", + "description": "provided thickness remains, taking volume starting at the provided face, leaving it open in that direction.", + "tags": [], + "args": [ + { + "name": "data", + "type": "ShellData", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "ShellData", + "description": "Data for shells.", + "type": "object", + "required": [ + "faces", + "thickness" + ], + "properties": { + "thickness": { + "description": "The thickness of the shell.", + "type": "number", + "format": "double" + }, + "faces": { + "description": "The faces you want removed.", + "type": "array", + "items": { + "$ref": "#/components/schemas/FaceTag" + } + } + }, + "definitions": { + "FaceTag": { + "description": "A tag for a face.", + "anyOf": [ + { + "$ref": "#/components/schemas/StartOrEnd" + }, + { + "description": "A tag for the face.", + "allOf": [ + { + "$ref": "#/components/schemas/TagIdentifier" + } + ] + } + ] + }, + "StartOrEnd": { + "oneOf": [ + { + "description": "The start face as in before you extruded. This could also be known as the bottom face. But we do not call it bottom because it would be the top face if you extruded it in the opposite direction or flipped the camera.", + "type": "string", + "enum": [ + "start" + ] + }, + { + "description": "The end face after you extruded. This could also be known as the top face. But we do not call it top because it would be the bottom face if you extruded it in the opposite direction or flipped the camera.", + "type": "string", + "enum": [ + "end" + ] + } + ] + }, + "TagIdentifier": { + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + "TagEngineInfo": { + "description": "Engine information for a tag.", + "type": "object", + "required": [ + "id", + "sketch" + ], + "properties": { + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" + }, + "sketch": { + "description": "The sketch the tag is on.", + "type": "string", + "format": "uuid" + }, + "path": { + "description": "The path the tag is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Path" + } + ], + "nullable": true + }, + "surface": { + "description": "The surface information for the tag.", + "allOf": [ + { + "$ref": "#/components/schemas/ExtrudeSurface" + } + ], + "nullable": true + } + } + }, + "Path": { + "description": "A path.", + "oneOf": [ + { + "description": "A path that goes to a point.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ToPoint" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment that goes to a point", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArcTo" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArc" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "a complete arc", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Circle" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "the arc's radius", + "type": "number", + "format": "double" + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A path that is horizontal.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type", + "x" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Horizontal" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "An angled line to.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "AngledLineTo" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Base" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A circular arc, not necessarily tangential to the current point.", + "type": "object", + "required": [ + "__geoMeta", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Arc" + ] + }, + "center": { + "description": "Center of the circle that this arc is drawn on.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "Radius of the circle that this arc is drawn on.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + } + ] + }, + "TagDeclarator": { + "type": "object", + "required": [ + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "GeoMeta": { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + "SourceRange": { + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 3, + "minItems": 3 + }, + "ExtrudeSurface": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + }, + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + }, + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "fillet" + ] + }, + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "tag": { + "description": "The tag.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + ] + }, + "Metadata": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + } + } + }, + "required": true + }, + { + "name": "solid_set", + "type": "SolidSet", + "schema": { + "$schema": "https://spec.openapis.org/oas/3.0/schema/2019-04-02#/definitions/Schema", + "title": "SolidSet", + "description": "A solid or a group of solids.", + "oneOf": [ + { + "description": "An solid is a collection of extrude surfaces.", + "type": "object", + "required": [ + "__meta", + "height", + "id", + "sketch", + "type", + "value" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "solid" + ] + }, + "id": { + "description": "The id of the solid.", + "type": "string", + "format": "uuid" + }, + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "$ref": "#/components/schemas/ExtrudeSurface" + } + }, + "sketch": { + "description": "The sketch.", + "allOf": [ + { + "$ref": "#/components/schemas/Sketch" + } + ] + }, + "height": { + "description": "The height of the solid.", + "type": "number", + "format": "double" + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "edgeCuts": { + "description": "Chamfers or fillets on this solid.", + "type": "array", + "items": { + "$ref": "#/components/schemas/EdgeCut" + } + }, + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + { + "type": [ + "object", + "array" + ], + "items": { + "$ref": "#/components/schemas/Solid" + }, + "required": [ + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "solids" + ] + } + } + } + ], + "definitions": { + "FaceTag": { + "description": "A tag for a face.", + "anyOf": [ + { + "$ref": "#/components/schemas/StartOrEnd" + }, + { + "description": "A tag for the face.", + "allOf": [ + { + "$ref": "#/components/schemas/TagIdentifier" + } + ] + } + ] + }, + "StartOrEnd": { + "oneOf": [ + { + "description": "The start face as in before you extruded. This could also be known as the bottom face. But we do not call it bottom because it would be the top face if you extruded it in the opposite direction or flipped the camera.", + "type": "string", + "enum": [ + "start" + ] + }, + { + "description": "The end face after you extruded. This could also be known as the top face. But we do not call it top because it would be the bottom face if you extruded it in the opposite direction or flipped the camera.", + "type": "string", + "enum": [ + "end" + ] + } + ] + }, + "TagIdentifier": { + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "info": { + "allOf": [ + { + "$ref": "#/components/schemas/TagEngineInfo" + } + ], + "nullable": true + }, + "__meta": { + "type": "array", + "items": { + "$ref": "#/components/schemas/Metadata" + } + } + } + }, + "TagEngineInfo": { + "description": "Engine information for a tag.", + "type": "object", + "required": [ + "id", + "sketch" + ], + "properties": { + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" + }, + "sketch": { + "description": "The sketch the tag is on.", + "type": "string", + "format": "uuid" + }, + "path": { + "description": "The path the tag is on.", + "allOf": [ + { + "$ref": "#/components/schemas/Path" + } + ], + "nullable": true + }, + "surface": { + "description": "The surface information for the tag.", + "allOf": [ + { + "$ref": "#/components/schemas/ExtrudeSurface" + } + ], + "nullable": true + } + } + }, + "Path": { + "description": "A path.", + "oneOf": [ + { + "description": "A path that goes to a point.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "ToPoint" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment that goes to a point", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArcTo" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "TangentialArc" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "a complete arc", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Circle" + ] + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "the arc's radius", + "type": "number", + "format": "double" + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A path that is horizontal.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type", + "x" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Horizontal" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "An angled line to.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "AngledLineTo" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Base" + ] + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + }, + { + "description": "A circular arc, not necessarily tangential to the current point.", + "type": "object", + "required": [ + "__geoMeta", + "center", + "from", + "radius", + "to", + "type" + ], + "properties": { + "type": { + "type": "string", + "enum": [ + "Arc" + ] + }, + "center": { + "description": "Center of the circle that this arc is drawn on.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "radius": { + "description": "Radius of the circle that this arc is drawn on.", + "type": "number", + "format": "double" + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "allOf": [ + { + "$ref": "#/components/schemas/TagDeclarator" + } + ], + "nullable": true + }, + "__geoMeta": { + "description": "Metadata.", + "allOf": [ + { + "$ref": "#/components/schemas/GeoMeta" + } + ] + } + } + } + ] + }, + "TagDeclarator": { + "type": "object", + "required": [ + "value" + ], + "properties": { + "value": { + "type": "string" + }, + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + } + } + }, + "GeoMeta": { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "allOf": [ + { + "$ref": "#/components/schemas/SourceRange" + } + ] + } + } + }, + "SourceRange": { + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 3, + "minItems": 3 }, "ExtrudeSurface": { "description": "An extrude surface.", @@ -165516,8 +169098,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -166918,8 +170500,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Solid": { "description": "An solid is a collection of extrude surfaces.", @@ -168236,8 +171818,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Solid": { "description": "An solid is a collection of extrude surfaces.", @@ -170132,8 +173714,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -171528,8 +175110,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -172677,8 +176259,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -174095,8 +177677,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Sketch": { "description": "A sketch is a collection of paths.", @@ -175338,8 +178920,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "Solid": { "description": "An solid is a collection of extrude surfaces.", @@ -177316,8 +180898,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -178634,8 +182216,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -180000,8 +183582,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -181393,8 +184975,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -182711,8 +186293,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -184077,8 +187659,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -185470,8 +189052,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -186788,8 +190370,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -188154,8 +191736,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -189641,8 +193223,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -190959,8 +194541,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -192325,8 +195907,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -193713,8 +197295,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -195031,8 +198613,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -196397,8 +199979,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -197785,8 +201367,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -199103,8 +202685,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -200469,8 +204051,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -201857,8 +205439,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -203175,8 +206757,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", @@ -204541,8 +208123,8 @@ "format": "uint", "minimum": 0.0 }, - "maxItems": 2, - "minItems": 2 + "maxItems": 3, + "minItems": 3 }, "SketchSurface": { "description": "A sketch type.", diff --git a/e2e/playwright/editor-tests.spec.ts b/e2e/playwright/editor-tests.spec.ts index 8a42d026ee..05beac6876 100644 --- a/e2e/playwright/editor-tests.spec.ts +++ b/e2e/playwright/editor-tests.spec.ts @@ -632,16 +632,18 @@ test.describe('Editor tests', () => { await u.waitForAuthSkipAppStart() - // this test might be brittle as we add and remove functions - // but should also be easy to update. // tests clicking on an option, selection the first option // and arrowing down to an option await u.codeLocator.click() await page.keyboard.type('sketch001 = start') - // expect there to be six auto complete options - await expect(page.locator('.cm-completionLabel')).toHaveCount(8) + // expect there to be some auto complete options + // exact number depends on the KCL stdlib, so let's just check it's > 0 for now. + await expect(async () => { + const children = await page.locator('.cm-completionLabel').count() + expect(children).toBeGreaterThan(0) + }).toPass() // this makes sure we can accept a completion with click await page.getByText('startSketchOn').click() await page.keyboard.type("'XZ'") diff --git a/e2e/playwright/file-tree.spec.ts b/e2e/playwright/file-tree.spec.ts index 384358708d..ece92ac676 100644 --- a/e2e/playwright/file-tree.spec.ts +++ b/e2e/playwright/file-tree.spec.ts @@ -26,10 +26,6 @@ test.describe('integrations tests', () => { 'Creating a new file or switching file while in sketchMode should exit sketchMode', { tag: '@electron' }, async ({ tronApp, homePage, scene, editor, toolbar }) => { - test.skip( - process.platform === 'win32', - 'windows times out will waiting for the execution indicator?' - ) await tronApp.initialise({ fixtures: { homePage, scene, editor, toolbar }, folderSetupFn: async (dir) => { @@ -55,7 +51,6 @@ test.describe('integrations tests', () => { sortBy: 'last-modified-desc', }) await homePage.openProject('test-sample') - // windows times out here, hence the skip above await scene.waitForExecutionDone() }) await test.step('enter sketch mode', async () => { @@ -71,10 +66,13 @@ test.describe('integrations tests', () => { await toolbar.editSketch() await expect(toolbar.exitSketchBtn).toBeVisible() }) + + const fileName = 'Untitled.kcl' await test.step('check sketch mode is exited when creating new file', async () => { await toolbar.fileTreeBtn.click() await toolbar.expectFileTreeState(['main.kcl']) - await toolbar.createFile({ wait: true }) + + await toolbar.createFile({ fileName, waitForToastToDisappear: true }) // check we're out of sketch mode await expect(toolbar.exitSketchBtn).not.toBeVisible() @@ -93,10 +91,10 @@ test.describe('integrations tests', () => { }) await toolbar.editSketch() await expect(toolbar.exitSketchBtn).toBeVisible() - await toolbar.expectFileTreeState(['main.kcl', 'Untitled.kcl']) + await toolbar.expectFileTreeState(['main.kcl', fileName]) }) await test.step('check sketch mode is exited when opening a different file', async () => { - await toolbar.openFile('untitled.kcl', { wait: false }) + await toolbar.openFile(fileName, { wait: false }) // check we're out of sketch mode await expect(toolbar.exitSketchBtn).not.toBeVisible() diff --git a/e2e/playwright/fixtures/sceneFixture.ts b/e2e/playwright/fixtures/sceneFixture.ts index 3baee5dc87..798868ec53 100644 --- a/e2e/playwright/fixtures/sceneFixture.ts +++ b/e2e/playwright/fixtures/sceneFixture.ts @@ -195,7 +195,7 @@ export class SceneFixture { } waitForExecutionDone = async () => { - await expect(this.exeIndicator).toBeVisible() + await expect(this.exeIndicator).toBeVisible({ timeout: 30000 }) } expectPixelColor = async ( diff --git a/e2e/playwright/fixtures/toolbarFixture.ts b/e2e/playwright/fixtures/toolbarFixture.ts index fb8a908ada..3d303be8a2 100644 --- a/e2e/playwright/fixtures/toolbarFixture.ts +++ b/e2e/playwright/fixtures/toolbarFixture.ts @@ -16,6 +16,7 @@ export class ToolbarFixture { fileCreateToast!: Locator filePane!: Locator exeIndicator!: Locator + treeInputField!: Locator constructor(page: Page) { this.page = page @@ -31,6 +32,7 @@ export class ToolbarFixture { this.editSketchBtn = page.getByText('Edit Sketch') this.fileTreeBtn = page.locator('[id="files-button-holder"]') this.createFileBtn = page.getByTestId('create-file-button') + this.treeInputField = page.getByTestId('tree-input-field') this.filePane = page.locator('#files-pane') this.fileCreateToast = page.getByText('Successfully created') @@ -59,10 +61,15 @@ export class ToolbarFixture { expectFileTreeState = async (expected: string[]) => { await expect.poll(this._serialiseFileTree).toEqual(expected) } - createFile = async ({ wait }: { wait: boolean } = { wait: false }) => { + createFile = async (args: { + fileName: string + waitForToastToDisappear: boolean + }) => { await this.createFileBtn.click() + await this.treeInputField.fill(args.fileName) + await this.treeInputField.press('Enter') await expect(this.fileCreateToast).toBeVisible() - if (wait) { + if (args.waitForToastToDisappear) { await this.fileCreateToast.waitFor({ state: 'detached' }) } } diff --git a/e2e/playwright/lib/console-error-whitelist.ts b/e2e/playwright/lib/console-error-whitelist.ts index 721f1683a2..176c335ef0 100644 --- a/e2e/playwright/lib/console-error-whitelist.ts +++ b/e2e/playwright/lib/console-error-whitelist.ts @@ -18,7 +18,7 @@ export const isErrorWhitelisted = (exception: Error) => { { name: '"{"kind"', message: - '"engine","sourceRanges":[[0,0]],"msg":"Failed to get string from response from engine: `JsValue(undefined)`"}"', + '"engine","sourceRanges":[[0,0,0]],"msg":"Failed to get string from response from engine: `JsValue(undefined)`"}"', stack: '', foundInSpec: 'e2e/playwright/testing-settings.spec.ts', project: 'Google Chrome', @@ -156,8 +156,8 @@ export const isErrorWhitelisted = (exception: Error) => { { name: 'Unhandled Promise Rejection', message: - '{"kind":"engine","sourceRanges":[[0,0]],"msg":"Failed to get string from response from engine: `JsValue(undefined)`"}', - stack: `Unhandled Promise Rejection: {"kind":"engine","sourceRanges":[[0,0]],"msg":"Failed to get string from response from engine: \`JsValue(undefined)\`"} + '{"kind":"engine","sourceRanges":[[0,0,0]],"msg":"Failed to get string from response from engine: `JsValue(undefined)`"}', + stack: `Unhandled Promise Rejection: {"kind":"engine","sourceRanges":[[0,0,0]],"msg":"Failed to get string from response from engine: \`JsValue(undefined)\`"} at unknown (http://localhost:3000/src/lang/std/engineConnection.ts:1245:26)`, foundInSpec: 'e2e/playwright/onboarding-tests.spec.ts Click through each onboarding step', @@ -253,7 +253,7 @@ export const isErrorWhitelisted = (exception: Error) => { { name: '{"kind"', stack: ``, - message: `engine","sourceRanges":[[0,0]],"msg":"Failed to wait for promise from engine: JsValue(\\"Force interrupt, executionIsStale, new AST requested\\")"}`, + message: `engine","sourceRanges":[[0,0,0]],"msg":"Failed to wait for promise from engine: JsValue(\\"Force interrupt, executionIsStale, new AST requested\\")"}`, project: 'Google Chrome', foundInSpec: 'e2e/playwright/testing-settings.spec.ts', }, diff --git a/e2e/playwright/projects.spec.ts b/e2e/playwright/projects.spec.ts index 5b1b6a6eed..1aab2fa7b7 100644 --- a/e2e/playwright/projects.spec.ts +++ b/e2e/playwright/projects.spec.ts @@ -1669,7 +1669,8 @@ test( } ) -test( +// Flaky +test.fixme( 'Original project name persist after onboarding', { tag: '@electron' }, async ({ browserName }, testInfo) => { diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-linux.png index 78e1c50726..2450b64d13 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-win32.png index 5ff8e179df..3c526ec8a5 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-linux.png index fe1a02df5c..fa7c3287f6 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-win32.png index 7e774fc588..cdc1bdcbf2 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Inch-scale-2-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png index 2e8e827a6e..ab801a5987 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-win32.png index 0c5da89c12..b17b2b5662 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png index 1ea5160b96..c448e34f9e 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-win32.png index ab82fb0aa4..ade086c0df 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-linux.png index 22ad3f3e29..5df4be739c 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-win32.png index 1a21406c02..b481cb02af 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-circle-should-look-right-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png index e3a43017c5..96422ab7ad 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png index 25a04b3f20..8fdf3768c2 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-linux.png index ddc31ae4ee..38d1cb522e 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-win32.png index c6a6c0b53b..fe490db6f6 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-linux.png index 2a73202357..a3a2f4088d 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-win32.png index 6e2b1727aa..0a26cefc24 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-segments-should-look-right-2-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-linux.png index d06e5b7b9c..378c43abbb 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-win32.png index 3656ad8693..be55c744a1 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-off-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-linux.png index 66fcb1ddc3..0fc0607366 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-win32.png index d61edb3340..83a3c1d397 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Grid-visibility-Grid-turned-on-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-linux.png index e01b575a0b..140271cb28 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-win32.png index ba30cd7e1a..110ae2f07c 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Sketch-on-face-with-none-z-up-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-linux.png index e609971d98..13b7f6b5ff 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-win32.png index 2a783e73d8..5ec8c667f8 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-2d-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-linux.png index 8fa0f8e247..2ad93f6433 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png index 458fc92bb0..3fe83caa8c 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-linux.png index 9c629d2980..06b144cfc4 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png index 90ba2f4bc6..7cfed13e4a 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-linux.png index 6496ad08cf..6414f5ec17 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-win32.png index f1762c6a20..ed11f9eec1 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XZ-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-linux.png index 306e5a29bf..9b6fb6c3ae 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-win32.png index c82d2fa126..2fd9e09826 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--YZ-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-linux.png index 18a5dd34c7..c14c85ee27 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png index 4b9c82f7c6..7e1c933405 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-linux.png index 4aa0095170..af3ff5c42a 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-win32.png index e8e2ee414b..434e5cc14d 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XZ-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-linux.png index 4c52ae90d2..2e8fff59e1 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-win32.png index a94a0b8c4e..f205ab39b6 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-YZ-1-Google-Chrome-win32.png differ diff --git a/interface.d.ts b/interface.d.ts index 8c06070fa4..af41a148de 100644 --- a/interface.d.ts +++ b/interface.d.ts @@ -78,6 +78,7 @@ export interface IElectronAPI { ) => Electron.IpcRenderer onUpdateError: (callback: (value: { error: Error }) => void) => Electron appRestart: () => void + getArgvParsed: () => any } declare global { diff --git a/package.json b/package.json index 64ba09b975..39732d1323 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "zoo-modeling-app", - "version": "0.26.2", + "version": "0.26.3", "private": true, "productName": "Zoo Modeling App", "author": { @@ -40,7 +40,7 @@ "codemirror": "^6.0.1", "decamelize": "^6.0.0", "electron-squirrel-startup": "^1.0.1", - "electron-updater": "^6.3.0", + "electron-updater": "^6.3.9", "fuse.js": "^7.0.0", "html2canvas-pro": "^1.5.8", "isomorphic-fetch": "^3.0.0", @@ -60,12 +60,13 @@ "sketch-helpers": "^0.0.4", "three": "^0.166.1", "ua-parser-js": "^1.0.37", - "uuid": "^9.0.1", + "uuid": "^11.0.2", "vscode-jsonrpc": "^8.2.1", "vscode-languageserver-protocol": "^3.17.5", "vscode-uri": "^3.0.8", "web-vitals": "^3.5.2", - "xstate": "^5.17.4" + "xstate": "^5.17.4", + "yargs": "^17.7.2" }, "scripts": { "start": "vite", @@ -105,7 +106,8 @@ "tronb:package": "electron-builder --config electron-builder.yml", "test-setup": "yarn install && yarn build:wasm", "test": "vitest --mode development", - "test:unit": "vitest run --mode development", + "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:browser:chrome": "playwright test --project='Google Chrome' --config=playwright.ci.config.ts --grep-invert='@snapshot|@electron'", "test:playwright:browser:chrome:windows": "playwright test --project=\"Google Chrome\" --config=playwright.ci.config.ts --grep-invert=\"@snapshot|@electron|@skipWin\"", "test:playwright:browser:chrome:ubuntu": "playwright test --project='Google Chrome' --config=playwright.ci.config.ts --grep-invert='@snapshot|@electron|@skipLinux'", @@ -117,7 +119,8 @@ "test:playwright:electron:windows:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipWin", "test:playwright:electron:macos:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipMacos", "test:playwright:electron:ubuntu:local": "yarn tron:package && NODE_ENV=development playwright test --config=playwright.electron.config.ts --grep=@electron --grep-invert=@skipLinux", - "test:unit:local": "yarn simpleserver:bg && yarn test:unit; kill-port 3000" + "test:unit:local": "yarn simpleserver:bg && yarn test:unit; kill-port 3000", + "test:unit:kcl-samples:local": "yarn simpleserver:bg && yarn test:unit:kcl-samples; kill-port 3000" }, "prettier": { "trailingComma": "es5", @@ -144,8 +147,8 @@ "@electron-forge/maker-deb": "^7.4.0", "@electron-forge/maker-rpm": "^7.4.0", "@electron-forge/maker-squirrel": "^7.4.0", - "@electron-forge/maker-wix": "^7.4.0", - "@electron-forge/maker-zip": "^7.4.0", + "@electron-forge/maker-wix": "^7.5.0", + "@electron-forge/maker-zip": "^7.5.0", "@electron-forge/plugin-auto-unpack-natives": "^7.4.0", "@electron-forge/plugin-fuses": "^7.4.0", "@electron-forge/plugin-vite": "^7.4.0", @@ -171,7 +174,7 @@ "@types/ua-parser-js": "^0.7.39", "@types/uuid": "^9.0.8", "@types/wicg-file-system-access": "^2023.10.5", - "@types/ws": "^8.5.10", + "@types/ws": "^8.5.13", "@typescript-eslint/eslint-plugin": "^5.0.0", "@typescript-eslint/parser": "^5.0.0", "@vitejs/plugin-react": "^4.3.0", @@ -187,7 +190,7 @@ "eslint-plugin-css-modules": "^2.12.0", "eslint-plugin-import": "^2.30.0", "eslint-plugin-suggest-no-throw": "^1.0.0", - "happy-dom": "^14.3.10", + "happy-dom": "^15.10.2", "http-server": "^14.1.1", "husky": "^9.1.5", "kill-port": "^2.0.1", diff --git a/src/App.tsx b/src/App.tsx index 558ee67ef8..e0beee00da 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -22,6 +22,10 @@ import Gizmo from 'components/Gizmo' import { CoreDumpManager } from 'lib/coredump' import { UnitsMenu } from 'components/UnitsMenu' import { CameraProjectionToggle } from 'components/CameraProjectionToggle' +import { maybeWriteToDisk } from 'lib/telemetry' +maybeWriteToDisk() + .then(() => {}) + .catch(() => {}) export function App() { const { project, file } = useLoaderData() as IndexLoaderData diff --git a/src/Router.tsx b/src/Router.tsx index d135cfc76b..82d42762e2 100644 --- a/src/Router.tsx +++ b/src/Router.tsx @@ -8,6 +8,7 @@ import { } from 'react-router-dom' import { ErrorPage } from './components/ErrorPage' import { Settings } from './routes/Settings' +import { Telemetry } from './routes/Telemetry' import Onboarding, { onboardingRoutes } from './routes/Onboarding' import SignIn from './routes/SignIn' import { Auth } from './Auth' @@ -28,6 +29,7 @@ import { homeLoader, onboardingRedirectLoader, settingsLoader, + telemetryLoader, } from 'lib/routeLoaders' import { CommandBarProvider } from 'components/CommandBar/CommandBarProvider' import SettingsAuthProvider from 'components/SettingsAuthProvider' @@ -43,6 +45,7 @@ import { coreDump } from 'lang/wasm' import { useMemo } from 'react' import { AppStateProvider } from 'AppState' import { reportRejection } from 'lib/trap' +import { RouteProvider } from 'components/RouteProvider' import { ProjectsContextProvider } from 'components/ProjectsContextProvider' const createRouter = isDesktop() ? createHashRouter : createBrowserRouter @@ -56,19 +59,21 @@ const router = createRouter([ * inefficient re-renders, use the react profiler to see. */ element: ( - - - - - - - - - - - - - + + + + + + + + + + + + + + + ), errorElement: , @@ -124,6 +129,16 @@ const router = createRouter([ }, ], }, + { + id: PATHS.FILE + 'TELEMETRY', + loader: telemetryLoader, + children: [ + { + path: makeUrlPathRelative(PATHS.TELEMETRY), + element: , + }, + ], + }, ], }, { @@ -149,6 +164,11 @@ const router = createRouter([ loader: settingsLoader, element: , }, + { + path: makeUrlPathRelative(PATHS.TELEMETRY), + loader: telemetryLoader, + element: , + }, ], }, { diff --git a/src/commandLineArgs.ts b/src/commandLineArgs.ts new file mode 100644 index 0000000000..658364f665 --- /dev/null +++ b/src/commandLineArgs.ts @@ -0,0 +1,12 @@ +import yargs from 'yargs' +import { hideBin } from 'yargs/helpers' + +const argv = yargs(hideBin(process.argv)) + .option('telemetry', { + alias: 't', + type: 'boolean', + description: 'Writes startup telemetry to file on disk.', + }) + .parse() + +export default argv diff --git a/src/components/CustomIcon.tsx b/src/components/CustomIcon.tsx index b99ec4fdb8..17a2700f3e 100644 --- a/src/components/CustomIcon.tsx +++ b/src/components/CustomIcon.tsx @@ -1161,6 +1161,29 @@ const CustomIconMap = { /> ), + stopwatch: ( + + + + + + ), } as const export type CustomIconName = keyof typeof CustomIconMap diff --git a/src/components/FileMachineProvider.tsx b/src/components/FileMachineProvider.tsx index 9c6788e892..3f08619e59 100644 --- a/src/components/FileMachineProvider.tsx +++ b/src/components/FileMachineProvider.tsx @@ -29,6 +29,7 @@ import { KclSamplesManifestItem, } from 'lib/getKclSamplesManifest' import { useSettingsAuthContext } from 'hooks/useSettingsAuthContext' +import { markOnce } from 'lib/performance' type MachineContext = { state: StateFrom @@ -54,6 +55,7 @@ export const FileMachineProvider = ({ ) useEffect(() => { + markOnce('code/didLoadFile') async function fetchKclSamples() { setKclSamples(await getKclSamplesManifest()) } diff --git a/src/components/FileTree.tsx b/src/components/FileTree.tsx index 70610549e8..2eecccfc78 100644 --- a/src/components/FileTree.tsx +++ b/src/components/FileTree.tsx @@ -6,10 +6,10 @@ import { Dispatch, useCallback, useRef, useState } from 'react' import { useNavigate, useRouteLoaderData } from 'react-router-dom' import { Disclosure } from '@headlessui/react' import { FontAwesomeIcon } from '@fortawesome/react-fontawesome' -import { faChevronRight } from '@fortawesome/free-solid-svg-icons' +import { faChevronRight, faPencil } from '@fortawesome/free-solid-svg-icons' import { useFileContext } from 'hooks/useFileContext' import styles from './FileTree.module.css' -import { sortProject } from 'lib/desktopFS' +import { sortFilesAndDirectories } from 'lib/desktopFS' import { FILE_EXT } from 'lib/constants' import { CustomIcon } from './CustomIcon' import { codeManager, kclManager } from 'lib/singletons' @@ -27,6 +27,36 @@ function getIndentationCSS(level: number) { return `calc(1rem * ${level + 1})` } +function TreeEntryInput(props: { + level: number + onSubmit: (value: string) => void +}) { + const [value, setValue] = useState('') + const onKeyPress = (e: React.KeyboardEvent) => { + if (e.key !== 'Enter') return + props.onSubmit(value) + } + + return ( + + ) +} + function RenameForm({ fileOrDir, onSubmit, @@ -113,23 +143,44 @@ function DeleteFileTreeItemDialog({ } const FileTreeItem = ({ + parentDir, project, currentFile, + lastDirectoryClicked, fileOrDir, onNavigateToFile, + onClickDirectory, + onCreateFile, + onCreateFolder, + newTreeEntry, level = 0, + treeSelection, + setTreeSelection, }: { + parentDir: FileEntry | undefined project?: IndexLoaderData['project'] currentFile?: IndexLoaderData['file'] + lastDirectoryClicked?: FileEntry fileOrDir: FileEntry onNavigateToFile?: () => void + onClickDirectory: ( + open: boolean, + path: FileEntry, + parentDir: FileEntry | undefined + ) => void + onCreateFile: (name: string) => void + onCreateFolder: (name: string) => void + newTreeEntry: TreeEntry level?: number + treeSelection: FileEntry | undefined + setTreeSelection: Dispatch> }) => { const { send: fileSend, context: fileContext } = useFileContext() const { onFileOpen, onFileClose } = useLspContext() const navigate = useNavigate() const [isConfirmingDelete, setIsConfirmingDelete] = useState(false) const isCurrentFile = fileOrDir.path === currentFile?.path + const isFileOrDirHighlighted = treeSelection?.path === fileOrDir?.path const itemRef = useRef(null) // Since every file or directory gets its own FileTreeItem, we can do this. @@ -156,6 +207,10 @@ const FileTreeItem = ({ [fileOrDir.path] ) + const showNewTreeEntry = + newTreeEntry !== undefined && + fileOrDir.path === fileContext.selectedDirectory.path + const isRenaming = fileContext.itemsBeingRenamed.includes(fileOrDir.path) const removeCurrentItemFromRenaming = useCallback( () => @@ -179,13 +234,6 @@ const FileTreeItem = ({ }) }, [fileContext.itemsBeingRenamed, fileOrDir.path, fileSend]) - const clickDirectory = () => { - fileSend({ - type: 'Set selected directory', - directory: fileOrDir, - }) - } - function handleKeyUp(e: React.KeyboardEvent) { if (e.metaKey && e.key === 'Backspace') { // Open confirmation dialog @@ -199,6 +247,8 @@ const FileTreeItem = ({ } async function handleClick() { + setTreeSelection(fileOrDir) + if (fileOrDir.children !== null) return // Don't open directories if (fileOrDir.name?.endsWith(FILE_EXT) === false && project?.path) { @@ -220,16 +270,19 @@ const FileTreeItem = ({ // Open kcl files navigate(`${PATHS.FILE}/${encodeURIComponent(fileOrDir.path)}`) } + onNavigateToFile?.() } + // The below handles both the "root" of all directories and all subs. It's + // why some code is duplicated. return (
{fileOrDir.children === null ? (
  • e.currentTarget.focus()} - onClickCapture={clickDirectory} - onFocusCapture={clickDirectory} + onClick={(e) => { + e.stopPropagation() + onClickDirectory(open, fileOrDir, parentDir) + }} onKeyDown={(e) => e.key === 'Enter' && e.preventDefault()} onKeyUp={handleKeyUp} > @@ -315,35 +367,69 @@ const FileTreeItem = ({ >
      { - fileSend({ - type: 'Set selected directory', - directory: fileOrDir, - }) + onClick={(e) => { + e.stopPropagation() + onClickDirectory(open, fileOrDir, parentDir) }} - onFocusCapture={(e) => - fileSend({ - type: 'Set selected directory', - directory: fileOrDir, - }) - } > - {fileOrDir.children?.map((child) => ( - - ))} + {showNewTreeEntry && ( +
      + + + newTreeEntry === 'file' + ? onCreateFile(value) + : onCreateFolder(value) + } + /> +
      + )} + {sortFilesAndDirectories(fileOrDir.children || []).map( + (child) => ( + + ) + )} + {!showNewTreeEntry && fileOrDir.children?.length === 0 && ( +
      +
      No files
      +
      + )}
  • )} )} + {isConfirmingDelete && ( void } -export const FileTreeMenu = () => { - const { send } = useFileContext() - const { send: modelingSend } = useModelingContext() - - function createFile() { - send({ - type: 'Create file', - data: { name: '', makeDir: false, shouldSetToRename: true }, - }) - modelingSend({ type: 'Cancel' }) - } - - function createFolder() { - send({ - type: 'Create file', - data: { name: '', makeDir: true, shouldSetToRename: true }, - }) - } - - useHotkeyWrapper(['mod + n'], createFile) - useHotkeyWrapper(['mod + shift + n'], createFolder) +export const FileTreeMenu = ({ + onCreateFile, + onCreateFolder, +}: { + onCreateFile: () => void + onCreateFolder: () => void +}) => { + useHotkeyWrapper(['mod + n'], onCreateFile) + useHotkeyWrapper(['mod + shift + n'], onCreateFolder) return ( <> @@ -440,7 +514,7 @@ export const FileTreeMenu = () => { bgClassName: 'bg-transparent', }} className="!p-0 !bg-transparent hover:text-primary border-transparent hover:border-primary !outline-none" - onClick={createFile} + onClick={onCreateFile} > Create file @@ -456,7 +530,7 @@ export const FileTreeMenu = () => { bgClassName: 'bg-transparent', }} className="!p-0 !bg-transparent hover:text-primary border-transparent hover:border-primary !outline-none" - onClick={createFolder} + onClick={onCreateFolder} > Create folder @@ -466,30 +540,110 @@ export const FileTreeMenu = () => { ) } +type TreeEntry = 'file' | 'folder' | undefined + +export const useFileTreeOperations = () => { + const { send } = useFileContext() + const { send: modelingSend } = useModelingContext() + + // As long as this is undefined, a new "file tree entry prompt" is not shown. + const [newTreeEntry, setNewTreeEntry] = useState(undefined) + + function createFile(args: { dryRun: boolean; name?: string }) { + if (args.dryRun) { + setNewTreeEntry('file') + return + } + + // Clear so that the entry prompt goes away. + setNewTreeEntry(undefined) + + if (!args.name) return + + send({ + type: 'Create file', + data: { name: args.name, makeDir: false, shouldSetToRename: false }, + }) + modelingSend({ type: 'Cancel' }) + } + + function createFolder(args: { dryRun: boolean; name?: string }) { + if (args.dryRun) { + setNewTreeEntry('folder') + return + } + + setNewTreeEntry(undefined) + + if (!args.name) return + + send({ + type: 'Create file', + data: { name: args.name, makeDir: true, shouldSetToRename: false }, + }) + } + + return { + createFile, + createFolder, + newTreeEntry, + } +} + export const FileTree = ({ className = '', onNavigateToFile: closePanel, }: FileTreeProps) => { + const { createFile, createFolder, newTreeEntry } = useFileTreeOperations() + return (

    Files

    - + createFile({ dryRun: true })} + onCreateFolder={() => createFolder({ dryRun: true })} + />
    - + createFile({ dryRun: false, name })} + onCreateFolder={(name: string) => createFolder({ dryRun: false, name })} + />
    ) } export const FileTreeInner = ({ onNavigateToFile, + onCreateFile, + onCreateFolder, + newTreeEntry, }: { + onCreateFile: (name: string) => void + onCreateFolder: (name: string) => void + newTreeEntry: TreeEntry onNavigateToFile?: () => void }) => { const loaderData = useRouteLoaderData(PATHS.FILE) as IndexLoaderData const { send: fileSend, context: fileContext } = useFileContext() const { send: modelingSend } = useModelingContext() + const [lastDirectoryClicked, setLastDirectoryClicked] = useState< + FileEntry | undefined + >(undefined) + + const [treeSelection, setTreeSelection] = useState( + loaderData.file + ) + + const onNavigateToFile_ = () => { + // Reset modeling state when navigating to a new file + onNavigateToFile?.() + modelingSend({ type: 'Cancel' }) + } + // Refresh the file tree when there are changes. useFileSystemWatcher( async (eventType, path) => { @@ -513,33 +667,81 @@ export const FileTreeInner = ({ ) ) - const clickDirectory = () => { + const onTreeEntryInputSubmit = (value: string) => { + if (newTreeEntry === 'file') { + onCreateFile(value) + onNavigateToFile_() + } else { + onCreateFolder(value) + } + } + + const onClickDirectory = ( + open_: boolean, + fileOrDir: FileEntry, + parentDir: FileEntry | undefined + ) => { + // open true is closed... it's broken. Save me. I've inverted it here for + // sanity. + const open = !open_ + + const target = open ? fileOrDir : parentDir + + // We're at the root, can't select anything further + if (!target) return + + setTreeSelection(target) + setLastDirectoryClicked(target) fileSend({ type: 'Set selected directory', - directory: fileContext.project, + directory: target, }) } + const showNewTreeEntry = + newTreeEntry !== undefined && + fileContext.selectedDirectory.path === loaderData.project?.path + return ( -
    -
      - {sortProject(fileContext.project?.children || []).map((fileOrDir) => ( - { - // Reset modeling state when navigating to a new file - modelingSend({ type: 'Cancel' }) - onNavigateToFile?.() - }} - key={fileOrDir.path} - /> - ))} -
    +
    +
    +
      + {showNewTreeEntry && ( +
      + + +
      + )} + {sortFilesAndDirectories(fileContext.project?.children || []).map( + (fileOrDir) => ( + + ) + )} +
    +
    ) } diff --git a/src/components/LowerRightControls.tsx b/src/components/LowerRightControls.tsx index ad3ddb8082..41531ef42f 100644 --- a/src/components/LowerRightControls.tsx +++ b/src/components/LowerRightControls.tsx @@ -96,6 +96,23 @@ export function LowerRightControls({ Report a bug + + + Telemetry + + Telemetry + + { editorManager.setCopilotEnabled(true) }, - 'sketch exit execute': ({ context: { store } }) => { - ;(async () => { - // When cancelling the sketch mode we should disable sketch mode within the engine. - await engineCommandManager.sendSceneCommand({ - type: 'modeling_cmd_req', - cmd_id: uuidv4(), - cmd: { type: 'sketch_mode_disable' }, - }) + // tsc reports this typing as perfectly fine, but eslint is complaining. + // It's actually nonsensical, so I'm quieting. + // eslint-disable-next-line @typescript-eslint/no-misused-promises + 'sketch exit execute': async ({ + context: { store }, + }): Promise => { + // When cancelling the sketch mode we should disable sketch mode within the engine. + await engineCommandManager.sendSceneCommand({ + type: 'modeling_cmd_req', + cmd_id: uuidv4(), + cmd: { type: 'sketch_mode_disable' }, + }) - sceneInfra.camControls.syncDirection = 'clientToEngine' + sceneInfra.camControls.syncDirection = 'clientToEngine' - if (cameraProjection.current === 'perspective') { - await sceneInfra.camControls.snapToPerspectiveBeforeHandingBackControlToEngine() - } + if (cameraProjection.current === 'perspective') { + await sceneInfra.camControls.snapToPerspectiveBeforeHandingBackControlToEngine() + } - sceneInfra.camControls.syncDirection = 'engineToClient' + sceneInfra.camControls.syncDirection = 'engineToClient' - store.videoElement?.pause() + store.videoElement?.pause() - kclManager - .executeCode() - .then(() => { - if (engineCommandManager.engineConnection?.idleMode) return + return kclManager + .executeCode() + .then(() => { + if (engineCommandManager.engineConnection?.idleMode) return - store.videoElement?.play().catch((e) => { - console.warn('Video playing was prevented', e) - }) + store.videoElement?.play().catch((e) => { + console.warn('Video playing was prevented', e) }) - .catch(reportRejection) - })().catch(reportRejection) + }) + .catch(reportRejection) }, 'Set mouse state': assign(({ context, event }) => { if (event.type !== 'Set mouse state') return {} diff --git a/src/components/ModelingSidebar/ModelingPane.tsx b/src/components/ModelingSidebar/ModelingPane.tsx index a5ffc079f5..2f959b4192 100644 --- a/src/components/ModelingSidebar/ModelingPane.tsx +++ b/src/components/ModelingSidebar/ModelingPane.tsx @@ -48,7 +48,7 @@ export const ModelingPaneHeader = ({ bgClassName: 'bg-transparent dark:bg-transparent', }} className="!p-0 !bg-transparent hover:text-primary border-transparent dark:!border-transparent hover:!border-primary dark:hover:!border-chalkboard-70 !outline-none" - onClick={onClose} + onClick={() => onClose()} > Close @@ -59,14 +59,12 @@ export const ModelingPaneHeader = ({ } export const ModelingPane = ({ - title, - icon, id, children, className, - Menu, detailsTestId, onClose, + title, ...props }: ModelingPaneProps) => { const { settings } = useSettingsAuthContext() @@ -78,6 +76,7 @@ export const ModelingPane = ({ return (
    - -
    {children}
    + {children}
    ) } diff --git a/src/components/ModelingSidebar/ModelingPanes/DebugPane.tsx b/src/components/ModelingSidebar/ModelingPanes/DebugPane.tsx index f96cb9ef69..c98810ded6 100644 --- a/src/components/ModelingSidebar/ModelingPanes/DebugPane.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/DebugPane.tsx @@ -5,16 +5,18 @@ import { CamDebugSettings } from 'clientSideScene/ClientSideSceneComp' export const DebugPane = () => { return ( -
    -
    - - - - -
    -
    +
    +
    +
    + + + + +
    +
    +
    ) } diff --git a/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx b/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx index f27f412267..e43132958d 100644 --- a/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/KclEditorPane.tsx @@ -4,7 +4,7 @@ import { Themes, getSystemTheme } from 'lib/theme' import { useMemo, useRef } from 'react' import { highlightSelectionMatches, searchKeymap } from '@codemirror/search' import { lineHighlightField } from 'editor/highlightextension' -import { roundOff } from 'lib/utils' +import { onMouseDragMakeANewNumber, onMouseDragRegex } from 'lib/utils' import { lineNumbers, rectangularSelection, @@ -129,7 +129,9 @@ export const KclEditorPane = () => { closeBrackets(), highlightActiveLine(), highlightSelectionMatches(), - syntaxHighlighting(defaultHighlightStyle, { fallback: true }), + syntaxHighlighting(defaultHighlightStyle, { + fallback: true, + }), rectangularSelection(), dropCursor(), interact({ @@ -137,29 +139,12 @@ export const KclEditorPane = () => { // a rule for a number dragger { // the regexp matching the value - regexp: /-?\b\d+\.?\d*\b/g, + regexp: onMouseDragRegex, // set cursor to "ew-resize" on hover cursor: 'ew-resize', // change number value based on mouse X movement on drag onDrag: (text, setText, e) => { - const multiplier = - e.shiftKey && e.metaKey - ? 0.01 - : e.metaKey - ? 0.1 - : e.shiftKey - ? 10 - : 1 - - const delta = e.movementX * multiplier - - const newVal = roundOff( - Number(text) + delta, - multiplier === 0.01 ? 2 : multiplier === 0.1 ? 1 : 0 - ) - - if (isNaN(newVal)) return - setText(newVal.toString()) + onMouseDragMakeANewNumber(text, setText, e) }, }, ], @@ -174,27 +159,31 @@ export const KclEditorPane = () => { const initialCode = useRef(codeManager.code) return ( -
    - { - if (_editorView === null) return +
    +
    + { + if (_editorView === null) return - editorManager.setEditorView(_editorView) + editorManager.setEditorView(_editorView) - // On first load of this component, ensure we show the current errors - // in the editor. - // Make sure we don't add them twice. - if (diagnosticCount(_editorView.state) === 0) { - kclManager.setDiagnosticsForCurrentErrors() - } - }} - /> + // On first load of this component, ensure we show the current errors + // in the editor. + // Make sure we don't add them twice. + if (diagnosticCount(_editorView.state) === 0) { + kclManager.setDiagnosticsForCurrentErrors() + } + }} + /> +
    ) } diff --git a/src/components/ModelingSidebar/ModelingPanes/MemoryPane.test.tsx b/src/components/ModelingSidebar/ModelingPanes/MemoryPane.test.tsx index 6b1931875e..6b3adfbb2e 100644 --- a/src/components/ModelingSidebar/ModelingPanes/MemoryPane.test.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/MemoryPane.test.tsx @@ -43,14 +43,14 @@ describe('processMemory', () => { tag: null, id: expect.any(String), faceId: expect.any(String), - sourceRange: [170, 194], + sourceRange: [170, 194, 0], }, { type: 'extrudePlane', tag: null, id: expect.any(String), faceId: expect.any(String), - sourceRange: [202, 230], + sourceRange: [202, 230, 0], }, ], theSketch: [ diff --git a/src/components/ModelingSidebar/ModelingPanes/index.tsx b/src/components/ModelingSidebar/ModelingPanes/index.tsx index f48c16b0ca..3284275343 100644 --- a/src/components/ModelingSidebar/ModelingPanes/index.tsx +++ b/src/components/ModelingSidebar/ModelingPanes/index.tsx @@ -2,11 +2,17 @@ import { IconDefinition, faBugSlash } from '@fortawesome/free-solid-svg-icons' import { KclEditorMenu } from 'components/ModelingSidebar/ModelingPanes/KclEditorMenu' import { CustomIconName } from 'components/CustomIcon' import { KclEditorPane } from 'components/ModelingSidebar/ModelingPanes/KclEditorPane' +import { ModelingPaneHeader } from 'components/ModelingSidebar/ModelingPane' import { MouseEventHandler, ReactNode } from 'react' import { MemoryPane, MemoryPaneMenu } from './MemoryPane' import { LogsPane } from './LoggingPanes' import { DebugPane } from './DebugPane' -import { FileTreeInner, FileTreeMenu, FileTreeRoot } from 'components/FileTree' +import { + FileTreeInner, + FileTreeMenu, + FileTreeRoot, + useFileTreeOperations, +} from 'components/FileTree' import { useKclContext } from 'lang/KclProvider' import { editorManager } from 'lib/singletons' import { ContextFrom } from 'xstate' @@ -38,20 +44,19 @@ interface PaneCallbackProps { export type SidebarPane = { id: SidebarType - title: ReactNode - sidebarName?: string + sidebarName: string icon: CustomIconName | IconDefinition keybinding: string - Content: ReactNode | React.FC - Menu?: ReactNode | React.FC + Content: React.FC<{ id: SidebarType; onClose: () => void }> hide?: boolean | ((props: PaneCallbackProps) => boolean) showBadge?: BadgeInfo } export type SidebarAction = { id: string - title: ReactNode + sidebarName: string icon: CustomIconName + title: ReactNode iconClassName?: string // Just until we get rid of FontAwesome icons keybinding: string action: () => void @@ -59,14 +64,30 @@ export type SidebarAction = { disable?: () => string | undefined } +// For now a lot of icons are the same but the reality is they could totally +// be different, like an icon based on some data for the pane, or the icon +// changes to be a spinning loader on loading. + export const sidebarPanes: SidebarPane[] = [ { id: 'code', - title: 'KCL Code', icon: 'code', - Content: KclEditorPane, + sidebarName: 'KCL Code', + Content: (props: { id: SidebarType; onClose: () => void }) => { + return ( + <> + } + onClose={props.onClose} + /> + + + ) + }, keybinding: 'Shift + C', - Menu: KclEditorMenu, showBadge: { value: ({ kclContext }) => { return kclContext.errors.length @@ -79,34 +100,96 @@ export const sidebarPanes: SidebarPane[] = [ }, { id: 'files', - title: , - sidebarName: 'Project Files', icon: 'folder', - Content: FileTreeInner, + sidebarName: 'Project Files', + Content: (props: { id: SidebarType; onClose: () => void }) => { + const { createFile, createFolder, newTreeEntry } = useFileTreeOperations() + + return ( + <> + } + Menu={ + createFile({ dryRun: true })} + onCreateFolder={() => createFolder({ dryRun: true })} + /> + } + onClose={props.onClose} + /> + createFile({ dryRun: false, name })} + onCreateFolder={(name: string) => + createFolder({ dryRun: false, name }) + } + newTreeEntry={newTreeEntry} + /> + + ) + }, keybinding: 'Shift + F', - Menu: FileTreeMenu, hide: ({ platform }) => platform === 'web', }, { id: 'variables', - title: 'Variables', icon: 'make-variable', - Content: MemoryPane, - Menu: MemoryPaneMenu, + sidebarName: 'Variables', + Content: (props: { id: SidebarType; onClose: () => void }) => { + return ( + <> + } + onClose={props.onClose} + /> + + + ) + }, keybinding: 'Shift + V', }, { id: 'logs', - title: 'Logs', icon: 'logs', - Content: LogsPane, + sidebarName: 'Logs', + Content: (props: { id: SidebarType; onClose: () => void }) => { + return ( + <> + + + + ) + }, keybinding: 'Shift + L', }, { id: 'debug', - title: 'Debug', icon: faBugSlash, - Content: DebugPane, + sidebarName: 'Debug', + Content: (props: { id: SidebarType; onClose: () => void }) => { + return ( + <> + + + + ) + }, keybinding: 'Shift + D', hide: ({ settings }) => !settings.modeling.showDebugPanel.current, }, diff --git a/src/components/ModelingSidebar/ModelingSidebar.module.css b/src/components/ModelingSidebar/ModelingSidebar.module.css index c2bda853d7..e69de29bb2 100644 --- a/src/components/ModelingSidebar/ModelingSidebar.module.css +++ b/src/components/ModelingSidebar/ModelingSidebar.module.css @@ -1,11 +0,0 @@ -.grid { - display: grid; - grid-template-columns: auto 1fr; - grid-template-rows: 1fr 1fr; - row-gap: 0.25rem; - align-items: stretch; - position: relative; - padding-block: 1px; - max-width: 100%; - flex: 1 1 0; -} diff --git a/src/components/ModelingSidebar/ModelingSidebar.tsx b/src/components/ModelingSidebar/ModelingSidebar.tsx index 81a3ee2026..77eeca84bb 100644 --- a/src/components/ModelingSidebar/ModelingSidebar.tsx +++ b/src/components/ModelingSidebar/ModelingSidebar.tsx @@ -5,14 +5,12 @@ import { useCallback, useEffect, useMemo, - ReactNode, useContext, } from 'react' import { useHotkeys } from 'react-hotkeys-hook' import { SidebarAction, SidebarType, sidebarPanes } from './ModelingPanes' import Tooltip from 'components/Tooltip' import { ActionIcon } from 'components/ActionIcon' -import styles from './ModelingSidebar.module.css' import { ModelingPane } from './ModelingPane' import { isDesktop } from 'lib/isDesktop' import { useModelingContext } from 'hooks/useModelingContext' @@ -62,6 +60,7 @@ export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { { id: 'export', title: 'Export part', + sidebarName: 'Export part', icon: 'floppyDiskArrow', keybinding: 'Ctrl + Shift + E', action: () => @@ -73,6 +72,7 @@ export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { { id: 'make', title: 'Make part', + sidebarName: 'Make part', icon: 'printer3d', keybinding: 'Ctrl + Shift + M', // eslint-disable-next-line @typescript-eslint/no-misused-promises @@ -182,7 +182,7 @@ export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { bottomRight: 'hidden', }} > -
    +
      = 1 - ? `row-start-1 row-end-3` - : `hidden`) + 'ml-[-1px] col-start-2 col-span-1 flex flex-col items-stretch gap-2 ' + + (context.store?.openPanes.length >= 1 ? `w-full` : `hidden`) } > {filteredPanes @@ -249,13 +247,15 @@ export function ModelingSidebar({ paneOpacity }: ModelingSidebarProps) { {}} id={`${pane.id}-pane`} - title={pane.title} - Menu={pane.Menu} - onClose={() => togglePane(pane.id)} > {pane.Content instanceof Function ? ( - + togglePane(pane.id)} + /> ) : ( pane.Content )} @@ -271,8 +271,7 @@ interface ModelingPaneButtonProps extends React.HTMLAttributes { paneConfig: { id: string - title: ReactNode - sidebarName?: string + sidebarName: string icon: CustomIconName | IconDefinition keybinding: string iconClassName?: string @@ -301,10 +300,7 @@ function ModelingPaneButton({ +
    +
    +
    + +
    + + + + + ) +} diff --git a/src/wasm-lib/Cargo.lock b/src/wasm-lib/Cargo.lock index 2076424b10..57b368856e 100644 --- a/src/wasm-lib/Cargo.lock +++ b/src/wasm-lib/Cargo.lock @@ -121,9 +121,9 @@ dependencies = [ [[package]] name = "anyhow" -version = "1.0.92" +version = "1.0.93" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "74f37166d7d48a0284b99dd824694c26119c700b53bf0d1540cdb147dbdaaf13" +checksum = "4c95c10ba0b00a02636238b814946408b1322d5ac4760326e6fb8ec956d85775" dependencies = [ "backtrace", ] @@ -165,7 +165,7 @@ dependencies = [ "futures-sink", "log", "pin-project-lite", - "thiserror", + "thiserror 1.0.68", ] [[package]] @@ -1140,7 +1140,7 @@ dependencies = [ "pest_derive", "serde", "serde_json", - "thiserror", + "thiserror 1.0.68", ] [[package]] @@ -1377,6 +1377,124 @@ dependencies = [ "cc", ] +[[package]] +name = "icu_collections" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "db2fa452206ebee18c4b5c2274dbf1de17008e874b4dc4f0aea9d01ca79e4526" +dependencies = [ + "displaydoc", + "yoke", + "zerofrom", + "zerovec", +] + +[[package]] +name = "icu_locid" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "13acbb8371917fc971be86fc8057c41a64b521c184808a698c02acc242dbf637" +dependencies = [ + "displaydoc", + "litemap", + "tinystr", + "writeable", + "zerovec", +] + +[[package]] +name = "icu_locid_transform" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "01d11ac35de8e40fdeda00d9e1e9d92525f3f9d887cdd7aa81d727596788b54e" +dependencies = [ + "displaydoc", + "icu_locid", + "icu_locid_transform_data", + "icu_provider", + "tinystr", + "zerovec", +] + +[[package]] +name = "icu_locid_transform_data" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "fdc8ff3388f852bede6b579ad4e978ab004f139284d7b28715f773507b946f6e" + +[[package]] +name = "icu_normalizer" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "19ce3e0da2ec68599d193c93d088142efd7f9c5d6fc9b803774855747dc6a84f" +dependencies = [ + "displaydoc", + "icu_collections", + "icu_normalizer_data", + "icu_properties", + "icu_provider", + "smallvec", + "utf16_iter", + "utf8_iter", + "write16", + "zerovec", +] + +[[package]] +name = "icu_normalizer_data" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "f8cafbf7aa791e9b22bec55a167906f9e1215fd475cd22adfcf660e03e989516" + +[[package]] +name = "icu_properties" +version = "1.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "93d6020766cfc6302c15dbbc9c8778c37e62c14427cb7f6e601d849e092aeef5" +dependencies = [ + "displaydoc", + "icu_collections", + "icu_locid_transform", + "icu_properties_data", + "icu_provider", + "tinystr", + "zerovec", +] + +[[package]] +name = "icu_properties_data" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "67a8effbc3dd3e4ba1afa8ad918d5684b8868b3b26500753effea8d2eed19569" + +[[package]] +name = "icu_provider" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6ed421c8a8ef78d3e2dbc98a973be2f3770cb42b606e3ab18d6237c4dfde68d9" +dependencies = [ + "displaydoc", + "icu_locid", + "icu_provider_macros", + "stable_deref_trait", + "tinystr", + "writeable", + "yoke", + "zerofrom", + "zerovec", +] + +[[package]] +name = "icu_provider_macros" +version = "1.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "ident_case" version = "1.0.1" @@ -1385,19 +1503,30 @@ checksum = "b9e0384b61958566e926dc50660321d12159025e767c18e043daf26b70104c39" [[package]] name = "idna" -version = "0.5.0" +version = "1.0.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "686f825264d630750a544639377bae737628043f20d38bbc029e8f29ea968a7e" +dependencies = [ + "idna_adapter", + "smallvec", + "utf8_iter", +] + +[[package]] +name = "idna_adapter" +version = "1.2.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "634d9b1461af396cad843f47fdba5597a4f9e6ddd4bfb6ff5d85028c25cb12f6" +checksum = "daca1df1c957320b2cf139ac61e7bd64fed304c5040df000a745aa1de3b4ef71" dependencies = [ - "unicode-bidi", - "unicode-normalization", + "icu_normalizer", + "icu_properties", ] [[package]] name = "image" -version = "0.25.3" +version = "0.25.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "d97eb9a8e0cd5b76afea91d7eecd5cf8338cd44ced04256cf1f800474b227c52" +checksum = "cd6f44aed642f18953a158afeb30206f4d50da59fbc66ecb53c66488de73563b" dependencies = [ "bytemuck", "byteorder-lite", @@ -1414,7 +1543,7 @@ dependencies = [ "image", "itertools 0.12.1", "rayon", - "thiserror", + "thiserror 1.0.68", ] [[package]] @@ -1453,9 +1582,9 @@ checksum = "a257582fdcde896fd96463bf2d40eefea0580021c0712a0e2b028b60b47a837a" [[package]] name = "insta" -version = "1.41.0" +version = "1.41.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a1f72d3e19488cf7d8ea52d2fc0f8754fc933398b337cd3cbdb28aaeb35159ef" +checksum = "7e9ffc4d4892617c50a928c52b2961cb5174b6fc6ebf252b2fac9d21955c48b8" dependencies = [ "console", "lazy_static", @@ -1587,7 +1716,7 @@ dependencies = [ "serde_json", "sha2", "tabled", - "thiserror", + "thiserror 2.0.0", "tokio", "tokio-tungstenite", "toml", @@ -1677,7 +1806,7 @@ dependencies = [ "serde_bytes", "serde_json", "serde_urlencoded", - "thiserror", + "thiserror 1.0.68", "tokio", "tracing", "url", @@ -1686,9 +1815,9 @@ dependencies = [ [[package]] name = "kittycad-modeling-cmds" -version = "0.2.71" +version = "0.2.72" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c6d2160dcb0e5373b1242a760dbf17fb9c12de523c410c11b145381c852b377b" +checksum = "e41880dbe19df3150992988f438c9ba9022ea822d6e4e5ed53d28965de198ec7" dependencies = [ "anyhow", "chrono", @@ -1774,6 +1903,12 @@ version = "0.4.14" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "78b3ae25bc7c8c38cec158d1f2757ee79e9b3740fbc7ccf0e59e4b08d793fa89" +[[package]] +name = "litemap" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "643cb0b8d4fcc284004d5fd0d67ccf61dfffadb7f75e1e71bc420f4688a3a704" + [[package]] name = "lock_api" version = "0.4.12" @@ -2016,7 +2151,7 @@ dependencies = [ "js-sys", "once_cell", "pin-project-lite", - "thiserror", + "thiserror 1.0.68", ] [[package]] @@ -2034,7 +2169,7 @@ dependencies = [ "opentelemetry", "percent-encoding", "rand 0.8.5", - "thiserror", + "thiserror 1.0.68", ] [[package]] @@ -2159,7 +2294,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "fdbef9d1d47087a895abd220ed25eb4ad973a5e26f6a4367b038c25e28dfc2d9" dependencies = [ "memchr", - "thiserror", + "thiserror 1.0.68", "ucd-trie", ] @@ -2215,7 +2350,7 @@ dependencies = [ "serde", "serde_derive", "strum", - "thiserror", + "thiserror 1.0.68", ] [[package]] @@ -2352,6 +2487,28 @@ dependencies = [ "version_check", ] +[[package]] +name = "proc-macro-error-attr2" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "96de42df36bb9bba5542fe9f1a054b8cc87e172759a1868aa05c1f3acc89dfc5" +dependencies = [ + "proc-macro2", + "quote", +] + +[[package]] +name = "proc-macro-error2" +version = "2.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "11ec05c52be0a07b08061f7dd003e7d7092e0472bc731b4af7bb1ef876109802" +dependencies = [ + "proc-macro-error-attr2", + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "proc-macro2" version = "1.0.89" @@ -2363,9 +2520,9 @@ dependencies = [ [[package]] name = "pyo3" -version = "0.22.5" +version = "0.22.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "3d922163ba1f79c04bc49073ba7b32fd5a8d3b76a87c955921234b8e77333c51" +checksum = "f402062616ab18202ae8319da13fa4279883a2b8a9d9f83f20dbade813ce1884" dependencies = [ "cfg-if", "indoc", @@ -2381,9 +2538,9 @@ dependencies = [ [[package]] name = "pyo3-build-config" -version = "0.22.5" +version = "0.22.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bc38c5feeb496c8321091edf3d63e9a6829eab4b863b4a6a65f26f3e9cc6b179" +checksum = "b14b5775b5ff446dd1056212d778012cbe8a0fbffd368029fd9e25b514479c38" dependencies = [ "once_cell", "target-lexicon", @@ -2391,9 +2548,9 @@ dependencies = [ [[package]] name = "pyo3-ffi" -version = "0.22.5" +version = "0.22.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "94845622d88ae274d2729fcefc850e63d7a3ddff5e3ce11bd88486db9f1d357d" +checksum = "9ab5bcf04a2cdcbb50c7d6105de943f543f9ed92af55818fd17b660390fc8636" dependencies = [ "libc", "pyo3-build-config", @@ -2401,9 +2558,9 @@ dependencies = [ [[package]] name = "pyo3-macros" -version = "0.22.5" +version = "0.22.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e655aad15e09b94ffdb3ce3d217acf652e26bbc37697ef012f5e5e348c716e5e" +checksum = "0fd24d897903a9e6d80b968368a34e1525aeb719d568dba8b3d4bfa5dc67d453" dependencies = [ "proc-macro2", "pyo3-macros-backend", @@ -2413,9 +2570,9 @@ dependencies = [ [[package]] name = "pyo3-macros-backend" -version = "0.22.5" +version = "0.22.6" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae1e3f09eecd94618f60a455a23def79f79eba4dc561a97324bf9ac8c6df30ce" +checksum = "36c011a03ba1e50152b4b394b479826cad97e7a21eb52df179cd91ac411cbfbe" dependencies = [ "heck 0.5.0", "proc-macro2", @@ -2446,7 +2603,7 @@ dependencies = [ "rustc-hash", "rustls", "socket2", - "thiserror", + "thiserror 1.0.68", "tokio", "tracing", ] @@ -2463,7 +2620,7 @@ dependencies = [ "rustc-hash", "rustls", "slab", - "thiserror", + "thiserror 1.0.68", "tinyvec", "tracing", ] @@ -2650,9 +2807,9 @@ checksum = "2b15c43186be67a4fd63bee50d0303afffcef381492ebe2c5d87f324e1b8815c" [[package]] name = "reqwest" -version = "0.12.8" +version = "0.12.9" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f713147fbe92361e52392c73b8c9e48c04c6625bce969ef54dc901e58e042a7b" +checksum = "a77c62af46e79de0a562e1a9849205ffcb7fc1238876e9bd743357570e04046f" dependencies = [ "base64 0.22.1", "bytes", @@ -2717,7 +2874,7 @@ dependencies = [ "http 1.1.0", "reqwest", "serde", - "thiserror", + "thiserror 1.0.68", "tower-service", ] @@ -2814,7 +2971,7 @@ checksum = "f1adc9dfed5cc999077978cc7163b9282c5751c8d39827c4ea8c8c220ca5a440" dependencies = [ "serde", "tempfile", - "thiserror", + "thiserror 1.0.68", "toml", "toolchain_find", ] @@ -2926,6 +3083,7 @@ dependencies = [ "chrono", "dyn-clone", "indexmap 1.9.3", + "indexmap 2.6.0", "schemars_derive", "serde", "serde_json", @@ -3178,6 +3336,12 @@ version = "0.9.8" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "6980e8d7511241f8acf4aebddbb1ff938df5eebe98691418c4468d0b72a96a67" +[[package]] +name = "stable_deref_trait" +version = "1.2.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a8f112729512f8e442d81f95a8a7ddf2b7c6b8a1a6f509a95864142b30cab2d3" + [[package]] name = "str_indices" version = "0.4.3" @@ -3343,18 +3507,38 @@ dependencies = [ [[package]] name = "thiserror" -version = "1.0.65" +version = "1.0.68" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "02dd99dc800bbb97186339685293e1cc5d9df1f8fae2d0aecd9ff1c77efea892" +dependencies = [ + "thiserror-impl 1.0.68", +] + +[[package]] +name = "thiserror" +version = "2.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15291287e9bff1bc6f9ff3409ed9af665bec7a5fc8ac079ea96be07bca0e2668" +dependencies = [ + "thiserror-impl 2.0.0", +] + +[[package]] +name = "thiserror-impl" +version = "1.0.68" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5d11abd9594d9b38965ef50805c5e469ca9cc6f197f883f717e0269a3057b3d5" +checksum = "a7c61ec9a6f64d2793d8a45faba21efbe3ced62a886d44c36a009b2b519b4c7e" dependencies = [ - "thiserror-impl", + "proc-macro2", + "quote", + "syn 2.0.87", ] [[package]] name = "thiserror-impl" -version = "1.0.65" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ae71770322cbd277e69d762a16c444af02aa0575ac0d174f0b9562d3b37f8602" +checksum = "22efd00f33f93fa62848a7cab956c3d38c8d43095efda1decfc2b3a5dc0b8972" dependencies = [ "proc-macro2", "quote", @@ -3402,6 +3586,16 @@ dependencies = [ "time-core", ] +[[package]] +name = "tinystr" +version = "0.7.6" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9117f5d4db391c1cf6927e7bea3db74b9a1c1add8f7eda9ffd5364f40f57b82f" +dependencies = [ + "displaydoc", + "zerovec", +] + [[package]] name = "tinytemplate" version = "1.2.1" @@ -3690,9 +3884,10 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "3a2f31991cee3dce1ca4f929a8a04fdd11fd8801aac0f2030b0fa8a0a3fef6b9" dependencies = [ "chrono", + "indexmap 2.6.0", "lazy_static", "serde_json", - "thiserror", + "thiserror 1.0.68", "ts-rs-macros", "url", "uuid", @@ -3726,7 +3921,7 @@ dependencies = [ "rustls", "rustls-pki-types", "sha1", - "thiserror", + "thiserror 1.0.68", "utf-8", ] @@ -3763,27 +3958,12 @@ dependencies = [ "version_check", ] -[[package]] -name = "unicode-bidi" -version = "0.3.15" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "08f95100a766bf4f8f28f90d77e0a5461bbdb219042e7679bebe79004fed8d75" - [[package]] name = "unicode-ident" version = "1.0.13" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "e91b56cd4cadaeb79bbf1a5645f6b4f8dc5bde8834ad5894a8db35fda9efa1fe" -[[package]] -name = "unicode-normalization" -version = "0.1.24" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "5033c97c4262335cded6d6fc3e5c18ab755e1a3dc96376350f3d8e9f009ad956" -dependencies = [ - "tinyvec", -] - [[package]] name = "unicode-segmentation" version = "1.12.0" @@ -3816,9 +3996,9 @@ checksum = "8ecb6da28b8a351d773b68d5825ac39017e680750f980f3a1a85cd8dd28a47c1" [[package]] name = "url" -version = "2.5.2" +version = "2.5.3" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "22784dbdf76fdde8af1aeda5622b546b422b6fc585325248a2bf9f5e41e94d6c" +checksum = "8d157f1b96d14500ffdc1f10ba712e780825526c03d9a49b4d0324b0d9113ada" dependencies = [ "form_urlencoded", "idna", @@ -3838,6 +4018,18 @@ version = "0.7.6" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "09cc8ee72d2a9becf2f2febe0205bbed8fc6615b7cb429ad062dc7b7ddd036a9" +[[package]] +name = "utf16_iter" +version = "1.0.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c8232dd3cdaed5356e0f716d285e4b40b932ac434100fe9b7e0e8e935b9e6246" + +[[package]] +name = "utf8_iter" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "b6c140620e7ffbb22c2dee59cafe6084a59b5ffc27a8859a5f0d494b5d52b6be" + [[package]] name = "utf8parse" version = "0.2.2" @@ -3857,9 +4049,9 @@ dependencies = [ [[package]] name = "validator" -version = "0.18.1" +version = "0.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "db79c75af171630a3148bd3e6d7c4f42b6a9a014c2945bc5ed0020cbb8d9478e" +checksum = "d0b4a29d8709210980a09379f27ee31549b73292c87ab9899beee1c0d3be6303" dependencies = [ "idna", "once_cell", @@ -3873,13 +4065,13 @@ dependencies = [ [[package]] name = "validator_derive" -version = "0.18.2" +version = "0.19.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "df0bcf92720c40105ac4b2dda2a4ea3aa717d4d6a862cc217da653a4bd5c6b10" +checksum = "bac855a2ce6f843beb229757e6e570a42e837bcb15e5f449dd48d5747d41bf77" dependencies = [ "darling", "once_cell", - "proc-macro-error", + "proc-macro-error2", "proc-macro2", "quote", "syn 2.0.87", @@ -4303,6 +4495,18 @@ dependencies = [ "memchr", ] +[[package]] +name = "write16" +version = "1.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "d1890f4022759daae28ed4fe62859b1236caebfc61ede2f63ed4e695f3f6d936" + +[[package]] +name = "writeable" +version = "0.5.5" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1e9df38ee2d2c3c5948ea468a8406ff0db0b29ae1ffde1bcf20ef305bcc95c51" + [[package]] name = "wyz" version = "0.5.1" @@ -4327,6 +4531,30 @@ version = "1.0.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "cfe53a6657fd280eaa890a3bc59152892ffa3e30101319d168b781ed6529b049" +[[package]] +name = "yoke" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6c5b1314b079b0930c31e3af543d8ee1757b1951ae1e1565ec704403a7240ca5" +dependencies = [ + "serde", + "stable_deref_trait", + "yoke-derive", + "zerofrom", +] + +[[package]] +name = "yoke-derive" +version = "0.7.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "28cc31741b18cb6f1d5ff12f5b7523e3d6eb0852bbbad19d73905511d9849b95" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", + "synstructure", +] + [[package]] name = "zerocopy" version = "0.7.35" @@ -4348,12 +4576,55 @@ dependencies = [ "syn 2.0.87", ] +[[package]] +name = "zerofrom" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "91ec111ce797d0e0784a1116d0ddcdbea84322cd79e5d5ad173daeba4f93ab55" +dependencies = [ + "zerofrom-derive", +] + +[[package]] +name = "zerofrom-derive" +version = "0.1.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0ea7b4a3637ea8669cedf0f1fd5c286a17f3de97b8dd5a70a6c167a1730e63a5" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", + "synstructure", +] + [[package]] name = "zeroize" version = "1.8.1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" +[[package]] +name = "zerovec" +version = "0.10.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "aa2b893d79df23bfb12d5461018d408ea19dfafe76c2c7ef6d4eba614f8ff079" +dependencies = [ + "yoke", + "zerofrom", + "zerovec-derive", +] + +[[package]] +name = "zerovec-derive" +version = "0.10.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6eafa6dfb17584ea3e2bd6e76e0cc15ad7af12b09abdd1ca55961bed9b1063c6" +dependencies = [ + "proc-macro2", + "quote", + "syn 2.0.87", +] + [[package]] name = "zip" version = "2.2.0" @@ -4366,5 +4637,5 @@ dependencies = [ "displaydoc", "indexmap 2.6.0", "memchr", - "thiserror", + "thiserror 1.0.68", ] diff --git a/src/wasm-lib/Cargo.toml b/src/wasm-lib/Cargo.toml index d0579c67b9..413bc69519 100644 --- a/src/wasm-lib/Cargo.toml +++ b/src/wasm-lib/Cargo.toml @@ -24,7 +24,7 @@ wasm-bindgen-futures = "0.4.44" [dev-dependencies] anyhow = "1" -image = { version = "0.25.3", default-features = false, features = ["png"] } +image = { version = "0.25.5", default-features = false, features = ["png"] } kittycad = { workspace = true, default-features = true } kittycad-modeling-cmds = { workspace = true } pretty_assertions = "1.4.1" @@ -76,7 +76,7 @@ members = [ [workspace.dependencies] http = "1" kittycad = { version = "0.3.25", default-features = false, features = ["js", "requests"] } -kittycad-modeling-cmds = { version = "0.2.71", features = ["websocket"] } +kittycad-modeling-cmds = { version = "0.2.72", features = ["websocket"] } [[test]] name = "executor" diff --git a/src/wasm-lib/derive-docs/Cargo.toml b/src/wasm-lib/derive-docs/Cargo.toml index 099460356e..5c198853d6 100644 --- a/src/wasm-lib/derive-docs/Cargo.toml +++ b/src/wasm-lib/derive-docs/Cargo.toml @@ -23,7 +23,7 @@ serde_tokenstream = "0.2" syn = { version = "2.0.87", features = ["full"] } [dev-dependencies] -anyhow = "1.0.92" +anyhow = "1.0.93" expectorate = "1.1.0" pretty_assertions = "1.4.1" rustfmt-wrapper = "0.2.1" diff --git a/src/wasm-lib/derive-docs/src/lib.rs b/src/wasm-lib/derive-docs/src/lib.rs index 8281ed24ba..d9887c9bee 100644 --- a/src/wasm-lib/derive-docs/src/lib.rs +++ b/src/wasm-lib/derive-docs/src/lib.rs @@ -173,9 +173,7 @@ fn do_stdlib_inner( quote! { let code_blocks = vec![#(#cb),*]; code_blocks.iter().map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; @@ -750,9 +748,7 @@ fn generate_code_block_test(fn_name: &str, code_block: &str, index: usize) -> pr quote! { #[tokio::test(flavor = "multi_thread")] async fn #test_name_mock() { - let tokens = crate::token::lexer(#code_block).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(#code_block).unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new(crate::engine::conn_mock::EngineConnection::new().await.unwrap())), diff --git a/src/wasm-lib/derive-docs/tests/args_with_lifetime.gen b/src/wasm-lib/derive-docs/tests/args_with_lifetime.gen index a826f845af..9a3f6cef74 100644 --- a/src/wasm-lib/derive-docs/tests/args_with_lifetime.gen +++ b/src/wasm-lib/derive-docs/tests/args_with_lifetime.gen @@ -2,9 +2,7 @@ mod test_examples_someFn { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_someFn0() { - let tokens = crate::token::lexer("someFn()").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse("someFn()").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +111,7 @@ impl crate::docs::StdLibFn for SomeFn { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/args_with_refs.gen b/src/wasm-lib/derive-docs/tests/args_with_refs.gen index f75f6dff18..7a7f70e34f 100644 --- a/src/wasm-lib/derive-docs/tests/args_with_refs.gen +++ b/src/wasm-lib/derive-docs/tests/args_with_refs.gen @@ -2,9 +2,7 @@ mod test_examples_someFn { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_someFn0() { - let tokens = crate::token::lexer("someFn()").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse("someFn()").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +111,7 @@ impl crate::docs::StdLibFn for SomeFn { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/array.gen b/src/wasm-lib/derive-docs/tests/array.gen index d9dec25e8c..2f6069e55d 100644 --- a/src/wasm-lib/derive-docs/tests/array.gen +++ b/src/wasm-lib/derive-docs/tests/array.gen @@ -2,9 +2,9 @@ mod test_examples_show { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_show0() { - let tokens = crate::token::lexer("This is another code block.\nyes sirrr.\nshow").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is another code block.\nyes sirrr.\nshow") + .unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -36,9 +36,8 @@ mod test_examples_show { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_show1() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nshow").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nshow").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -150,9 +149,7 @@ impl crate::docs::StdLibFn for Show { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/box.gen b/src/wasm-lib/derive-docs/tests/box.gen index 1ec101210b..6ff618161f 100644 --- a/src/wasm-lib/derive-docs/tests/box.gen +++ b/src/wasm-lib/derive-docs/tests/box.gen @@ -2,9 +2,8 @@ mod test_examples_show { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_show0() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nshow").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nshow").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +112,7 @@ impl crate::docs::StdLibFn for Show { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/doc_comment_with_code.gen b/src/wasm-lib/derive-docs/tests/doc_comment_with_code.gen index 03bf0f2240..3c5d416cf0 100644 --- a/src/wasm-lib/derive-docs/tests/doc_comment_with_code.gen +++ b/src/wasm-lib/derive-docs/tests/doc_comment_with_code.gen @@ -2,10 +2,9 @@ mod test_examples_my_func { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_my_func0() { - let tokens = - crate::token::lexer("This is another code block.\nyes sirrr.\nmyFunc").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is another code block.\nyes sirrr.\nmyFunc") + .unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -37,9 +36,8 @@ mod test_examples_my_func { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_my_func1() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nmyFunc").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nmyFunc").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -151,9 +149,7 @@ impl crate::docs::StdLibFn for MyFunc { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/lineTo.gen b/src/wasm-lib/derive-docs/tests/lineTo.gen index c699a8a490..013b722f2c 100644 --- a/src/wasm-lib/derive-docs/tests/lineTo.gen +++ b/src/wasm-lib/derive-docs/tests/lineTo.gen @@ -2,10 +2,9 @@ mod test_examples_line_to { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_line_to0() { - let tokens = - crate::token::lexer("This is another code block.\nyes sirrr.\nlineTo").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is another code block.\nyes sirrr.\nlineTo") + .unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -37,9 +36,8 @@ mod test_examples_line_to { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_line_to1() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nlineTo").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nlineTo").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -159,9 +157,7 @@ impl crate::docs::StdLibFn for LineTo { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/min.gen b/src/wasm-lib/derive-docs/tests/min.gen index 63c776961f..076679a873 100644 --- a/src/wasm-lib/derive-docs/tests/min.gen +++ b/src/wasm-lib/derive-docs/tests/min.gen @@ -2,9 +2,8 @@ mod test_examples_min { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_min0() { - let tokens = crate::token::lexer("This is another code block.\nyes sirrr.\nmin").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is another code block.\nyes sirrr.\nmin").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -36,9 +35,8 @@ mod test_examples_min { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_min1() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nmin").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nmin").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -150,9 +148,7 @@ impl crate::docs::StdLibFn for Min { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/option.gen b/src/wasm-lib/derive-docs/tests/option.gen index a3624d9eed..23208fda81 100644 --- a/src/wasm-lib/derive-docs/tests/option.gen +++ b/src/wasm-lib/derive-docs/tests/option.gen @@ -2,9 +2,8 @@ mod test_examples_show { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_show0() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nshow").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nshow").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +112,7 @@ impl crate::docs::StdLibFn for Show { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/option_input_format.gen b/src/wasm-lib/derive-docs/tests/option_input_format.gen index 0473f1ebec..5ae970f03e 100644 --- a/src/wasm-lib/derive-docs/tests/option_input_format.gen +++ b/src/wasm-lib/derive-docs/tests/option_input_format.gen @@ -2,9 +2,8 @@ mod test_examples_import { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_import0() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nimport").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nimport").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +112,7 @@ impl crate::docs::StdLibFn for Import { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/return_vec_box_sketch.gen b/src/wasm-lib/derive-docs/tests/return_vec_box_sketch.gen index 83b21aa2ad..ce8838efeb 100644 --- a/src/wasm-lib/derive-docs/tests/return_vec_box_sketch.gen +++ b/src/wasm-lib/derive-docs/tests/return_vec_box_sketch.gen @@ -2,9 +2,8 @@ mod test_examples_import { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_import0() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nimport").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nimport").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +112,7 @@ impl crate::docs::StdLibFn for Import { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/return_vec_sketch.gen b/src/wasm-lib/derive-docs/tests/return_vec_sketch.gen index 1f0670cd52..c7075331a8 100644 --- a/src/wasm-lib/derive-docs/tests/return_vec_sketch.gen +++ b/src/wasm-lib/derive-docs/tests/return_vec_sketch.gen @@ -2,9 +2,8 @@ mod test_examples_import { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_import0() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nimport").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nimport").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +112,7 @@ impl crate::docs::StdLibFn for Import { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/show.gen b/src/wasm-lib/derive-docs/tests/show.gen index 822c93c2ae..600f581a28 100644 --- a/src/wasm-lib/derive-docs/tests/show.gen +++ b/src/wasm-lib/derive-docs/tests/show.gen @@ -2,9 +2,8 @@ mod test_examples_show { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_show0() { - let tokens = crate::token::lexer("This is code.\nIt does other shit.\nshow").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = + crate::parser::top_level_parse("This is code.\nIt does other shit.\nshow").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -113,9 +112,7 @@ impl crate::docs::StdLibFn for Show { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/derive-docs/tests/test_args_with_exec_state.gen b/src/wasm-lib/derive-docs/tests/test_args_with_exec_state.gen index 87d2c4b3c2..8e4661bbe1 100644 --- a/src/wasm-lib/derive-docs/tests/test_args_with_exec_state.gen +++ b/src/wasm-lib/derive-docs/tests/test_args_with_exec_state.gen @@ -2,9 +2,7 @@ mod test_examples_some_function { #[tokio::test(flavor = "multi_thread")] async fn test_mock_example_some_function0() { - let tokens = crate::token::lexer("someFunction()").unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse("someFunction()").unwrap(); let id_generator = crate::executor::IdGenerator::default(); let ctx = crate::executor::ExecutorContext { engine: std::sync::Arc::new(Box::new( @@ -108,9 +106,7 @@ impl crate::docs::StdLibFn for SomeFunction { code_blocks .iter() .map(|cb| { - let tokens = crate::token::lexer(cb).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(cb).unwrap(); let mut options: crate::ast::types::FormatOptions = Default::default(); options.insert_final_newline = false; program.recast(&options, 0) diff --git a/src/wasm-lib/kcl-macros/src/lib.rs b/src/wasm-lib/kcl-macros/src/lib.rs index 611731325c..d19cf98e44 100644 --- a/src/wasm-lib/kcl-macros/src/lib.rs +++ b/src/wasm-lib/kcl-macros/src/lib.rs @@ -16,8 +16,7 @@ use syn::{parse_macro_input, LitStr}; pub fn parse(input: TokenStream) -> TokenStream { let input = parse_macro_input!(input as LitStr); let kcl_src = input.value(); - let tokens = kcl_lib::token::lexer(&kcl_src).unwrap(); - let ast = kcl_lib::parser::Parser::new(tokens).ast().unwrap(); + let ast = kcl_lib::parser::top_level_parse(&kcl_src).unwrap(); let ast_struct = ast.bake(&Default::default()); quote!(#ast_struct).into() } diff --git a/src/wasm-lib/kcl-macros/tests/macro_test.rs b/src/wasm-lib/kcl-macros/tests/macro_test.rs index 19408eb7ee..bf39ea1ebb 100644 --- a/src/wasm-lib/kcl-macros/tests/macro_test.rs +++ b/src/wasm-lib/kcl-macros/tests/macro_test.rs @@ -1,6 +1,6 @@ extern crate alloc; use kcl_lib::ast::types::{ - BodyItem, Expr, Identifier, ItemVisibility, Literal, LiteralValue, Node, Program, VariableDeclaration, + BodyItem, Expr, Identifier, ItemVisibility, Literal, LiteralValue, ModuleId, Node, Program, VariableDeclaration, VariableDeclarator, VariableKind, }; use kcl_macros::parse; @@ -9,6 +9,7 @@ use pretty_assertions::assert_eq; #[test] fn basic() { let actual = parse!("const y = 4"); + let module_id = ModuleId::default(); let expected = Node { inner: Program { body: vec![BodyItem::VariableDeclaration(Box::new(Node::new( @@ -22,6 +23,7 @@ fn basic() { }, 6, 7, + module_id, ), init: Expr::Literal(Box::new(Node::new( Literal { @@ -31,11 +33,13 @@ fn basic() { }, 10, 11, + module_id, ))), digest: None, }, 6, 11, + module_id, )], visibility: ItemVisibility::Default, kind: VariableKind::Const, @@ -43,12 +47,14 @@ fn basic() { }, 0, 11, + module_id, )))], non_code_meta: Default::default(), digest: None, }, start: 0, end: 11, + module_id, }; assert_eq!(expected, actual); } diff --git a/src/wasm-lib/kcl-test-server/Cargo.toml b/src/wasm-lib/kcl-test-server/Cargo.toml index b48a63f53d..bfb5d02b08 100644 --- a/src/wasm-lib/kcl-test-server/Cargo.toml +++ b/src/wasm-lib/kcl-test-server/Cargo.toml @@ -6,7 +6,7 @@ edition = "2021" license = "MIT" [dependencies] -anyhow = "1.0.92" +anyhow = "1.0.93" hyper = { version = "0.14.29", features = ["http1", "server", "tcp"] } kcl-lib = { version = "0.2", path = "../kcl" } pico-args = "0.5.0" diff --git a/src/wasm-lib/kcl-test-server/src/lib.rs b/src/wasm-lib/kcl-test-server/src/lib.rs index 45de4eec23..54e1dbfa9e 100644 --- a/src/wasm-lib/kcl-test-server/src/lib.rs +++ b/src/wasm-lib/kcl-test-server/src/lib.rs @@ -15,7 +15,7 @@ use hyper::{ service::{make_service_fn, service_fn}, Body, Error, Response, Server, }; -use kcl_lib::{executor::ExecutorContext, settings::types::UnitLength, test_server::RequestBody}; +use kcl_lib::{ast::types::ModuleId, executor::ExecutorContext, settings::types::UnitLength, test_server::RequestBody}; use tokio::{ sync::{mpsc, oneshot}, task::JoinHandle, @@ -157,7 +157,8 @@ async fn snapshot_endpoint(body: Bytes, state: ExecutorContext) -> Response return bad_request(format!("Invalid request JSON: {e}")), }; let RequestBody { kcl_program, test_name } = body; - let parser = match kcl_lib::token::lexer(&kcl_program) { + let module_id = ModuleId::default(); + let parser = match kcl_lib::token::lexer(&kcl_program, module_id) { Ok(ts) => kcl_lib::parser::Parser::new(ts), Err(e) => return bad_request(format!("tokenization error: {e}")), }; diff --git a/src/wasm-lib/kcl-to-core/src/lib.rs b/src/wasm-lib/kcl-to-core/src/lib.rs index ce007d177c..1b6b19a161 100644 --- a/src/wasm-lib/kcl-to-core/src/lib.rs +++ b/src/wasm-lib/kcl-to-core/src/lib.rs @@ -7,9 +7,7 @@ mod conn_mock_core; ///Converts the given kcl code to an engine test pub async fn kcl_to_engine_core(code: &str) -> Result { - let tokens = kcl_lib::token::lexer(code)?; - let parser = kcl_lib::parser::Parser::new(tokens); - let program = parser.ast()?; + let program = kcl_lib::parser::top_level_parse(code)?; let result = Arc::new(Mutex::new("".into())); let ref_result = Arc::clone(&result); diff --git a/src/wasm-lib/kcl/Cargo.toml b/src/wasm-lib/kcl/Cargo.toml index 74756ff142..5a134bd70b 100644 --- a/src/wasm-lib/kcl/Cargo.toml +++ b/src/wasm-lib/kcl/Cargo.toml @@ -11,7 +11,7 @@ keywords = ["kcl", "KittyCAD", "CAD"] # See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html [dependencies] -anyhow = { version = "1.0.92", features = ["backtrace"] } +anyhow = { version = "1.0.93", features = ["backtrace"] } async-recursion = "1.1.1" async-trait = "0.1.83" base64 = "0.22.1" @@ -26,7 +26,7 @@ futures = { version = "0.3.31" } git_rev = "0.1.0" gltf-json = "1.4.1" http = { workspace = true } -image = { version = "0.25.3", default-features = false, features = ["png"] } +image = { version = "0.25.5", default-features = false, features = ["png"] } indexmap = { version = "2.6.0", features = ["serde"] } kittycad = { workspace = true } kittycad-modeling-cmds = { workspace = true } @@ -34,21 +34,21 @@ lazy_static = "1.5.0" measurements = "0.11.0" mime_guess = "2.0.5" parse-display = "0.9.1" -pyo3 = { version = "0.22.5", optional = true } +pyo3 = { version = "0.22.6", optional = true } reqwest = { version = "0.12", default-features = false, features = ["stream", "rustls-tls"] } ropey = "1.6.1" -schemars = { version = "0.8.17", features = ["impl_json_schema", "url", "uuid1", "preserve_order"] } +schemars = { version = "0.8.17", features = ["impl_json_schema", "indexmap2", "url", "uuid1", "preserve_order"] } serde = { version = "1.0.214", features = ["derive"] } serde_json = "1.0.128" sha2 = "0.10.8" tabled = { version = "0.15.0", optional = true } -thiserror = "1.0.65" +thiserror = "2.0.0" toml = "0.8.19" -ts-rs = { version = "10.0.0", features = ["uuid-impl", "url-impl", "chrono-impl", "no-serde-warnings", "serde-json-impl"] } -url = { version = "2.5.2", features = ["serde"] } +ts-rs = { version = "10.0.0", features = ["uuid-impl", "url-impl", "chrono-impl", "indexmap-impl", "no-serde-warnings", "serde-json-impl"] } +url = { version = "2.5.3", features = ["serde"] } urlencoding = "2.1.3" uuid = { version = "1.11.0", features = ["v4", "js", "serde"] } -validator = { version = "0.18.1", features = ["derive"] } +validator = { version = "0.19.0", features = ["derive"] } winnow = "0.6.18" zip = { version = "2.0.0", default-features = false } @@ -85,8 +85,8 @@ criterion = { version = "0.5.1", features = ["async_tokio"] } expectorate = "1.1.0" handlebars = "6.2.0" iai = "0.1" -image = { version = "0.25.3", default-features = false, features = ["png"] } -insta = { version = "1.41.0", features = ["json", "filters"] } +image = { version = "0.25.5", default-features = false, features = ["png"] } +insta = { version = "1.41.1", features = ["json", "filters"] } itertools = "0.13.0" pretty_assertions = "1.4.1" tokio = { version = "1.40.0", features = ["rt-multi-thread", "macros", "time"] } diff --git a/src/wasm-lib/kcl/benches/compiler_benchmark_criterion.rs b/src/wasm-lib/kcl/benches/compiler_benchmark_criterion.rs index f913a26658..6205afc6f4 100644 --- a/src/wasm-lib/kcl/benches/compiler_benchmark_criterion.rs +++ b/src/wasm-lib/kcl/benches/compiler_benchmark_criterion.rs @@ -1,9 +1,10 @@ use criterion::{black_box, criterion_group, criterion_main, Criterion}; pub fn bench_lex(c: &mut Criterion) { - c.bench_function("lex_cube", |b| b.iter(|| lex(CUBE_PROGRAM))); - c.bench_function("lex_big_kitt", |b| b.iter(|| lex(KITT_PROGRAM))); - c.bench_function("lex_pipes_on_pipes", |b| b.iter(|| lex(PIPES_PROGRAM))); + let module_id = kcl_lib::ast::types::ModuleId::default(); + c.bench_function("lex_cube", |b| b.iter(|| lex(CUBE_PROGRAM, module_id))); + c.bench_function("lex_big_kitt", |b| b.iter(|| lex(KITT_PROGRAM, module_id))); + c.bench_function("lex_pipes_on_pipes", |b| b.iter(|| lex(PIPES_PROGRAM, module_id))); } pub fn bench_parse(c: &mut Criterion) { @@ -15,7 +16,8 @@ pub fn bench_parse(c: &mut Criterion) { ("mike_stress_test", MIKE_STRESS_TEST_PROGRAM), ("koch snowflake", LSYSTEM_KOCH_SNOWFLAKE_PROGRAM), ] { - let tokens = kcl_lib::token::lexer(file).unwrap(); + let module_id = kcl_lib::ast::types::ModuleId::default(); + let tokens = kcl_lib::token::lexer(file, module_id).unwrap(); c.bench_function(&format!("parse_{name}"), move |b| { let tok = tokens.clone(); b.iter(move || { @@ -26,8 +28,8 @@ pub fn bench_parse(c: &mut Criterion) { } } -fn lex(program: &str) { - black_box(kcl_lib::token::lexer(program).unwrap()); +fn lex(program: &str, module_id: kcl_lib::ast::types::ModuleId) { + black_box(kcl_lib::token::lexer(program, module_id).unwrap()); } criterion_group!(benches, bench_lex, bench_parse); diff --git a/src/wasm-lib/kcl/benches/compiler_benchmark_iai.rs b/src/wasm-lib/kcl/benches/compiler_benchmark_iai.rs index c1d27c8d4c..ca28a1fec8 100644 --- a/src/wasm-lib/kcl/benches/compiler_benchmark_iai.rs +++ b/src/wasm-lib/kcl/benches/compiler_benchmark_iai.rs @@ -1,26 +1,32 @@ use iai::black_box; pub fn parse(program: &str) { - let tokens = kcl_lib::token::lexer(program).unwrap(); + let module_id = kcl_lib::ast::types::ModuleId::default(); + let tokens = kcl_lib::token::lexer(program, module_id).unwrap(); let tok = tokens.clone(); let parser = kcl_lib::parser::Parser::new(tok.clone()); black_box(parser.ast().unwrap()); } fn lex_kitt() { - black_box(kcl_lib::token::lexer(KITT_PROGRAM).unwrap()); + let module_id = kcl_lib::ast::types::ModuleId::default(); + black_box(kcl_lib::token::lexer(KITT_PROGRAM, module_id).unwrap()); } fn lex_pipes() { - black_box(kcl_lib::token::lexer(PIPES_PROGRAM).unwrap()); + let module_id = kcl_lib::ast::types::ModuleId::default(); + black_box(kcl_lib::token::lexer(PIPES_PROGRAM, module_id).unwrap()); } fn lex_cube() { - black_box(kcl_lib::token::lexer(CUBE_PROGRAM).unwrap()); + let module_id = kcl_lib::ast::types::ModuleId::default(); + black_box(kcl_lib::token::lexer(CUBE_PROGRAM, module_id).unwrap()); } fn lex_math() { - black_box(kcl_lib::token::lexer(MATH_PROGRAM).unwrap()); + let module_id = kcl_lib::ast::types::ModuleId::default(); + black_box(kcl_lib::token::lexer(MATH_PROGRAM, module_id).unwrap()); } fn lex_lsystem() { - black_box(kcl_lib::token::lexer(LSYSTEM_PROGRAM).unwrap()); + let module_id = kcl_lib::ast::types::ModuleId::default(); + black_box(kcl_lib::token::lexer(LSYSTEM_PROGRAM, module_id).unwrap()); } fn parse_kitt() { diff --git a/src/wasm-lib/kcl/benches/digest_benchmark.rs b/src/wasm-lib/kcl/benches/digest_benchmark.rs index 91b8e949c4..33bdbd14a1 100644 --- a/src/wasm-lib/kcl/benches/digest_benchmark.rs +++ b/src/wasm-lib/kcl/benches/digest_benchmark.rs @@ -9,8 +9,7 @@ pub fn bench_digest(c: &mut Criterion) { ("mike_stress_test", MIKE_STRESS_TEST_PROGRAM), ("lsystem", LSYSTEM_PROGRAM), ] { - let tokens = kcl_lib::token::lexer(file).unwrap(); - let prog = kcl_lib::parser::Parser::new(tokens).ast().unwrap(); + let prog = kcl_lib::parser::top_level_parse(file).unwrap(); c.bench_function(&format!("digest_{name}"), move |b| { let prog = prog.clone(); diff --git a/src/wasm-lib/kcl/src/ast/modify.rs b/src/wasm-lib/kcl/src/ast/modify.rs index 432bb8cc64..35f00c9c0a 100644 --- a/src/wasm-lib/kcl/src/ast/modify.rs +++ b/src/wasm-lib/kcl/src/ast/modify.rs @@ -16,7 +16,7 @@ use crate::{ executor::{Point2d, SourceRange}, }; -use super::types::Node; +use super::types::{ModuleId, Node}; type Point3d = kcmc::shared::Point3d; @@ -38,6 +38,7 @@ const EPSILON: f64 = 0.015625; // or 2^-6 pub async fn modify_ast_for_sketch( engine: &Arc>, program: &mut Node, + module_id: ModuleId, // The name of the sketch. sketch_name: &str, // The type of plane the sketch is on. `XY` or `XZ`, etc @@ -183,9 +184,7 @@ pub async fn modify_ast_for_sketch( let recasted = program.recast(&FormatOptions::default(), 0); // Re-parse the ast so we get the correct source ranges. - let tokens = crate::token::lexer(&recasted)?; - let parser = crate::parser::Parser::new(tokens); - *program = parser.ast()?; + *program = crate::parser::parse(&recasted, module_id)?; Ok(recasted) } diff --git a/src/wasm-lib/kcl/src/ast/types.rs b/src/wasm-lib/kcl/src/ast/types.rs index e197f149d6..01a2d99377 100644 --- a/src/wasm-lib/kcl/src/ast/types.rs +++ b/src/wasm-lib/kcl/src/ast/types.rs @@ -37,6 +37,7 @@ pub(crate) mod digest; pub(crate) mod execute; mod literal_value; mod none; +pub(crate) mod source_range; use digest::Digest; @@ -48,11 +49,14 @@ pub enum Definition<'a> { #[derive(Debug, Default, Clone, Deserialize, Serialize, PartialEq, Eq, ts_rs::TS, Bake)] #[databake(path = kcl_lib::ast::types)] #[ts(export)] +#[serde(rename_all = "camelCase")] pub struct Node { #[serde(flatten)] pub inner: T, pub start: usize, pub end: usize, + #[serde(default, skip_serializing_if = "ModuleId::is_top_level")] + pub module_id: ModuleId, } impl schemars::JsonSchema for Node { @@ -78,8 +82,13 @@ impl schemars::JsonSchema for Node { } impl Node { - pub fn new(inner: T, start: usize, end: usize) -> Self { - Self { inner, start, end } + pub fn new(inner: T, start: usize, end: usize, module_id: ModuleId) -> Self { + Self { + inner, + start, + end, + module_id, + } } pub fn no_src(inner: T) -> Self { @@ -87,15 +96,21 @@ impl Node { inner, start: 0, end: 0, + module_id: ModuleId::default(), } } - pub fn boxed(inner: T, start: usize, end: usize) -> BoxNode { - Box::new(Node { inner, start, end }) + pub fn boxed(inner: T, start: usize, end: usize, module_id: ModuleId) -> BoxNode { + Box::new(Node { + inner, + start, + end, + module_id, + }) } pub fn as_source_ranges(&self) -> Vec { - vec![SourceRange([self.start, self.end])] + vec![SourceRange([self.start, self.end, self.module_id.as_usize()])] } } @@ -121,19 +136,19 @@ impl fmt::Display for Node { impl From> for crate::executor::SourceRange { fn from(v: Node) -> Self { - Self([v.start, v.end]) + Self([v.start, v.end, v.module_id.as_usize()]) } } impl From<&Node> for crate::executor::SourceRange { fn from(v: &Node) -> Self { - Self([v.start, v.end]) + Self([v.start, v.end, v.module_id.as_usize()]) } } impl From<&BoxNode> for crate::executor::SourceRange { fn from(v: &BoxNode) -> Self { - Self([v.start, v.end]) + Self([v.start, v.end, v.module_id.as_usize()]) } } @@ -217,6 +232,7 @@ impl Node { crate::lint::checks::lint_variables, crate::lint::checks::lint_object_properties, crate::lint::checks::lint_call_expressions, + crate::lint::checks::lint_should_be_offset_plane, ]; let mut findings = vec![]; @@ -504,6 +520,29 @@ impl Program { } } +/// Identifier of a source file. Uses a u32 to keep the size small. +#[derive(Debug, Default, Clone, Copy, PartialEq, Eq, Hash, Deserialize, Serialize, ts_rs::TS, JsonSchema, Bake)] +#[cfg_attr(feature = "pyo3", pyo3::pyclass)] +#[databake(path = kcl_lib::ast::types)] +#[ts(export)] +pub struct ModuleId(pub u32); + +impl ModuleId { + pub fn from_usize(id: usize) -> Self { + Self(u32::try_from(id).expect("module ID should fit in a u32")) + } + + pub fn as_usize(&self) -> usize { + usize::try_from(self.0).expect("module ID should fit in a usize") + } + + /// Top-level file is the one being executed. + /// Represented by module ID of 0, i.e. the default value. + pub fn is_top_level(&self) -> bool { + *self == Self::default() + } +} + #[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema, Bake)] #[databake(path = kcl_lib::ast::types)] #[ts(export)] @@ -537,13 +576,13 @@ impl BodyItem { impl From for SourceRange { fn from(item: BodyItem) -> Self { - Self([item.start(), item.end()]) + Self([item.start(), item.end(), item.module_id().as_usize()]) } } impl From<&BodyItem> for SourceRange { fn from(item: &BodyItem) -> Self { - Self([item.start(), item.end()]) + Self([item.start(), item.end(), item.module_id().as_usize()]) } } @@ -567,7 +606,7 @@ pub enum Expr { MemberExpression(BoxNode), UnaryExpression(BoxNode), IfExpression(BoxNode), - None(KclNone), + None(Node), } impl Expr { @@ -757,13 +796,13 @@ impl Expr { impl From for SourceRange { fn from(value: Expr) -> Self { - Self([value.start(), value.end()]) + Self([value.start(), value.end(), value.module_id().as_usize()]) } } impl From<&Expr> for SourceRange { fn from(value: &Expr) -> Self { - Self([value.start(), value.end()]) + Self([value.start(), value.end(), value.module_id().as_usize()]) } } @@ -783,13 +822,13 @@ pub enum BinaryPart { impl From for SourceRange { fn from(value: BinaryPart) -> Self { - Self([value.start(), value.end()]) + Self([value.start(), value.end(), value.module_id().as_usize()]) } } impl From<&BinaryPart> for SourceRange { fn from(value: &BinaryPart) -> Self { - Self([value.start(), value.end()]) + Self([value.start(), value.end(), value.module_id().as_usize()]) } } @@ -2153,13 +2192,13 @@ impl MemberObject { impl From for SourceRange { fn from(obj: MemberObject) -> Self { - Self([obj.start(), obj.end()]) + Self([obj.start(), obj.end(), obj.module_id().as_usize()]) } } impl From<&MemberObject> for SourceRange { fn from(obj: &MemberObject) -> Self { - Self([obj.start(), obj.end()]) + Self([obj.start(), obj.end(), obj.module_id().as_usize()]) } } @@ -2190,13 +2229,13 @@ impl LiteralIdentifier { impl From for SourceRange { fn from(id: LiteralIdentifier) -> Self { - Self([id.start(), id.end()]) + Self([id.start(), id.end(), id.module_id().as_usize()]) } } impl From<&LiteralIdentifier> for SourceRange { fn from(id: &LiteralIdentifier) -> Self { - Self([id.start(), id.end()]) + Self([id.start(), id.end(), id.module_id().as_usize()]) } } @@ -3017,9 +3056,7 @@ fn ghi = (x) => { ghi("things") "#; - let tokens = crate::token::lexer(code).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(code).unwrap(); let folding_ranges = program.get_lsp_folding_ranges(); assert_eq!(folding_ranges.len(), 3); assert_eq!(folding_ranges[0].start_line, 29); @@ -3055,9 +3092,7 @@ fn ghi = (x) => { return x } "#; - let tokens = crate::token::lexer(code).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(code).unwrap(); let symbols = program.get_lsp_symbols(code).unwrap(); assert_eq!(symbols.len(), 7); } @@ -3077,9 +3112,7 @@ const cylinder = startSketchOn('-XZ') }, %) |> extrude(h, %) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let value = program.get_non_code_meta_for_position(50); @@ -3102,9 +3135,7 @@ const cylinder = startSketchOn('-XZ') }, %) |> extrude(h, %) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let value = program.get_non_code_meta_for_position(124); @@ -3117,9 +3148,7 @@ const cylinder = startSketchOn('-XZ') |> startProfileAt([0,0], %) |> xLine(5, %) // lin "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let value = program.get_non_code_meta_for_position(86); @@ -3131,9 +3160,7 @@ const cylinder = startSketchOn('-XZ') let some_program_string = r#"fn thing = (arg0: number, arg1: string, tag?: string) => { return arg0 }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); // Check the program output for the types of the parameters. let function = program.body.first().unwrap(); @@ -3155,9 +3182,7 @@ const cylinder = startSketchOn('-XZ') let some_program_string = r#"fn thing = (arg0: number[], arg1: string[], tag?: string) => { return arg0 }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); // Check the program output for the types of the parameters. let function = program.body.first().unwrap(); @@ -3179,9 +3204,8 @@ const cylinder = startSketchOn('-XZ') let some_program_string = r#"fn thing = (arg0: number[], arg1: {thing: number, things: string[], more?: string}, tag?: string) => { return arg0 }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let module_id = ModuleId::default(); + let program = crate::parser::parse(some_program_string, module_id).unwrap(); // Check the program output for the types of the parameters. let function = program.body.first().unwrap(); @@ -3206,6 +3230,7 @@ const cylinder = startSketchOn('-XZ') }, 35, 40, + module_id, ), type_: Some(FnArgType::Primitive(FnArgPrimitive::Number)), optional: false, @@ -3219,6 +3244,7 @@ const cylinder = startSketchOn('-XZ') }, 50, 56, + module_id, ), type_: Some(FnArgType::Array(FnArgPrimitive::String)), optional: false, @@ -3232,6 +3258,7 @@ const cylinder = startSketchOn('-XZ') }, 68, 72, + module_id, ), type_: Some(FnArgType::Primitive(FnArgPrimitive::String)), optional: true, @@ -3248,9 +3275,8 @@ const cylinder = startSketchOn('-XZ') let some_program_string = r#"fn thing = () => {thing: number, things: string[], more?: string} { return 1 }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let module_id = ModuleId::default(); + let program = crate::parser::parse(some_program_string, module_id).unwrap(); // Check the program output for the types of the parameters. let function = program.body.first().unwrap(); @@ -3274,6 +3300,7 @@ const cylinder = startSketchOn('-XZ') }, 18, 23, + module_id, ), type_: Some(FnArgType::Primitive(FnArgPrimitive::Number)), optional: false, @@ -3287,6 +3314,7 @@ const cylinder = startSketchOn('-XZ') }, 33, 39, + module_id, ), type_: Some(FnArgType::Array(FnArgPrimitive::String)), optional: false, @@ -3300,6 +3328,7 @@ const cylinder = startSketchOn('-XZ') }, 51, 55, + module_id, ), type_: Some(FnArgType::Primitive(FnArgPrimitive::String)), optional: true, @@ -3348,6 +3377,7 @@ const cylinder = startSketchOn('-XZ') }, start: 0, end: 0, + module_id: ModuleId::default(), }, return_type: None, digest: None, @@ -3374,6 +3404,7 @@ const cylinder = startSketchOn('-XZ') }, start: 0, end: 0, + module_id: ModuleId::default(), }, return_type: None, digest: None, @@ -3411,6 +3442,7 @@ const cylinder = startSketchOn('-XZ') }, start: 0, end: 0, + module_id: ModuleId::default(), }, return_type: None, digest: None, @@ -3428,9 +3460,7 @@ const cylinder = startSketchOn('-XZ') #[tokio::test(flavor = "multi_thread")] async fn test_parse_object_bool() { let some_program_string = r#"some_func({thing: true, other_thing: false})"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); // We want to get the bool and verify it is a bool. @@ -3478,14 +3508,12 @@ const cylinder = startSketchOn('-XZ') |> startProfileAt([0, 0], %) |> line([5, 5], %, $xLine) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let result = parser.ast(); + let result = crate::parser::top_level_parse(some_program_string); assert!(result.is_err()); assert_eq!( result.unwrap_err().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([76, 82])], message: "Cannot assign a tag to a reserved keyword: xLine" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([76, 82, 0])], message: "Cannot assign a tag to a reserved keyword: xLine" }"# ); } @@ -3495,14 +3523,12 @@ const cylinder = startSketchOn('-XZ') |> startProfileAt([0, 0], %) |> line([5, 5], %, $) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let result = parser.ast(); + let result = crate::parser::top_level_parse(some_program_string); assert!(result.is_err()); assert_eq!( result.unwrap_err().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([57, 59])], message: "Unexpected token: |>" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([57, 59, 0])], message: "Unexpected token: |>" }"# ); } @@ -3512,17 +3538,13 @@ const cylinder = startSketchOn('-XZ') |> startProfileAt([0, 0], %) |> line([5, 5], %) "#; - let prog1_tokens = crate::token::lexer(prog1_string).unwrap(); - let prog1_parser = crate::parser::Parser::new(prog1_tokens); - let prog1_digest = prog1_parser.ast().unwrap().compute_digest(); + let prog1_digest = crate::parser::top_level_parse(prog1_string).unwrap().compute_digest(); let prog2_string = r#"startSketchOn('XY') |> startProfileAt([0, 2], %) |> line([5, 5], %) "#; - let prog2_tokens = crate::token::lexer(prog2_string).unwrap(); - let prog2_parser = crate::parser::Parser::new(prog2_tokens); - let prog2_digest = prog2_parser.ast().unwrap().compute_digest(); + let prog2_digest = crate::parser::top_level_parse(prog2_string).unwrap().compute_digest(); assert!(prog1_digest != prog2_digest); @@ -3530,9 +3552,7 @@ const cylinder = startSketchOn('-XZ') |> startProfileAt([0, 0], %) |> line([5, 5], %) "#; - let prog3_tokens = crate::token::lexer(prog3_string).unwrap(); - let prog3_parser = crate::parser::Parser::new(prog3_tokens); - let prog3_digest = prog3_parser.ast().unwrap().compute_digest(); + let prog3_digest = crate::parser::top_level_parse(prog3_string).unwrap().compute_digest(); assert_eq!(prog1_digest, prog3_digest); } diff --git a/src/wasm-lib/kcl/src/ast/types/condition.rs b/src/wasm-lib/kcl/src/ast/types/condition.rs index 9244a41305..ce6e0c90a4 100644 --- a/src/wasm-lib/kcl/src/ast/types/condition.rs +++ b/src/wasm-lib/kcl/src/ast/types/condition.rs @@ -50,7 +50,7 @@ impl Node { impl Node { #[allow(dead_code)] fn source_ranges(&self) -> Vec { - vec![SourceRange([self.start, self.end])] + vec![SourceRange([self.start, self.end, self.module_id.as_usize()])] } } diff --git a/src/wasm-lib/kcl/src/ast/types/execute.rs b/src/wasm-lib/kcl/src/ast/types/execute.rs index d2934305d4..b90ee9da75 100644 --- a/src/wasm-lib/kcl/src/ast/types/execute.rs +++ b/src/wasm-lib/kcl/src/ast/types/execute.rs @@ -235,7 +235,7 @@ pub(crate) async fn execute_pipe_body( // they use the % from the parent. After all, this pipe expression hasn't been executed yet, so it doesn't have any % value // of its own. let meta = Metadata { - source_range: SourceRange([first.start(), first.end()]), + source_range: SourceRange::from(first), }; let output = ctx .execute_expr(first, exec_state, &meta, StatementKind::Expression) @@ -285,7 +285,7 @@ async fn inner_execute_pipe_body( | Expr::None(_) => {} }; let metadata = Metadata { - source_range: SourceRange([expression.start(), expression.end()]), + source_range: SourceRange::from(expression), }; let output = ctx .execute_expr(expression, exec_state, &metadata, StatementKind::Expression) diff --git a/src/wasm-lib/kcl/src/ast/types/none.rs b/src/wasm-lib/kcl/src/ast/types/none.rs index ea4bd7fc81..6e60243311 100644 --- a/src/wasm-lib/kcl/src/ast/types/none.rs +++ b/src/wasm-lib/kcl/src/ast/types/none.rs @@ -6,9 +6,11 @@ use serde::{Deserialize, Serialize}; use crate::{ ast::types::ConstraintLevel, - executor::{KclValue, SourceRange, UserVal}, + executor::{KclValue, UserVal}, }; +use super::Node; + const KCL_NONE_ID: &str = "KCL_NONE_ID"; /// KCL value for an optional parameter which was not given an argument. @@ -19,9 +21,6 @@ const KCL_NONE_ID: &str = "KCL_NONE_ID"; #[ts(export)] #[serde(tag = "type")] pub struct KclNone { - // TODO: Convert this to be an Option. - pub start: usize, - pub end: usize, #[serde(deserialize_with = "deser_private")] #[ts(skip)] #[schemars(skip)] @@ -29,12 +28,8 @@ pub struct KclNone { } impl KclNone { - pub fn new(start: usize, end: usize) -> Self { - Self { - start, - end, - __private: Private {}, - } + pub fn new() -> Self { + Self { __private: Private {} } } } @@ -63,12 +58,6 @@ where } } -impl From<&KclNone> for SourceRange { - fn from(v: &KclNone) -> Self { - Self([v.start, v.end]) - } -} - impl From<&KclNone> for UserVal { fn from(none: &KclNone) -> Self { UserVal { @@ -85,16 +74,18 @@ impl From<&KclNone> for KclValue { } } -impl KclNone { - pub fn source_range(&self) -> SourceRange { - SourceRange([self.start, self.end]) +impl From<&Node> for KclValue { + fn from(none: &Node) -> Self { + Self::from(&none.inner) } +} +impl Node { /// Get the constraint level. /// KCL None is never constrained. pub fn get_constraint_level(&self) -> ConstraintLevel { ConstraintLevel::None { - source_ranges: vec![self.source_range()], + source_ranges: self.as_source_ranges(), } } } diff --git a/src/wasm-lib/kcl/src/ast/types/source_range.rs b/src/wasm-lib/kcl/src/ast/types/source_range.rs new file mode 100644 index 0000000000..f68210d030 --- /dev/null +++ b/src/wasm-lib/kcl/src/ast/types/source_range.rs @@ -0,0 +1,66 @@ +use super::{BinaryPart, BodyItem, Expr, LiteralIdentifier, MemberObject, ModuleId}; + +impl BodyItem { + pub fn module_id(&self) -> ModuleId { + match self { + BodyItem::ImportStatement(stmt) => stmt.module_id, + BodyItem::ExpressionStatement(expression_statement) => expression_statement.module_id, + BodyItem::VariableDeclaration(variable_declaration) => variable_declaration.module_id, + BodyItem::ReturnStatement(return_statement) => return_statement.module_id, + } + } +} + +impl Expr { + pub fn module_id(&self) -> ModuleId { + match self { + Expr::Literal(literal) => literal.module_id, + Expr::Identifier(identifier) => identifier.module_id, + Expr::TagDeclarator(tag) => tag.module_id, + Expr::BinaryExpression(binary_expression) => binary_expression.module_id, + Expr::FunctionExpression(function_expression) => function_expression.module_id, + Expr::CallExpression(call_expression) => call_expression.module_id, + Expr::PipeExpression(pipe_expression) => pipe_expression.module_id, + Expr::PipeSubstitution(pipe_substitution) => pipe_substitution.module_id, + Expr::ArrayExpression(array_expression) => array_expression.module_id, + Expr::ArrayRangeExpression(array_range) => array_range.module_id, + Expr::ObjectExpression(object_expression) => object_expression.module_id, + Expr::MemberExpression(member_expression) => member_expression.module_id, + Expr::UnaryExpression(unary_expression) => unary_expression.module_id, + Expr::IfExpression(expr) => expr.module_id, + Expr::None(none) => none.module_id, + } + } +} + +impl BinaryPart { + pub fn module_id(&self) -> ModuleId { + match self { + BinaryPart::Literal(literal) => literal.module_id, + BinaryPart::Identifier(identifier) => identifier.module_id, + BinaryPart::BinaryExpression(binary_expression) => binary_expression.module_id, + BinaryPart::CallExpression(call_expression) => call_expression.module_id, + BinaryPart::UnaryExpression(unary_expression) => unary_expression.module_id, + BinaryPart::MemberExpression(member_expression) => member_expression.module_id, + BinaryPart::IfExpression(e) => e.module_id, + } + } +} + +impl MemberObject { + pub fn module_id(&self) -> ModuleId { + match self { + MemberObject::MemberExpression(member_expression) => member_expression.module_id, + MemberObject::Identifier(identifier) => identifier.module_id, + } + } +} + +impl LiteralIdentifier { + pub fn module_id(&self) -> ModuleId { + match self { + LiteralIdentifier::Identifier(identifier) => identifier.module_id, + LiteralIdentifier::Literal(literal) => literal.module_id, + } + } +} diff --git a/src/wasm-lib/kcl/src/errors.rs b/src/wasm-lib/kcl/src/errors.rs index 9f8dea6915..b0caaeb3ba 100644 --- a/src/wasm-lib/kcl/src/errors.rs +++ b/src/wasm-lib/kcl/src/errors.rs @@ -2,7 +2,7 @@ use serde::{Deserialize, Serialize}; use thiserror::Error; use tower_lsp::lsp_types::{Diagnostic, DiagnosticSeverity}; -use crate::{executor::SourceRange, lsp::IntoDiagnostic}; +use crate::{ast::types::ModuleId, executor::SourceRange, lsp::IntoDiagnostic}; #[derive(Error, Debug, Serialize, Deserialize, ts_rs::TS, Clone, PartialEq, Eq)] #[ts(export)] @@ -147,6 +147,13 @@ impl IntoDiagnostic for KclError { let message = self.get_message(); let source_ranges = self.source_ranges(); + // Limit to only errors in the top-level file. + let module_id = ModuleId::default(); + let source_ranges = source_ranges + .iter() + .filter(|r| r.module_id() == module_id) + .collect::>(); + Diagnostic { range: source_ranges.first().map(|r| r.to_lsp_range(code)).unwrap_or_default(), severity: Some(self.severity()), diff --git a/src/wasm-lib/kcl/src/executor.rs b/src/wasm-lib/kcl/src/executor.rs index 3e817d4455..ecbd4f4b97 100644 --- a/src/wasm-lib/kcl/src/executor.rs +++ b/src/wasm-lib/kcl/src/executor.rs @@ -7,6 +7,7 @@ use std::{ use anyhow::Result; use async_recursion::async_recursion; +use indexmap::IndexMap; use kcmc::{ each_cmd as mcmd, ok_response::{output::TakeSnapshot, OkModelingCmdResponse}, @@ -26,8 +27,8 @@ type Point3D = kcmc::shared::Point3d; use crate::{ ast::types::{ - human_friendly_type, BodyItem, Expr, FunctionExpression, ItemVisibility, KclNone, Node, NodeRef, Program, - TagDeclarator, TagNode, + human_friendly_type, BodyItem, Expr, FunctionExpression, ItemVisibility, KclNone, ModuleId, Node, NodeRef, + Program, TagDeclarator, TagNode, }, engine::{EngineManager, ExecutionKind}, errors::{KclError, KclErrorDetails}, @@ -55,11 +56,32 @@ pub struct ExecState { /// The stack of import statements for detecting circular module imports. /// If this is empty, we're not currently executing an import statement. pub import_stack: Vec, + /// Map from source file absolute path to module ID. + pub path_to_source_id: IndexMap, + /// Map from module ID to module info. + pub module_infos: IndexMap, /// The directory of the current project. This is used for resolving import /// paths. If None is given, the current working directory is used. pub project_directory: Option, } +impl ExecState { + pub fn add_module(&mut self, path: std::path::PathBuf) -> ModuleId { + // Need to avoid borrowing self in the closure. + let new_module_id = ModuleId::from_usize(self.path_to_source_id.len()); + let mut is_new = false; + let id = *self.path_to_source_id.entry(path.clone()).or_insert_with(|| { + is_new = true; + new_module_id + }); + if is_new { + let module_info = ModuleInfo { id, path }; + self.module_infos.insert(id, module_info); + } + id + } +} + #[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema)] #[ts(export)] #[serde(rename_all = "camelCase")] @@ -1373,21 +1395,33 @@ pub enum BodyType { Block, } +/// Info about a module. Right now, this is pretty minimal. We hope to cache +/// modules here in the future. +#[derive(Debug, Default, Clone, PartialEq, Eq, Deserialize, Serialize, ts_rs::TS, JsonSchema)] +#[cfg_attr(feature = "pyo3", pyo3::pyclass)] +#[ts(export)] +pub struct ModuleInfo { + /// The ID of the module. + id: ModuleId, + /// Absolute path of the module's source file. + path: std::path::PathBuf, +} + #[derive(Debug, Default, Deserialize, Serialize, PartialEq, Copy, Clone, ts_rs::TS, JsonSchema, Hash, Eq)] #[cfg_attr(feature = "pyo3", pyo3::pyclass)] #[ts(export)] -pub struct SourceRange(#[ts(type = "[number, number]")] pub [usize; 2]); +pub struct SourceRange(#[ts(type = "[number, number]")] pub [usize; 3]); -impl From<[usize; 2]> for SourceRange { - fn from(value: [usize; 2]) -> Self { +impl From<[usize; 3]> for SourceRange { + fn from(value: [usize; 3]) -> Self { Self(value) } } impl SourceRange { /// Create a new source range. - pub fn new(start: usize, end: usize) -> Self { - Self([start, end]) + pub fn new(start: usize, end: usize, module_id: ModuleId) -> Self { + Self([start, end, module_id.as_usize()]) } /// Get the start of the range. @@ -1400,6 +1434,11 @@ impl SourceRange { self.0[1] } + /// Get the module ID of the range. + pub fn module_id(&self) -> ModuleId { + ModuleId::from_usize(self.0[2]) + } + /// Check if the range contains a position. pub fn contains(&self, pos: usize) -> bool { pos >= self.start() && pos <= self.end() @@ -1533,7 +1572,7 @@ impl From for Metadata { impl From> for Metadata { fn from(node: NodeRef<'_, T>) -> Self { Self { - source_range: SourceRange::new(node.start, node.end), + source_range: SourceRange::new(node.start, node.end, node.module_id), } } } @@ -2171,6 +2210,8 @@ impl ExecutorContext { project_directory, ..Default::default() }; + // TODO: Use the top-level file's path. + exec_state.add_module(std::path::PathBuf::from("")); // Before we even start executing the program, set the units. self.engine .batch_modeling_cmd( @@ -2210,6 +2251,13 @@ impl ExecutorContext { BodyItem::ImportStatement(import_stmt) => { let source_range = SourceRange::from(import_stmt); let path = import_stmt.path.clone(); + // Empty path is used by the top-level module. + if path.is_empty() { + return Err(KclError::Semantic(KclErrorDetails { + message: "import path cannot be empty".to_owned(), + source_ranges: vec![source_range], + })); + } let resolved_path = if let Some(project_dir) = &exec_state.project_directory { std::path::PathBuf::from(project_dir).join(&path) } else { @@ -2230,8 +2278,9 @@ impl ExecutorContext { source_ranges: vec![import_stmt.into()], })); } + let module_id = exec_state.add_module(resolved_path.clone()); let source = self.fs.read_to_string(&resolved_path, source_range).await?; - let program = crate::parser::parse(&source)?; + let program = crate::parser::parse(&source, module_id)?; let (module_memory, module_exports) = { exec_state.import_stack.push(resolved_path.clone()); let original_execution = self.engine.replace_execution_kind(ExecutionKind::Isolated); @@ -2359,7 +2408,7 @@ impl ExecutorContext { // True here tells the engine to flush all the end commands as well like fillets // and chamfers where the engine would otherwise eat the ID of the segments. true, - SourceRange([program.end, program.end]), + SourceRange([program.end, program.end, program.module_id.as_usize()]), ) .await?; } @@ -2525,7 +2574,12 @@ fn assign_args_to_params( if param.optional { // If the corresponding parameter is optional, // then it's fine, the user doesn't need to supply it. - let none = KclNone::new(param.identifier.start, param.identifier.end); + let none = Node { + inner: KclNone::new(), + start: param.identifier.start, + end: param.identifier.end, + module_id: param.identifier.module_id, + }; fn_memory.add( ¶m.identifier.name, KclValue::from(&none), @@ -2586,9 +2640,8 @@ mod tests { use crate::ast::types::{Identifier, Node, Parameter}; pub async fn parse_execute(code: &str) -> Result { - let tokens = crate::token::lexer(code)?; - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast()?; + let program = crate::parser::top_level_parse(code)?; + let ctx = ExecutorContext { engine: Arc::new(Box::new(crate::engine::conn_mock::EngineConnection::new().await?)), fs: Arc::new(crate::fs::FileManager::new()), @@ -3027,7 +3080,7 @@ const answer = returnX()"#; err, KclError::UndefinedValue(KclErrorDetails { message: "memory item key `x` is not defined".to_owned(), - source_ranges: vec![SourceRange([64, 65]), SourceRange([97, 106])], + source_ranges: vec![SourceRange([64, 65, 0]), SourceRange([97, 106, 0])], }), ); } @@ -3062,7 +3115,7 @@ let shape = layer() |> patternTransform(10, transform, %) err, KclError::UndefinedValue(KclErrorDetails { message: "memory item key `x` is not defined".to_owned(), - source_ranges: vec![SourceRange([80, 81])], + source_ranges: vec![SourceRange([80, 81, 0])], }), ); } @@ -3317,7 +3370,7 @@ let notNull = !myNull parse_execute(code1).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { message: "Cannot apply unary operator ! to non-boolean value: null".to_owned(), - source_ranges: vec![SourceRange([56, 63])], + source_ranges: vec![SourceRange([56, 63, 0])], }) ); @@ -3326,7 +3379,7 @@ let notNull = !myNull parse_execute(code2).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { message: "Cannot apply unary operator ! to non-boolean value: 0".to_owned(), - source_ranges: vec![SourceRange([14, 16])], + source_ranges: vec![SourceRange([14, 16, 0])], }) ); @@ -3337,7 +3390,7 @@ let notEmptyString = !"" parse_execute(code3).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { message: "Cannot apply unary operator ! to non-boolean value: \"\"".to_owned(), - source_ranges: vec![SourceRange([22, 25])], + source_ranges: vec![SourceRange([22, 25, 0])], }) ); @@ -3349,7 +3402,7 @@ let notMember = !obj.a parse_execute(code4).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { message: "Cannot apply unary operator ! to non-boolean value: 1".to_owned(), - source_ranges: vec![SourceRange([36, 42])], + source_ranges: vec![SourceRange([36, 42, 0])], }) ); @@ -3360,7 +3413,7 @@ let notArray = !a"; parse_execute(code5).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { message: "Cannot apply unary operator ! to non-boolean value: []".to_owned(), - source_ranges: vec![SourceRange([27, 29])], + source_ranges: vec![SourceRange([27, 29, 0])], }) ); @@ -3371,7 +3424,7 @@ let notObject = !x"; parse_execute(code6).await.unwrap_err().downcast::().unwrap(), KclError::Semantic(KclErrorDetails { message: "Cannot apply unary operator ! to non-boolean value: {}".to_owned(), - source_ranges: vec![SourceRange([28, 30])], + source_ranges: vec![SourceRange([28, 30, 0])], }) ); @@ -3424,7 +3477,7 @@ let notTagIdentifier = !myTag"; parse_execute(code10).await.unwrap_err().downcast::().unwrap(), KclError::Syntax(KclErrorDetails { message: "Unexpected token: !".to_owned(), - source_ranges: vec![SourceRange([14, 15])], + source_ranges: vec![SourceRange([14, 15, 0])], }) ); @@ -3437,7 +3490,7 @@ let notPipeSub = 1 |> identity(!%))"; parse_execute(code11).await.unwrap_err().downcast::().unwrap(), KclError::Syntax(KclErrorDetails { message: "Unexpected token: |>".to_owned(), - source_ranges: vec![SourceRange([54, 56])], + source_ranges: vec![SourceRange([54, 56, 0])], }) ); @@ -3483,7 +3536,7 @@ test([0, 0]) assert!(result.is_err()); assert_eq!( result.unwrap_err().to_string(), - r#"undefined value: KclErrorDetails { source_ranges: [SourceRange([10, 34])], message: "Result of user-defined function test is undefined" }"#.to_owned() + r#"undefined value: KclErrorDetails { source_ranges: [SourceRange([10, 34, 0])], message: "Result of user-defined function test is undefined" }"#.to_owned() ); } @@ -3600,7 +3653,7 @@ let w = f() + f() vec![req_param("x")], vec![], Err(KclError::Semantic(KclErrorDetails { - source_ranges: vec![SourceRange([0, 0])], + source_ranges: vec![SourceRange([0, 0, 0])], message: "Expected 1 arguments, got 0".to_owned(), })), ), @@ -3618,7 +3671,7 @@ let w = f() + f() vec![req_param("x"), opt_param("y")], vec![], Err(KclError::Semantic(KclErrorDetails { - source_ranges: vec![SourceRange([0, 0])], + source_ranges: vec![SourceRange([0, 0, 0])], message: "Expected 1-2 arguments, got 0".to_owned(), })), ), @@ -3645,7 +3698,7 @@ let w = f() + f() vec![req_param("x"), opt_param("y")], vec![mem(1), mem(2), mem(3)], Err(KclError::Semantic(KclErrorDetails { - source_ranges: vec![SourceRange([0, 0])], + source_ranges: vec![SourceRange([0, 0, 0])], message: "Expected 1-2 arguments, got 3".to_owned(), })), ), @@ -3661,6 +3714,7 @@ let w = f() + f() }, start: 0, end: 0, + module_id: ModuleId::default(), }, return_type: None, digest: None, diff --git a/src/wasm-lib/kcl/src/lint/checks/camel_case.rs b/src/wasm-lib/kcl/src/lint/checks/camel_case.rs index 872a39539d..1f68c1eff8 100644 --- a/src/wasm-lib/kcl/src/lint/checks/camel_case.rs +++ b/src/wasm-lib/kcl/src/lint/checks/camel_case.rs @@ -29,7 +29,10 @@ fn lint_lower_camel_case_var(decl: &VariableDeclarator) -> Result Result Result> { + let Node::CallExpression(call) = node else { + return Ok(vec![]); + }; + + if call.inner.callee.inner.name != "startSketchOn" { + return Ok(vec![]); + } + + if call.arguments.len() != 1 { + // we only look for single-argument object patterns, if there's more + // than that we don't have a plane decl + return Ok(vec![]); + } + + let Expr::ObjectExpression(arg) = &call.arguments[0] else { + return Ok(vec![]); + }; + + let Some(plane) = arg.inner.properties.iter().find(|v| v.key.inner.name == "plane") else { + return Ok(vec![]); + }; + + let Expr::ObjectExpression(ref plane) = plane.inner.value else { + return Ok(vec![]); + }; + + let mut origin: Option<(f64, f64, f64)> = None; + let mut x_vec: Option<(f64, f64, f64)> = None; + let mut y_vec: Option<(f64, f64, f64)> = None; + + for property in &plane.inner.properties { + let Expr::ObjectExpression(ref point) = property.inner.value else { + return Ok(vec![]); + }; + + let Some((x, y, z)) = get_xyz(&point.inner) else { + return Ok(vec![]); + }; + + let property_name = &property.inner.key.inner.name; + + match property_name.as_str() { + "origin" => origin = Some((x, y, z)), + "xAxis" => x_vec = Some((x, y, z)), + "yAxis" => y_vec = Some((x, y, z)), + _ => { + continue; + } + }; + } + + let Some(origin) = origin else { return Ok(vec![]) }; + let Some(x_vec) = x_vec else { return Ok(vec![]) }; + let Some(y_vec) = y_vec else { return Ok(vec![]) }; + + if [origin.0, origin.1, origin.2].iter().filter(|v| **v == 0.0).count() < 2 { + return Ok(vec![]); + } + // two of the origin values are 0, 0; let's work it out and check + // what's **up** + + /// This will attempt to very poorly translate orientation to a letter + /// if it's possible to do so. The engine will norm these vectors, so + /// we'll just use logic off 0 for now, but this sucks, generally speaking. + fn vector_to_letter<'a>(x: f64, y: f64, z: f64) -> Option<&'a str> { + if x > 0.0 && y == 0.0 && z == 0.0 { + return Some("X"); + } + if x < 0.0 && y == 0.0 && z == 0.0 { + return Some("-X"); + } + + if x == 0.0 && y > 0.0 && z == 0.0 { + return Some("Y"); + } + if x == 0.0 && y < 0.0 && z == 0.0 { + return Some("-Y"); + } + + if x == 0.0 && y == 0.0 && z > 0.0 { + return Some("Z"); + } + if x == 0.0 && y == 0.0 && z < 0.0 { + return Some("-Z"); + } + + None + } + + let allowed_planes = HashMap::from([ + // allowed built-in planes + ("XY".to_owned(), true), + ("-XY".to_owned(), true), + ("XZ".to_owned(), true), + ("-XZ".to_owned(), true), + ("YZ".to_owned(), true), + ("-YZ".to_owned(), true), + ]); + // Currently, the engine **ONLY** accepts[1] the following: + // + // XY + // -XY + // XZ + // -XZ + // YZ + // -YZ + // + // [1]: https://zoo.dev/docs/kcl/types/PlaneData + + let plane_name = format!( + "{}{}", + vector_to_letter(x_vec.0, x_vec.1, x_vec.2).unwrap_or(""), + vector_to_letter(y_vec.0, y_vec.1, y_vec.2).unwrap_or(""), + ); + + if !allowed_planes.contains_key(&plane_name) { + return Ok(vec![]); + }; + + let call_source_range = SourceRange::new(call.start, call.end, call.module_id); + Ok(vec![Z0003.at( + format!( + "custom plane in startSketchOn; offsetPlane from {} would work here", + plane_name + ), + call_source_range, + )]) +} + +fn get_xyz(point: &ObjectExpression) -> Option<(f64, f64, f64)> { + let mut x: Option = None; + let mut y: Option = None; + let mut z: Option = None; + + fn unlitafy(lit: &LiteralValue) -> Option { + Some(match lit { + LiteralValue::IInteger(value) => *value as f64, + LiteralValue::Fractional(value) => *value, + _ => { + return None; + } + }) + } + + for property in &point.properties { + let Some(value) = (match &property.value { + Expr::UnaryExpression(ref value) => { + if value.operator != UnaryOperator::Neg { + continue; + } + let BinaryPart::Literal(ref value) = &value.inner.argument else { + continue; + }; + unlitafy(&value.inner.value).map(|v| -v) + } + Expr::Literal(ref value) => unlitafy(&value.value), + _ => { + continue; + } + }) else { + continue; + }; + + match property.key.inner.name.as_str() { + "x" => x = Some(value), + "y" => y = Some(value), + "z" => z = Some(value), + _ => {} + } + } + + Some((x?, y?, z?)) +} + +#[cfg(test)] +mod tests { + use super::{lint_should_be_offset_plane, Z0003}; + use crate::lint::rule::{test_finding, test_no_finding}; + + test_finding!( + z0003_bad_sketch_on, + lint_should_be_offset_plane, + Z0003, + "\ +startSketchOn({ + plane: { + origin: { x: 0, y: -14.3, z: 0 }, + xAxis: { x: 1, y: 0, z: 0 }, + yAxis: { x: 0, y: 0, z: 1 }, + zAxis: { x: 0, y: -1, z: 0 } + } +}) +" + ); + + test_no_finding!( + z0003_good_sketch_on, + lint_should_be_offset_plane, + Z0003, + "\ +startSketchOn({ + plane: { + origin: { x: 10, y: -14.3, z: 0 }, + xAxis: { x: 1, y: 0, z: 0 }, + yAxis: { x: 0, y: 0, z: 1 }, + zAxis: { x: 0, y: -1, z: 0 } + } +}) +" + ); +} diff --git a/src/wasm-lib/kcl/src/lint/checks/std_lib_args.rs b/src/wasm-lib/kcl/src/lint/checks/std_lib_args.rs index 6b12626f25..0a8f185e72 100644 --- a/src/wasm-lib/kcl/src/lint/checks/std_lib_args.rs +++ b/src/wasm-lib/kcl/src/lint/checks/std_lib_args.rs @@ -28,7 +28,7 @@ fn lint_too_many_args_std_lib_function( if exp.arguments.len() != 2 { findings.push(Z0002.at( format!("expected 2 arguments, found {}", exp.arguments.len()), - SourceRange::new(exp.start, exp.end), + SourceRange::new(exp.start, exp.end, exp.module_id), )); } return Ok(findings); @@ -38,7 +38,7 @@ fn lint_too_many_args_std_lib_function( if exp.arguments.len() < 2 { findings.push(Z0002.at( format!("expected at least 2 arguments, found {}", exp.arguments.len()), - SourceRange::new(exp.start, exp.end), + SourceRange::new(exp.start, exp.end, exp.module_id), )); } return Ok(findings); @@ -48,7 +48,7 @@ fn lint_too_many_args_std_lib_function( if exp.arguments.len() > fn_args_len { findings.push(Z0002.at( format!("expected {} arguments, found {}", fn_args_len, exp.arguments.len()), - SourceRange::new(exp.start, exp.end), + SourceRange::new(exp.start, exp.end, exp.module_id), )); } diff --git a/src/wasm-lib/kcl/src/lint/rule.rs b/src/wasm-lib/kcl/src/lint/rule.rs index 1b11e605ff..014d90f6fe 100644 --- a/src/wasm-lib/kcl/src/lint/rule.rs +++ b/src/wasm-lib/kcl/src/lint/rule.rs @@ -182,9 +182,7 @@ mod test { macro_rules! assert_no_finding { ( $check:expr, $finding:expr, $kcl:expr ) => { - let tokens = $crate::token::lexer($kcl).unwrap(); - let parser = $crate::parser::Parser::new(tokens); - let prog = parser.ast().unwrap(); + let prog = $crate::parser::top_level_parse($kcl).unwrap(); for discovered_finding in prog.lint($check).unwrap() { if discovered_finding.finding == $finding { assert!(false, "Finding {:?} was emitted", $finding.code); @@ -195,9 +193,7 @@ mod test { macro_rules! assert_finding { ( $check:expr, $finding:expr, $kcl:expr ) => { - let tokens = $crate::token::lexer($kcl).unwrap(); - let parser = $crate::parser::Parser::new(tokens); - let prog = parser.ast().unwrap(); + let prog = $crate::parser::top_level_parse($kcl).unwrap(); for discovered_finding in prog.lint($check).unwrap() { if discovered_finding.finding == $finding { diff --git a/src/wasm-lib/kcl/src/lsp/kcl/mod.rs b/src/wasm-lib/kcl/src/lsp/kcl/mod.rs index 3d3948e157..44c50ab6d1 100644 --- a/src/wasm-lib/kcl/src/lsp/kcl/mod.rs +++ b/src/wasm-lib/kcl/src/lsp/kcl/mod.rs @@ -40,7 +40,7 @@ use tower_lsp::{ }; use crate::{ - ast::types::{Expr, Node, NodeRef, VariableKind}, + ast::types::{Expr, ModuleId, Node, NodeRef, VariableKind}, executor::{IdGenerator, SourceRange}, lsp::{backend::Backend as _, util::IntoDiagnostic}, parser::PIPE_OPERATOR, @@ -188,7 +188,8 @@ impl crate::lsp::backend::Backend for Backend { // We already updated the code map in the shared backend. // Lets update the tokens. - let tokens = match crate::token::lexer(¶ms.text) { + let module_id = ModuleId::default(); + let tokens = match crate::token::lexer(¶ms.text, module_id) { Ok(tokens) => tokens, Err(err) => { self.add_to_diagnostics(¶ms, &[err], true).await; @@ -1235,7 +1236,8 @@ impl LanguageServer for Backend { // Parse the ast. // I don't know if we need to do this again since it should be updated in the context. // But I figure better safe than sorry since this will write back out to the file. - let Ok(tokens) = crate::token::lexer(current_code) else { + let module_id = ModuleId::default(); + let Ok(tokens) = crate::token::lexer(current_code, module_id) else { return Ok(None); }; let parser = crate::parser::Parser::new(tokens); @@ -1251,7 +1253,7 @@ impl LanguageServer for Backend { }, 0, ); - let source_range = SourceRange([0, current_code.len()]); + let source_range = SourceRange::new(0, current_code.len(), module_id); let range = source_range.to_lsp_range(current_code); Ok(Some(vec![TextEdit { new_text: recast, @@ -1272,7 +1274,8 @@ impl LanguageServer for Backend { // Parse the ast. // I don't know if we need to do this again since it should be updated in the context. // But I figure better safe than sorry since this will write back out to the file. - let Ok(tokens) = crate::token::lexer(current_code) else { + let module_id = ModuleId::default(); + let Ok(tokens) = crate::token::lexer(current_code, module_id) else { return Ok(None); }; let parser = crate::parser::Parser::new(tokens); @@ -1286,7 +1289,7 @@ impl LanguageServer for Backend { ast.rename_symbol(¶ms.new_name, pos); // Now recast it. let recast = ast.recast(&Default::default(), 0); - let source_range = SourceRange([0, current_code.len() - 1]); + let source_range = SourceRange::new(0, current_code.len() - 1, module_id); let range = source_range.to_lsp_range(current_code); Ok(Some(WorkspaceEdit { changes: Some(HashMap::from([( diff --git a/src/wasm-lib/kcl/src/parser.rs b/src/wasm-lib/kcl/src/parser.rs index 9fa74fcf9f..f0de970b34 100644 --- a/src/wasm-lib/kcl/src/parser.rs +++ b/src/wasm-lib/kcl/src/parser.rs @@ -1,5 +1,5 @@ use crate::{ - ast::types::{Node, Program}, + ast::types::{ModuleId, Node, Program}, errors::{KclError, KclErrorDetails}, executor::SourceRange, token::{Token, TokenType}, @@ -12,9 +12,15 @@ pub(crate) mod parser_impl; pub const PIPE_SUBSTITUTION_OPERATOR: &str = "%"; pub const PIPE_OPERATOR: &str = "|>"; +/// Parse the given KCL code into an AST. This is the top-level. +pub fn top_level_parse(code: &str) -> Result, KclError> { + let module_id = ModuleId::default(); + parse(code, module_id) +} + /// Parse the given KCL code into an AST. -pub fn parse(code: &str) -> Result, KclError> { - let tokens = crate::token::lexer(code)?; +pub fn parse(code: &str, module_id: ModuleId) -> Result, KclError> { + let tokens = crate::token::lexer(code, module_id)?; let parser = Parser::new(tokens); parser.ast() } diff --git a/src/wasm-lib/kcl/src/parser/bad_inputs.rs b/src/wasm-lib/kcl/src/parser/bad_inputs.rs index 8fa31fb4f0..e416564665 100644 --- a/src/wasm-lib/kcl/src/parser/bad_inputs.rs +++ b/src/wasm-lib/kcl/src/parser/bad_inputs.rs @@ -5,7 +5,8 @@ mod tests { ($func_name:ident, $test_kcl_program:expr) => { #[test] fn $func_name() { - if let Ok(v) = $crate::token::lexer($test_kcl_program) { + let module_id = $crate::parser::ModuleId::default(); + if let Ok(v) = $crate::token::lexer($test_kcl_program, module_id) { let _ = $crate::parser::Parser::new(v).ast(); } } diff --git a/src/wasm-lib/kcl/src/parser/math.rs b/src/wasm-lib/kcl/src/parser/math.rs index e4744ed3d0..cec3cbd97b 100644 --- a/src/wasm-lib/kcl/src/parser/math.rs +++ b/src/wasm-lib/kcl/src/parser/math.rs @@ -30,6 +30,7 @@ fn evaluate(rpn: Vec) -> Result, K }; let start = left.start(); let end = right.end(); + let module_id = left.module_id(); BinaryPart::BinaryExpression(Node::boxed( BinaryExpression { @@ -40,6 +41,7 @@ fn evaluate(rpn: Vec) -> Result, K }, start, end, + module_id, )) } BinaryExpressionToken::Operand(o) => o, @@ -60,11 +62,11 @@ fn source_range(tokens: &[BinaryExpressionToken]) -> Vec { .iter() .filter_map(|op| match op { BinaryExpressionToken::Operator(_) => None, - BinaryExpressionToken::Operand(o) => Some((o.start(), o.end())), + BinaryExpressionToken::Operand(o) => Some((o.start(), o.end(), o.module_id())), }) .collect(); match (sources.first(), sources.last()) { - (Some((start, _)), Some((_, end))) => vec![SourceRange([*start, *end])], + (Some((start, _, module_id)), Some((_, end, _))) => vec![SourceRange([*start, *end, module_id.as_usize()])], _ => Vec::new(), } } @@ -124,7 +126,7 @@ impl From for BinaryExpressionToken { #[cfg(test)] mod tests { use super::*; - use crate::ast::types::Literal; + use crate::ast::types::{Literal, ModuleId}; #[test] fn parse_and_evaluate() { @@ -138,6 +140,7 @@ mod tests { }, 0, 0, + ModuleId::default(), ))) } let tests: Vec> = vec![ @@ -158,6 +161,7 @@ mod tests { }, 0, 0, + ModuleId::default(), )) .into(), BinaryOperator::Pow.into(), diff --git a/src/wasm-lib/kcl/src/parser/parser_impl.rs b/src/wasm-lib/kcl/src/parser/parser_impl.rs index 9e63a8f919..339eef9d44 100644 --- a/src/wasm-lib/kcl/src/parser/parser_impl.rs +++ b/src/wasm-lib/kcl/src/parser/parser_impl.rs @@ -91,6 +91,7 @@ fn non_code_node(i: TokenSlice) -> PResult> { }, leading_whitespace.start, node.end + 1, + node.module_id, )), _ => None, }) @@ -124,7 +125,12 @@ fn non_code_node_no_leading_whitespace(i: TokenSlice) -> PResult return None, }; - Some(Node::new(NonCodeNode { value, digest: None }, token.start, token.end)) + Some(Node::new( + NonCodeNode { value, digest: None }, + token.start, + token.end, + token.module_id, + )) } }) .context(expected("Non-code token (comments or whitespace)")) @@ -194,6 +200,7 @@ fn pipe_expression(i: TokenSlice) -> PResult> { Ok(Node { start: values.first().unwrap().start(), end: values.last().unwrap().end().max(max_noncode_end), + module_id: values.first().unwrap().module_id(), inner: PipeExpression { body: values, non_code_meta, @@ -222,6 +229,7 @@ fn bool_value(i: TokenSlice) -> PResult> { }, token.start, token.end, + token.module_id, ))) } @@ -255,6 +263,7 @@ pub fn string_literal(i: TokenSlice) -> PResult> { }, token.start, token.end, + token.module_id, )) } @@ -290,6 +299,7 @@ pub(crate) fn unsigned_number_literal(i: TokenSlice) -> PResult> { }, token.start, token.end, + token.module_id, )) } @@ -333,7 +343,7 @@ fn operand(i: TokenSlice) -> PResult { const TODO_783: &str = "found a value, but this kind of value cannot be used as the operand to an operator yet (see https://github.com/KittyCAD/modeling-app/issues/783)"; let op = possible_operands .try_map(|part| { - let source_ranges = vec![SourceRange([part.start(), part.end()])]; + let source_ranges = vec![SourceRange::from(&part)]; let expr = match part { // TODO: these should be valid operands eventually, // users should be able to run "let x = f() + g()" @@ -458,6 +468,7 @@ fn shebang(i: TokenSlice) -> PResult> { }, 0, tokens.last().unwrap().end, + tokens.first().unwrap().module_id, )) } @@ -486,7 +497,8 @@ fn array(i: TokenSlice) -> PResult { /// Match an empty array. fn array_empty(i: TokenSlice) -> PResult> { - let start = open_bracket(i)?.start; + let open = open_bracket(i)?; + let start = open.start; ignore_whitespace(i); let end = close_bracket(i)?.end; Ok(Node::new( @@ -497,6 +509,7 @@ fn array_empty(i: TokenSlice) -> PResult> { }, start, end, + open.module_id, )) } @@ -512,7 +525,8 @@ fn array_separator(i: TokenSlice) -> PResult<()> { } pub(crate) fn array_elem_by_elem(i: TokenSlice) -> PResult> { - let start = open_bracket(i)?.start; + let open = open_bracket(i)?; + let start = open.start; ignore_whitespace(i); let elements: Vec<_> = repeat( 0.., @@ -554,11 +568,13 @@ pub(crate) fn array_elem_by_elem(i: TokenSlice) -> PResult }, start, end, + open.module_id, )) } fn array_end_start(i: TokenSlice) -> PResult> { - let start = open_bracket(i)?.start; + let open = open_bracket(i)?; + let start = open.start; ignore_whitespace(i); let start_element = expression.parse_next(i)?; ignore_whitespace(i); @@ -576,6 +592,7 @@ fn array_end_start(i: TokenSlice) -> PResult> { }, start, end, + open.module_id, )) } @@ -596,6 +613,7 @@ fn object_property(i: TokenSlice) -> PResult> { Ok(Node { start: key.start, end: expr.end(), + module_id: key.module_id, inner: ObjectProperty { key, value: expr, @@ -617,7 +635,8 @@ fn property_separator(i: TokenSlice) -> PResult<()> { /// Parse a KCL object value. pub(crate) fn object(i: TokenSlice) -> PResult> { - let start = open_brace(i)?.start; + let open = open_brace(i)?; + let start = open.start; ignore_whitespace(i); let properties: Vec<_> = repeat( 0.., @@ -661,6 +680,7 @@ pub(crate) fn object(i: TokenSlice) -> PResult> { }, start, end, + open.module_id, )) } @@ -668,7 +688,12 @@ pub(crate) fn object(i: TokenSlice) -> PResult> { fn pipe_sub(i: TokenSlice) -> PResult> { any.try_map(|token: Token| { if matches!(token.token_type, TokenType::Operator) && token.value == PIPE_SUBSTITUTION_OPERATOR { - Ok(Node::new(PipeSubstitution { digest: None }, token.start, token.end)) + Ok(Node::new( + PipeSubstitution { digest: None }, + token.start, + token.end, + token.module_id, + )) } else { Err(KclError::Syntax(KclErrorDetails { source_ranges: token.as_source_ranges(), @@ -684,10 +709,10 @@ fn pipe_sub(i: TokenSlice) -> PResult> { } fn else_if(i: TokenSlice) -> PResult> { - let start = any + let else_ = any .try_map(|token: Token| { if matches!(token.token_type, TokenType::Keyword) && token.value == "else" { - Ok(token.start) + Ok(token) } else { Err(KclError::Syntax(KclErrorDetails { source_ranges: token.as_source_ranges(), @@ -728,16 +753,17 @@ fn else_if(i: TokenSlice) -> PResult> { then_val, digest: Default::default(), }, - start, + else_.start, end, + else_.module_id, )) } fn if_expr(i: TokenSlice) -> PResult> { - let start = any + let if_ = any .try_map(|token: Token| { if matches!(token.token_type, TokenType::Keyword) && token.value == "if" { - Ok(token.start) + Ok(token) } else { Err(KclError::Syntax(KclErrorDetails { source_ranges: token.as_source_ranges(), @@ -795,8 +821,9 @@ fn if_expr(i: TokenSlice) -> PResult> { final_else, digest: Default::default(), }, - start, + if_.start, end, + if_.module_id, )) } @@ -806,7 +833,8 @@ fn if_expr(i: TokenSlice) -> PResult> { // return x // } fn function_expression(i: TokenSlice) -> PResult> { - let start = open_paren(i)?.start; + let open = open_paren(i)?; + let start = open.start; let params = parameters(i)?; close_paren(i)?; ignore_whitespace(i); @@ -827,6 +855,7 @@ fn function_expression(i: TokenSlice) -> PResult> { }, start, end, + open.module_id, )) } @@ -874,6 +903,7 @@ fn member_expression(i: TokenSlice) -> PResult> { // which is guaranteed to have >=1 elements. let (property, end, computed) = members.remove(0); let start = id.start; + let module_id = id.module_id; let initial_member_expression = Node::new( MemberExpression { object: MemberObject::Identifier(Box::new(id)), @@ -883,6 +913,7 @@ fn member_expression(i: TokenSlice) -> PResult> { }, start, end, + module_id, ); // Each remaining member wraps the current member expression inside another member expression. @@ -900,6 +931,7 @@ fn member_expression(i: TokenSlice) -> PResult> { }, start, end, + module_id, ) })) } @@ -934,7 +966,12 @@ fn noncode_just_after_code(i: TokenSlice) -> PResult> { x @ NonCodeValue::NewLineBlockComment { .. } => x, x @ NonCodeValue::NewLine => x, }; - Node::new(NonCodeNode { value, ..nc.inner }, nc.start.saturating_sub(1), nc.end) + Node::new( + NonCodeNode { value, ..nc.inner }, + nc.start.saturating_sub(1), + nc.end, + nc.module_id, + ) } else if has_newline { // Nothing has to change, a single newline does not need preserving. nc @@ -950,10 +987,10 @@ fn noncode_just_after_code(i: TokenSlice) -> PResult> { x @ NonCodeValue::NewLineBlockComment { .. } => x, x @ NonCodeValue::NewLine => x, }; - Node::new(NonCodeNode { value, ..nc.inner }, nc.start, nc.end) + Node::new(NonCodeNode { value, ..nc.inner }, nc.start, nc.end, nc.module_id) } }) - .map(|nc| Node::new(nc.inner, nc.start.saturating_sub(1), nc.end)) + .map(|nc| Node::new(nc.inner, nc.start.saturating_sub(1), nc.end, nc.module_id)) .parse_next(i)?; Ok(nc) } @@ -1015,7 +1052,7 @@ pub fn function_body(i: TokenSlice) -> PResult> { // Subtract 1 from `t.start` to match behaviour of the old parser. // Consider removing the -1 in the future because I think it's inaccurate, but for now, // I prefer to match the old parser exactly when I can. - opt(whitespace).map(|tok| tok.and_then(|t| t.first().map(|t| t.start.saturating_sub(1)))), + opt(whitespace).map(|tok| tok.and_then(|t| t.first().map(|t| (t.start.saturating_sub(1), t.module_id)))), )) .parse_next(i)?; @@ -1061,6 +1098,7 @@ pub fn function_body(i: TokenSlice) -> PResult> { }, ws_token.start, ws_token.end, + ws_token.module_id, ))); } } @@ -1103,7 +1141,7 @@ pub fn function_body(i: TokenSlice) -> PResult> { match thing_in_body { WithinFunction::BodyItem((b, maybe_noncode)) => { if start.is_none() { - start = Some(b.start()); + start = Some((b.start(), b.module_id())); } end = b.end(); body.push(b); @@ -1114,7 +1152,7 @@ pub fn function_body(i: TokenSlice) -> PResult> { } WithinFunction::NonCode(nc) => { if start.is_none() { - start = Some(nc.start); + start = Some((nc.start, nc.module_id)); } end = nc.end; if body.is_empty() { @@ -1143,8 +1181,9 @@ pub fn function_body(i: TokenSlice) -> PResult> { non_code_meta, digest: None, }, - start, + start.0, end, + start.1, )) } @@ -1200,7 +1239,7 @@ fn import_stmt(i: TokenSlice) -> PResult> { { return Err(ErrMode::Cut( KclError::Syntax(KclErrorDetails { - source_ranges: vec![SourceRange::new(path.start, path.end)], + source_ranges: vec![SourceRange::new(path.start, path.end, path.module_id)], message: "import path may only contain alphanumeric characters, underscore, hyphen, and period. Files in other directories are not yet supported.".to_owned(), }) .into(), @@ -1215,12 +1254,14 @@ fn import_stmt(i: TokenSlice) -> PResult> { }, start, end, + import_token.module_id, )) } fn import_item(i: TokenSlice) -> PResult> { let name = identifier.context(expected("an identifier to import")).parse_next(i)?; let start = name.start; + let module_id = name.module_id; let alias = opt(preceded( (whitespace, import_as_keyword, whitespace), identifier.context(expected("an identifier to alias the import")), @@ -1239,6 +1280,7 @@ fn import_item(i: TokenSlice) -> PResult> { }, start, end, + module_id, )) } @@ -1259,10 +1301,10 @@ fn import_as_keyword(i: TokenSlice) -> PResult { /// Parse a return statement of a user-defined function, e.g. `return x`. pub fn return_stmt(i: TokenSlice) -> PResult> { - let start = any + let ret = any .try_map(|token: Token| { if matches!(token.token_type, TokenType::Keyword) && token.value == "return" { - Ok(token.start) + Ok(token) } else { Err(KclError::Syntax(KclErrorDetails { source_ranges: token.as_source_ranges(), @@ -1277,8 +1319,9 @@ pub fn return_stmt(i: TokenSlice) -> PResult> { require_whitespace(i)?; let argument = expression(i)?; Ok(Node { - start, + start: ret.start, end: argument.end(), + module_id: ret.module_id, inner: ReturnStatement { argument, digest: None }, }) } @@ -1385,10 +1428,10 @@ fn declaration(i: TokenSlice) -> PResult> { "an identifier, which becomes name you're binding the value to", )) .parse_next(i)?; - let (kind, mut start, dec_end) = if let Some((kind, token)) = &decl_token { - (*kind, token.start, token.end) + let (kind, mut start, dec_end, module_id) = if let Some((kind, token)) = &decl_token { + (*kind, token.start, token.end, token.module_id) } else { - (VariableKind::Const, id.start, id.end) + (VariableKind::Const, id.start, id.end, id.module_id) }; if let Some(token) = visibility_token { start = token.start; @@ -1419,7 +1462,7 @@ fn declaration(i: TokenSlice) -> PResult> { // Check the 'if' direction: if matches!(val, Expr::FunctionExpression(_)) { return Err(KclError::Syntax(KclErrorDetails { - source_ranges: vec![SourceRange([start, dec_end])], + source_ranges: vec![SourceRange([start, dec_end, module_id.as_usize()])], message: format!("Expected a `fn` variable kind, found: `{}`", kind), })); } @@ -1436,6 +1479,7 @@ fn declaration(i: TokenSlice) -> PResult> { declarations: vec![Node { start: id.start, end, + module_id, inner: VariableDeclarator { id, init: val, @@ -1448,6 +1492,7 @@ fn declaration(i: TokenSlice) -> PResult> { }, start, end, + module_id, })) } @@ -1463,6 +1508,7 @@ impl TryFrom for Node { }, token.start, token.end, + token.module_id, )) } else { Err(KclError::Syntax(KclErrorDetails { @@ -1493,6 +1539,7 @@ fn sketch_keyword(i: TokenSlice) -> PResult> { }, token.start, token.end, + token.module_id, )) } else { Err(KclError::Syntax(KclErrorDetails { @@ -1518,6 +1565,7 @@ impl TryFrom for Node { }, token.start - 1, token.end, + token.module_id, )) } else { Err(KclError::Syntax(KclErrorDetails { @@ -1533,7 +1581,7 @@ impl Node { // Make sure they are not assigning a variable to a stdlib function. if crate::std::name_in_stdlib(&self.name) { return Err(KclError::Syntax(KclErrorDetails { - source_ranges: vec![SourceRange([self.start, self.end])], + source_ranges: vec![SourceRange::from(&self)], message: format!("Cannot assign a tag to a reserved keyword: {}", self.name), })); } @@ -1588,6 +1636,7 @@ fn unary_expression(i: TokenSlice) -> PResult> { Ok(Node { start: op_token.start, end: argument.end(), + module_id: op_token.module_id, inner: UnaryExpression { operator, argument, @@ -1669,6 +1718,7 @@ fn expression_stmt(i: TokenSlice) -> PResult> { Ok(Node { start: val.start(), end: val.end(), + module_id: val.module_id(), inner: ExpressionStatement { expression: val, digest: None, @@ -1907,7 +1957,7 @@ impl Node { // Make sure they are not assigning a variable to a stdlib function. if crate::std::name_in_stdlib(&self.name) { return Err(KclError::Syntax(KclErrorDetails { - source_ranges: vec![SourceRange([self.start, self.end])], + source_ranges: vec![SourceRange::from(&self)], message: format!("Cannot assign a variable to a reserved keyword: {}", self.name), })); } @@ -1950,7 +2000,7 @@ fn fn_call(i: TokenSlice) -> PResult> { e => { return Err(ErrMode::Cut( KclError::Syntax(KclErrorDetails { - source_ranges: vec![SourceRange([arg.start(), arg.end()])], + source_ranges: vec![SourceRange::from(*arg)], message: format!("Expected a tag declarator like `$name`, found {:?}", e), }) .into(), @@ -1963,7 +2013,7 @@ fn fn_call(i: TokenSlice) -> PResult> { e => { return Err(ErrMode::Cut( KclError::Syntax(KclErrorDetails { - source_ranges: vec![SourceRange([arg.start(), arg.end()])], + source_ranges: vec![SourceRange::from(*arg)], message: format!("Expected a tag identifier like `tagName`, found {:?}", e), }) .into(), @@ -1978,6 +2028,7 @@ fn fn_call(i: TokenSlice) -> PResult> { Ok(Node { start: fn_name.start, end, + module_id: fn_name.module_id, inner: CallExpression { callee: fn_name, arguments: args, @@ -1992,12 +2043,12 @@ mod tests { use pretty_assertions::assert_eq; use super::*; - use crate::ast::types::{BodyItem, Expr, VariableKind}; + use crate::ast::types::{BodyItem, Expr, ModuleId, VariableKind}; #[test] fn parse_args() { for (i, (test, expected_len)) in [("someVar", 1), ("5, 3", 2), (r#""a""#, 1)].into_iter().enumerate() { - let tokens = crate::token::lexer(test).unwrap(); + let tokens = crate::token::lexer(test, ModuleId::default()).unwrap(); let actual = match arguments.parse(&tokens) { Ok(x) => x, Err(e) => panic!("Failed test {i}, could not parse function arguments from \"{test}\": {e:?}"), @@ -2008,7 +2059,7 @@ mod tests { #[test] fn weird_program_unclosed_paren() { - let tokens = crate::token::lexer("fn firstPrime=(").unwrap(); + let tokens = crate::token::lexer("fn firstPrime=(", ModuleId::default()).unwrap(); let last = tokens.last().unwrap(); let err: KclError = program.parse(&tokens).unwrap_err().into(); assert_eq!(err.source_ranges(), last.as_source_ranges()); @@ -2019,16 +2070,16 @@ mod tests { #[test] fn weird_program_just_a_pipe() { - let tokens = crate::token::lexer("|").unwrap(); + let tokens = crate::token::lexer("|", ModuleId::default()).unwrap(); let err: KclError = program.parse(&tokens).unwrap_err().into(); - assert_eq!(err.source_ranges(), vec![SourceRange([0, 1])]); + assert_eq!(err.source_ranges(), vec![SourceRange([0, 1, 0])]); assert_eq!(err.message(), "Unexpected token: |"); } #[test] fn parse_binary_expressions() { for (i, test_program) in ["1 + 2 + 3"].into_iter().enumerate() { - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let mut slice = tokens.as_slice(); let _actual = match binary_expression.parse_next(&mut slice) { Ok(x) => x, @@ -2039,7 +2090,7 @@ mod tests { #[test] fn test_vardec_no_keyword() { - let tokens = crate::token::lexer("x = 4").unwrap(); + let tokens = crate::token::lexer("x = 4", ModuleId::default()).unwrap(); let vardec = declaration(&mut tokens.as_slice()).unwrap(); assert_eq!(vardec.inner.kind, VariableKind::Const); let vardec = vardec.declarations.first().unwrap(); @@ -2052,7 +2103,7 @@ mod tests { #[test] fn test_negative_operands() { - let tokens = crate::token::lexer("-leg2").unwrap(); + let tokens = crate::token::lexer("-leg2", ModuleId::default()).unwrap(); let _s = operand.parse_next(&mut tokens.as_slice()).unwrap(); } @@ -2066,7 +2117,7 @@ mod tests { // comment 2 return 1 }"#; - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let mut slice = tokens.as_slice(); let expr = function_expression.parse_next(&mut slice).unwrap(); assert_eq!(expr.params, vec![]); @@ -2084,7 +2135,7 @@ mod tests { const yo = { a: { b: { c: '123' } } } /* block comment */ }"#; - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let mut slice = tokens.as_slice(); let expr = function_expression.parse_next(&mut slice).unwrap(); let comment0 = &expr.body.non_code_meta.non_code_nodes.get(&0).unwrap()[0]; @@ -2097,7 +2148,7 @@ comment */ /* comment at start */ const mySk1 = startSketchAt([0, 0])"#; - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let program = program.parse(&tokens).unwrap(); let mut starting_comments = program.inner.non_code_meta.start_nodes; assert_eq!(starting_comments.len(), 2); @@ -2115,7 +2166,7 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_comment_in_pipe() { - let tokens = crate::token::lexer(r#"const x = y() |> /*hi*/ z(%)"#).unwrap(); + let tokens = crate::token::lexer(r#"const x = y() |> /*hi*/ z(%)"#, ModuleId::default()).unwrap(); let mut body = program.parse(&tokens).unwrap().inner.body; let BodyItem::VariableDeclaration(mut item) = body.remove(0) else { panic!("expected vardec"); @@ -2142,7 +2193,7 @@ const mySk1 = startSketchAt([0, 0])"#; return sg return sg }"#; - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let mut slice = tokens.as_slice(); let _expr = function_expression.parse_next(&mut slice).unwrap(); } @@ -2153,7 +2204,8 @@ const mySk1 = startSketchAt([0, 0])"#; return 2 }"; - let tokens = crate::token::lexer(test_program).unwrap(); + let module_id = ModuleId::from_usize(1); + let tokens = crate::token::lexer(test_program, module_id).unwrap(); let mut slice = tokens.as_slice(); let expr = function_expression.parse_next(&mut slice).unwrap(); assert_eq!( @@ -2173,11 +2225,13 @@ const mySk1 = startSketchAt([0, 0])"#; }, 32, 33, + module_id, ))), digest: None, }, 25, 33, + module_id, ))], non_code_meta: NonCodeMeta { non_code_nodes: Default::default(), @@ -2188,6 +2242,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 7, 25, + module_id, )], digest: None, }, @@ -2195,12 +2250,14 @@ const mySk1 = startSketchAt([0, 0])"#; }, 7, 47, + module_id, ), return_type: None, digest: None, }, 0, 47, + module_id, ) ); } @@ -2212,7 +2269,7 @@ const mySk1 = startSketchAt([0, 0])"#; |> c(%) // inline-comment |> d(%)"#; - let tokens = crate::token::lexer(test_input).unwrap(); + let tokens = crate::token::lexer(test_input, ModuleId::default()).unwrap(); let mut slice = tokens.as_slice(); let Node { inner: PipeExpression { @@ -2242,7 +2299,8 @@ const mySk1 = startSketchAt([0, 0])"#; return things "#; - let tokens = crate::token::lexer(test_program).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(test_program, module_id).unwrap(); let Program { non_code_meta, .. } = function_body.parse(&tokens).unwrap().inner; assert_eq!( vec![Node::new( @@ -2255,6 +2313,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 20, + module_id, )], non_code_meta.start_nodes, ); @@ -2271,6 +2330,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 60, 82, + module_id, ), Node::new( NonCodeNode { @@ -2279,6 +2339,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 82, 86, + module_id, ) ]), non_code_meta.non_code_nodes.get(&0), @@ -2295,6 +2356,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 103, 129, + module_id, )]), non_code_meta.non_code_nodes.get(&1), ); @@ -2306,7 +2368,7 @@ const mySk1 = startSketchAt([0, 0])"#; comment */ return 1"#; - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let actual = program.parse(&tokens).unwrap(); assert_eq!(actual.non_code_meta.non_code_nodes.len(), 1); assert_eq!( @@ -2321,7 +2383,7 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_bracketed_binary_expression() { let input = "(2 - 3)"; - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); let actual = match binary_expr_in_parens.parse(&tokens) { Ok(x) => x, Err(e) => panic!("{e:?}"), @@ -2336,7 +2398,7 @@ const mySk1 = startSketchAt([0, 0])"#; "6 / ( sigmaAllow * width )", "sqrt(distance * p * FOS * 6 / ( sigmaAllow * width ))", ] { - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); let _actual = match expression.parse(&tokens) { Ok(x) => x, Err(e) => panic!("{e:?}"), @@ -2347,7 +2409,7 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_arithmetic() { let input = "1 * (2 - 3)"; - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); // The RHS should be a binary expression. let actual = binary_expression.parse(&tokens).unwrap(); assert_eq!(actual.operator, BinaryOperator::Mul); @@ -2375,7 +2437,7 @@ const mySk1 = startSketchAt([0, 0])"#; .into_iter() .enumerate() { - let tokens = crate::token::lexer(test_input).unwrap(); + let tokens = crate::token::lexer(test_input, ModuleId::default()).unwrap(); let mut actual = match declaration.parse(&tokens) { Err(e) => panic!("Could not parse test {i}: {e:#?}"), Ok(a) => a, @@ -2393,7 +2455,7 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_function_call() { for (i, test_input) in ["const x = f(1)", "const x = f( 1 )"].into_iter().enumerate() { - let tokens = crate::token::lexer(test_input).unwrap(); + let tokens = crate::token::lexer(test_input, ModuleId::default()).unwrap(); let _actual = match declaration.parse(&tokens) { Err(e) => panic!("Could not parse test {i}: {e:#?}"), Ok(a) => a, @@ -2404,7 +2466,7 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_nested_arithmetic() { let input = "1 * ((2 - 3) / 4)"; - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); // The RHS should be a binary expression. let outer = binary_expression.parse(&tokens).unwrap(); assert_eq!(outer.operator, BinaryOperator::Mul); @@ -2423,7 +2485,7 @@ const mySk1 = startSketchAt([0, 0])"#; fn binary_expression_ignores_whitespace() { let tests = ["1 - 2", "1- 2", "1 -2", "1-2"]; for test in tests { - let tokens = crate::token::lexer(test).unwrap(); + let tokens = crate::token::lexer(test, ModuleId::default()).unwrap(); let actual = binary_expression.parse(&tokens).unwrap(); assert_eq!(actual.operator, BinaryOperator::Sub); let BinaryPart::Literal(left) = actual.inner.left else { @@ -2444,7 +2506,7 @@ const mySk1 = startSketchAt([0, 0])"#; a comment spanning a few lines */ |> z(%)"#; - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let actual = pipe_expression.parse(&tokens).unwrap(); let n = actual.non_code_meta.non_code_nodes.len(); assert_eq!(n, 1, "expected one comment in pipe expression but found {n}"); @@ -2472,7 +2534,7 @@ const mySk1 = startSketchAt([0, 0])"#; .into_iter() .enumerate() { - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let actual = pipe_expression.parse(&tokens); assert!(actual.is_ok(), "could not parse test {i}, '{test_program}'"); let actual = actual.unwrap(); @@ -2483,6 +2545,7 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn comments() { + let module_id = ModuleId::from_usize(1); for (i, (test_program, expected)) in [ ( "//hi", @@ -2496,6 +2559,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 4, + module_id, ), ), ( @@ -2510,6 +2574,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 9, + module_id, ), ), ( @@ -2524,6 +2589,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 11, + module_id, ), ), ( @@ -2538,6 +2604,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 12, + module_id, ), ), ( @@ -2553,6 +2620,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 29, + module_id, ), ), ( @@ -2570,6 +2638,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 32, + module_id, ), ), ( @@ -2587,6 +2656,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 30, + module_id, ), ), ( @@ -2602,13 +2672,14 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 39, + module_id, ), ), ] .into_iter() .enumerate() { - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, module_id).unwrap(); let actual = non_code_node.parse(&tokens); assert!(actual.is_ok(), "could not parse test {i}: {actual:#?}"); let actual = actual.unwrap(); @@ -2619,11 +2690,12 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn recognize_invalid_params() { let test_fn = "(let) => { return 1 }"; - let tokens = crate::token::lexer(test_fn).unwrap(); + let module_id = ModuleId::from_usize(2); + let tokens = crate::token::lexer(test_fn, module_id).unwrap(); let err = function_expression.parse(&tokens).unwrap_err().into_inner(); let cause = err.cause.unwrap(); // This is the token `let` - assert_eq!(cause.source_ranges(), vec![SourceRange([1, 4])]); + assert_eq!(cause.source_ranges(), vec![SourceRange([1, 4, 2])]); assert_eq!(cause.message(), "Cannot assign a variable to a reserved keyword: let"); } @@ -2632,7 +2704,7 @@ const mySk1 = startSketchAt([0, 0])"#; let string_literal = r#"" // a comment ""#; - let tokens = crate::token::lexer(string_literal).unwrap(); + let tokens = crate::token::lexer(string_literal, ModuleId::default()).unwrap(); let parsed_literal = literal.parse(&tokens).unwrap(); assert_eq!( parsed_literal.value, @@ -2649,7 +2721,7 @@ const mySk1 = startSketchAt([0, 0])"#; |> lineTo([0, -0], %) // MoveRelative "#; - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let mut slice = &tokens[..]; let _actual = pipe_expression.parse_next(&mut slice).unwrap(); assert_eq!(slice[0].token_type, TokenType::Whitespace); @@ -2658,14 +2730,14 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_pipes_on_pipes() { let test_program = include_str!("../../../tests/executor/inputs/pipes_on_pipes.kcl"); - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); let _actual = program.parse(&tokens).unwrap(); } #[test] fn test_cube() { let test_program = include_str!("../../../tests/executor/inputs/cube.kcl"); - let tokens = crate::token::lexer(test_program).unwrap(); + let tokens = crate::token::lexer(test_program, ModuleId::default()).unwrap(); match program.parse(&tokens) { Ok(_) => {} Err(e) => { @@ -2683,7 +2755,7 @@ const mySk1 = startSketchAt([0, 0])"#; ("a,b", vec!["a", "b"]), ]; for (i, (input, expected)) in tests.into_iter().enumerate() { - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); let actual = parameters.parse(&tokens); assert!(actual.is_ok(), "could not parse test {i}"); let actual_ids: Vec<_> = actual.unwrap().into_iter().map(|p| p.identifier.inner.name).collect(); @@ -2697,7 +2769,7 @@ const mySk1 = startSketchAt([0, 0])"#; return 2 }"; - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); let actual = function_expression.parse(&tokens); assert!(actual.is_ok(), "could not parse test function"); } @@ -2707,7 +2779,7 @@ const mySk1 = startSketchAt([0, 0])"#; let tests = ["const myVar = 5", "const myVar=5", "const myVar =5", "const myVar= 5"]; for test in tests { // Run the original parser - let tokens = crate::token::lexer(test).unwrap(); + let tokens = crate::token::lexer(test, ModuleId::default()).unwrap(); let mut expected_body = crate::parser::Parser::new(tokens.clone()).ast().unwrap().inner.body; assert_eq!(expected_body.len(), 1); let BodyItem::VariableDeclaration(expected) = expected_body.pop().unwrap() else { @@ -2734,7 +2806,8 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_math_parse() { - let tokens = crate::token::lexer(r#"5 + "a""#).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(r#"5 + "a""#, module_id).unwrap(); let actual = crate::parser::Parser::new(tokens).ast().unwrap().inner.body; let expr = Node::boxed( BinaryExpression { @@ -2747,6 +2820,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 1, + module_id, ))), right: BinaryPart::Literal(Box::new(Node::new( Literal { @@ -2756,11 +2830,13 @@ const mySk1 = startSketchAt([0, 0])"#; }, 4, 7, + module_id, ))), digest: None, }, 0, 7, + module_id, ); let expected = vec![BodyItem::ExpressionStatement(Node::new( ExpressionStatement { @@ -2769,53 +2845,62 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 7, + module_id, ))]; assert_eq!(expected, actual); } #[test] fn test_is_code_token() { + let module_id = ModuleId::default(); let tokens = [ Token { token_type: TokenType::Word, start: 0, end: 3, + module_id, value: "log".to_string(), }, Token { token_type: TokenType::Brace, start: 3, end: 4, + module_id, value: "(".to_string(), }, Token { token_type: TokenType::Number, start: 4, end: 5, + module_id, value: "5".to_string(), }, Token { token_type: TokenType::Comma, start: 5, end: 6, + module_id, value: ",".to_string(), }, Token { token_type: TokenType::String, start: 7, end: 14, + module_id, value: "\"hello\"".to_string(), }, Token { token_type: TokenType::Word, start: 16, end: 27, + module_id, value: "aIdentifier".to_string(), }, Token { token_type: TokenType::Brace, start: 27, end: 28, + module_id, value: ")".to_string(), }, ]; @@ -2826,23 +2911,27 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_is_not_code_token() { + let module_id = ModuleId::default(); let tokens = [ Token { token_type: TokenType::Whitespace, start: 6, end: 7, + module_id, value: " ".to_string(), }, Token { token_type: TokenType::BlockComment, start: 28, end: 30, + module_id, value: "/* abte */".to_string(), }, Token { token_type: TokenType::LineComment, start: 30, end: 33, + module_id, value: "// yoyo a line".to_string(), }, ]; @@ -2854,7 +2943,8 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_abstract_syntax_tree() { let code = "5 +6"; - let parser = crate::parser::Parser::new(crate::token::lexer(code).unwrap()); + let module_id = ModuleId::default(); + let parser = crate::parser::Parser::new(crate::token::lexer(code, module_id).unwrap()); let result = parser.ast().unwrap(); let expected_result = Node::new( Program { @@ -2870,6 +2960,7 @@ const mySk1 = startSketchAt([0, 0])"#; }, 0, 1, + module_id, ))), operator: BinaryOperator::Add, right: BinaryPart::Literal(Box::new(Node::new( @@ -2880,22 +2971,26 @@ const mySk1 = startSketchAt([0, 0])"#; }, 3, 4, + module_id, ))), digest: None, }, 0, 4, + module_id, )), digest: None, }, 0, 4, + module_id, ))], non_code_meta: NonCodeMeta::default(), digest: None, }, 0, 4, + module_id, ); assert_eq!(result, expected_result); @@ -2904,22 +2999,16 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_empty_file() { let some_program_string = r#""#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let result = parser.ast(); + let result = crate::parser::top_level_parse(some_program_string); assert!(result.is_ok()); } #[test] fn test_parse_half_pipe_small() { - let tokens = crate::token::lexer( - "const secondExtrude = startSketchOn('XY') + let code = "const secondExtrude = startSketchOn('XY') |> startProfileAt([0,0], %) - |", - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - let result = parser.ast(); + |"; + let result = crate::parser::top_level_parse(code); assert!(result.is_err()); let actual = result.err().unwrap().to_string(); assert!(actual.contains("Unexpected token: |"), "actual={actual:?}"); @@ -2927,93 +3016,63 @@ const mySk1 = startSketchAt([0, 0])"#; #[test] fn test_parse_member_expression_double_nested_braces() { - let tokens = crate::token::lexer(r#"const prop = yo["one"][two]"#).unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + let code = r#"const prop = yo["one"][two]"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_member_expression_binary_expression_period_number_first() { - let tokens = crate::token::lexer( - r#"const obj = { a: 1, b: 2 } -const height = 1 - obj.a"#, - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + let code = r#"const obj = { a: 1, b: 2 } +const height = 1 - obj.a"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_member_expression_allowed_type_in_expression() { - let tokens = crate::token::lexer( - r#"const obj = { thing: 1 } -startSketchOn(obj.sketch)"#, - ) - .unwrap(); + let code = r#"const obj = { thing: 1 } +startSketchOn(obj.sketch)"#; - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_member_expression_binary_expression_brace_number_first() { - let tokens = crate::token::lexer( - r#"const obj = { a: 1, b: 2 } -const height = 1 - obj["a"]"#, - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + let code = r#"const obj = { a: 1, b: 2 } +const height = 1 - obj["a"]"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_member_expression_binary_expression_brace_number_second() { - let tokens = crate::token::lexer( - r#"const obj = { a: 1, b: 2 } -const height = obj["a"] - 1"#, - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + let code = r#"const obj = { a: 1, b: 2 } +const height = obj["a"] - 1"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_member_expression_binary_expression_in_array_number_first() { - let tokens = crate::token::lexer( - r#"const obj = { a: 1, b: 2 } -const height = [1 - obj["a"], 0]"#, - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + let code = r#"const obj = { a: 1, b: 2 } +const height = [1 - obj["a"], 0]"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_member_expression_binary_expression_in_array_number_second() { - let tokens = crate::token::lexer( - r#"const obj = { a: 1, b: 2 } -const height = [obj["a"] - 1, 0]"#, - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + let code = r#"const obj = { a: 1, b: 2 } +const height = [obj["a"] - 1, 0]"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_member_expression_binary_expression_in_array_number_second_missing_space() { - let tokens = crate::token::lexer( - r#"const obj = { a: 1, b: 2 } -const height = [obj["a"] -1, 0]"#, - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + let code = r#"const obj = { a: 1, b: 2 } +const height = [obj["a"] -1, 0]"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_half_pipe() { - let tokens = crate::token::lexer( - "const height = 10 + let code = "const height = 10 const firstExtrude = startSketchOn('XY') |> startProfileAt([0,0], %) @@ -3025,70 +3084,67 @@ const firstExtrude = startSketchOn('XY') const secondExtrude = startSketchOn('XY') |> startProfileAt([0,0], %) - |", - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - let result = parser.ast(); + |"; + let result = crate::parser::top_level_parse(code); assert!(result.is_err()); assert!(result.err().unwrap().to_string().contains("Unexpected token: |")); } #[test] fn test_parse_greater_bang() { - let tokens = crate::token::lexer(">!").unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(">!", module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let err = parser.ast().unwrap_err(); assert_eq!( err.to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([0, 1])], message: "Unexpected token: >" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([0, 1, 0])], message: "Unexpected token: >" }"# ); } #[test] fn test_parse_z_percent_parens() { - let tokens = crate::token::lexer("z%)").unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer("z%)", module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([1, 2])], message: "Unexpected token: %" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([1, 2, 0])], message: "Unexpected token: %" }"# ); } #[test] fn test_parse_parens_unicode() { - let result = crate::token::lexer("(ޜ"); + let module_id = ModuleId::default(); + let result = crate::token::lexer("(ޜ", module_id); // TODO: Better errors when program cannot tokenize. // https://github.com/KittyCAD/modeling-app/issues/696 assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"lexical: KclErrorDetails { source_ranges: [SourceRange([1, 2])], message: "found unknown token 'ޜ'" }"# + r#"lexical: KclErrorDetails { source_ranges: [SourceRange([1, 2, 0])], message: "found unknown token 'ޜ'" }"# ); } #[test] fn test_parse_negative_in_array_binary_expression() { - let tokens = crate::token::lexer( - r#"const leg1 = 5 + let code = r#"const leg1 = 5 const thickness = 0.56 const bracket = [-leg2 + thickness, 0] -"#, - ) - .unwrap(); - let parser = crate::parser::Parser::new(tokens); - let result = parser.ast(); - assert!(result.is_ok()); +"#; + crate::parser::top_level_parse(code).unwrap(); } #[test] fn test_parse_nested_open_brackets() { + let module_id = ModuleId::default(); let tokens = crate::token::lexer( r#" z(-[["#, + module_id, ) .unwrap(); let parser = crate::parser::Parser::new(tokens); @@ -3098,9 +3154,11 @@ z(-[["#, #[test] fn test_parse_weird_new_line_function() { + let module_id = ModuleId::default(); let tokens = crate::token::lexer( r#"z (--#"#, + module_id, ) .unwrap(); let parser = crate::parser::Parser::new(tokens); @@ -3108,28 +3166,31 @@ z(-[["#, assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([3, 4])], message: "Unexpected token: (" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([3, 4, 0])], message: "Unexpected token: (" }"# ); } #[test] fn test_parse_weird_lots_of_fancy_brackets() { - let tokens = crate::token::lexer(r#"zz({{{{{{{{)iegAng{{{{{{{##"#).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(r#"zz({{{{{{{{)iegAng{{{{{{{##"#, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([2, 3])], message: "Unexpected token: (" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([2, 3, 0])], message: "Unexpected token: (" }"# ); } #[test] fn test_parse_weird_close_before_open() { + let module_id = ModuleId::default(); let tokens = crate::token::lexer( r#"fn)n e ["#, + module_id, ) .unwrap(); let parser = crate::parser::Parser::new(tokens); @@ -3144,7 +3205,8 @@ e #[test] fn test_parse_weird_close_before_nada() { - let tokens = crate::token::lexer(r#"fn)n-"#).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(r#"fn)n-"#, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); @@ -3157,9 +3219,11 @@ e #[test] fn test_parse_weird_lots_of_slashes() { + let module_id = ModuleId::default(); let tokens = crate::token::lexer( r#"J///////////o//+///////////P++++*++++++P///////˟ ++4"#, + module_id, ) .unwrap(); let parser = crate::parser::Parser::new(tokens); @@ -3254,26 +3318,28 @@ e #[test] fn test_error_keyword_in_variable() { let some_program_string = r#"const let = "thing""#; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([6, 9])], message: "Cannot assign a variable to a reserved keyword: let" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([6, 9, 0])], message: "Cannot assign a variable to a reserved keyword: let" }"# ); } #[test] fn test_error_keyword_in_fn_name() { let some_program_string = r#"fn let = () {}"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([3, 6])], message: "Cannot assign a variable to a reserved keyword: let" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([3, 6, 0])], message: "Cannot assign a variable to a reserved keyword: let" }"# ); } @@ -3282,13 +3348,14 @@ e let some_program_string = r#"fn cos = () => { return 1 }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([3, 6])], message: "Cannot assign a variable to a reserved keyword: cos" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([3, 6, 0])], message: "Cannot assign a variable to a reserved keyword: cos" }"# ); } @@ -3297,13 +3364,14 @@ e let some_program_string = r#"fn thing = (let) => { return 1 }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([12, 15])], message: "Cannot assign a variable to a reserved keyword: let" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([12, 15, 0])], message: "Cannot assign a variable to a reserved keyword: let" }"# ); } @@ -3312,33 +3380,33 @@ e let some_program_string = r#"fn thing = (cos) => { return 1 }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([12, 15])], message: "Cannot assign a variable to a reserved keyword: cos" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([12, 15, 0])], message: "Cannot assign a variable to a reserved keyword: cos" }"# ); } #[test] fn zero_param_function() { - let program = r#" + let code = r#" fn firstPrimeNumber = () => { return 2 } firstPrimeNumber() "#; - let tokens = crate::token::lexer(program).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let _ast = parser.ast().unwrap(); + let _ast = crate::parser::top_level_parse(code).unwrap(); } #[test] fn array() { let program = r#"[1, 2, 3]"#; - let tokens = crate::token::lexer(program).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(program, module_id).unwrap(); let mut sl: &[Token] = &tokens; let _arr = array_elem_by_elem(&mut sl).unwrap(); } @@ -3350,7 +3418,8 @@ e 2, 3, ]"#; - let tokens = crate::token::lexer(program).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(program, module_id).unwrap(); let mut sl: &[Token] = &tokens; let _arr = array_elem_by_elem(&mut sl).unwrap(); } @@ -3363,7 +3432,8 @@ e 2, 3 ]"#; - let tokens = crate::token::lexer(program).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(program, module_id).unwrap(); let mut sl: &[Token] = &tokens; let _arr = array_elem_by_elem(&mut sl).unwrap(); } @@ -3375,7 +3445,8 @@ e } else { 4 }"; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let mut sl: &[Token] = &tokens; let _res = if_expr(&mut sl).unwrap(); } @@ -3385,7 +3456,8 @@ e let some_program_string = "else if true { 4 }"; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let mut sl: &[Token] = &tokens; let _res = else_if(&mut sl).unwrap(); } @@ -3399,7 +3471,8 @@ e } else { 5 }"; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let mut sl: &[Token] = &tokens; let _res = if_expr(&mut sl).unwrap(); } @@ -3412,9 +3485,7 @@ e thing(false) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + crate::parser::top_level_parse(some_program_string).unwrap(); } #[test] @@ -3429,14 +3500,15 @@ thing(false) "#, name ); - let tokens = crate::token::lexer(&some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(&some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), format!( - r#"syntax: KclErrorDetails {{ source_ranges: [SourceRange([0, {}])], message: "Expected a `fn` variable kind, found: `const`" }}"#, + r#"syntax: KclErrorDetails {{ source_ranges: [SourceRange([0, {}, 0])], message: "Expected a `fn` variable kind, found: `const`" }}"#, name.len(), ) ); @@ -3446,7 +3518,8 @@ thing(false) #[test] fn test_error_define_var_as_function() { let some_program_string = r#"fn thing = "thing""#; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); assert!(result.is_err()); @@ -3455,7 +3528,7 @@ thing(false) // It should say that the compiler is expecting a function expression on the RHS. assert_eq!( result.err().unwrap().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([11, 18])], message: "Unexpected token: \"thing\"" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([11, 18, 0])], message: "Unexpected token: \"thing\"" }"# ); } @@ -3469,7 +3542,8 @@ thing(false) |> line([-5.09, 12.33], %) asdasd "#; - let tokens = crate::token::lexer(test_program).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(test_program, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let result = parser.ast(); let _e = result.unwrap_err(); @@ -3493,18 +3567,14 @@ const b2 = cube([3,3], 4) const pt1 = b1[0] const pt2 = b2[0] "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + crate::parser::top_level_parse(some_program_string).unwrap(); } #[test] fn test_math_with_stdlib() { let some_program_string = r#"const d2r = pi() / 2 let other_thing = 2 * cos(3)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + crate::parser::top_level_parse(some_program_string).unwrap(); } #[test] @@ -3522,9 +3592,7 @@ let other_thing = 2 * cos(3)"#; } let myBox = box([0,0], -3, -16, -10) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - parser.ast().unwrap(); + crate::parser::top_level_parse(some_program_string).unwrap(); } #[test] fn must_use_percent_in_pipeline_fn() { @@ -3532,12 +3600,13 @@ let myBox = box([0,0], -3, -16, -10) foo() |> bar(2) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); + let module_id = ModuleId::default(); + let tokens = crate::token::lexer(some_program_string, module_id).unwrap(); let parser = crate::parser::Parser::new(tokens); let err = parser.ast().unwrap_err(); assert_eq!( err.to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([30, 36])], message: "All expressions in a pipeline must use the % (substitution operator)" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([30, 36, 0])], message: "All expressions in a pipeline must use the % (substitution operator)" }"# ); } } @@ -3553,7 +3622,8 @@ mod snapshot_math_tests { ($func_name:ident, $test_kcl_program:expr) => { #[test] fn $func_name() { - let tokens = crate::token::lexer($test_kcl_program).unwrap(); + let module_id = crate::ast::types::ModuleId::default(); + let tokens = crate::token::lexer($test_kcl_program, module_id).unwrap(); let actual = match binary_expression.parse(&tokens) { Ok(x) => x, Err(_e) => panic!("could not parse test"), @@ -3587,7 +3657,8 @@ mod snapshot_tests { ($func_name:ident, $test_kcl_program:expr) => { #[test] fn $func_name() { - let tokens = crate::token::lexer($test_kcl_program).unwrap(); + let module_id = crate::ast::types::ModuleId::default(); + let tokens = crate::token::lexer($test_kcl_program, module_id).unwrap(); print_tokens(&tokens); let actual = match program.parse(&tokens) { Ok(x) => x, diff --git a/src/wasm-lib/kcl/src/parser/parser_impl/error.rs b/src/wasm-lib/kcl/src/parser/parser_impl/error.rs index 11dea7c3fd..a4697076b8 100644 --- a/src/wasm-lib/kcl/src/parser/parser_impl/error.rs +++ b/src/wasm-lib/kcl/src/parser/parser_impl/error.rs @@ -1,13 +1,12 @@ use winnow::{ error::{ErrorKind, ParseError, StrContext}, stream::Stream, - Located, }; use crate::{ errors::{KclError, KclErrorDetails}, executor::SourceRange, - token::Token, + token::{Input, Token}, }; /// Accumulate context while backtracking errors @@ -20,9 +19,10 @@ pub struct ContextError { pub cause: Option, } -impl From, winnow::error::ContextError>> for KclError { - fn from(err: ParseError, winnow::error::ContextError>) -> Self { +impl From, winnow::error::ContextError>> for KclError { + fn from(err: ParseError, winnow::error::ContextError>) -> Self { let (input, offset): (Vec, usize) = (err.input().chars().collect(), err.offset()); + let module_id = err.input().state.module_id; if offset >= input.len() { // From the winnow docs: @@ -31,7 +31,7 @@ impl From, winnow::error::ContextError>> for KclError { // the end of input (input.len()) on eof errors. return KclError::Lexical(KclErrorDetails { - source_ranges: vec![SourceRange([offset, offset])], + source_ranges: vec![SourceRange([offset, offset, module_id.as_usize()])], message: "unexpected EOF while parsing".to_string(), }); } @@ -42,7 +42,7 @@ impl From, winnow::error::ContextError>> for KclError { // TODO: Add the Winnow parser context to the error. // See https://github.com/KittyCAD/modeling-app/issues/784 KclError::Lexical(KclErrorDetails { - source_ranges: vec![SourceRange([offset, offset + 1])], + source_ranges: vec![SourceRange([offset, offset + 1, module_id.as_usize()])], message: format!("found unknown token '{}'", bad_token), }) } diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__a.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__a.snap index f30645a1cf..27911dd0d0 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__a.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__a.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__b.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__b.snap index bffcf481b9..2bc325a6bc 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__b.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__b.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__c.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__c.snap index 9cc43dfca6..5b4b0b8976 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__c.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__c.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__d.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__d.snap index 283ec78bfc..f81155de65 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__d.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__d.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__e.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__e.snap index c83a2eebe9..7ff5d38e47 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__e.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__e.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__f.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__f.snap index 3baab161c3..471e22080f 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__f.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__f.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__g.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__g.snap index 46150903db..9dc60e7db7 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__g.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__g.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__h.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__h.snap index 76742272bc..ca065c4319 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__h.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__h.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__i.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__i.snap index 42083c553f..a71017b623 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__i.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__i.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__j.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__j.snap index 4f95eca1e1..a518eec2be 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__j.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__j.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__k.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__k.snap index 67132df9b3..9f68b8df84 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__k.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_math_tests__k.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "type": "BinaryExpression", diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__a.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__a.snap index dc9ab6cf09..ce731c13a8 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__a.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__a.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aa.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aa.snap index 834ec68887..275e95265a 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aa.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aa.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ab.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ab.snap index 3f4b17b81e..7718496469 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ab.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ab.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ac.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ac.snap index 8832057bef..6d0e9fb304 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ac.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ac.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ad.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ad.snap index e5ad22512d..56c13cdf90 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ad.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ad.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ae.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ae.snap index 0e3165d349..9b441c29ad 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ae.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ae.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__af.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__af.snap index 4273e1a871..cbf5b29148 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__af.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__af.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ag.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ag.snap index 7ed1857221..87318a5110 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ag.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ag.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ah.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ah.snap index 5fa8bf2db1..b7585eb89c 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ah.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ah.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ai.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ai.snap index e6ebdbdaff..544fcab0f4 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ai.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ai.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aj.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aj.snap index e8b7a8301f..3e4a3746e4 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aj.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aj.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ak.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ak.snap index 081fa3c9f3..da75a88893 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ak.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ak.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__al.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__al.snap index 4933fbb4cf..d4df1ac8cc 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__al.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__al.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__am.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__am.snap index f7682b4bc4..3312f11517 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__am.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__am.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__an.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__an.snap index 9732c22f51..16ff0a44f7 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__an.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__an.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ao.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ao.snap index d7192b4815..051096d0fb 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ao.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ao.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ap.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ap.snap index 5e99e3f379..5e33c7d5bf 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ap.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ap.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aq.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aq.snap index 7ed31b5658..f3b1a0fc24 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aq.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aq.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ar.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ar.snap index 0f91fbc8da..1d1598936a 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ar.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ar.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__at.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__at.snap index a8fb77a34f..2c705f15c9 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__at.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__at.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__au.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__au.snap index 3c08a0bf08..685bf9547e 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__au.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__au.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__av.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__av.snap index a631e0954e..333a63dde8 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__av.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__av.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aw.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aw.snap index db78d6395a..be04b7bac1 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aw.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__aw.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ax.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ax.snap index 5598578f6e..78be4f7f55 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ax.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ax.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ay.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ay.snap index 4599b186de..ef20b8c121 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ay.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ay.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__az.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__az.snap index fc06a858dd..fc12943fec 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__az.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__az.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__b.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__b.snap index e54a3e6136..ded0b975a8 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__b.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__b.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ba.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ba.snap index 0a9ac7936f..1badc4e75c 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ba.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__ba.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bb.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bb.snap index 780dc200d2..e23fc1dfb6 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bb.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bb.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bc.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bc.snap index f666a67f38..033c322f0a 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bc.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bc.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bd.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bd.snap index e27be60c5a..4029ff529c 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bd.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bd.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__be.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__be.snap index 2eaa1e886b..96d898516c 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__be.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__be.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bf.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bf.snap index 3ef1d35f1b..e0b5362593 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bf.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bf.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bg.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bg.snap index c7bd3c4ebd..cfecc70ab0 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bg.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bg.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bh.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bh.snap index 3f909f8cc4..02a350ca82 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bh.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__bh.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__c.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__c.snap index 5a53a6c8d7..5a4848bef8 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__c.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__c.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d.snap index a8de2c89c1..1a3aa3be1f 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d2.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d2.snap index afd750d991..72e8c8f0d4 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d2.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__d2.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__e.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__e.snap index b11920a867..b27b1739cf 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__e.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__e.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__f.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__f.snap index e9608c3f78..49d6f14ae2 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__f.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__f.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__g.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__g.snap index 2a72694874..88263fb773 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__g.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__g.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__h.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__h.snap index 9092210533..1323f91697 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__h.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__h.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__i.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__i.snap index e348e7d25a..f3dbba2234 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__i.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__i.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__j.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__j.snap index 70ec94c42b..e39bc8ad5b 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__j.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__j.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__k.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__k.snap index a5b9e8f5a2..b9396fc63b 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__k.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__k.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__l.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__l.snap index 6781f3e4de..c012455f8a 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__l.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__l.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__m.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__m.snap index df9258851f..26d38c83aa 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__m.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__m.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__n.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__n.snap index 2e194c1f85..905b668428 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__n.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__n.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__o.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__o.snap index 47c66b9cc3..9ebd3732a7 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__o.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__o.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__p.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__p.snap index 6802bdc534..6cc8f4ac3b 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__p.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__p.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__q.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__q.snap index 1951dcf4d8..b4101a984c 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__q.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__q.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__r.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__r.snap index c6ab62448a..6e7d889297 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__r.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__r.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__s.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__s.snap index 935901238f..57cbc351fd 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__s.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__s.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__t.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__t.snap index a45fae8a82..261d1d28bc 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__t.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__t.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__u.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__u.snap index 7ace17471c..36b7438b36 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__u.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__u.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__v.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__v.snap index d654143f32..68fee419a5 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__v.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__v.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__w.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__w.snap index 5e3e22268d..f7458327c8 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__w.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__w.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__x.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__x.snap index 12f97d6889..754dd975f1 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__x.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__x.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__y.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__y.snap index d9e1e0f3ac..eae62204ba 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__y.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__y.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__z.snap b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__z.snap index f13e29ef28..42b432e6aa 100644 --- a/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__z.snap +++ b/src/wasm-lib/kcl/src/parser/snapshots/kcl_lib__parser__parser_impl__snapshot_tests__z.snap @@ -1,6 +1,7 @@ --- source: kcl/src/parser/parser_impl.rs expression: actual +snapshot_kind: text --- { "body": [ diff --git a/src/wasm-lib/kcl/src/simulation_tests.rs b/src/wasm-lib/kcl/src/simulation_tests.rs index cd7b75aa92..4a7c67e006 100644 --- a/src/wasm-lib/kcl/src/simulation_tests.rs +++ b/src/wasm-lib/kcl/src/simulation_tests.rs @@ -1,5 +1,5 @@ use crate::{ - ast::types::{Node, Program}, + ast::types::{ModuleId, Node, Program}, errors::KclError, parser::Parser, token::Token, @@ -44,7 +44,7 @@ fn read(filename: &'static str, test_name: &str) -> String { fn tokenize(test_name: &str) { let input = read("input.kcl", test_name); - let token_res = crate::token::lexer(&input); + let token_res = crate::token::lexer(&input, ModuleId::default()); assert_snapshot(test_name, "Result of tokenizing", || { insta::assert_json_snapshot!("tokens", token_res); diff --git a/src/wasm-lib/kcl/src/std/args.rs b/src/wasm-lib/kcl/src/std/args.rs index b3ef1c442f..34c233c5ab 100644 --- a/src/wasm-lib/kcl/src/std/args.rs +++ b/src/wasm-lib/kcl/src/std/args.rs @@ -181,47 +181,39 @@ impl Args { Ok(()) } - fn make_user_val_from_json(&self, j: serde_json::Value) -> Result { - Ok(KclValue::UserVal(crate::executor::UserVal { + fn make_user_val_from_json(&self, j: serde_json::Value) -> KclValue { + KclValue::UserVal(crate::executor::UserVal { value: j, meta: vec![Metadata { source_range: self.source_range, }], - })) + }) } - pub(crate) fn make_null_user_val(&self) -> Result { + pub(crate) fn make_null_user_val(&self) -> KclValue { self.make_user_val_from_json(serde_json::Value::Null) } - pub(crate) fn make_user_val_from_i64(&self, n: i64) -> Result { + pub(crate) fn make_user_val_from_i64(&self, n: i64) -> KclValue { self.make_user_val_from_json(serde_json::Value::Number(serde_json::Number::from(n))) } pub(crate) fn make_user_val_from_f64(&self, f: f64) -> Result { - self.make_user_val_from_json(serde_json::Value::Number(serde_json::Number::from_f64(f).ok_or_else( - || { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert `{}` to a number", f), - source_ranges: vec![self.source_range], - }) - }, - )?)) + f64_to_jnum(f, vec![self.source_range]).map(|x| self.make_user_val_from_json(x)) + } + + pub(crate) fn make_user_val_from_point(&self, p: [f64; 2]) -> Result { + let x = f64_to_jnum(p[0], vec![self.source_range])?; + let y = f64_to_jnum(p[1], vec![self.source_range])?; + let array = serde_json::Value::Array(vec![x, y]); + Ok(self.make_user_val_from_json(array)) } pub(crate) fn make_user_val_from_f64_array(&self, f: Vec) -> Result { - let mut arr = Vec::new(); - for n in f { - arr.push(serde_json::Value::Number(serde_json::Number::from_f64(n).ok_or_else( - || { - KclError::Type(KclErrorDetails { - message: format!("Failed to convert `{}` to a number", n), - source_ranges: vec![self.source_range], - }) - }, - )?)); - } - self.make_user_val_from_json(serde_json::Value::Array(arr)) + f.into_iter() + .map(|n| f64_to_jnum(n, vec![self.source_range])) + .collect::, _>>() + .map(|arr| self.make_user_val_from_json(serde_json::Value::Array(arr))) } pub(crate) fn get_number(&self) -> Result { @@ -750,3 +742,14 @@ impl<'a> FromKclValue<'a> for SketchSurface { } } } + +fn f64_to_jnum(f: f64, source_ranges: Vec) -> Result { + serde_json::Number::from_f64(f) + .ok_or_else(|| { + KclError::Type(KclErrorDetails { + message: format!("Failed to convert `{f}` to a number"), + source_ranges, + }) + }) + .map(serde_json::Value::Number) +} diff --git a/src/wasm-lib/kcl/src/std/assert.rs b/src/wasm-lib/kcl/src/std/assert.rs index f4cb15c359..5a71efe73c 100644 --- a/src/wasm-lib/kcl/src/std/assert.rs +++ b/src/wasm-lib/kcl/src/std/assert.rs @@ -24,7 +24,7 @@ async fn _assert(value: bool, message: &str, args: &Args) -> Result<(), KclError pub async fn assert(_exec_state: &mut ExecState, args: Args) -> Result { let (data, description): (bool, String) = args.get_data()?; inner_assert(data, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check a value at runtime, and raise an error if the argument provided @@ -44,7 +44,7 @@ async fn inner_assert(data: bool, message: &str, args: &Args) -> Result<(), KclE pub async fn assert_lt(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_lt(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is less than to another at runtime, @@ -63,7 +63,7 @@ async fn inner_assert_lt(left: f64, right: f64, message: &str, args: &Args) -> R pub async fn assert_gt(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_gt(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value equals another at runtime, @@ -96,7 +96,7 @@ async fn inner_assert_equal(left: f64, right: f64, epsilon: f64, message: &str, pub async fn assert_equal(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, epsilon, description): (f64, f64, f64, String) = args.get_data()?; inner_assert_equal(left, right, epsilon, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is greater than another at runtime, @@ -115,7 +115,7 @@ async fn inner_assert_gt(left: f64, right: f64, message: &str, args: &Args) -> R pub async fn assert_lte(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_lte(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is less than or equal to another at runtime, @@ -135,7 +135,7 @@ async fn inner_assert_lte(left: f64, right: f64, message: &str, args: &Args) -> pub async fn assert_gte(_exec_state: &mut ExecState, args: Args) -> Result { let (left, right, description): (f64, f64, String) = args.get_data()?; inner_assert_gte(left, right, &description, &args).await?; - args.make_null_user_val() + Ok(args.make_null_user_val()) } /// Check that a numerical value is greater than or equal to another at runtime, diff --git a/src/wasm-lib/kcl/src/std/convert.rs b/src/wasm-lib/kcl/src/std/convert.rs index 69e8729ff2..777eb0fe50 100644 --- a/src/wasm-lib/kcl/src/std/convert.rs +++ b/src/wasm-lib/kcl/src/std/convert.rs @@ -34,7 +34,7 @@ pub async fn int(_exec_state: &mut ExecState, args: Args) -> Result { /// Return the program and its single function. /// Return None if those expectations aren't met. pub fn extract_function(source: &str) -> Option<(Node, crate::ast::types::BoxNode)> { - let tokens = lexer(source).unwrap(); - let src = crate::parser::Parser::new(tokens).ast().ok()?; + let src = crate::parser::top_level_parse(source).ok()?; assert_eq!(src.body.len(), 1); let BodyItem::ExpressionStatement(expr) = src.body.last()? else { panic!("expected expression statement"); diff --git a/src/wasm-lib/kcl/src/std/math.rs b/src/wasm-lib/kcl/src/std/math.rs index 36989ac46b..a81c1ce835 100644 --- a/src/wasm-lib/kcl/src/std/math.rs +++ b/src/wasm-lib/kcl/src/std/math.rs @@ -16,7 +16,7 @@ pub async fn rem(_exec_state: &mut ExecState, args: Args) -> Result Result { + let tag: TagIdentifier = args.get_data()?; + let result = inner_segment_end(&tag, exec_state, args.clone())?; + + args.make_user_val_from_point(result) +} + +/// Compute the ending point of the provided line segment. +/// +/// ```no_run +/// w = 15 +/// cube = startSketchAt([0, 0]) +/// |> line([w, 0], %, $line1) +/// |> line([0, w], %, $line2) +/// |> line([-w, 0], %, $line3) +/// |> line([0, -w], %, $line4) +/// |> close(%) +/// |> extrude(5, %) +/// +/// fn cylinder = (radius, tag) => { +/// return startSketchAt([0, 0]) +/// |> circle({ radius: radius, center: segEnd(tag) }, %) +/// |> extrude(radius, %) +/// } +/// +/// cylinder(1, line1) +/// cylinder(2, line2) +/// cylinder(3, line3) +/// cylinder(4, line4) +/// ``` +#[stdlib { + name = "segEnd", +}] +fn inner_segment_end(tag: &TagIdentifier, exec_state: &mut ExecState, args: Args) -> Result<[f64; 2], KclError> { + let line = args.get_tag_engine_info(exec_state, tag)?; + let path = line.path.clone().ok_or_else(|| { + KclError::Type(KclErrorDetails { + message: format!("Expected a line segment with a path, found `{:?}`", line), + source_ranges: vec![args.source_range], + }) + })?; + + Ok(path.get_base().to) +} + /// Returns the segment end of x. pub async fn segment_end_x(exec_state: &mut ExecState, args: Args) -> Result { let tag: TagIdentifier = args.get_data()?; @@ -82,6 +128,124 @@ fn inner_segment_end_y(tag: &TagIdentifier, exec_state: &mut ExecState, args: Ar Ok(path.get_to()[1]) } +/// Returns the point at the start of the given segment. +pub async fn segment_start(exec_state: &mut ExecState, args: Args) -> Result { + let tag: TagIdentifier = args.get_data()?; + let result = inner_segment_start(&tag, exec_state, args.clone())?; + + args.make_user_val_from_point(result) +} + +/// Compute the starting point of the provided line segment. +/// +/// ```no_run +/// w = 15 +/// cube = startSketchAt([0, 0]) +/// |> line([w, 0], %, $line1) +/// |> line([0, w], %, $line2) +/// |> line([-w, 0], %, $line3) +/// |> line([0, -w], %, $line4) +/// |> close(%) +/// |> extrude(5, %) +/// +/// fn cylinder = (radius, tag) => { +/// return startSketchAt([0, 0]) +/// |> circle({ radius: radius, center: segStart(tag) }, %) +/// |> extrude(radius, %) +/// } +/// +/// cylinder(1, line1) +/// cylinder(2, line2) +/// cylinder(3, line3) +/// cylinder(4, line4) +/// ``` +#[stdlib { + name = "segStart", +}] +fn inner_segment_start(tag: &TagIdentifier, exec_state: &mut ExecState, args: Args) -> Result<[f64; 2], KclError> { + let line = args.get_tag_engine_info(exec_state, tag)?; + let path = line.path.clone().ok_or_else(|| { + KclError::Type(KclErrorDetails { + message: format!("Expected a line segment with a path, found `{:?}`", line), + source_ranges: vec![args.source_range], + }) + })?; + + Ok(path.get_from().to_owned()) +} + +/// Returns the segment start of x. +pub async fn segment_start_x(exec_state: &mut ExecState, args: Args) -> Result { + let tag: TagIdentifier = args.get_data()?; + let result = inner_segment_start_x(&tag, exec_state, args.clone())?; + + args.make_user_val_from_f64(result) +} + +/// Compute the starting point of the provided line segment along the 'x' axis. +/// +/// ```no_run +/// const exampleSketch = startSketchOn('XZ') +/// |> startProfileAt([0, 0], %) +/// |> line([20, 0], %, $thing) +/// |> line([0, 5], %) +/// |> line([20 - segStartX(thing), 0], %) +/// |> line([-20, 10], %) +/// |> close(%) +/// +/// const example = extrude(5, exampleSketch) +/// ``` +#[stdlib { + name = "segStartX", +}] +fn inner_segment_start_x(tag: &TagIdentifier, exec_state: &mut ExecState, args: Args) -> Result { + let line = args.get_tag_engine_info(exec_state, tag)?; + let path = line.path.clone().ok_or_else(|| { + KclError::Type(KclErrorDetails { + message: format!("Expected a line segment with a path, found `{:?}`", line), + source_ranges: vec![args.source_range], + }) + })?; + + Ok(path.get_from()[0]) +} + +/// Returns the segment start of y. +pub async fn segment_start_y(exec_state: &mut ExecState, args: Args) -> Result { + let tag: TagIdentifier = args.get_data()?; + let result = inner_segment_start_y(&tag, exec_state, args.clone())?; + + args.make_user_val_from_f64(result) +} + +/// Compute the starting point of the provided line segment along the 'y' axis. +/// +/// ```no_run +/// const exampleSketch = startSketchOn('XZ') +/// |> startProfileAt([0, 0], %) +/// |> line([20, 0], %) +/// |> line([0, 3], %, $thing) +/// |> line([-10, 0], %) +/// |> line([0, 20-segStartY(thing)], %) +/// |> line([-10, 0], %) +/// |> close(%) +/// +/// const example = extrude(5, exampleSketch) +/// ``` +#[stdlib { + name = "segStartY", +}] +fn inner_segment_start_y(tag: &TagIdentifier, exec_state: &mut ExecState, args: Args) -> Result { + let line = args.get_tag_engine_info(exec_state, tag)?; + let path = line.path.clone().ok_or_else(|| { + KclError::Type(KclErrorDetails { + message: format!("Expected a line segment with a path, found `{:?}`", line), + source_ranges: vec![args.source_range], + }) + })?; + + Ok(path.get_from()[1]) +} /// Returns the last segment of x. pub async fn last_segment_x(_exec_state: &mut ExecState, args: Args) -> Result { let sketch = args.get_sketch()?; diff --git a/src/wasm-lib/kcl/src/test_server.rs b/src/wasm-lib/kcl/src/test_server.rs index d54bc511e8..c75780a36d 100644 --- a/src/wasm-lib/kcl/src/test_server.rs +++ b/src/wasm-lib/kcl/src/test_server.rs @@ -17,9 +17,7 @@ pub struct RequestBody { /// This returns the bytes of the snapshot. pub async fn execute_and_snapshot(code: &str, units: UnitLength) -> anyhow::Result { let ctx = new_context(units, true).await?; - let tokens = crate::token::lexer(code)?; - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast()?; + let program = crate::parser::top_level_parse(code)?; do_execute_and_snapshot(&ctx, program).await.map(|(_state, snap)| snap) } @@ -37,9 +35,7 @@ pub async fn execute_and_snapshot_ast( pub async fn execute_and_snapshot_no_auth(code: &str, units: UnitLength) -> anyhow::Result { let ctx = new_context(units, false).await?; - let tokens = crate::token::lexer(code)?; - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast()?; + let program = crate::parser::top_level_parse(code)?; do_execute_and_snapshot(&ctx, program).await.map(|(_state, snap)| snap) } diff --git a/src/wasm-lib/kcl/src/token.rs b/src/wasm-lib/kcl/src/token.rs index 7533ffda34..a4b7131b79 100644 --- a/src/wasm-lib/kcl/src/token.rs +++ b/src/wasm-lib/kcl/src/token.rs @@ -8,13 +8,16 @@ use tower_lsp::lsp_types::SemanticTokenType; use winnow::stream::ContainsToken; use crate::{ - ast::types::{ItemVisibility, VariableKind}, + ast::types::{ItemVisibility, ModuleId, VariableKind}, errors::KclError, executor::SourceRange, }; mod tokeniser; +// Re-export +pub use tokeniser::Input; + /// The types of tokens. #[derive(Debug, PartialEq, Eq, Copy, Clone, Deserialize, Serialize, ts_rs::TS, JsonSchema, FromStr, Display)] #[cfg_attr(feature = "pyo3", pyo3::pyclass(eq, eq_int))] @@ -161,6 +164,8 @@ pub struct Token { pub start: usize, /// Offset in the source code where this token ends. pub end: usize, + #[serde(default, skip_serializing_if = "ModuleId::is_top_level")] + pub module_id: ModuleId, pub value: String, } @@ -177,10 +182,16 @@ impl ContainsToken for TokenType { } impl Token { - pub fn from_range(range: std::ops::Range, token_type: TokenType, value: String) -> Self { + pub fn from_range( + range: std::ops::Range, + module_id: ModuleId, + token_type: TokenType, + value: String, + ) -> Self { Self { start: range.start, end: range.end, + module_id, value, token_type, } @@ -193,7 +204,7 @@ impl Token { } pub fn as_source_range(&self) -> SourceRange { - SourceRange([self.start, self.end]) + SourceRange([self.start, self.end, self.module_id.as_usize()]) } pub fn as_source_ranges(&self) -> Vec { @@ -227,18 +238,18 @@ impl Token { impl From for SourceRange { fn from(token: Token) -> Self { - Self([token.start, token.end]) + Self([token.start, token.end, token.module_id.as_usize()]) } } impl From<&Token> for SourceRange { fn from(token: &Token) -> Self { - Self([token.start, token.end]) + Self([token.start, token.end, token.module_id.as_usize()]) } } -pub fn lexer(s: &str) -> Result, KclError> { - tokeniser::lexer(s).map_err(From::from) +pub fn lexer(s: &str, module_id: ModuleId) -> Result, KclError> { + tokeniser::lexer(s, module_id).map_err(From::from) } #[cfg(test)] diff --git a/src/wasm-lib/kcl/src/token/tokeniser.rs b/src/wasm-lib/kcl/src/token/tokeniser.rs index 589a8d478e..8837055953 100644 --- a/src/wasm-lib/kcl/src/token/tokeniser.rs +++ b/src/wasm-lib/kcl/src/token/tokeniser.rs @@ -5,16 +5,37 @@ use winnow::{ prelude::*, stream::{Location, Stream}, token::{any, none_of, one_of, take_till, take_until}, - Located, + Located, Stateful, }; -use crate::token::{Token, TokenType}; +use crate::{ + ast::types::ModuleId, + token::{Token, TokenType}, +}; -pub fn lexer(i: &str) -> Result, ParseError, ContextError>> { - repeat(0.., token).parse(Located::new(i)) +pub fn lexer(i: &str, module_id: ModuleId) -> Result, ParseError, ContextError>> { + let state = State::new(module_id); + let input = Input { + input: Located::new(i), + state, + }; + repeat(0.., token).parse(input) } -pub fn token(i: &mut Located<&str>) -> PResult { +pub type Input<'a> = Stateful, State>; + +#[derive(Debug, Clone)] +pub struct State { + pub module_id: ModuleId, +} + +impl State { + fn new(module_id: ModuleId) -> Self { + Self { module_id } + } +} + +pub fn token(i: &mut Input<'_>) -> PResult { match winnow::combinator::dispatch! {peek(any); '"' | '\'' => string, '/' => alt((line_comment, block_comment, operator)), @@ -42,6 +63,7 @@ pub fn token(i: &mut Located<&str>) -> PResult { Ok(Token::from_range( i.location()..i.location() + 1, + i.state.module_id, TokenType::Unknown, i.next_slice(1).to_string(), )) @@ -49,19 +71,29 @@ pub fn token(i: &mut Located<&str>) -> PResult { } } -fn block_comment(i: &mut Located<&str>) -> PResult { +fn block_comment(i: &mut Input<'_>) -> PResult { let inner = ("/*", take_until(0.., "*/"), "*/").take(); let (value, range) = inner.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::BlockComment, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::BlockComment, + value.to_string(), + )) } -fn line_comment(i: &mut Located<&str>) -> PResult { +fn line_comment(i: &mut Input<'_>) -> PResult { let inner = (r#"//"#, take_till(0.., ['\n', '\r'])).take(); let (value, range) = inner.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::LineComment, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::LineComment, + value.to_string(), + )) } -fn number(i: &mut Located<&str>) -> PResult { +fn number(i: &mut Input<'_>) -> PResult { let number_parser = alt(( // Digits before the decimal point. (digit1, opt(('.', digit1))).map(|_| ()), @@ -69,113 +101,193 @@ fn number(i: &mut Located<&str>) -> PResult { ('.', digit1).map(|_| ()), )); let (value, range) = number_parser.take().with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Number, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Number, + value.to_string(), + )) } -fn whitespace(i: &mut Located<&str>) -> PResult { +fn whitespace(i: &mut Input<'_>) -> PResult { let (value, range) = multispace1.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Whitespace, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Whitespace, + value.to_string(), + )) } -fn inner_word(i: &mut Located<&str>) -> PResult<()> { +fn inner_word(i: &mut Input<'_>) -> PResult<()> { one_of(('a'..='z', 'A'..='Z', '_')).parse_next(i)?; repeat::<_, _, (), _, _>(0.., one_of(('a'..='z', 'A'..='Z', '0'..='9', '_'))).parse_next(i)?; Ok(()) } -fn word(i: &mut Located<&str>) -> PResult { +fn word(i: &mut Input<'_>) -> PResult { let (value, range) = inner_word.take().with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Word, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Word, + value.to_string(), + )) } -fn operator(i: &mut Located<&str>) -> PResult { +fn operator(i: &mut Input<'_>) -> PResult { let (value, range) = alt(( ">=", "<=", "==", "=>", "!=", "|>", "*", "+", "-", "/", "%", "=", "<", ">", r"\", "|", "^", )) .with_span() .parse_next(i)?; - Ok(Token::from_range(range, TokenType::Operator, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Operator, + value.to_string(), + )) } -fn brace_start(i: &mut Located<&str>) -> PResult { +fn brace_start(i: &mut Input<'_>) -> PResult { let (value, range) = alt(('{', '(', '[')).with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Brace, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Brace, + value.to_string(), + )) } -fn brace_end(i: &mut Located<&str>) -> PResult { +fn brace_end(i: &mut Input<'_>) -> PResult { let (value, range) = alt(('}', ')', ']')).with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Brace, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Brace, + value.to_string(), + )) } -fn comma(i: &mut Located<&str>) -> PResult { +fn comma(i: &mut Input<'_>) -> PResult { let (value, range) = ','.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Comma, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Comma, + value.to_string(), + )) } -fn hash(i: &mut Located<&str>) -> PResult { +fn hash(i: &mut Input<'_>) -> PResult { let (value, range) = '#'.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Hash, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Hash, + value.to_string(), + )) } -fn bang(i: &mut Located<&str>) -> PResult { +fn bang(i: &mut Input<'_>) -> PResult { let (value, range) = '!'.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Bang, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Bang, + value.to_string(), + )) } -fn dollar(i: &mut Located<&str>) -> PResult { +fn dollar(i: &mut Input<'_>) -> PResult { let (value, range) = '$'.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Dollar, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Dollar, + value.to_string(), + )) } -fn question_mark(i: &mut Located<&str>) -> PResult { +fn question_mark(i: &mut Input<'_>) -> PResult { let (value, range) = '?'.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::QuestionMark, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::QuestionMark, + value.to_string(), + )) } -fn colon(i: &mut Located<&str>) -> PResult { +fn colon(i: &mut Input<'_>) -> PResult { let (value, range) = ':'.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Colon, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Colon, + value.to_string(), + )) } -fn period(i: &mut Located<&str>) -> PResult { +fn period(i: &mut Input<'_>) -> PResult { let (value, range) = '.'.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Period, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Period, + value.to_string(), + )) } -fn double_period(i: &mut Located<&str>) -> PResult { +fn double_period(i: &mut Input<'_>) -> PResult { let (value, range) = "..".with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::DoublePeriod, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::DoublePeriod, + value.to_string(), + )) } /// Zero or more of either: /// 1. Any character except " or \ /// 2. Any character preceded by \ -fn inner_double_quote(i: &mut Located<&str>) -> PResult<()> { +fn inner_double_quote(i: &mut Input<'_>) -> PResult<()> { repeat(0.., alt((none_of(('"', '\\')), preceded('\\', winnow::token::any)))).parse_next(i) } /// Zero or more of either: /// 1. Any character except ' or \ /// 2. Any character preceded by \ -fn inner_single_quote(i: &mut Located<&str>) -> PResult<()> { +fn inner_single_quote(i: &mut Input<'_>) -> PResult<()> { repeat(0.., alt((none_of(('\'', '\\')), preceded('\\', winnow::token::any)))).parse_next(i) } -fn string(i: &mut Located<&str>) -> PResult { +fn string(i: &mut Input<'_>) -> PResult { let single_quoted_string = ('\'', inner_single_quote.take(), '\''); let double_quoted_string = ('"', inner_double_quote.take(), '"'); let either_quoted_string = alt((single_quoted_string.take(), double_quoted_string.take())); let (value, range): (&str, _) = either_quoted_string.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::String, value.to_string())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::String, + value.to_string(), + )) } -fn import_keyword(i: &mut Located<&str>) -> PResult { +fn import_keyword(i: &mut Input<'_>) -> PResult { let (value, range) = "import".with_span().parse_next(i)?; let token_type = peek(alt((' '.map(|_| TokenType::Keyword), '('.map(|_| TokenType::Word)))).parse_next(i)?; - Ok(Token::from_range(range, token_type, value.to_owned())) + Ok(Token::from_range( + range, + i.state.module_id, + token_type, + value.to_owned(), + )) } -fn unambiguous_keywords(i: &mut Located<&str>) -> PResult { +fn unambiguous_keywords(i: &mut Input<'_>) -> PResult { // These are the keywords themselves. let keyword_candidates = alt(( "if", "else", "for", "while", "return", "break", "continue", "fn", "let", "mut", "loop", "true", "false", @@ -188,14 +300,19 @@ fn unambiguous_keywords(i: &mut Located<&str>) -> PResult { peek(none_of(('a'..='z', 'A'..='Z', '-', '_', '0'..='9'))), ); let (value, range) = keyword.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Keyword, value.to_owned())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Keyword, + value.to_owned(), + )) } -fn keyword(i: &mut Located<&str>) -> PResult { +fn keyword(i: &mut Input<'_>) -> PResult { alt((import_keyword, unambiguous_keywords)).parse_next(i) } -fn type_(i: &mut Located<&str>) -> PResult { +fn type_(i: &mut Input<'_>) -> PResult { // These are the types themselves. let type_candidates = alt(("string", "number", "bool", "sketch", "sketch_surface", "solid")); // Look ahead. If any of these characters follow the type, then it's not a type, it's just @@ -205,7 +322,12 @@ fn type_(i: &mut Located<&str>) -> PResult { peek(none_of(('a'..='z', 'A'..='Z', '-', '_', '0'..='9'))), ); let (value, range) = type_.with_span().parse_next(i)?; - Ok(Token::from_range(range, TokenType::Type, value.to_owned())) + Ok(Token::from_range( + range, + i.state.module_id, + TokenType::Type, + value.to_owned(), + )) } #[cfg(test)] @@ -216,21 +338,28 @@ mod tests { fn assert_parse_err<'i, P, O, E>(mut p: P, s: &'i str) where O: std::fmt::Debug, - P: Parser, O, E>, + P: Parser, O, E>, { - assert!( - p.parse_next(&mut Located::new(s)).is_err(), - "parsed {s} but should have failed" - ); + let state = State::new(ModuleId::default()); + let mut input = Input { + input: Located::new(s), + state, + }; + assert!(p.parse_next(&mut input).is_err(), "parsed {s} but should have failed"); } fn assert_parse_ok<'i, P, O, E>(mut p: P, s: &'i str) where E: std::fmt::Debug, O: std::fmt::Debug, - P: Parser, O, E>, + P: Parser, O, E>, { - let res = p.parse_next(&mut Located::new(s)); + let state = State::new(ModuleId::default()); + let mut input = Input { + input: Located::new(s), + state, + }; + let res = p.parse_next(&mut input); assert!(res.is_ok(), "failed to parse {s}, got {}", res.unwrap_err()); } @@ -246,10 +375,13 @@ mod tests { assert_parse_err(number, invalid); } - assert_eq!( - number.parse(Located::new("0.0000000000")).unwrap().value, - "0.0000000000" - ); + let module_id = ModuleId::from_usize(1); + let input = Input { + input: Located::new("0.0000000000"), + state: State::new(module_id), + }; + + assert_eq!(number.parse(input).unwrap().value, "0.0000000000"); } #[test] @@ -318,37 +450,43 @@ mod tests { #[test] fn test_program0() { let program = "const a=5"; - let actual = lexer(program).unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer(program, module_id).unwrap(); let expected = vec![ Token { token_type: TokenType::Keyword, value: "const".to_string(), start: 0, end: 5, + module_id, }, Token { token_type: TokenType::Whitespace, value: " ".to_string(), start: 5, end: 6, + module_id, }, Token { token_type: TokenType::Word, value: "a".to_string(), start: 6, end: 7, + module_id, }, Token { token_type: TokenType::Operator, value: "=".to_string(), start: 7, end: 8, + module_id, }, Token { token_type: TokenType::Number, value: "5".to_string(), start: 8, end: 9, + module_id, }, ]; assert_tokens(expected, actual); @@ -357,61 +495,71 @@ mod tests { #[test] fn test_program1() { let program = "54 + 22500 + 6"; - let actual = lexer(program).unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer(program, module_id).unwrap(); let expected = vec![ Token { token_type: TokenType::Number, value: "54".to_string(), start: 0, end: 2, + module_id, }, Token { token_type: TokenType::Whitespace, value: " ".to_string(), start: 2, end: 3, + module_id, }, Token { token_type: TokenType::Operator, value: "+".to_string(), start: 3, end: 4, + module_id, }, Token { token_type: TokenType::Whitespace, value: " ".to_string(), start: 4, end: 5, + module_id, }, Token { token_type: TokenType::Number, value: "22500".to_string(), start: 5, end: 10, + module_id, }, Token { token_type: TokenType::Whitespace, value: " ".to_string(), start: 10, end: 11, + module_id, }, Token { token_type: TokenType::Operator, value: "+".to_string(), start: 11, end: 12, + module_id, }, Token { token_type: TokenType::Whitespace, value: " ".to_string(), start: 12, end: 13, + module_id, }, Token { token_type: TokenType::Number, value: "6".to_string(), start: 13, end: 14, + module_id, }, ]; assert_tokens(expected, actual); @@ -432,6 +580,7 @@ fn ghi = (part001) => { } show(part001)"#; + let module_id = ModuleId::from_usize(1); use TokenType::*; @@ -440,676 +589,788 @@ show(part001)"#; token_type: Keyword, start: 0, end: 5, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 5, end: 6, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 6, end: 13, + module_id, value: "part001".to_owned(), }, Token { token_type: Whitespace, start: 13, end: 14, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 14, end: 15, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 15, end: 16, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 16, end: 29, + module_id, value: "startSketchAt".to_owned(), }, Token { token_type: Brace, start: 29, end: 30, + module_id, value: "(".to_owned(), }, Token { token_type: Brace, start: 30, end: 31, + module_id, value: "[".to_owned(), }, Token { token_type: Number, start: 31, end: 43, + module_id, value: "0.0000000000".to_owned(), }, Token { token_type: Comma, start: 43, end: 44, + module_id, value: ",".to_owned(), }, Token { token_type: Whitespace, start: 44, end: 45, + module_id, value: " ".to_owned(), }, Token { token_type: Number, start: 45, end: 57, + module_id, value: "5.0000000000".to_owned(), }, Token { token_type: Brace, start: 57, end: 58, + module_id, value: "]".to_owned(), }, Token { token_type: Brace, start: 58, end: 59, + module_id, value: ")".to_owned(), }, Token { token_type: Whitespace, start: 59, end: 64, + module_id, value: "\n ".to_owned(), }, Token { token_type: Operator, start: 64, end: 66, + module_id, value: "|>".to_owned(), }, Token { token_type: Whitespace, start: 66, end: 67, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 67, end: 71, + module_id, value: "line".to_owned(), }, Token { token_type: Brace, start: 71, end: 72, + module_id, value: "(".to_owned(), }, Token { token_type: Brace, start: 72, end: 73, + module_id, value: "[".to_owned(), }, Token { token_type: Number, start: 73, end: 85, + module_id, value: "0.4900857016".to_owned(), }, Token { token_type: Comma, start: 85, end: 86, + module_id, value: ",".to_owned(), }, Token { token_type: Whitespace, start: 86, end: 87, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 87, end: 88, + module_id, value: "-".to_owned(), }, Token { token_type: Number, start: 88, end: 100, + module_id, value: "0.0240763666".to_owned(), }, Token { token_type: Brace, start: 100, end: 101, + module_id, value: "]".to_owned(), }, Token { token_type: Comma, start: 101, end: 102, + module_id, value: ",".to_owned(), }, Token { token_type: Whitespace, start: 102, end: 103, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 103, end: 104, + module_id, value: "%".to_owned(), }, Token { token_type: Brace, start: 104, end: 105, + module_id, value: ")".to_owned(), }, Token { token_type: Whitespace, start: 105, end: 107, + module_id, value: "\n\n".to_owned(), }, Token { token_type: Keyword, start: 107, end: 112, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 112, end: 113, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 113, end: 120, + module_id, value: "part002".to_owned(), }, Token { token_type: Whitespace, start: 120, end: 121, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 121, end: 122, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 122, end: 123, + module_id, value: " ".to_owned(), }, Token { token_type: String, start: 123, end: 132, + module_id, value: "\"part002\"".to_owned(), }, Token { token_type: Whitespace, start: 132, end: 133, + module_id, value: "\n".to_owned(), }, Token { token_type: Keyword, start: 133, end: 138, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 138, end: 139, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 139, end: 145, + module_id, value: "things".to_owned(), }, Token { token_type: Whitespace, start: 145, end: 146, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 146, end: 147, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 147, end: 148, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 148, end: 149, + module_id, value: "[".to_owned(), }, Token { token_type: Word, start: 149, end: 156, + module_id, value: "part001".to_owned(), }, Token { token_type: Comma, start: 156, end: 157, + module_id, value: ",".to_owned(), }, Token { token_type: Whitespace, start: 157, end: 158, + module_id, value: " ".to_owned(), }, Token { token_type: Number, start: 158, end: 161, + module_id, value: "0.0".to_owned(), }, Token { token_type: Brace, start: 161, end: 162, + module_id, value: "]".to_owned(), }, Token { token_type: Whitespace, start: 162, end: 163, + module_id, value: "\n".to_owned(), }, Token { token_type: Keyword, start: 163, end: 166, + module_id, value: "let".to_owned(), }, Token { token_type: Whitespace, start: 166, end: 167, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 167, end: 171, + module_id, value: "blah".to_owned(), }, Token { token_type: Whitespace, start: 171, end: 172, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 172, end: 173, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 173, end: 174, + module_id, value: " ".to_owned(), }, Token { token_type: Number, start: 174, end: 175, + module_id, value: "1".to_owned(), }, Token { token_type: Whitespace, start: 175, end: 176, + module_id, value: "\n".to_owned(), }, Token { token_type: Keyword, start: 176, end: 181, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 181, end: 182, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 182, end: 185, + module_id, value: "foo".to_owned(), }, Token { token_type: Whitespace, start: 185, end: 186, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 186, end: 187, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 187, end: 188, + module_id, value: " ".to_owned(), }, Token { token_type: Keyword, start: 188, end: 193, + module_id, value: "false".to_owned(), }, Token { token_type: Whitespace, start: 193, end: 194, + module_id, value: "\n".to_owned(), }, Token { token_type: Keyword, start: 194, end: 197, + module_id, value: "let".to_owned(), }, Token { token_type: Whitespace, start: 197, end: 198, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 198, end: 201, + module_id, value: "baz".to_owned(), }, Token { token_type: Whitespace, start: 201, end: 202, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 202, end: 203, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 203, end: 204, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 204, end: 205, + module_id, value: "{".to_owned(), }, Token { token_type: Word, start: 205, end: 206, + module_id, value: "a".to_owned(), }, Token { token_type: Colon, start: 206, end: 207, + module_id, value: ":".to_owned(), }, Token { token_type: Whitespace, start: 207, end: 208, + module_id, value: " ".to_owned(), }, Token { token_type: Number, start: 208, end: 209, + module_id, value: "1".to_owned(), }, Token { token_type: Comma, start: 209, end: 210, + module_id, value: ",".to_owned(), }, Token { token_type: Whitespace, start: 210, end: 211, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 211, end: 218, + module_id, value: "part001".to_owned(), }, Token { token_type: Colon, start: 218, end: 219, + module_id, value: ":".to_owned(), }, Token { token_type: Whitespace, start: 219, end: 220, + module_id, value: " ".to_owned(), }, Token { token_type: String, start: 220, end: 227, + module_id, value: "\"thing\"".to_owned(), }, Token { token_type: Brace, start: 227, end: 228, + module_id, value: "}".to_owned(), }, Token { token_type: Whitespace, start: 228, end: 230, + module_id, value: "\n\n".to_owned(), }, Token { token_type: Keyword, start: 230, end: 232, + module_id, value: "fn".to_owned(), }, Token { token_type: Whitespace, start: 232, end: 233, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 233, end: 236, + module_id, value: "ghi".to_owned(), }, Token { token_type: Whitespace, start: 236, end: 237, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 237, end: 238, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 238, end: 239, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 239, end: 240, + module_id, value: "(".to_owned(), }, Token { token_type: Word, start: 240, end: 247, + module_id, value: "part001".to_owned(), }, Token { token_type: Brace, start: 247, end: 248, + module_id, value: ")".to_owned(), }, Token { token_type: Whitespace, start: 248, end: 249, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 249, end: 251, + module_id, value: "=>".to_owned(), }, Token { token_type: Whitespace, start: 251, end: 252, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 252, end: 253, + module_id, value: "{".to_owned(), }, Token { token_type: Whitespace, start: 253, end: 256, + module_id, value: "\n ".to_owned(), }, Token { token_type: Keyword, start: 256, end: 262, + module_id, value: "return".to_owned(), }, Token { token_type: Whitespace, start: 262, end: 263, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 263, end: 270, + module_id, value: "part001".to_owned(), }, Token { token_type: Whitespace, start: 270, end: 271, + module_id, value: "\n".to_owned(), }, Token { token_type: Brace, start: 271, end: 272, + module_id, value: "}".to_owned(), }, Token { token_type: Whitespace, start: 272, end: 274, + module_id, value: "\n\n".to_owned(), }, Token { token_type: Word, start: 274, end: 278, + module_id, value: "show".to_owned(), }, Token { token_type: Brace, start: 278, end: 279, + module_id, value: "(".to_owned(), }, Token { token_type: Word, start: 279, end: 286, + module_id, value: "part001".to_owned(), }, Token { token_type: Brace, start: 286, end: 287, + module_id, value: ")".to_owned(), }, ]; - let actual = lexer(program).unwrap(); + let actual = lexer(program, module_id).unwrap(); assert_tokens(expected, actual); } @@ -1123,301 +1384,351 @@ const key = 'c' const things = "things" // this is also a comment"#; - let actual = lexer(program).unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer(program, module_id).unwrap(); use TokenType::*; let expected = vec![ Token { token_type: Whitespace, start: 0, end: 1, + module_id, value: "\n".to_owned(), }, Token { token_type: LineComment, start: 1, end: 21, + module_id, value: "// this is a comment".to_owned(), }, Token { token_type: Whitespace, start: 21, end: 22, + module_id, value: "\n".to_owned(), }, Token { token_type: Keyword, start: 22, end: 27, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 27, end: 28, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 28, end: 30, + module_id, value: "yo".to_owned(), }, Token { token_type: Whitespace, start: 30, end: 31, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 31, end: 32, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 32, end: 33, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 33, end: 34, + module_id, value: "{".to_owned(), }, Token { token_type: Whitespace, start: 34, end: 35, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 35, end: 36, + module_id, value: "a".to_owned(), }, Token { token_type: Colon, start: 36, end: 37, + module_id, value: ":".to_owned(), }, Token { token_type: Whitespace, start: 37, end: 38, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 38, end: 39, + module_id, value: "{".to_owned(), }, Token { token_type: Whitespace, start: 39, end: 40, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 40, end: 41, + module_id, value: "b".to_owned(), }, Token { token_type: Colon, start: 41, end: 42, + module_id, value: ":".to_owned(), }, Token { token_type: Whitespace, start: 42, end: 43, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 43, end: 44, + module_id, value: "{".to_owned(), }, Token { token_type: Whitespace, start: 44, end: 45, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 45, end: 46, + module_id, value: "c".to_owned(), }, Token { token_type: Colon, start: 46, end: 47, + module_id, value: ":".to_owned(), }, Token { token_type: Whitespace, start: 47, end: 48, + module_id, value: " ".to_owned(), }, Token { token_type: String, start: 48, end: 53, + module_id, value: "'123'".to_owned(), }, Token { token_type: Whitespace, start: 53, end: 54, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 54, end: 55, + module_id, value: "}".to_owned(), }, Token { token_type: Whitespace, start: 55, end: 56, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 56, end: 57, + module_id, value: "}".to_owned(), }, Token { token_type: Whitespace, start: 57, end: 58, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 58, end: 59, + module_id, value: "}".to_owned(), }, Token { token_type: Whitespace, start: 59, end: 61, + module_id, value: "\n\n".to_owned(), }, Token { token_type: Keyword, start: 61, end: 66, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 66, end: 67, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 67, end: 70, + module_id, value: "key".to_owned(), }, Token { token_type: Whitespace, start: 70, end: 71, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 71, end: 72, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 72, end: 73, + module_id, value: " ".to_owned(), }, Token { token_type: String, start: 73, end: 76, + module_id, value: "'c'".to_owned(), }, Token { token_type: Whitespace, start: 76, end: 77, + module_id, value: "\n".to_owned(), }, Token { token_type: Keyword, start: 77, end: 82, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 82, end: 83, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 83, end: 89, + module_id, value: "things".to_owned(), }, Token { token_type: Whitespace, start: 89, end: 90, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 90, end: 91, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 91, end: 92, + module_id, value: " ".to_owned(), }, Token { token_type: String, start: 92, end: 100, + module_id, value: "\"things\"".to_owned(), }, Token { token_type: Whitespace, start: 100, end: 102, + module_id, value: "\n\n".to_owned(), }, Token { token_type: LineComment, start: 102, end: 127, + module_id, value: "// this is also a comment".to_owned(), }, ]; @@ -1427,106 +1738,121 @@ const things = "things" #[test] fn test_program4() { let program = "const myArray = [0..10]"; + let module_id = ModuleId::from_usize(1); use TokenType::*; let expected = vec![ Token { token_type: Keyword, start: 0, end: 5, + module_id, value: "const".to_owned(), }, Token { token_type: Whitespace, start: 5, end: 6, + module_id, value: " ".to_owned(), }, Token { token_type: Word, start: 6, end: 13, + module_id, value: "myArray".to_owned(), }, Token { token_type: Whitespace, start: 13, end: 14, + module_id, value: " ".to_owned(), }, Token { token_type: Operator, start: 14, end: 15, + module_id, value: "=".to_owned(), }, Token { token_type: Whitespace, start: 15, end: 16, + module_id, value: " ".to_owned(), }, Token { token_type: Brace, start: 16, end: 17, + module_id, value: "[".to_owned(), }, Token { token_type: Number, start: 17, end: 18, + module_id, value: "0".to_owned(), }, Token { token_type: DoublePeriod, start: 18, end: 20, + module_id, value: "..".to_owned(), }, Token { token_type: Number, start: 20, end: 22, + module_id, value: "10".to_owned(), }, Token { token_type: Brace, start: 22, end: 23, + module_id, value: "]".to_owned(), }, ]; - let actual = lexer(program).unwrap(); + let actual = lexer(program, module_id).unwrap(); assert_tokens(expected, actual); } #[test] fn test_kitt() { let program = include_str!("../../../tests/executor/inputs/kittycad_svg.kcl"); - let actual = lexer(program).unwrap(); + let actual = lexer(program, ModuleId::default()).unwrap(); assert_eq!(actual.len(), 5103); } #[test] fn test_pipes_on_pipes() { let program = include_str!("../../../tests/executor/inputs/pipes_on_pipes.kcl"); - let actual = lexer(program).unwrap(); + let actual = lexer(program, ModuleId::default()).unwrap(); assert_eq!(actual.len(), 17841); } #[test] fn test_lexer_negative_word() { - let actual = lexer("-legX").unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer("-legX", module_id).unwrap(); let expected = vec![ Token { token_type: TokenType::Operator, value: "-".to_string(), start: 0, end: 1, + module_id, }, Token { token_type: TokenType::Word, value: "legX".to_string(), start: 1, end: 5, + module_id, }, ]; assert_tokens(expected, actual); @@ -1534,49 +1860,57 @@ const things = "things" #[test] fn not_eq() { - let actual = lexer("!=").unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer("!=", module_id).unwrap(); let expected = vec![Token { token_type: TokenType::Operator, value: "!=".to_owned(), start: 0, end: 2, + module_id, }]; assert_eq!(actual, expected); } #[test] fn test_unrecognized_token() { - let actual = lexer("12 ; 8").unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer("12 ; 8", module_id).unwrap(); let expected = vec![ Token { token_type: TokenType::Number, value: "12".to_string(), start: 0, end: 2, + module_id, }, Token { token_type: TokenType::Whitespace, value: " ".to_string(), start: 2, end: 3, + module_id, }, Token { token_type: TokenType::Unknown, value: ";".to_string(), start: 3, end: 4, + module_id, }, Token { token_type: TokenType::Whitespace, value: " ".to_string(), start: 4, end: 5, + module_id, }, Token { token_type: TokenType::Number, value: "8".to_string(), start: 5, end: 6, + module_id, }, ]; @@ -1585,24 +1919,28 @@ const things = "things" #[test] fn import_keyword() { - let actual = lexer("import foo").unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer("import foo", module_id).unwrap(); let expected = Token { token_type: TokenType::Keyword, value: "import".to_owned(), start: 0, end: 6, + module_id, }; assert_eq!(actual[0], expected); } #[test] fn import_function() { - let actual = lexer("import(3)").unwrap(); + let module_id = ModuleId::from_usize(1); + let actual = lexer("import(3)", module_id).unwrap(); let expected = Token { token_type: TokenType::Word, value: "import".to_owned(), start: 0, end: 6, + module_id, }; assert_eq!(actual[0], expected); } diff --git a/src/wasm-lib/kcl/src/unparser.rs b/src/wasm-lib/kcl/src/unparser.rs index 4ada1b8c31..8396262474 100644 --- a/src/wasm-lib/kcl/src/unparser.rs +++ b/src/wasm-lib/kcl/src/unparser.rs @@ -573,7 +573,7 @@ impl FunctionExpression { mod tests { use pretty_assertions::assert_eq; - use crate::ast::types::FormatOptions; + use crate::ast::types::{FormatOptions, ModuleId}; #[test] fn test_recast_if_else_if_same() { @@ -585,9 +585,7 @@ mod tests { 5 } "#; - let tokens = crate::token::lexer(input).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(input).unwrap(); let output = program.recast(&Default::default(), 0); assert_eq!(output, input); } @@ -600,9 +598,7 @@ mod tests { 5 } "#; - let tokens = crate::token::lexer(input).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(input).unwrap(); let output = program.recast(&Default::default(), 0); assert_eq!(output, input); } @@ -616,9 +612,7 @@ import a as aaa, b from "a.kcl" import a, b as bbb from "a.kcl" import a as aaa, b as bbb from "a.kcl" "#; - let tokens = crate::token::lexer(input).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(input).unwrap(); let output = program.recast(&Default::default(), 0); assert_eq!(output, input); } @@ -627,7 +621,7 @@ import a as aaa, b as bbb from "a.kcl" fn test_recast_import_as_same_name() { let input = r#"import a as a from "a.kcl" "#; - let program = crate::parser::parse(input).unwrap(); + let program = crate::parser::top_level_parse(input).unwrap(); let output = program.recast(&Default::default(), 0); let expected = r#"import a from "a.kcl" "#; @@ -640,9 +634,7 @@ import a as aaa, b as bbb from "a.kcl" return 0 } "#; - let tokens = crate::token::lexer(input).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(input).unwrap(); let output = program.recast(&Default::default(), 0); assert_eq!(output, input); } @@ -765,9 +757,7 @@ fn zoo = (x0, y0) => { zoo(zoo_x, zoo_y) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted, some_program_string); @@ -836,9 +826,7 @@ outsideRevolve = startSketchOn('XZ') |> line([overHangLength - thickness, 0], %) |> close(%) |> revolve({ axis: 'y' }, %)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -914,9 +902,7 @@ outsideRevolve = startSketchOn('XZ') let some_program_string = r#"bing = { yo: 55 } myNestedVar = [{ prop: callExp(bing.yo) }] "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted, some_program_string); @@ -927,9 +913,7 @@ myNestedVar = [{ prop: callExp(bing.yo) }] let some_program_string = r#"bing = { yo: 55 } myNestedVar = [callExp(bing.yo)] "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted, some_program_string); @@ -941,9 +925,7 @@ myNestedVar = [callExp(bing.yo)] ten = 10 bar = [0 + 1 .. ten] "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted, some_program_string); @@ -957,9 +939,7 @@ bar = [0 + 1 .. ten] thing ( 1 ) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -982,9 +962,7 @@ myNestedVar = [ } ] "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1000,9 +978,7 @@ myNestedVar = [ #[test] fn test_recast_empty_file() { let some_program_string = r#""#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); // Its VERY important this comes back with zero new lines. @@ -1013,9 +989,7 @@ myNestedVar = [ fn test_recast_empty_file_new_line() { let some_program_string = r#" "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); // Its VERY important this comes back with zero new lines. @@ -1026,14 +1000,12 @@ myNestedVar = [ fn test_recast_shebang_only() { let some_program_string = r#"#!/usr/local/env zoo kcl"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let result = parser.ast(); + let result = crate::parser::top_level_parse(some_program_string); assert!(result.is_err()); assert_eq!( result.unwrap_err().to_string(), - r#"syntax: KclErrorDetails { source_ranges: [SourceRange([21, 24])], message: "Unexpected end of file. The compiler expected a function body items (functions are made up of variable declarations, expressions, and return statements, each of those is a possible body item" }"# + r#"syntax: KclErrorDetails { source_ranges: [SourceRange([21, 24, 0])], message: "Unexpected end of file. The compiler expected a function body items (functions are made up of variable declarations, expressions, and return statements, each of those is a possible body item" }"# ); } @@ -1048,9 +1020,7 @@ part001 = startSketchOn('XY') |> close(%) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1081,9 +1051,7 @@ part001 = startSketchOn('XY') |> close(%) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1113,9 +1081,7 @@ part001 = startSketchOn('XY') |> close(%) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1253,9 +1219,7 @@ tabs_l = startSketchOn({ distance: length - 10 }, %) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); // Its VERY important this comes back with zero new lines. @@ -1393,9 +1357,7 @@ tabs_l = startSketchOn({ |> close(%) |> extrude(scale, %) }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1419,9 +1381,7 @@ tabs_l = startSketchOn({ |> startProfileAt([0.0, 5.0], %) |> line([0.4900857016, -0.0240763666], %) |> line([0.6804562304, 0.9087880491], %)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1440,9 +1400,7 @@ tabs_l = startSketchOn({ |> startProfileAt([0.0, 5.0], %) |> line([0.4900857016, -0.0240763666], %) // hello world |> line([0.6804562304, 0.9087880491], %)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1461,9 +1419,7 @@ tabs_l = startSketchOn({ |> line([0.4900857016, -0.0240763666], %) // hello world |> line([0.6804562304, 0.9087880491], %)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1488,9 +1444,7 @@ tabs_l = startSketchOn({ // this is also a comment return things }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1514,9 +1468,7 @@ tabs_l = startSketchOn({ // this is also a comment thing = 'foo' "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1537,9 +1489,7 @@ key = 'c' // hello thing = 'foo' "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1567,9 +1517,7 @@ thing = 'c' foo = 'bar' // "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1595,9 +1543,7 @@ foo = 'bar' // // hello thing = 'foo' "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1616,9 +1562,7 @@ thing = 'foo' /* comment at start */ mySk1 = startSketchAt([0, 0])"#; - let tokens = crate::token::lexer(test_program).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(test_program).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1648,9 +1592,7 @@ mySk1 = startSketchOn('XY') |> ry(45, %) |> rx(45, %) // one more for good measure"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1686,9 +1628,7 @@ mySk1 = startSketchOn('XY') intersectTag: seg01 }, %) |> line([-0.42, -1.72], %)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted.trim(), some_program_string); @@ -1712,9 +1652,7 @@ yo = [ " hey oooooo really long long long" ] "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted, some_program_string); @@ -1730,9 +1668,7 @@ key = 'c' things = "things" // this is also a comment"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); let expected = some_program_string.trim(); @@ -1751,9 +1687,7 @@ things = "things" // a comment " }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted.trim(), some_program_string.trim()); @@ -1777,9 +1711,7 @@ part001 = startSketchOn('XY') -angleToMatchLengthY(seg01, myVar, %), myVar ], %) // ln-lineTo-yAbsolute should use angleToMatchLengthY helper"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted.trim(), some_program_string); @@ -1804,9 +1736,7 @@ part001 = startSketchOn('XY') myVar ], %) // ln-lineTo-yAbsolute should use angleToMatchLengthY helper "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast( &FormatOptions { @@ -1835,9 +1765,7 @@ fn ghi = (part001) => { return part001 } "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let mut program = parser.ast().unwrap(); + let mut program = crate::parser::top_level_parse(some_program_string).unwrap(); program.rename_symbol("mySuperCoolPart", 6); let recasted = program.recast(&Default::default(), 0); @@ -1865,9 +1793,7 @@ fn ghi = (part001) => { let some_program_string = r#"fn ghi = (x, y, z) => { return x }"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let mut program = parser.ast().unwrap(); + let mut program = crate::parser::top_level_parse(some_program_string).unwrap(); program.rename_symbol("newName", 10); let recasted = program.recast(&Default::default(), 0); @@ -1889,9 +1815,7 @@ fn ghi = (part001) => { angle_start: 0, angle_end: 180, }, %)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1921,9 +1845,7 @@ firstExtrude = startSketchOn('XY') |> close(%) |> extrude(h, %) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1960,9 +1882,7 @@ firstExtrude = startSketchOn('XY') |> close(%) |> extrude(h, %) "#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!( @@ -1988,9 +1908,7 @@ firstExtrude = startSketchOn('XY') #[tokio::test(flavor = "multi_thread")] async fn test_recast_math_start_negative() { let some_program_string = r#"myVar = -5 + 6"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted.trim(), some_program_string); @@ -2007,9 +1925,7 @@ startSketchOn('XY') |> line([0, -(5 - thickness)], %) |> line([0, -(5 - 1)], %) |> line([0, -(-5 - 1)], %)"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted.trim(), some_program_string); @@ -2023,9 +1939,7 @@ FOS = 2 sigmaAllow = 8 width = 20 thickness = sqrt(distance * p * FOS * 6 / (sigmaAllow * width))"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted.trim(), some_program_string); @@ -2034,9 +1948,7 @@ thickness = sqrt(distance * p * FOS * 6 / (sigmaAllow * width))"#; #[tokio::test(flavor = "multi_thread")] async fn no_vardec_keyword() { let some_program_string = r#"distance = 5"#; - let tokens = crate::token::lexer(some_program_string).unwrap(); - let parser = crate::parser::Parser::new(tokens); - let program = parser.ast().unwrap(); + let program = crate::parser::top_level_parse(some_program_string).unwrap(); let recasted = program.recast(&Default::default(), 0); assert_eq!(recasted.trim(), some_program_string); @@ -2066,7 +1978,7 @@ thickness = sqrt(distance * p * FOS * 6 / (sigmaAllow * width))"#; .into_iter() .enumerate() { - let tokens = crate::token::lexer(raw).unwrap(); + let tokens = crate::token::lexer(raw, ModuleId::default()).unwrap(); let literal = crate::parser::parser_impl::unsigned_number_literal .parse(&tokens) .unwrap(); @@ -2099,9 +2011,7 @@ sketch002 = startSketchOn({ } }) "#; - let tokens = crate::token::lexer(input).unwrap(); - let p = crate::parser::Parser::new(tokens); - let ast = p.ast().unwrap(); + let ast = crate::parser::top_level_parse(input).unwrap(); let actual = ast.recast(&FormatOptions::new(), 0); assert_eq!(actual, expected); } @@ -2127,7 +2037,7 @@ sketch002 = startSketchOn({ .into_iter() .enumerate() { - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); crate::parser::parser_impl::print_tokens(&tokens); let expr = crate::parser::parser_impl::object.parse(&tokens).unwrap(); assert_eq!( @@ -2225,7 +2135,7 @@ sketch002 = startSketchOn({ .into_iter() .enumerate() { - let tokens = crate::token::lexer(input).unwrap(); + let tokens = crate::token::lexer(input, ModuleId::default()).unwrap(); let expr = crate::parser::parser_impl::array_elem_by_elem.parse(&tokens).unwrap(); assert_eq!( expr.recast(&FormatOptions::new(), 0, false), diff --git a/src/wasm-lib/kcl/src/walk/ast_node.rs b/src/wasm-lib/kcl/src/walk/ast_node.rs index 4b76f26865..293e7c42fb 100644 --- a/src/wasm-lib/kcl/src/walk/ast_node.rs +++ b/src/wasm-lib/kcl/src/walk/ast_node.rs @@ -42,30 +42,30 @@ pub enum Node<'a> { impl From<&Node<'_>> for SourceRange { fn from(node: &Node) -> Self { match node { - Node::Program(p) => SourceRange([p.start, p.end]), - Node::ImportStatement(e) => SourceRange([e.start, e.end]), - Node::ExpressionStatement(e) => SourceRange([e.start, e.end]), - Node::VariableDeclaration(v) => SourceRange([v.start, v.end]), - Node::ReturnStatement(r) => SourceRange([r.start, r.end]), - Node::VariableDeclarator(v) => SourceRange([v.start, v.end]), - Node::Literal(l) => SourceRange([l.start, l.end]), - Node::TagDeclarator(t) => SourceRange([t.start, t.end]), - Node::Identifier(i) => SourceRange([i.start, i.end]), - Node::BinaryExpression(b) => SourceRange([b.start, b.end]), - Node::FunctionExpression(f) => SourceRange([f.start, f.end]), - Node::CallExpression(c) => SourceRange([c.start, c.end]), - Node::PipeExpression(p) => SourceRange([p.start, p.end]), - Node::PipeSubstitution(p) => SourceRange([p.start, p.end]), - Node::ArrayExpression(a) => SourceRange([a.start, a.end]), - Node::ArrayRangeExpression(a) => SourceRange([a.start, a.end]), - Node::ObjectExpression(o) => SourceRange([o.start, o.end]), - Node::MemberExpression(m) => SourceRange([m.start, m.end]), - Node::UnaryExpression(u) => SourceRange([u.start, u.end]), - Node::Parameter(p) => SourceRange([p.identifier.start, p.identifier.end]), - Node::ObjectProperty(o) => SourceRange([o.start, o.end]), - Node::MemberObject(m) => SourceRange([m.start(), m.end()]), - Node::IfExpression(m) => SourceRange([m.start, m.end]), - Node::LiteralIdentifier(l) => SourceRange([l.start(), l.end()]), + Node::Program(n) => SourceRange::from(*n), + Node::ImportStatement(n) => SourceRange::from(*n), + Node::ExpressionStatement(n) => SourceRange::from(*n), + Node::VariableDeclaration(n) => SourceRange::from(*n), + Node::ReturnStatement(n) => SourceRange::from(*n), + Node::VariableDeclarator(n) => SourceRange::from(*n), + Node::Literal(n) => SourceRange::from(*n), + Node::TagDeclarator(n) => SourceRange::from(*n), + Node::Identifier(n) => SourceRange::from(*n), + Node::BinaryExpression(n) => SourceRange::from(*n), + Node::FunctionExpression(n) => SourceRange::from(*n), + Node::CallExpression(n) => SourceRange::from(*n), + Node::PipeExpression(n) => SourceRange::from(*n), + Node::PipeSubstitution(n) => SourceRange::from(*n), + Node::ArrayExpression(n) => SourceRange::from(*n), + Node::ArrayRangeExpression(n) => SourceRange::from(*n), + Node::ObjectExpression(n) => SourceRange::from(*n), + Node::MemberExpression(n) => SourceRange::from(*n), + Node::UnaryExpression(n) => SourceRange::from(*n), + Node::Parameter(p) => SourceRange::from(&p.identifier), + Node::ObjectProperty(n) => SourceRange::from(*n), + Node::MemberObject(m) => SourceRange([m.start(), m.end(), m.module_id().as_usize()]), + Node::IfExpression(n) => SourceRange::from(*n), + Node::LiteralIdentifier(l) => SourceRange([l.start(), l.end(), l.module_id().as_usize()]), } } } diff --git a/src/wasm-lib/kcl/src/walk/ast_walk.rs b/src/wasm-lib/kcl/src/walk/ast_walk.rs index f5836b94ef..7e467b63be 100644 --- a/src/wasm-lib/kcl/src/walk/ast_walk.rs +++ b/src/wasm-lib/kcl/src/walk/ast_walk.rs @@ -315,9 +315,7 @@ mod tests { macro_rules! kcl { ( $kcl:expr ) => {{ - let tokens = $crate::token::lexer($kcl).unwrap(); - let parser = $crate::parser::Parser::new(tokens); - parser.ast().unwrap() + $crate::parser::top_level_parse($kcl).unwrap() }}; } diff --git a/src/wasm-lib/kcl/tests/add_lots/program_memory.snap b/src/wasm-lib/kcl/tests/add_lots/program_memory.snap index 24acbd15d3..d15686eaca 100644 --- a/src/wasm-lib/kcl/tests/add_lots/program_memory.snap +++ b/src/wasm-lib/kcl/tests/add_lots/program_memory.snap @@ -123,7 +123,8 @@ snapshot_kind: text { "sourceRange": [ 7, - 32 + 32, + 0 ] } ] @@ -136,7 +137,8 @@ snapshot_kind: text { "sourceRange": [ 38, - 834 + 834, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap b/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap index 9ecdf9a841..7eb65a5fca 100644 --- a/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap +++ b/src/wasm-lib/kcl/tests/array_elem_push/program_memory.snap @@ -43,7 +43,8 @@ snapshot_kind: text { "sourceRange": [ 6, - 15 + 15, + 0 ] } ] @@ -61,7 +62,8 @@ snapshot_kind: text { "sourceRange": [ 27, - 39 + 39, + 0 ] } ] @@ -80,7 +82,8 @@ snapshot_kind: text { "sourceRange": [ 51, - 68 + 68, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap b/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap index 815317548c..90b6736b8a 100644 --- a/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap +++ b/src/wasm-lib/kcl/tests/array_range_expr/program_memory.snap @@ -39,7 +39,8 @@ snapshot_kind: text { "sourceRange": [ 175, - 188 + 188, + 0 ] } ] @@ -52,7 +53,8 @@ snapshot_kind: text { "sourceRange": [ 79, - 80 + 80, + 0 ] } ] @@ -71,7 +73,8 @@ snapshot_kind: text { "sourceRange": [ 5, - 11 + 11, + 0 ] } ] @@ -90,7 +93,8 @@ snapshot_kind: text { "sourceRange": [ 95, - 107 + 107, + 0 ] } ] @@ -110,7 +114,8 @@ snapshot_kind: text { "sourceRange": [ 194, - 206 + 206, + 0 ] } ] @@ -128,7 +133,8 @@ snapshot_kind: text { "sourceRange": [ 341, - 373 + 373, + 0 ] } ] @@ -141,7 +147,8 @@ snapshot_kind: text { "sourceRange": [ 88, - 89 + 89, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap b/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap index bcc0be7448..b62ff7e493 100644 --- a/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap +++ b/src/wasm-lib/kcl/tests/array_range_negative_expr/program_memory.snap @@ -51,7 +51,8 @@ snapshot_kind: text { "sourceRange": [ 5, - 19 + 19, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/cube/program_memory.snap b/src/wasm-lib/kcl/tests/cube/program_memory.snap index 6e8b18ae96..5f76caf6be 100644 --- a/src/wasm-lib/kcl/tests/cube/program_memory.snap +++ b/src/wasm-lib/kcl/tests/cube/program_memory.snap @@ -751,7 +751,8 @@ snapshot_kind: text { "sourceRange": [ 10, - 316 + 316, + 0 ] } ] @@ -766,7 +767,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 206, - 219 + 219, + 0 ], "tag": null, "type": "extrudePlane" @@ -776,7 +778,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 225, - 238 + 238, + 0 ], "tag": null, "type": "extrudePlane" @@ -786,7 +789,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 244, - 257 + 257, + 0 ], "tag": null, "type": "extrudePlane" @@ -796,7 +800,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 263, - 276 + 276, + 0 ], "tag": null, "type": "extrudePlane" @@ -811,7 +816,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 206, - 219 + 219, + 0 ] }, "from": [ @@ -830,7 +836,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 225, - 238 + 238, + 0 ] }, "from": [ @@ -849,7 +856,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 244, - 257 + 257, + 0 ] }, "from": [ @@ -868,7 +876,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 263, - 276 + 276, + 0 ] }, "from": [ @@ -887,7 +896,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 282, - 290 + 290, + 0 ] }, "from": [ @@ -942,7 +952,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 183, - 200 + 200, + 0 ] } }, @@ -950,7 +961,8 @@ snapshot_kind: text { "sourceRange": [ 183, - 200 + 200, + 0 ] } ] @@ -962,7 +974,8 @@ snapshot_kind: text { "sourceRange": [ 183, - 200 + 200, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/cube/tokens.snap b/src/wasm-lib/kcl/tests/cube/tokens.snap index 6136c04861..ed630c9a77 100644 --- a/src/wasm-lib/kcl/tests/cube/tokens.snap +++ b/src/wasm-lib/kcl/tests/cube/tokens.snap @@ -1,5 +1,5 @@ --- -source: kcl/src/tests.rs +source: kcl/src/simulation_tests.rs description: Result of tokenizing cube.kcl snapshot_kind: text --- diff --git a/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap b/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap index 3e588ea413..6ac6c1eaf0 100644 --- a/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap +++ b/src/wasm-lib/kcl/tests/double_map_fn/program_memory.snap @@ -123,7 +123,8 @@ snapshot_kind: text { "sourceRange": [ 15, - 40 + 40, + 0 ] } ] @@ -140,7 +141,8 @@ snapshot_kind: text { "sourceRange": [ 47, - 53 + 53, + 0 ] } ] @@ -157,7 +159,8 @@ snapshot_kind: text { "sourceRange": [ 90, - 107 + 107, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap b/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap index 3d55ba52af..6824b22cb0 100644 --- a/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap +++ b/src/wasm-lib/kcl/tests/helix_ccw/program_memory.snap @@ -41,7 +41,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 35, - 76 + 76, + 0 ], "tag": null, "type": "extrudeArc" @@ -56,7 +57,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 35, - 76 + 76, + 0 ] }, "ccw": true, @@ -117,7 +119,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 35, - 76 + 76, + 0 ] } }, @@ -125,7 +128,8 @@ snapshot_kind: text { "sourceRange": [ 35, - 76 + 76, + 0 ] } ] @@ -137,7 +141,8 @@ snapshot_kind: text { "sourceRange": [ 35, - 76 + 76, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/if_else/program_memory.snap b/src/wasm-lib/kcl/tests/if_else/program_memory.snap index c396ab169e..1a6a881f8f 100644 --- a/src/wasm-lib/kcl/tests/if_else/program_memory.snap +++ b/src/wasm-lib/kcl/tests/if_else/program_memory.snap @@ -39,7 +39,8 @@ snapshot_kind: text { "sourceRange": [ 64, - 65 + 65, + 0 ] } ] @@ -52,7 +53,8 @@ snapshot_kind: text { "sourceRange": [ 199, - 200 + 200, + 0 ] } ] @@ -65,7 +67,8 @@ snapshot_kind: text { "sourceRange": [ 332, - 333 + 333, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap b/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap index b679079fcf..06c3f72782 100644 --- a/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap +++ b/src/wasm-lib/kcl/tests/index_of_array/program_memory.snap @@ -43,7 +43,8 @@ snapshot_kind: text { "sourceRange": [ 43, - 55 + 55, + 0 ] } ] @@ -56,7 +57,8 @@ snapshot_kind: text { "sourceRange": [ 256, - 266 + 266, + 0 ] } ] @@ -69,7 +71,8 @@ snapshot_kind: text { "sourceRange": [ 93, - 101 + 101, + 0 ] } ] @@ -82,7 +85,8 @@ snapshot_kind: text { "sourceRange": [ 277, - 285 + 285, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_end0.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_end0.png new file mode 100644 index 0000000000..f4a3e8c7ec Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_end0.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start0.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start0.png new file mode 100644 index 0000000000..39763be301 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start0.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start_x0.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start_x0.png new file mode 100644 index 0000000000..9b186e485b Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start_x0.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start_y0.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start_y0.png new file mode 100644 index 0000000000..3592ea952b Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_segment_start_y0.png differ diff --git a/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap b/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap index 10acfd7ae2..35a8f4ef15 100644 --- a/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap +++ b/src/wasm-lib/kcl/tests/property_of_object/program_memory.snap @@ -42,7 +42,8 @@ snapshot_kind: text { "sourceRange": [ 56, - 74 + 74, + 0 ] } ] @@ -60,7 +61,8 @@ snapshot_kind: text { "sourceRange": [ 529, - 543 + 543, + 0 ] } ] @@ -73,7 +75,8 @@ snapshot_kind: text { "sourceRange": [ 122, - 132 + 132, + 0 ] } ] @@ -86,7 +89,8 @@ snapshot_kind: text { "sourceRange": [ 356, - 362 + 362, + 0 ] } ] @@ -99,7 +103,8 @@ snapshot_kind: text { "sourceRange": [ 553, - 570 + 570, + 0 ] } ] @@ -112,7 +117,8 @@ snapshot_kind: text { "sourceRange": [ 757, - 770 + 770, + 0 ] } ] @@ -125,7 +131,8 @@ snapshot_kind: text { "sourceRange": [ 342, - 347 + 347, + 0 ] } ] diff --git a/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap b/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap index 93d2a364a7..7ae5862740 100644 --- a/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap +++ b/src/wasm-lib/kcl/tests/sketch_in_object/program_memory.snap @@ -329,7 +329,8 @@ snapshot_kind: text { "sourceRange": [ 10, - 157 + 157, + 0 ] } ] @@ -957,7 +958,8 @@ snapshot_kind: text { "sourceRange": [ 10, - 157 + 157, + 0 ] } ] @@ -973,7 +975,8 @@ snapshot_kind: text { "sourceRange": [ 170, - 369 + 369, + 0 ] } ] @@ -986,7 +989,8 @@ snapshot_kind: text { "sourceRange": [ 52, - 77 + 77, + 0 ] } ], @@ -1023,7 +1027,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 83, - 98 + 98, + 0 ] }, "from": [ @@ -1042,7 +1047,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 104, - 119 + 119, + 0 ] }, "from": [ @@ -1061,7 +1067,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 125, - 141 + 141, + 0 ] }, "from": [ @@ -1080,7 +1087,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 147, - 155 + 155, + 0 ] }, "from": [ @@ -1100,7 +1108,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 52, - 77 + 77, + 0 ] }, "from": [ @@ -1119,7 +1128,8 @@ snapshot_kind: text { "sourceRange": [ 52, - 77 + 77, + 0 ] } ] @@ -1134,7 +1144,8 @@ snapshot_kind: text { "sourceRange": [ 242, - 267 + 267, + 0 ] } ], @@ -1171,7 +1182,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 277, - 292 + 292, + 0 ] }, "from": [ @@ -1190,7 +1202,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 302, - 317 + 317, + 0 ] }, "from": [ @@ -1209,7 +1222,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 327, - 343 + 343, + 0 ] }, "from": [ @@ -1228,7 +1242,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 353, - 361 + 361, + 0 ] }, "from": [ @@ -1248,7 +1263,8 @@ snapshot_kind: text "id": "[uuid]", "sourceRange": [ 242, - 267 + 267, + 0 ] }, "from": [ @@ -1269,7 +1285,8 @@ snapshot_kind: text { "sourceRange": [ 187, - 367 + 367, + 0 ] } ] diff --git a/src/wasm-lib/src/wasm.rs b/src/wasm-lib/src/wasm.rs index 5d5393182e..741c9a86b7 100644 --- a/src/wasm-lib/src/wasm.rs +++ b/src/wasm-lib/src/wasm.rs @@ -8,7 +8,7 @@ use std::{ use futures::stream::TryStreamExt; use gloo_utils::format::JsValueSerdeExt; use kcl_lib::{ - ast::types::{Node, Program}, + ast::types::{ModuleId, Node, Program}, coredump::CoreDump, engine::EngineManager, executor::ExecutorSettings, @@ -153,9 +153,11 @@ pub async fn modify_ast_for_sketch_wasm( .map_err(|e| format!("{:?}", e))?, )); + let module_id = ModuleId::default(); let _ = kcl_lib::ast::modify::modify_ast_for_sketch( &engine, &mut program, + module_id, sketch_name, plane, uuid::Uuid::parse_str(sketch_id).map_err(|e| e.to_string())?, @@ -193,7 +195,8 @@ pub fn deserialize_files(data: &[u8]) -> Result { pub fn lexer_wasm(js: &str) -> Result { console_error_panic_hook::set_once(); - let tokens = kcl_lib::token::lexer(js).map_err(JsError::from)?; + let module_id = ModuleId::default(); + let tokens = kcl_lib::token::lexer(js, module_id).map_err(JsError::from)?; Ok(JsValue::from_serde(&tokens)?) } @@ -201,7 +204,8 @@ pub fn lexer_wasm(js: &str) -> Result { pub fn parse_wasm(js: &str) -> Result { console_error_panic_hook::set_once(); - let tokens = kcl_lib::token::lexer(js).map_err(String::from)?; + let module_id = ModuleId::default(); + let tokens = kcl_lib::token::lexer(js, module_id).map_err(String::from)?; let parser = kcl_lib::parser::Parser::new(tokens); let program = parser.ast().map_err(String::from)?; // The serde-wasm-bindgen does not work here because of weird HashMap issues so we use the diff --git a/src/wasm-lib/tests/executor/main.rs b/src/wasm-lib/tests/executor/main.rs index 5e46ce0845..034adb81d3 100644 --- a/src/wasm-lib/tests/executor/main.rs +++ b/src/wasm-lib/tests/executor/main.rs @@ -28,7 +28,7 @@ async fn kcl_test_fillet_duplicate_tags() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([203, 249])], message: "Duplicate tags are not allowed." }"#, + r#"type: KclErrorDetails { source_ranges: [SourceRange([203, 249, 0])], message: "Duplicate tags are not allowed." }"#, ); } @@ -83,7 +83,7 @@ async fn kcl_test_execute_engine_error_return() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"engine: KclErrorDetails { source_ranges: [SourceRange([216, 229])], message: "Modeling command failed: [ApiError { error_code: BadRequest, message: \"The path is not closed. Solid2D construction requires a closed path!\" }]" }"#, + r#"engine: KclErrorDetails { source_ranges: [SourceRange([216, 229, 0])], message: "Modeling command failed: [ApiError { error_code: BadRequest, message: \"The path is not closed. Solid2D construction requires a closed path!\" }]" }"#, ); } @@ -515,7 +515,7 @@ async fn kcl_test_import_file_doesnt_exist() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([8, 27])], message: "File `thing.obj` does not exist." }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([8, 27, 0])], message: "File `thing.obj` does not exist." }"# ); } @@ -583,7 +583,7 @@ async fn kcl_test_import_ext_doesnt_match() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([8, 76])], message: "The given format does not match the file extension. Expected: `gltf`, Given: `obj`" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([8, 76, 0])], message: "The given format does not match the file extension. Expected: `gltf`, Given: `obj`" }"# ); } @@ -742,7 +742,7 @@ part002 = startSketchOn(part001, part001.sketch.tags.here) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([88, 133]), SourceRange([210, 226])], message: "could not sketch tangential arc, because its center would be infinitely far away in the X direction" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([88, 133, 0]), SourceRange([210, 226, 0])], message: "could not sketch tangential arc, because its center would be infinitely far away in the X direction" }"# ); } @@ -799,7 +799,7 @@ async fn kcl_test_stdlib_kcl_error_right_code_path() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([151, 189])], message: "Expected an argument at index 1" }"#, + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([151, 189, 0])], message: "Expected an argument at index 1" }"#, ); } @@ -869,7 +869,7 @@ part = rectShape([0, 0], 20, 20) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([863, 912])], message: "Argument at index 0 was supposed to be type kcl_lib::std::shapes::CircleData but found string (text)" }"#, + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([863, 912, 0])], message: "Argument at index 0 was supposed to be type kcl_lib::std::shapes::CircleData but found string (text)" }"#, ); } @@ -954,7 +954,7 @@ async fn kcl_test_revolve_bad_angle_low() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([272, 308])], message: "Expected angle to be between -360 and 360 and not 0, found `-455`" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([272, 308, 0])], message: "Expected angle to be between -360 and 360 and not 0, found `-455`" }"# ); } @@ -979,7 +979,7 @@ async fn kcl_test_revolve_bad_angle_high() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([272, 307])], message: "Expected angle to be between -360 and 360 and not 0, found `455`" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([272, 307, 0])], message: "Expected angle to be between -360 and 360 and not 0, found `455`" }"# ); } @@ -1073,7 +1073,7 @@ sketch001 = startSketchOn(box, revolveAxis) //this fails right now, but slightly differently, lets just say its enough for it to fail - mike //assert_eq!( // result.err().unwrap().to_string(), - // r#"engine: KclErrorDetails { source_ranges: [SourceRange([346, 390])], message: "Modeling command failed: [ApiError { error_code: InternalEngine, message: \"Solid3D revolve failed: sketch profile must lie entirely on one side of the revolution axis\" }]" }"# + // r#"engine: KclErrorDetails { source_ranges: [SourceRange([346, 390, 0])], message: "Modeling command failed: [ApiError { error_code: InternalEngine, message: \"Solid3D revolve failed: sketch profile must lie entirely on one side of the revolution axis\" }]" }"# //); } @@ -1354,7 +1354,7 @@ secondSketch = startSketchOn(part001, '') assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([260, 286])], message: "Argument at index 1 was supposed to be type kcl_lib::std::sketch::FaceTag but found string (text)" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([260, 286, 0])], message: "Argument at index 1 was supposed to be type kcl_lib::std::sketch::FaceTag but found string (text)" }"# ); } @@ -1385,7 +1385,7 @@ extrusion = startSketchOn('XY') assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([68, 334]), SourceRange([428, 461])], message: "Expected 2 arguments, got 3" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([68, 334, 0]), SourceRange([428, 461, 0])], message: "Expected 2 arguments, got 3" }"# ); } @@ -1681,7 +1681,7 @@ part001 = cube([0,0], 20) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([259, 345])], message: "You can only tag one edge at a time with a tagged chamfer. Either delete the tag for the chamfer fn if you don't need it OR separate into individual chamfer functions for each tag." }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([259, 345, 0])], message: "You can only tag one edge at a time with a tagged chamfer. Either delete the tag for the chamfer fn if you don't need it OR separate into individual chamfer functions for each tag." }"# ); } @@ -1708,7 +1708,7 @@ let p = triangle(200) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"value already defined: KclErrorDetails { source_ranges: [SourceRange([311, 313]), SourceRange([326, 339])], message: "Cannot redefine `a`" }"# + r#"value already defined: KclErrorDetails { source_ranges: [SourceRange([311, 313, 0]), SourceRange([326, 339, 0])], message: "Cannot redefine `a`" }"# ); } @@ -1783,7 +1783,7 @@ async fn kcl_test_arc_error_same_start_end() { assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([57, 140])], message: "Arc start and end angles must be different" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([57, 140, 0])], message: "Arc start and end angles must be different" }"# ); } @@ -1803,7 +1803,7 @@ example = extrude(10, exampleSketch) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 111])], message: "Cannot have an x constrained angle of 90 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 111, 0])], message: "Cannot have an x constrained angle of 90 degrees" }"# ); } @@ -1823,7 +1823,7 @@ example = extrude(10, exampleSketch) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 112])], message: "Cannot have an x constrained angle of 270 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 112, 0])], message: "Cannot have an x constrained angle of 270 degrees" }"# ); } @@ -1843,7 +1843,7 @@ example = extrude(10, exampleSketch) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 110])], message: "Cannot have a y constrained angle of 0 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 110, 0])], message: "Cannot have a y constrained angle of 0 degrees" }"# ); } @@ -1863,7 +1863,7 @@ example = extrude(10, exampleSketch) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 112])], message: "Cannot have a y constrained angle of 180 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([72, 112, 0])], message: "Cannot have a y constrained angle of 180 degrees" }"# ); } @@ -1883,7 +1883,7 @@ extrusion = extrude(10, sketch001) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([68, 125])], message: "Cannot have an x constrained angle of 90 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([68, 125, 0])], message: "Cannot have an x constrained angle of 90 degrees" }"# ); } @@ -1903,7 +1903,7 @@ extrusion = extrude(10, sketch001) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([68, 125])], message: "Cannot have an x constrained angle of 90 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([68, 125, 0])], message: "Cannot have an x constrained angle of 90 degrees" }"# ); } @@ -1925,7 +1925,7 @@ example = extrude(10, exampleSketch) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([94, 142])], message: "Cannot have a y constrained angle of 0 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([94, 142, 0])], message: "Cannot have a y constrained angle of 0 degrees" }"# ); } @@ -1947,7 +1947,7 @@ example = extrude(10, exampleSketch) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([94, 144])], message: "Cannot have a y constrained angle of 180 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([94, 144, 0])], message: "Cannot have a y constrained angle of 180 degrees" }"# ); } @@ -1969,7 +1969,7 @@ example = extrude(10, exampleSketch) assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"type: KclErrorDetails { source_ranges: [SourceRange([94, 145])], message: "Cannot have a y constrained angle of 180 degrees" }"# + r#"type: KclErrorDetails { source_ranges: [SourceRange([94, 145, 0])], message: "Cannot have a y constrained angle of 180 degrees" }"# ); } @@ -1986,7 +1986,7 @@ someFunction('INVALID') assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([37, 61]), SourceRange([65, 88])], message: "Argument at index 0 was supposed to be type kcl_lib::std::sketch::SketchData but found string (text)" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([37, 61, 0]), SourceRange([65, 88, 0])], message: "Argument at index 0 was supposed to be type kcl_lib::std::sketch::SketchData but found string (text)" }"# ); } @@ -2007,7 +2007,7 @@ someFunction('INVALID') assert!(result.is_err()); assert_eq!( result.err().unwrap().to_string(), - r#"semantic: KclErrorDetails { source_ranges: [SourceRange([89, 114]), SourceRange([126, 155]), SourceRange([159, 182])], message: "Argument at index 0 was supposed to be type kcl_lib::std::sketch::SketchData but found string (text)" }"# + r#"semantic: KclErrorDetails { source_ranges: [SourceRange([89, 114, 0]), SourceRange([126, 155, 0]), SourceRange([159, 182, 0])], message: "Argument at index 0 was supposed to be type kcl_lib::std::sketch::SketchData but found string (text)" }"# ); } diff --git a/src/wasm-lib/tests/executor/no_visuals.rs b/src/wasm-lib/tests/executor/no_visuals.rs index dd374a1350..8f573fbfcb 100644 --- a/src/wasm-lib/tests/executor/no_visuals.rs +++ b/src/wasm-lib/tests/executor/no_visuals.rs @@ -1,5 +1,5 @@ use kcl_lib::{ - ast::types::{Node, Program}, + ast::types::{ModuleId, Node, Program}, errors::KclError, executor::{ExecutorContext, IdGenerator}, parser, @@ -28,7 +28,8 @@ macro_rules! gen_test_parse_fail { } async fn setup(program: &str) -> (ExecutorContext, Node, IdGenerator) { - let tokens = kcl_lib::token::lexer(program).unwrap(); + let module_id = ModuleId::default(); + let tokens = kcl_lib::token::lexer(program, module_id).unwrap(); let parser = kcl_lib::parser::Parser::new(tokens); let program = parser.ast().unwrap(); let ctx = kcl_lib::executor::ExecutorContext { @@ -60,7 +61,7 @@ async fn run_fail(code: &str) -> KclError { } async fn run_parse_fail(code: &str) -> KclError { - let Err(e) = parser::parse(code) else { + let Err(e) = parser::top_level_parse(code) else { panic!("Expected this KCL program to fail to parse, but it (incorrectly) never threw an error."); }; e diff --git a/src/wasm-lib/tests/modify/main.rs b/src/wasm-lib/tests/modify/main.rs index bf8f20bab1..bf18aa734a 100644 --- a/src/wasm-lib/tests/modify/main.rs +++ b/src/wasm-lib/tests/modify/main.rs @@ -2,7 +2,7 @@ use anyhow::Result; use kcl_lib::{ ast::{ modify::modify_ast_for_sketch, - types::{Node, Program}, + types::{ModuleId, Node, Program}, }, executor::{ExecutorContext, IdGenerator, KclValue, PlaneType, Sketch, SourceRange}, }; @@ -10,10 +10,9 @@ use kittycad_modeling_cmds::{each_cmd as mcmd, length_unit::LengthUnit, shared:: use pretty_assertions::assert_eq; /// Setup the engine and parse code for an ast. -async fn setup(code: &str, name: &str) -> Result<(ExecutorContext, Node, uuid::Uuid)> { - let tokens = kcl_lib::token::lexer(code)?; - let parser = kcl_lib::parser::Parser::new(tokens); - let program = parser.ast()?; +async fn setup(code: &str, name: &str) -> Result<(ExecutorContext, Node, ModuleId, uuid::Uuid)> { + let module_id = ModuleId::default(); + let program = kcl_lib::parser::parse(code, module_id)?; let ctx = kcl_lib::executor::ExecutorContext::new_with_default_client(Default::default()).await?; let exec_state = ctx.run(&program, None, IdGenerator::default(), None).await?; @@ -60,7 +59,7 @@ async fn setup(code: &str, name: &str) -> Result<(ExecutorContext, Node ) .await?; - Ok((ctx, program, sketch_id)) + Ok((ctx, program, module_id, sketch_id)) } #[tokio::test(flavor = "multi_thread")] @@ -76,9 +75,9 @@ async fn kcl_test_modify_sketch_part001() { name ); - let (ctx, program, sketch_id) = setup(&code, name).await.unwrap(); + let (ctx, program, module_id, sketch_id) = setup(&code, name).await.unwrap(); let mut new_program = program.clone(); - let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id) + let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, module_id, name, PlaneType::XY, sketch_id) .await .unwrap(); @@ -101,9 +100,9 @@ async fn kcl_test_modify_sketch_part002() { name ); - let (ctx, program, sketch_id) = setup(&code, name).await.unwrap(); + let (ctx, program, module_id, sketch_id) = setup(&code, name).await.unwrap(); let mut new_program = program.clone(); - let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id) + let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, module_id, name, PlaneType::XY, sketch_id) .await .unwrap(); @@ -128,9 +127,9 @@ async fn kcl_test_modify_close_sketch() { name ); - let (ctx, program, sketch_id) = setup(&code, name).await.unwrap(); + let (ctx, program, module_id, sketch_id) = setup(&code, name).await.unwrap(); let mut new_program = program.clone(); - let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id) + let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, module_id, name, PlaneType::XY, sketch_id) .await .unwrap(); @@ -154,9 +153,9 @@ async fn kcl_test_modify_line_to_close_sketch() { name ); - let (ctx, program, sketch_id) = setup(&code, name).await.unwrap(); + let (ctx, program, module_id, sketch_id) = setup(&code, name).await.unwrap(); let mut new_program = program.clone(); - let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id) + let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, module_id, name, PlaneType::XY, sketch_id) .await .unwrap(); @@ -191,14 +190,14 @@ const {} = startSketchOn("XY") name ); - let (ctx, program, sketch_id) = setup(&code, name).await.unwrap(); + let (ctx, program, module_id, sketch_id) = setup(&code, name).await.unwrap(); let mut new_program = program.clone(); - let result = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id).await; + let result = modify_ast_for_sketch(&ctx.engine, &mut new_program, module_id, name, PlaneType::XY, sketch_id).await; assert!(result.is_err()); assert_eq!( result.unwrap_err().to_string(), - r#"engine: KclErrorDetails { source_ranges: [SourceRange([188, 193])], message: "Sketch part002 is constrained `partial` and cannot be modified" }"# + r#"engine: KclErrorDetails { source_ranges: [SourceRange([188, 193, 0])], message: "Sketch part002 is constrained `partial` and cannot be modified" }"# ); } @@ -216,9 +215,9 @@ async fn kcl_test_modify_line_should_close_sketch() { name ); - let (ctx, program, sketch_id) = setup(&code, name).await.unwrap(); + let (ctx, program, module_id, sketch_id) = setup(&code, name).await.unwrap(); let mut new_program = program.clone(); - let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, name, PlaneType::XY, sketch_id) + let new_code = modify_ast_for_sketch(&ctx.engine, &mut new_program, module_id, name, PlaneType::XY, sketch_id) .await .unwrap(); diff --git a/yarn.lock b/yarn.lock index 67e0684de8..56965dc6cb 100644 --- a/yarn.lock +++ b/yarn.lock @@ -1430,6 +1430,15 @@ fs-extra "^10.0.0" which "^2.0.2" +"@electron-forge/maker-base@7.5.0": + version "7.5.0" + resolved "https://registry.yarnpkg.com/@electron-forge/maker-base/-/maker-base-7.5.0.tgz#6626f33d5f1616b5f31eca72c6818806e3550909" + integrity sha512-+jluKW2UPxaI1+qQQ8fqaUVVbZohRjOSF0Iti7STRFbgJKJitzPB24Cjji9qJCKIx5klMeEiwp0YPAE/d9Xt8g== + dependencies: + "@electron-forge/shared-types" "7.5.0" + fs-extra "^10.0.0" + which "^2.0.2" + "@electron-forge/maker-deb@^7.4.0": version "7.4.0" resolved "https://registry.yarnpkg.com/@electron-forge/maker-deb/-/maker-deb-7.4.0.tgz#7787f525ab8c7ddcc3e9665e60d704179a92848a" @@ -1461,25 +1470,25 @@ optionalDependencies: electron-winstaller "^5.3.0" -"@electron-forge/maker-wix@^7.4.0": - version "7.4.0" - resolved "https://registry.yarnpkg.com/@electron-forge/maker-wix/-/maker-wix-7.4.0.tgz#4c271a70506247f07d704634ef6cbe0fe6060bdc" - integrity sha512-+a5zNh/e8/aguDT7Ya+hEsKkkV7VSSaaB45RaA4ahI91bx/mRAWEhGQjnqakGkSAZkRzM6n37Tedx3wz0/2H4A== +"@electron-forge/maker-wix@^7.5.0": + version "7.5.0" + resolved "https://registry.yarnpkg.com/@electron-forge/maker-wix/-/maker-wix-7.5.0.tgz#994e65e86a5bfab944a9847e146608d4b567197b" + integrity sha512-6OmaviVRixGJ8EYvWik76zWJBvKW8XkQZ/mpwd4JjYcVKudA0zWbzEw2xBzf62TaRoXtSHTxbpfJRDiHyJK4dg== dependencies: - "@electron-forge/maker-base" "7.4.0" - "@electron-forge/shared-types" "7.4.0" + "@electron-forge/maker-base" "7.5.0" + "@electron-forge/shared-types" "7.5.0" chalk "^4.0.0" electron-wix-msi "^5.1.3" log-symbols "^4.0.0" parse-author "^2.0.0" -"@electron-forge/maker-zip@^7.4.0": - version "7.4.0" - resolved "https://registry.yarnpkg.com/@electron-forge/maker-zip/-/maker-zip-7.4.0.tgz#e82ab6174344c43eb9a30b2fb5e2c2e32de2113d" - integrity sha512-UGbMdpuK/P29x1FFRWNOs3bNz+7QNFWVWyTM5hcWqib66cNuUmoaPifQyuwW2POIrIohrxlzLK87/i9Zc8g4dA== +"@electron-forge/maker-zip@^7.5.0": + version "7.5.0" + resolved "https://registry.yarnpkg.com/@electron-forge/maker-zip/-/maker-zip-7.5.0.tgz#e0d534cc5715055b611507e3754f9b4bbe31ffef" + integrity sha512-gIO3bEbubOJqWV6kd0b9nBwTrFfFQv/K8PAqg6e0uSZiy7QuSCFZVAZse02gO3AzxVDSVjjTQ4nmXBXC4Glh1A== dependencies: - "@electron-forge/maker-base" "7.4.0" - "@electron-forge/shared-types" "7.4.0" + "@electron-forge/maker-base" "7.5.0" + "@electron-forge/shared-types" "7.5.0" cross-zip "^4.0.0" fs-extra "^10.0.0" got "^11.8.5" @@ -1538,6 +1547,16 @@ "@electron/rebuild" "^3.2.10" listr2 "^7.0.2" +"@electron-forge/shared-types@7.5.0": + version "7.5.0" + resolved "https://registry.yarnpkg.com/@electron-forge/shared-types/-/shared-types-7.5.0.tgz#3e72aba15add1e5af45767640eeb43b59497f907" + integrity sha512-VXuLVGYa3ZulBlmjA40ZEpk+iPH5ebN0v7t27wDt3rm23bph2aQrL7uSTLXhobenXYBVKggXnQt6rJ9A7FCDNQ== + dependencies: + "@electron-forge/tracer" "7.5.0" + "@electron/packager" "^18.3.5" + "@electron/rebuild" "^3.2.10" + listr2 "^7.0.2" + "@electron-forge/template-base@7.4.0": version "7.4.0" resolved "https://registry.yarnpkg.com/@electron-forge/template-base/-/template-base-7.4.0.tgz#ef2e2fd9f3632860d7ad192dc4a2e09e98463d05" @@ -1592,6 +1611,13 @@ dependencies: chrome-trace-event "^1.0.3" +"@electron-forge/tracer@7.5.0": + version "7.5.0" + resolved "https://registry.yarnpkg.com/@electron-forge/tracer/-/tracer-7.5.0.tgz#45288de6a99923073e69c74ec46c63e26ce1ef66" + integrity sha512-1dE0wKCmv/K3BXCH70o2jp/y2kXgZQm73gIvzyadySXYwu2L4BWxhAO+Q+JsnbUk+nclHEup5ph4D0JoPIWLcQ== + dependencies: + chrome-trace-event "^1.0.3" + "@electron-forge/web-multi-logger@7.4.0": version "7.4.0" resolved "https://registry.yarnpkg.com/@electron-forge/web-multi-logger/-/web-multi-logger-7.4.0.tgz#13df3a84827d07fc41f53e83b8af3a639be2a03d" @@ -1612,6 +1638,15 @@ glob "^7.1.6" minimatch "^3.0.4" +"@electron/asar@^3.2.13": + version "3.2.14" + resolved "https://registry.yarnpkg.com/@electron/asar/-/asar-3.2.14.tgz#8ae3d49e8548aa0b6847c62e021f829a88794408" + integrity sha512-hc52QkesULqbxTRC1vOmSBN1YndUkieoNMfvpe988h0MEoGGqbijkOqv4/2M9PufBJxiTVoDdBmBFfXPowZDQg== + 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" @@ -1718,6 +1753,31 @@ semver "^7.1.3" yargs-parser "^21.1.1" +"@electron/packager@^18.3.5": + version "18.3.5" + resolved "https://registry.yarnpkg.com/@electron/packager/-/packager-18.3.5.tgz#571a8faa321bd9d9e5b97a9a71b8d40401980d8d" + integrity sha512-ClgTxXTt3MesWAcjIxIkgxELjTcllw1FRoVsihP7uT48kpDMqI71p4XvnMWbq8PvU57TcrKICAaLkxRhbc+/wQ== + 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.2.10", "@electron/rebuild@^3.6.0": version "3.6.0" resolved "https://registry.yarnpkg.com/@electron/rebuild/-/rebuild-3.6.0.tgz#60211375a5f8541a71eb07dd2f97354ad0b2b96f" @@ -2730,10 +2790,10 @@ resolved "https://registry.yarnpkg.com/@types/wicg-file-system-access/-/wicg-file-system-access-2023.10.5.tgz#14b3c25eb4d914b5734795bdea71da229f918b9d" integrity sha512-e9kZO9kCdLqT2h9Tw38oGv9UNzBBWaR1MzuAavxPcsV/7FJ3tWbU6RI3uB+yKIDPGLkGVbplS52ub0AcRLvrhA== -"@types/ws@^8.5.10": - version "8.5.12" - resolved "https://registry.yarnpkg.com/@types/ws/-/ws-8.5.12.tgz#619475fe98f35ccca2a2f6c137702d85ec247b7e" - integrity sha512-3tPRkv1EtkDpzlgyKyI8pGsGZAGPEaXeu0DOj5DI25Ja91bdAYddYHbADRYVrZMRbfW+1l5YwXVDKohDJNQxkQ== +"@types/ws@^8.5.13": + version "8.5.13" + resolved "https://registry.yarnpkg.com/@types/ws/-/ws-8.5.13.tgz#6414c280875e2691d0d1e080b05addbf5cb91e20" + integrity sha512-osM/gWBTPKgHV8XkTunnegTRIsvF6owmf5w+JtAfOw472dptdm0dlGv4xCt6GwQRcC2XVOvvRE/0bAoQcL2QkA== dependencies: "@types/node" "*" @@ -3599,18 +3659,18 @@ buffer@^5.1.0, buffer@^5.5.0: base64-js "^1.3.1" ieee754 "^1.1.13" -builder-util-runtime@9.2.4: - version "9.2.4" - resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-9.2.4.tgz#13cd1763da621e53458739a1e63f7fcba673c42a" - integrity sha512-upp+biKpN/XZMLim7aguUyW8s0FUpDvOtK6sbanMFDAMBzpHDqdhgVYm6zc9HJ6nWo7u2Lxk60i2M6Jd3aiNrA== +builder-util-runtime@9.2.10: + version "9.2.10" + resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-9.2.10.tgz#a0f7d9e214158402e78b74a745c8d9f870c604bc" + integrity sha512-6p/gfG1RJSQeIbz8TK5aPNkoztgY1q5TgmGFMAXcY8itsGW6Y2ld1ALsZ5UJn8rog7hKF3zHx5iQbNQ8uLcRlw== dependencies: debug "^4.3.4" sax "^1.2.4" -builder-util-runtime@9.2.5: - version "9.2.5" - resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-9.2.5.tgz#0afdffa0adb5c84c14926c7dd2cf3c6e96e9be83" - integrity sha512-HjIDfhvqx/8B3TDN4GbABQcgpewTU4LMRTQPkVpKYV3lsuxEJoIfvg09GyWTNmfVNSUAYf+fbTN//JX4TH20pg== +builder-util-runtime@9.2.4: + version "9.2.4" + resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-9.2.4.tgz#13cd1763da621e53458739a1e63f7fcba673c42a" + integrity sha512-upp+biKpN/XZMLim7aguUyW8s0FUpDvOtK6sbanMFDAMBzpHDqdhgVYm6zc9HJ6nWo7u2Lxk60i2M6Jd3aiNrA== dependencies: debug "^4.3.4" sax "^1.2.4" @@ -4567,18 +4627,18 @@ electron-to-chromium@^1.5.4: resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.5.13.tgz#1abf0410c5344b2b829b7247e031f02810d442e6" integrity sha512-lbBcvtIJ4J6sS4tb5TLp1b4LyfCdMkwStzXPyAgVgTRAsep4bvrAGaBOP7ZJtQMNJpSQ9SqG4brWOroNaQtm7Q== -electron-updater@^6.3.0: - version "6.3.0" - resolved "https://registry.yarnpkg.com/electron-updater/-/electron-updater-6.3.0.tgz#13a5c3c3f0b2b114fe33181e24a8270096734b3e" - integrity sha512-3Xlezhk+dKaSQrOnkQNqCGiuGSSUPO9BV9TQZ4Iig6AyTJ4FzJONE5gFFc382sY53Sh9dwJfzKsA3DxRHt2btw== +electron-updater@^6.3.9: + version "6.3.9" + resolved "https://registry.yarnpkg.com/electron-updater/-/electron-updater-6.3.9.tgz#e1e7f155624c58e6f3760f376c3a584028165ec4" + integrity sha512-2PJNONi+iBidkoC5D1nzT9XqsE8Q1X28Fn6xRQhO3YX8qRRyJ3mkV4F1aQsuRnYPqq6Hw+E51y27W75WgDoofw== dependencies: - builder-util-runtime "9.2.5" + builder-util-runtime "9.2.10" 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.3.8" + semver "^7.6.3" tiny-typed-emitter "^2.1.0" electron-winstaller@^5.3.0: @@ -5871,10 +5931,10 @@ graphemer@^1.4.0: resolved "https://registry.yarnpkg.com/graphemer/-/graphemer-1.4.0.tgz#fb2f1d55e0e3a1849aeffc90c4fa0dd53a0e66c6" integrity sha512-EtKwoO6kxCL9WO5xipiHTZlSzBm7WLT627TqC/uVRd0HKmq8NXyebnNYxDoBi7wt8eTWrUrKXCOVaFq9x1kgag== -happy-dom@^14.3.10: - version "14.12.3" - resolved "https://registry.yarnpkg.com/happy-dom/-/happy-dom-14.12.3.tgz#1b5892c670461fd1db041bee690981c22d3d521f" - integrity sha512-vsYlEs3E9gLwA1Hp+w3qzu+RUDFf4VTT8cyKqVICoZ2k7WM++Qyd2LwzyTi5bqMJFiIC/vNpTDYuxdreENRK/g== +happy-dom@^15.10.2: + version "15.10.2" + resolved "https://registry.yarnpkg.com/happy-dom/-/happy-dom-15.10.2.tgz#14ae6652d1a80d2611e3f5832cb61ab5e2d1b539" + integrity sha512-NbA5XrSovenJIIcfixCREX3ZnV7yHP4phhbfuxxf4CPn+LZpz/jIM9EqJ2DrPwgVDSMoAKH3pZwQvkbsSiCrUw== dependencies: entities "^4.5.0" webidl-conversions "^7.0.0" @@ -8502,7 +8562,7 @@ semver@^6.2.0, semver@^6.3.1: resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== -semver@^7.1.1, 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.3, semver@^7.6.0: +semver@^7.1.1, 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.3, semver@^7.6.0, semver@^7.6.3: version "7.6.3" resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.3.tgz#980f7b5550bc175fb4dc09403085627f9eb33143" integrity sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A== @@ -9475,10 +9535,10 @@ utrie@^1.0.2: dependencies: base64-arraybuffer "^1.0.2" -uuid@^9.0.1: - version "9.0.1" - resolved "https://registry.yarnpkg.com/uuid/-/uuid-9.0.1.tgz#e188d4c8853cc722220392c424cd637f32293f30" - integrity sha512-b+1eJOlsR9K8HJpow9Ok3fiWOWSIcIzXodvv0rQjVoOVNpWMpxf1wZNpt4y9h10odCNrqnYp1OBzRktckBe3sA== +uuid@^11.0.2: + version "11.0.2" + resolved "https://registry.yarnpkg.com/uuid/-/uuid-11.0.2.tgz#a8d68ba7347d051e7ea716cc8dcbbab634d66875" + integrity sha512-14FfcOJmqdjbBPdDjFQyk/SdT4NySW4eM0zcG+HqbHP5jzuH56xO3J1DGhgs/cEMCfwYi3HQI1gnTO62iaG+tQ== v8-compile-cache-lib@^3.0.1: version "3.0.1" @@ -9894,7 +9954,7 @@ yargs@^16.0.2: y18n "^5.0.5" yargs-parser "^20.2.2" -yargs@^17.0.1, yargs@^17.6.2: +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==