Compare commits
33 Commits
api-deux-p
...
franknoiro
Author | SHA1 | Date | |
---|---|---|---|
ec4fd08a7d | |||
40388b80e7 | |||
4b00d00977 | |||
fc574fa638 | |||
0bea2bcc98 | |||
a1059d547a | |||
e4b40f6d36 | |||
e5d082f441 | |||
d4d3e179b1 | |||
4f4c44e7c7 | |||
1b75020686 | |||
fba62dab98 | |||
a85a534d6b | |||
6ddbb7a31d | |||
051bb0589e | |||
7f9851ae28 | |||
fbcbb341e2 | |||
4a080d1583 | |||
85c721fb49 | |||
27af2d08a3 | |||
fb8b975b5e | |||
62d8d45a58 | |||
ae3440df0a | |||
af658c909d | |||
7ec11d23c8 | |||
30000a1eac | |||
cb3b45747c | |||
fe66310f2d | |||
fefb6cfe87 | |||
0f8375cbb4 | |||
107adc77b3 | |||
4356885aa2 | |||
6a2027cd51 |
@ -7,7 +7,6 @@ VITE_KC_API_WS_MODELING_URL=wss://api.dev.zoo.dev/ws/modeling/commands
|
|||||||
VITE_KC_API_BASE_URL=https://api.dev.zoo.dev
|
VITE_KC_API_BASE_URL=https://api.dev.zoo.dev
|
||||||
VITE_KC_SITE_BASE_URL=https://dev.zoo.dev
|
VITE_KC_SITE_BASE_URL=https://dev.zoo.dev
|
||||||
VITE_KC_SITE_APP_URL=https://app.dev.zoo.dev
|
VITE_KC_SITE_APP_URL=https://app.dev.zoo.dev
|
||||||
VITE_KC_SKIP_AUTH=false
|
|
||||||
VITE_KC_CONNECTION_TIMEOUT_MS=5000
|
VITE_KC_CONNECTION_TIMEOUT_MS=5000
|
||||||
#VITE_WASM_URL="optional way of overriding the wasm url, particular for unit tests which need this if you running not on the default 3000 port"
|
#VITE_WASM_URL="optional way of overriding the wasm url, particular for unit tests which need this if you running not on the default 3000 port"
|
||||||
#VITE_KC_DEV_TOKEN="optional token to skip auth in the app"
|
#VITE_KC_DEV_TOKEN="optional token to skip auth in the app"
|
||||||
|
@ -3,5 +3,4 @@ VITE_KC_API_WS_MODELING_URL=wss://api.zoo.dev/ws/modeling/commands
|
|||||||
VITE_KC_API_BASE_URL=https://api.zoo.dev
|
VITE_KC_API_BASE_URL=https://api.zoo.dev
|
||||||
VITE_KC_SITE_BASE_URL=https://zoo.dev
|
VITE_KC_SITE_BASE_URL=https://zoo.dev
|
||||||
VITE_KC_SITE_APP_URL=https://app.zoo.dev
|
VITE_KC_SITE_APP_URL=https://app.zoo.dev
|
||||||
VITE_KC_SKIP_AUTH=false
|
|
||||||
VITE_KC_CONNECTION_TIMEOUT_MS=15000
|
VITE_KC_CONNECTION_TIMEOUT_MS=15000
|
||||||
|
2
.github/workflows/build-apps.yml
vendored
@ -43,7 +43,7 @@ jobs:
|
|||||||
- name: Download Wasm Cache
|
- name: Download Wasm Cache
|
||||||
id: download-wasm
|
id: download-wasm
|
||||||
if: ${{ github.event_name == 'pull_request' && steps.filter.outputs.rust == 'false' }}
|
if: ${{ github.event_name == 'pull_request' && steps.filter.outputs.rust == 'false' }}
|
||||||
uses: dawidd6/action-download-artifact@v7
|
uses: dawidd6/action-download-artifact@v11
|
||||||
continue-on-error: true
|
continue-on-error: true
|
||||||
with:
|
with:
|
||||||
github_token: ${{secrets.GITHUB_TOKEN}}
|
github_token: ${{secrets.GITHUB_TOKEN}}
|
||||||
|
12
.github/workflows/cargo-test.yml
vendored
@ -25,8 +25,8 @@ jobs:
|
|||||||
- runner=8cpu-linux-x64
|
- runner=8cpu-linux-x64
|
||||||
- extras=s3-cache
|
- extras=s3-cache
|
||||||
steps:
|
steps:
|
||||||
- uses: runs-on/action@v1
|
- uses: runs-on/action@v2
|
||||||
- uses: actions/create-github-app-token@v1
|
- uses: actions/create-github-app-token@v2
|
||||||
id: app-token
|
id: app-token
|
||||||
with:
|
with:
|
||||||
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
||||||
@ -149,8 +149,8 @@ jobs:
|
|||||||
partitionIndex: [1, 2, 3, 4, 5, 6]
|
partitionIndex: [1, 2, 3, 4, 5, 6]
|
||||||
partitionTotal: [6]
|
partitionTotal: [6]
|
||||||
steps:
|
steps:
|
||||||
- uses: runs-on/action@v1
|
- uses: runs-on/action@v2
|
||||||
- uses: actions/create-github-app-token@v1
|
- uses: actions/create-github-app-token@v2
|
||||||
id: app-token
|
id: app-token
|
||||||
with:
|
with:
|
||||||
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
||||||
@ -207,8 +207,8 @@ jobs:
|
|||||||
- runner=32cpu-linux-x64
|
- runner=32cpu-linux-x64
|
||||||
- extras=s3-cache
|
- extras=s3-cache
|
||||||
steps:
|
steps:
|
||||||
- uses: runs-on/action@v1
|
- uses: runs-on/action@v2
|
||||||
- uses: actions/create-github-app-token@v1
|
- uses: actions/create-github-app-token@v2
|
||||||
id: app-token
|
id: app-token
|
||||||
with:
|
with:
|
||||||
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
||||||
|
6
.github/workflows/e2e-tests.yml
vendored
@ -46,7 +46,7 @@ jobs:
|
|||||||
- name: Download Wasm cache
|
- name: Download Wasm cache
|
||||||
id: download-wasm
|
id: download-wasm
|
||||||
if: ${{ github.event_name != 'schedule' && steps.filter.outputs.rust == 'false' }}
|
if: ${{ github.event_name != 'schedule' && steps.filter.outputs.rust == 'false' }}
|
||||||
uses: dawidd6/action-download-artifact@v7
|
uses: dawidd6/action-download-artifact@v11
|
||||||
continue-on-error: true
|
continue-on-error: true
|
||||||
with:
|
with:
|
||||||
github_token: ${{secrets.GITHUB_TOKEN}}
|
github_token: ${{secrets.GITHUB_TOKEN}}
|
||||||
@ -110,7 +110,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
|
||||||
- uses: actions/create-github-app-token@v1
|
- uses: actions/create-github-app-token@v2
|
||||||
id: app-token
|
id: app-token
|
||||||
with:
|
with:
|
||||||
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
||||||
@ -230,7 +230,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
|
||||||
- uses: actions/create-github-app-token@v1
|
- uses: actions/create-github-app-token@v2
|
||||||
id: app-token
|
id: app-token
|
||||||
with:
|
with:
|
||||||
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
app-id: ${{ secrets.MODELING_APP_GH_APP_ID }}
|
||||||
|
2
.github/workflows/generate-website-docs.yml
vendored
@ -20,7 +20,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- uses: actions/create-github-app-token@v1
|
- uses: actions/create-github-app-token@v2
|
||||||
id: app-token
|
id: app-token
|
||||||
with:
|
with:
|
||||||
# required
|
# required
|
||||||
|
6
.github/workflows/kcl-python-bindings.yml
vendored
@ -113,7 +113,7 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: Install uv
|
- name: Install uv
|
||||||
uses: astral-sh/setup-uv@v5
|
uses: astral-sh/setup-uv@v6
|
||||||
- uses: actions-rust-lang/setup-rust-toolchain@v1
|
- uses: actions-rust-lang/setup-rust-toolchain@v1
|
||||||
- uses: taiki-e/install-action@just
|
- uses: taiki-e/install-action@just
|
||||||
- name: Run tests
|
- name: Run tests
|
||||||
@ -130,7 +130,7 @@ jobs:
|
|||||||
steps:
|
steps:
|
||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: Install the latest version of uv
|
- name: Install the latest version of uv
|
||||||
uses: astral-sh/setup-uv@v5
|
uses: astral-sh/setup-uv@v6
|
||||||
- name: Install codespell
|
- name: Install codespell
|
||||||
run: |
|
run: |
|
||||||
uv venv .venv
|
uv venv .venv
|
||||||
@ -161,7 +161,7 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
path: rust/kcl-python-bindings
|
path: rust/kcl-python-bindings
|
||||||
- name: Install the latest version of uv
|
- name: Install the latest version of uv
|
||||||
uses: astral-sh/setup-uv@v5
|
uses: astral-sh/setup-uv@v6
|
||||||
- name: do uv things
|
- name: do uv things
|
||||||
run: |
|
run: |
|
||||||
cd rust/kcl-python-bindings
|
cd rust/kcl-python-bindings
|
||||||
|
30
.github/workflows/static-analysis.yml
vendored
@ -120,6 +120,36 @@ jobs:
|
|||||||
|
|
||||||
- run: npm run circular-deps:diff
|
- run: npm run circular-deps:diff
|
||||||
|
|
||||||
|
npm-url-checker:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: npm-build-wasm
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
|
- uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version-file: '.nvmrc'
|
||||||
|
cache: 'npm'
|
||||||
|
- run: npm install
|
||||||
|
|
||||||
|
- name: Download all artifacts
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
|
||||||
|
- name: Copy prepared wasm
|
||||||
|
run: |
|
||||||
|
ls -R prepared-wasm
|
||||||
|
cp prepared-wasm/kcl_wasm_lib_bg.wasm public
|
||||||
|
mkdir rust/kcl-wasm-lib/pkg
|
||||||
|
cp prepared-wasm/kcl_wasm_lib* rust/kcl-wasm-lib/pkg
|
||||||
|
|
||||||
|
- name: Copy prepared ts-rs bindings
|
||||||
|
run: |
|
||||||
|
ls -R prepared-ts-rs-bindings
|
||||||
|
mkdir rust/kcl-lib/bindings
|
||||||
|
cp -r prepared-ts-rs-bindings/* rust/kcl-lib/bindings/
|
||||||
|
|
||||||
|
- run: npm run url-checker:diff
|
||||||
|
|
||||||
python-codespell:
|
python-codespell:
|
||||||
runs-on: ubuntu-22.04
|
runs-on: ubuntu-22.04
|
||||||
steps:
|
steps:
|
||||||
|
@ -235,6 +235,47 @@ To display logging (to the terminal or console) set `ZOO_LOG=1`. This will log s
|
|||||||
|
|
||||||
To enable memory metrics, build with `--features dhat-heap`.
|
To enable memory metrics, build with `--features dhat-heap`.
|
||||||
|
|
||||||
|
## Running scripts
|
||||||
|
|
||||||
|
There are multiple scripts under the folder path `./scripts` which can be used in various settings.
|
||||||
|
|
||||||
|
### Pattern for a static file, npm run commands, and CI-CD checks
|
||||||
|
|
||||||
|
If you want to implement a static checker follow this pattern. Two static checkers we have are circular dependency checks in our typescript code and url checker to see if any hard coded URL is the typescript application 404s. We have a set of known files in `./scripts/known/*.txt` which is the baseline.
|
||||||
|
|
||||||
|
If you improve the baseline, run the overwrite command and commit the new smaller baseline. Try not to make the baseline bigger, the CI CD will complain.
|
||||||
|
These baselines are to hold us to higher standards and help implement automated testing against the repository
|
||||||
|
|
||||||
|
#### Output result to stdout
|
||||||
|
- `npm run circular-deps`
|
||||||
|
- `npm run url-checker`
|
||||||
|
|
||||||
|
- create a `<name>.sh` file that will run the static checker then output the result to `stdout`
|
||||||
|
|
||||||
|
#### Overwrite result to known .txt file on disk
|
||||||
|
|
||||||
|
If the application needs to overwrite the known file on disk use this pattern. This known .txt file will be source controlled as the baseline
|
||||||
|
|
||||||
|
- `npm run circular-deps:overwrite`
|
||||||
|
- `npm run url-checker:overwrite`
|
||||||
|
|
||||||
|
#### Diff baseline and current
|
||||||
|
|
||||||
|
These commands will write a /tmp/ file on disk and compare it to the known file in the repository. This command will also be used in the CI CD pipeline for automated checks
|
||||||
|
|
||||||
|
- create a `diff-<name>.sh` file that is the script to diff your tmp file to the baseline
|
||||||
|
e.g. `diff-url-checker.sh`
|
||||||
|
```bash
|
||||||
|
#!/bin/bash
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
npm run url-checker > /tmp/urls.txt
|
||||||
|
diff --ignore-blank-lines -w /tmp/urls.txt ./scripts/known/urls.txt
|
||||||
|
```
|
||||||
|
|
||||||
|
- `npm run circular-deps:diff`
|
||||||
|
- `npm run url-checker:diff`
|
||||||
|
|
||||||
## Proposing changes
|
## Proposing changes
|
||||||
|
|
||||||
Before you submit a contribution PR to this repo, please ensure that:
|
Before you submit a contribution PR to this repo, please ensure that:
|
||||||
|
48
docs/kcl-std/functions/std-sketch-planeOf.md
Normal file
55
docs/kcl-std/functions/std-sketch-rectangle.md
Normal file
@ -67,10 +67,12 @@ layout: manual
|
|||||||
* [`patternCircular2d`](/docs/kcl-std/functions/std-sketch-patternCircular2d)
|
* [`patternCircular2d`](/docs/kcl-std/functions/std-sketch-patternCircular2d)
|
||||||
* [`patternLinear2d`](/docs/kcl-std/functions/std-sketch-patternLinear2d)
|
* [`patternLinear2d`](/docs/kcl-std/functions/std-sketch-patternLinear2d)
|
||||||
* [`patternTransform2d`](/docs/kcl-std/functions/std-sketch-patternTransform2d)
|
* [`patternTransform2d`](/docs/kcl-std/functions/std-sketch-patternTransform2d)
|
||||||
|
* [`planeOf`](/docs/kcl-std/functions/std-sketch-planeOf)
|
||||||
* [`polygon`](/docs/kcl-std/functions/std-sketch-polygon)
|
* [`polygon`](/docs/kcl-std/functions/std-sketch-polygon)
|
||||||
* [`profileStart`](/docs/kcl-std/functions/std-sketch-profileStart)
|
* [`profileStart`](/docs/kcl-std/functions/std-sketch-profileStart)
|
||||||
* [`profileStartX`](/docs/kcl-std/functions/std-sketch-profileStartX)
|
* [`profileStartX`](/docs/kcl-std/functions/std-sketch-profileStartX)
|
||||||
* [`profileStartY`](/docs/kcl-std/functions/std-sketch-profileStartY)
|
* [`profileStartY`](/docs/kcl-std/functions/std-sketch-profileStartY)
|
||||||
|
* [`rectangle`](/docs/kcl-std/functions/std-sketch-rectangle)
|
||||||
* [`revolve`](/docs/kcl-std/functions/std-sketch-revolve)
|
* [`revolve`](/docs/kcl-std/functions/std-sketch-revolve)
|
||||||
* [`segAng`](/docs/kcl-std/functions/std-sketch-segAng)
|
* [`segAng`](/docs/kcl-std/functions/std-sketch-segAng)
|
||||||
* [`segEnd`](/docs/kcl-std/functions/std-sketch-segEnd)
|
* [`segEnd`](/docs/kcl-std/functions/std-sketch-segEnd)
|
||||||
|
@ -32,10 +32,12 @@ This module contains functions for creating and manipulating sketches, and makin
|
|||||||
* [`patternCircular2d`](/docs/kcl-std/functions/std-sketch-patternCircular2d)
|
* [`patternCircular2d`](/docs/kcl-std/functions/std-sketch-patternCircular2d)
|
||||||
* [`patternLinear2d`](/docs/kcl-std/functions/std-sketch-patternLinear2d)
|
* [`patternLinear2d`](/docs/kcl-std/functions/std-sketch-patternLinear2d)
|
||||||
* [`patternTransform2d`](/docs/kcl-std/functions/std-sketch-patternTransform2d)
|
* [`patternTransform2d`](/docs/kcl-std/functions/std-sketch-patternTransform2d)
|
||||||
|
* [`planeOf`](/docs/kcl-std/functions/std-sketch-planeOf)
|
||||||
* [`polygon`](/docs/kcl-std/functions/std-sketch-polygon)
|
* [`polygon`](/docs/kcl-std/functions/std-sketch-polygon)
|
||||||
* [`profileStart`](/docs/kcl-std/functions/std-sketch-profileStart)
|
* [`profileStart`](/docs/kcl-std/functions/std-sketch-profileStart)
|
||||||
* [`profileStartX`](/docs/kcl-std/functions/std-sketch-profileStartX)
|
* [`profileStartX`](/docs/kcl-std/functions/std-sketch-profileStartX)
|
||||||
* [`profileStartY`](/docs/kcl-std/functions/std-sketch-profileStartY)
|
* [`profileStartY`](/docs/kcl-std/functions/std-sketch-profileStartY)
|
||||||
|
* [`rectangle`](/docs/kcl-std/functions/std-sketch-rectangle)
|
||||||
* [`revolve`](/docs/kcl-std/functions/std-sketch-revolve)
|
* [`revolve`](/docs/kcl-std/functions/std-sketch-revolve)
|
||||||
* [`segAng`](/docs/kcl-std/functions/std-sketch-segAng)
|
* [`segAng`](/docs/kcl-std/functions/std-sketch-segAng)
|
||||||
* [`segEnd`](/docs/kcl-std/functions/std-sketch-segEnd)
|
* [`segEnd`](/docs/kcl-std/functions/std-sketch-segEnd)
|
||||||
|
@ -5,6 +5,7 @@ import { uuidv4 } from '@src/lib/utils'
|
|||||||
import type { HomePageFixture } from '@e2e/playwright/fixtures/homePageFixture'
|
import type { HomePageFixture } from '@e2e/playwright/fixtures/homePageFixture'
|
||||||
import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture'
|
import type { SceneFixture } from '@e2e/playwright/fixtures/sceneFixture'
|
||||||
import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture'
|
import type { ToolbarFixture } from '@e2e/playwright/fixtures/toolbarFixture'
|
||||||
|
import type { CmdBarFixture } from '@e2e/playwright/fixtures/cmdBarFixture'
|
||||||
import { getUtils } from '@e2e/playwright/test-utils'
|
import { getUtils } from '@e2e/playwright/test-utils'
|
||||||
import { expect, test } from '@e2e/playwright/zoo-test'
|
import { expect, test } from '@e2e/playwright/zoo-test'
|
||||||
|
|
||||||
@ -14,13 +15,18 @@ test.describe('Can create sketches on all planes and their back sides', () => {
|
|||||||
homePage: HomePageFixture,
|
homePage: HomePageFixture,
|
||||||
scene: SceneFixture,
|
scene: SceneFixture,
|
||||||
toolbar: ToolbarFixture,
|
toolbar: ToolbarFixture,
|
||||||
|
cmdBar: CmdBarFixture,
|
||||||
plane: string,
|
plane: string,
|
||||||
clickCoords: { x: number; y: number }
|
clickCoords: { x: number; y: number }
|
||||||
) => {
|
) => {
|
||||||
const u = await getUtils(page)
|
const u = await getUtils(page)
|
||||||
|
// await page.addInitScript(() => {
|
||||||
|
// localStorage.setItem('persistCode', '@settings(defaultLengthUnit = in)')
|
||||||
|
// })
|
||||||
await page.setBodyDimensions({ width: 1200, height: 500 })
|
await page.setBodyDimensions({ width: 1200, height: 500 })
|
||||||
|
|
||||||
await homePage.goToModelingScene()
|
await homePage.goToModelingScene()
|
||||||
|
// await scene.settled(cmdBar)
|
||||||
const XYPlanRed: [number, number, number] = [98, 50, 51]
|
const XYPlanRed: [number, number, number] = [98, 50, 51]
|
||||||
await scene.expectPixelColor(XYPlanRed, { x: 700, y: 300 }, 15)
|
await scene.expectPixelColor(XYPlanRed, { x: 700, y: 300 }, 15)
|
||||||
|
|
||||||
@ -119,12 +125,166 @@ test.describe('Can create sketches on all planes and their back sides', () => {
|
|||||||
]
|
]
|
||||||
|
|
||||||
for (const config of planeConfigs) {
|
for (const config of planeConfigs) {
|
||||||
test(config.plane, async ({ page, homePage, scene, toolbar }) => {
|
test(config.plane, async ({ page, homePage, scene, toolbar, cmdBar }) => {
|
||||||
await sketchOnPlaneAndBackSideTest(
|
await sketchOnPlaneAndBackSideTest(
|
||||||
page,
|
page,
|
||||||
homePage,
|
homePage,
|
||||||
scene,
|
scene,
|
||||||
toolbar,
|
toolbar,
|
||||||
|
cmdBar,
|
||||||
|
config.plane,
|
||||||
|
config.coords
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
test.describe('Can create sketches on offset planes and their back sides', () => {
|
||||||
|
const sketchOnPlaneAndBackSideTest = async (
|
||||||
|
page: Page,
|
||||||
|
homePage: HomePageFixture,
|
||||||
|
scene: SceneFixture,
|
||||||
|
toolbar: ToolbarFixture,
|
||||||
|
cmdbar: CmdBarFixture,
|
||||||
|
plane: string,
|
||||||
|
clickCoords: { x: number; y: number }
|
||||||
|
) => {
|
||||||
|
const u = await getUtils(page)
|
||||||
|
await page.addInitScript(() => {
|
||||||
|
localStorage.setItem(
|
||||||
|
'persistCode',
|
||||||
|
`@settings(defaultLengthUnit = in)
|
||||||
|
xyPlane = offsetPlane(XY, offset = 0.05)
|
||||||
|
xzPlane = offsetPlane(XZ, offset = 0.05)
|
||||||
|
yzPlane = offsetPlane(YZ, offset = 0.05)
|
||||||
|
`
|
||||||
|
)
|
||||||
|
})
|
||||||
|
await page.setBodyDimensions({ width: 1200, height: 500 })
|
||||||
|
|
||||||
|
await homePage.goToModelingScene()
|
||||||
|
// await scene.settled(cmdbar)
|
||||||
|
const XYPlanRed: [number, number, number] = [74, 74, 74]
|
||||||
|
await scene.expectPixelColor(XYPlanRed, { x: 700, y: 300 }, 15)
|
||||||
|
|
||||||
|
await u.openDebugPanel()
|
||||||
|
|
||||||
|
const coord =
|
||||||
|
plane === '-XY' || plane === '-YZ' || plane === 'XZ' ? -100 : 100
|
||||||
|
const camCommand: EngineCommand = {
|
||||||
|
type: 'modeling_cmd_req',
|
||||||
|
cmd_id: uuidv4(),
|
||||||
|
cmd: {
|
||||||
|
type: 'default_camera_look_at',
|
||||||
|
center: { x: 0, y: 0, z: 0 },
|
||||||
|
vantage: { x: coord, y: coord, z: coord },
|
||||||
|
up: { x: 0, y: 0, z: 1 },
|
||||||
|
},
|
||||||
|
}
|
||||||
|
const updateCamCommand: EngineCommand = {
|
||||||
|
type: 'modeling_cmd_req',
|
||||||
|
cmd_id: uuidv4(),
|
||||||
|
cmd: {
|
||||||
|
type: 'default_camera_get_settings',
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
const prefix = plane.length === 3 ? '-' : ''
|
||||||
|
const planeName = plane
|
||||||
|
.slice(plane.length === 3 ? 1 : 0)
|
||||||
|
.toLocaleLowerCase()
|
||||||
|
|
||||||
|
const codeLine1 = `sketch001 = startSketchOn(${prefix}${planeName}Plane)`
|
||||||
|
const codeLine2 = `profile001 = startProfile(sketch001, at = [${0.91 + (plane[0] === '-' ? 0.01 : 0)}, -${1.21 + (plane[0] === '-' ? 0.01 : 0)}])`
|
||||||
|
|
||||||
|
await u.openDebugPanel()
|
||||||
|
|
||||||
|
await u.clearCommandLogs()
|
||||||
|
await page.getByRole('button', { name: 'Start Sketch' }).click()
|
||||||
|
|
||||||
|
await u.sendCustomCmd(camCommand)
|
||||||
|
await page.waitForTimeout(100)
|
||||||
|
await u.sendCustomCmd(updateCamCommand)
|
||||||
|
|
||||||
|
await u.closeDebugPanel()
|
||||||
|
|
||||||
|
await toolbar.openFeatureTreePane()
|
||||||
|
await toolbar.getDefaultPlaneVisibilityButton('XY').click()
|
||||||
|
await toolbar.getDefaultPlaneVisibilityButton('XZ').click()
|
||||||
|
await toolbar.getDefaultPlaneVisibilityButton('YZ').click()
|
||||||
|
await expect(
|
||||||
|
toolbar
|
||||||
|
.getDefaultPlaneVisibilityButton('YZ')
|
||||||
|
.locator('[aria-label="eye crossed out"]')
|
||||||
|
).toBeVisible()
|
||||||
|
|
||||||
|
await page.mouse.click(clickCoords.x, clickCoords.y)
|
||||||
|
await page.waitForTimeout(600) // wait for animation
|
||||||
|
|
||||||
|
await toolbar.waitUntilSketchingReady()
|
||||||
|
|
||||||
|
await expect(
|
||||||
|
page.getByRole('button', { name: 'line Line', exact: true })
|
||||||
|
).toBeVisible()
|
||||||
|
|
||||||
|
await u.closeDebugPanel()
|
||||||
|
await page.mouse.click(707, 393)
|
||||||
|
|
||||||
|
await expect(page.locator('.cm-content')).toContainText(codeLine1)
|
||||||
|
await expect(page.locator('.cm-content')).toContainText(codeLine2)
|
||||||
|
|
||||||
|
await page
|
||||||
|
.getByRole('button', { name: 'line Line', exact: true })
|
||||||
|
.first()
|
||||||
|
.click()
|
||||||
|
await u.openAndClearDebugPanel()
|
||||||
|
await page.getByRole('button', { name: 'Exit Sketch' }).click()
|
||||||
|
await u.expectCmdLog('[data-message-type="execution-done"]')
|
||||||
|
|
||||||
|
await u.clearCommandLogs()
|
||||||
|
await u.removeCurrentCode()
|
||||||
|
}
|
||||||
|
|
||||||
|
const planeConfigs = [
|
||||||
|
{
|
||||||
|
plane: 'XY',
|
||||||
|
coords: { x: 600, y: 388 },
|
||||||
|
description: 'red plane',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
plane: 'YZ',
|
||||||
|
coords: { x: 700, y: 250 },
|
||||||
|
description: 'green plane',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
plane: 'XZ',
|
||||||
|
coords: { x: 684, y: 427 },
|
||||||
|
description: 'blue plane',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
plane: '-XY',
|
||||||
|
coords: { x: 600, y: 118 },
|
||||||
|
description: 'back of red plane',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
plane: '-YZ',
|
||||||
|
coords: { x: 700, y: 219 },
|
||||||
|
description: 'back of green plane',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
plane: '-XZ',
|
||||||
|
coords: { x: 700, y: 80 },
|
||||||
|
description: 'back of blue plane',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
for (const config of planeConfigs) {
|
||||||
|
test(config.plane, async ({ page, homePage, scene, toolbar, cmdBar }) => {
|
||||||
|
await sketchOnPlaneAndBackSideTest(
|
||||||
|
page,
|
||||||
|
homePage,
|
||||||
|
scene,
|
||||||
|
toolbar,
|
||||||
|
cmdBar,
|
||||||
config.plane,
|
config.plane,
|
||||||
config.coords
|
config.coords
|
||||||
)
|
)
|
||||||
|
@ -525,7 +525,9 @@ test.describe('Command bar tests', () => {
|
|||||||
const projectName = 'test'
|
const projectName = 'test'
|
||||||
const beforeKclCode = `a = 5
|
const beforeKclCode = `a = 5
|
||||||
b = a * a
|
b = a * a
|
||||||
c = 3 + a`
|
c = 3 + a
|
||||||
|
theta = 45deg
|
||||||
|
`
|
||||||
await context.folderSetupFn(async (dir) => {
|
await context.folderSetupFn(async (dir) => {
|
||||||
const testProject = join(dir, projectName)
|
const testProject = join(dir, projectName)
|
||||||
await fsp.mkdir(testProject, { recursive: true })
|
await fsp.mkdir(testProject, { recursive: true })
|
||||||
@ -615,9 +617,45 @@ c = 3 + a`
|
|||||||
stage: 'commandBarClosed',
|
stage: 'commandBarClosed',
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
await test.step(`Edit a parameter with explicit units via command bar`, async () => {
|
||||||
|
await cmdBar.cmdBarOpenBtn.click()
|
||||||
|
await cmdBar.chooseCommand('edit parameter')
|
||||||
|
await cmdBar
|
||||||
|
.selectOption({
|
||||||
|
name: 'theta',
|
||||||
|
})
|
||||||
|
.click()
|
||||||
|
await cmdBar.expectState({
|
||||||
|
stage: 'arguments',
|
||||||
|
commandName: 'Edit parameter',
|
||||||
|
currentArgKey: 'value',
|
||||||
|
currentArgValue: '45deg',
|
||||||
|
headerArguments: {
|
||||||
|
Name: 'theta',
|
||||||
|
Value: '',
|
||||||
|
},
|
||||||
|
highlightedHeaderArg: 'value',
|
||||||
|
})
|
||||||
|
await cmdBar.argumentInput
|
||||||
|
.locator('[contenteditable]')
|
||||||
|
.fill('45deg + 1deg')
|
||||||
|
await cmdBar.progressCmdBar()
|
||||||
|
await cmdBar.expectState({
|
||||||
|
stage: 'review',
|
||||||
|
commandName: 'Edit parameter',
|
||||||
|
headerArguments: {
|
||||||
|
Name: 'theta',
|
||||||
|
Value: '46deg',
|
||||||
|
},
|
||||||
|
})
|
||||||
|
await cmdBar.progressCmdBar()
|
||||||
|
await cmdBar.expectState({
|
||||||
|
stage: 'commandBarClosed',
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
await editor.expectEditor.toContain(
|
await editor.expectEditor.toContain(
|
||||||
`a = 5b = a * amyParameter001 = ${newValue}c = 3 + a`
|
`a = 5b = a * amyParameter001 = ${newValue}c = 3 + atheta = 45deg + 1deg`
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -274,6 +274,13 @@ export class ToolbarFixture {
|
|||||||
.nth(operationIndex)
|
.nth(operationIndex)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getDefaultPlaneVisibilityButton(plane: 'XY' | 'XZ' | 'YZ' = 'XY') {
|
||||||
|
const index = plane === 'XZ' ? 0 : plane === 'XY' ? 1 : 2
|
||||||
|
return this.featureTreePane
|
||||||
|
.getByTestId('feature-tree-visibility-toggle')
|
||||||
|
.nth(index)
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* View source on a specific operation in the Feature Tree pane.
|
* View source on a specific operation in the Feature Tree pane.
|
||||||
* @param operationName The name of the operation type
|
* @param operationName The name of the operation type
|
||||||
|
@ -136,17 +136,17 @@ test.describe('Point-and-click tests', () => {
|
|||||||
highlightedHeaderArg: 'length',
|
highlightedHeaderArg: 'length',
|
||||||
commandName: 'Extrude',
|
commandName: 'Extrude',
|
||||||
})
|
})
|
||||||
await page.keyboard.insertText('width - 0.001')
|
await page.keyboard.insertText('width - 0.001in')
|
||||||
await cmdBar.progressCmdBar()
|
await cmdBar.progressCmdBar()
|
||||||
await cmdBar.expectState({
|
await cmdBar.expectState({
|
||||||
stage: 'review',
|
stage: 'review',
|
||||||
headerArguments: {
|
headerArguments: {
|
||||||
Length: '4.999',
|
Length: '4.999in',
|
||||||
},
|
},
|
||||||
commandName: 'Extrude',
|
commandName: 'Extrude',
|
||||||
})
|
})
|
||||||
await cmdBar.progressCmdBar()
|
await cmdBar.progressCmdBar()
|
||||||
await editor.expectEditor.toContain('extrude(length = width - 0.001)')
|
await editor.expectEditor.toContain('extrude(length = width - 0.001in)')
|
||||||
})
|
})
|
||||||
|
|
||||||
await test.step(`Edit second extrude via feature tree`, async () => {
|
await test.step(`Edit second extrude via feature tree`, async () => {
|
||||||
|
Before Width: | Height: | Size: 56 KiB After Width: | Height: | Size: 56 KiB |
Before Width: | Height: | Size: 51 KiB After Width: | Height: | Size: 50 KiB |
Before Width: | Height: | Size: 58 KiB After Width: | Height: | Size: 58 KiB |
Before Width: | Height: | Size: 51 KiB After Width: | Height: | Size: 51 KiB |
Before Width: | Height: | Size: 50 KiB After Width: | Height: | Size: 49 KiB |
Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 37 KiB |
Before Width: | Height: | Size: 44 KiB After Width: | Height: | Size: 44 KiB |
Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 47 KiB |
Before Width: | Height: | Size: 52 KiB After Width: | Height: | Size: 52 KiB |
Before Width: | Height: | Size: 58 KiB After Width: | Height: | Size: 58 KiB |
Before Width: | Height: | Size: 32 KiB After Width: | Height: | Size: 38 KiB |
Before Width: | Height: | Size: 51 KiB After Width: | Height: | Size: 57 KiB |
Before Width: | Height: | Size: 49 KiB After Width: | Height: | Size: 54 KiB |
Before Width: | Height: | Size: 75 KiB After Width: | Height: | Size: 75 KiB |
Before Width: | Height: | Size: 48 KiB After Width: | Height: | Size: 48 KiB |
Before Width: | Height: | Size: 61 KiB After Width: | Height: | Size: 61 KiB |
Before Width: | Height: | Size: 135 KiB After Width: | Height: | Size: 132 KiB |
Before Width: | Height: | Size: 118 KiB After Width: | Height: | Size: 117 KiB |
Before Width: | Height: | Size: 135 KiB After Width: | Height: | Size: 132 KiB |
Before Width: | Height: | Size: 68 KiB After Width: | Height: | Size: 68 KiB |
Before Width: | Height: | Size: 71 KiB After Width: | Height: | Size: 71 KiB |
Before Width: | Height: | Size: 67 KiB After Width: | Height: | Size: 67 KiB |
Before Width: | Height: | Size: 69 KiB After Width: | Height: | Size: 69 KiB |
Before Width: | Height: | Size: 69 KiB After Width: | Height: | Size: 70 KiB |
Before Width: | Height: | Size: 65 KiB After Width: | Height: | Size: 66 KiB |
18
flake.lock
generated
@ -20,11 +20,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1745998881,
|
"lastModified": 1750865895,
|
||||||
"narHash": "sha256-vonyYAKJSlsX4n9GCsS0pHxR6yCrfqBIuGvANlkwG6U=",
|
"narHash": "sha256-p2dWAQcLVzquy9LxYCZPwyUdugw78Qv3ChvnX755qHA=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "423d2df5b04b4ee7688c3d71396e872afa236a89",
|
"rev": "61c0f513911459945e2cb8bf333dc849f1b976ff",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -36,11 +36,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs_2": {
|
"nixpkgs_2": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1745998881,
|
"lastModified": 1750865895,
|
||||||
"narHash": "sha256-vonyYAKJSlsX4n9GCsS0pHxR6yCrfqBIuGvANlkwG6U=",
|
"narHash": "sha256-p2dWAQcLVzquy9LxYCZPwyUdugw78Qv3ChvnX755qHA=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "423d2df5b04b4ee7688c3d71396e872afa236a89",
|
"rev": "61c0f513911459945e2cb8bf333dc849f1b976ff",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -78,11 +78,11 @@
|
|||||||
"nixpkgs": "nixpkgs_3"
|
"nixpkgs": "nixpkgs_3"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1745980514,
|
"lastModified": 1750964660,
|
||||||
"narHash": "sha256-CITAeiuXGjDvT5iZBXr6vKVWQwsUQLJUMFO91bfJFC4=",
|
"narHash": "sha256-YQ6EyFetjH1uy5JhdhRdPe6cuNXlYpMAQePFfZj4W7M=",
|
||||||
"owner": "oxalica",
|
"owner": "oxalica",
|
||||||
"repo": "rust-overlay",
|
"repo": "rust-overlay",
|
||||||
"rev": "7fbdae44b0f40ea432e46fd152ad8be0f8f41ad6",
|
"rev": "04f0fcfb1a50c63529805a798b4b5c21610ff390",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
|
2
interface.d.ts
vendored
@ -72,14 +72,12 @@ export interface IElectronAPI {
|
|||||||
}
|
}
|
||||||
process: {
|
process: {
|
||||||
env: {
|
env: {
|
||||||
BASE_URL: string
|
|
||||||
IS_PLAYWRIGHT: string
|
IS_PLAYWRIGHT: string
|
||||||
VITE_KC_DEV_TOKEN: string
|
VITE_KC_DEV_TOKEN: string
|
||||||
VITE_KC_API_WS_MODELING_URL: string
|
VITE_KC_API_WS_MODELING_URL: string
|
||||||
VITE_KC_API_BASE_URL: string
|
VITE_KC_API_BASE_URL: string
|
||||||
VITE_KC_SITE_BASE_URL: string
|
VITE_KC_SITE_BASE_URL: string
|
||||||
VITE_KC_SITE_APP_URL: string
|
VITE_KC_SITE_APP_URL: string
|
||||||
VITE_KC_SKIP_AUTH: string
|
|
||||||
VITE_KC_CONNECTION_TIMEOUT_MS: string
|
VITE_KC_CONNECTION_TIMEOUT_MS: string
|
||||||
VITE_KC_DEV_TOKEN: string
|
VITE_KC_DEV_TOKEN: string
|
||||||
NODE_ENV: string
|
NODE_ENV: string
|
||||||
|
@ -125,18 +125,57 @@ test('Shows a loading spinner when uninitialized credit count', async () => {
|
|||||||
await expect(queryByTestId('spinner')).toBeVisible()
|
await expect(queryByTestId('spinner')).toBeVisible()
|
||||||
})
|
})
|
||||||
|
|
||||||
test('Shows the total credits for Unknown subscription', async () => {
|
const unKnownTierData = {
|
||||||
const data = {
|
balance: {
|
||||||
balance: {
|
monthlyApiCreditsRemaining: 10,
|
||||||
monthlyApiCreditsRemaining: 10,
|
stableApiCreditsRemaining: 25,
|
||||||
stableApiCreditsRemaining: 25,
|
},
|
||||||
},
|
subscriptions: {
|
||||||
subscriptions: {
|
monthlyPayAsYouGoApiCreditsTotal: 20,
|
||||||
monthlyPayAsYouGoApiCreditsTotal: 20,
|
name: "unknown",
|
||||||
name: "unknown",
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const freeTierData = {
|
||||||
|
balance: {
|
||||||
|
monthlyApiCreditsRemaining: 10,
|
||||||
|
stableApiCreditsRemaining: 0,
|
||||||
|
},
|
||||||
|
subscriptions: {
|
||||||
|
monthlyPayAsYouGoApiCreditsTotal: 20,
|
||||||
|
name: "free",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const proTierData = {
|
||||||
|
// These are all ignored
|
||||||
|
balance: {
|
||||||
|
monthlyApiCreditsRemaining: 10,
|
||||||
|
stableApiCreditsRemaining: 0,
|
||||||
|
},
|
||||||
|
subscriptions: {
|
||||||
|
// This should be ignored because it's Pro tier.
|
||||||
|
monthlyPayAsYouGoApiCreditsTotal: 20,
|
||||||
|
name: "pro",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const enterpriseTierData = {
|
||||||
|
// These are all ignored, user is part of an org.
|
||||||
|
balance: {
|
||||||
|
monthlyApiCreditsRemaining: 10,
|
||||||
|
stableApiCreditsRemaining: 0,
|
||||||
|
},
|
||||||
|
subscriptions: {
|
||||||
|
// This should be ignored because it's Pro tier.
|
||||||
|
monthlyPayAsYouGoApiCreditsTotal: 20,
|
||||||
|
// This should be ignored because the user is part of an Org.
|
||||||
|
name: "free",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
test('Shows the total credits for Unknown subscription', async () => {
|
||||||
|
const data = unKnownTierData
|
||||||
server.use(
|
server.use(
|
||||||
http.get('*/user/payment/balance', (req, res, ctx) => {
|
http.get('*/user/payment/balance', (req, res, ctx) => {
|
||||||
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
||||||
@ -166,17 +205,7 @@ test('Shows the total credits for Unknown subscription', async () => {
|
|||||||
})
|
})
|
||||||
|
|
||||||
test('Progress bar reflects ratio left of Free subscription', async () => {
|
test('Progress bar reflects ratio left of Free subscription', async () => {
|
||||||
const data = {
|
const data = freeTierData
|
||||||
balance: {
|
|
||||||
monthlyApiCreditsRemaining: 10,
|
|
||||||
stableApiCreditsRemaining: 0,
|
|
||||||
},
|
|
||||||
subscriptions: {
|
|
||||||
monthlyPayAsYouGoApiCreditsTotal: 20,
|
|
||||||
name: "free",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
server.use(
|
server.use(
|
||||||
http.get('*/user/payment/balance', (req, res, ctx) => {
|
http.get('*/user/payment/balance', (req, res, ctx) => {
|
||||||
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
||||||
@ -212,19 +241,7 @@ test('Progress bar reflects ratio left of Free subscription', async () => {
|
|||||||
})
|
})
|
||||||
})
|
})
|
||||||
test('Shows infinite credits for Pro subscription', async () => {
|
test('Shows infinite credits for Pro subscription', async () => {
|
||||||
const data = {
|
const data = proTierData
|
||||||
// These are all ignored
|
|
||||||
balance: {
|
|
||||||
monthlyApiCreditsRemaining: 10,
|
|
||||||
stableApiCreditsRemaining: 0,
|
|
||||||
},
|
|
||||||
subscriptions: {
|
|
||||||
// This should be ignored because it's Pro tier.
|
|
||||||
monthlyPayAsYouGoApiCreditsTotal: 20,
|
|
||||||
name: "pro",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
server.use(
|
server.use(
|
||||||
http.get('*/user/payment/balance', (req, res, ctx) => {
|
http.get('*/user/payment/balance', (req, res, ctx) => {
|
||||||
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
||||||
@ -255,19 +272,7 @@ test('Shows infinite credits for Pro subscription', async () => {
|
|||||||
await expect(queryByTestId('billing-remaining-progress-bar-inline')).toBe(null)
|
await expect(queryByTestId('billing-remaining-progress-bar-inline')).toBe(null)
|
||||||
})
|
})
|
||||||
test('Shows infinite credits for Enterprise subscription', async () => {
|
test('Shows infinite credits for Enterprise subscription', async () => {
|
||||||
const data = {
|
const data = enterpriseTierData
|
||||||
// These are all ignored, user is part of an org.
|
|
||||||
balance: {
|
|
||||||
monthlyApiCreditsRemaining: 10,
|
|
||||||
stableApiCreditsRemaining: 0,
|
|
||||||
},
|
|
||||||
subscriptions: {
|
|
||||||
// This should be ignored because it's Pro tier.
|
|
||||||
monthlyPayAsYouGoApiCreditsTotal: 20,
|
|
||||||
// This should be ignored because the user is part of an Org.
|
|
||||||
name: "free",
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
server.use(
|
server.use(
|
||||||
http.get('*/user/payment/balance', (req, res, ctx) => {
|
http.get('*/user/payment/balance', (req, res, ctx) => {
|
||||||
@ -297,3 +302,58 @@ test('Shows infinite credits for Enterprise subscription', async () => {
|
|||||||
await expect(queryByTestId('infinity')).toBeVisible()
|
await expect(queryByTestId('infinity')).toBeVisible()
|
||||||
await expect(queryByTestId('billing-remaining-progress-bar-inline')).toBe(null)
|
await expect(queryByTestId('billing-remaining-progress-bar-inline')).toBe(null)
|
||||||
})
|
})
|
||||||
|
|
||||||
|
test('Show upgrade button if credits are not infinite', async () => {
|
||||||
|
const data = freeTierData
|
||||||
|
server.use(
|
||||||
|
http.get('*/user/payment/balance', (req, res, ctx) => {
|
||||||
|
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
||||||
|
}),
|
||||||
|
http.get('*/user/payment/subscriptions', (req, res, ctx) => {
|
||||||
|
return HttpResponse.json(createUserPaymentSubscriptionsResponse(data.subscriptions))
|
||||||
|
}),
|
||||||
|
http.get('*/org', (req, res, ctx) => {
|
||||||
|
return new HttpResponse(403)
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
const billingActor = createActor(billingMachine, { input: BILLING_CONTEXT_DEFAULTS }).start()
|
||||||
|
|
||||||
|
const { queryByTestId } = render(<BillingDialog
|
||||||
|
billingActor={billingActor}
|
||||||
|
/>)
|
||||||
|
|
||||||
|
await act(() => {
|
||||||
|
billingActor.send({ type: BillingTransition.Update, apiToken: "it doesn't matter wtf this is :)" })
|
||||||
|
})
|
||||||
|
|
||||||
|
await expect(queryByTestId('billing-upgrade-button')).toBeVisible()
|
||||||
|
})
|
||||||
|
|
||||||
|
test('Hide upgrade button if credits are infinite', async () => {
|
||||||
|
const data = enterpriseTierData
|
||||||
|
server.use(
|
||||||
|
http.get('*/user/payment/balance', (req, res, ctx) => {
|
||||||
|
return HttpResponse.json(createUserPaymentBalanceResponse(data.balance))
|
||||||
|
}),
|
||||||
|
http.get('*/user/payment/subscriptions', (req, res, ctx) => {
|
||||||
|
return HttpResponse.json(createUserPaymentSubscriptionsResponse(data.subscriptions))
|
||||||
|
}),
|
||||||
|
// Ok finally the first use of an org lol
|
||||||
|
http.get('*/org', (req, res, ctx) => {
|
||||||
|
return HttpResponse.json(createOrgResponse())
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
|
||||||
|
const billingActor = createActor(billingMachine, { input: BILLING_CONTEXT_DEFAULTS }).start()
|
||||||
|
|
||||||
|
const { queryByTestId } = render(<BillingDialog
|
||||||
|
billingActor={billingActor}
|
||||||
|
/>)
|
||||||
|
|
||||||
|
await act(() => {
|
||||||
|
billingActor.send({ type: BillingTransition.Update, apiToken: "it doesn't matter wtf this is :)" })
|
||||||
|
})
|
||||||
|
|
||||||
|
await expect(queryByTestId('billing-upgrade-button')).toBe(null)
|
||||||
|
})
|
||||||
|
1137
package-lock.json
generated
@ -110,8 +110,11 @@
|
|||||||
"remove-importmeta": "sed -i 's/import.meta.url/window.location.origin/g' \"./rust/kcl-wasm-lib/pkg/kcl_wasm_lib.js\"; sed -i '' 's/import.meta.url/window.location.origin/g' \"./rust/kcl-wasm-lib/pkg/kcl_wasm_lib.js\" || echo \"sed for both mac and linux\"",
|
"remove-importmeta": "sed -i 's/import.meta.url/window.location.origin/g' \"./rust/kcl-wasm-lib/pkg/kcl_wasm_lib.js\"; sed -i '' 's/import.meta.url/window.location.origin/g' \"./rust/kcl-wasm-lib/pkg/kcl_wasm_lib.js\" || echo \"sed for both mac and linux\"",
|
||||||
"lint-fix": "eslint --fix --ext .ts --ext .tsx src e2e packages/codemirror-lsp-client/src rust/kcl-language-server/client/src",
|
"lint-fix": "eslint --fix --ext .ts --ext .tsx src e2e packages/codemirror-lsp-client/src rust/kcl-language-server/client/src",
|
||||||
"lint": "eslint --max-warnings 0 --ext .ts --ext .tsx src e2e packages/codemirror-lsp-client/src rust/kcl-language-server/client/src",
|
"lint": "eslint --max-warnings 0 --ext .ts --ext .tsx src e2e packages/codemirror-lsp-client/src rust/kcl-language-server/client/src",
|
||||||
|
"url-checker":"./scripts/url-checker.sh",
|
||||||
|
"url-checker:overwrite":"npm run url-checker > scripts/known/urls.txt",
|
||||||
|
"url-checker:diff":"./scripts/diff-url-checker.sh",
|
||||||
"circular-deps": "dpdm --no-warning --no-tree -T --skip-dynamic-imports=circular src/index.tsx",
|
"circular-deps": "dpdm --no-warning --no-tree -T --skip-dynamic-imports=circular src/index.tsx",
|
||||||
"circular-deps:overwrite": "npm run circular-deps | sed '$d' | grep -v '^npm run' > known-circular.txt",
|
"circular-deps:overwrite": "npm run circular-deps | sed '$d' | grep -v '^npm run' > scripts/known/circular.txt",
|
||||||
"circular-deps:diff": "./scripts/diff-circular-deps.sh",
|
"circular-deps:diff": "./scripts/diff-circular-deps.sh",
|
||||||
"circular-deps:diff:nodejs": "npm run circular-deps:diff || node ./scripts/diff.js",
|
"circular-deps:diff:nodejs": "npm run circular-deps:diff || node ./scripts/diff.js",
|
||||||
"files:set-version": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json",
|
"files:set-version": "echo \"$(jq --arg v \"$VERSION\" '.version=$v' package.json --indent 2)\" > package.json",
|
||||||
|
@ -29,7 +29,7 @@
|
|||||||
"vscode-uri": "^3.1.0"
|
"vscode-uri": "^3.1.0"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/node": "^22.14.1",
|
"@types/node": "^24.0.7",
|
||||||
"ts-node": "^10.9.2"
|
"ts-node": "^10.9.2"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Before Width: | Height: | Size: 84 KiB After Width: | Height: | Size: 84 KiB |
144
rust/Cargo.lock
generated
@ -178,7 +178,7 @@ checksum = "3b43422f69d8ff38f95f1b2bb76517c91589a924d1559a0e935d7c8ce0274c11"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -189,7 +189,7 @@ checksum = "e539d3fca749fcee5236ab05e93a52867dd549cc157c8cb7f99595f3cedffdb5"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -211,7 +211,7 @@ checksum = "e12882f59de5360c748c4cbf569a042d5fb0eb515f7bea9c1f470b47f6ffbd73"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -514,7 +514,7 @@ dependencies = [
|
|||||||
"heck",
|
"heck",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -740,7 +740,7 @@ dependencies = [
|
|||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"strsim",
|
"strsim",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -751,7 +751,7 @@ checksum = "d336a2a514f6ccccaa3e09b02d41d35330c07ddf03a62165fcec10bb561c7806"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"darling_core",
|
"darling_core",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -810,7 +810,7 @@ checksum = "30542c1ad912e0e3d22a1935c290e12e8a29d704a420177a31faad4a601a0800"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -831,7 +831,7 @@ dependencies = [
|
|||||||
"darling",
|
"darling",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -841,7 +841,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index"
|
|||||||
checksum = "ab63b0e2bf4d5928aff72e83a7dace85d7bba5fe12dcc3c5a572d78caffd3f3c"
|
checksum = "ab63b0e2bf4d5928aff72e83a7dace85d7bba5fe12dcc3c5a572d78caffd3f3c"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"derive_builder_core",
|
"derive_builder_core",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -906,7 +906,7 @@ checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -944,7 +944,7 @@ checksum = "a1ab991c1362ac86c61ab6f556cff143daa22e5a15e4e189df818b2fd19fe65b"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -1119,7 +1119,7 @@ checksum = "162ee34ebcb7c64a8abebc059ce0fee27c2262618d7b60ed8faf72fef13c3650"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -1223,7 +1223,7 @@ dependencies = [
|
|||||||
"inflections",
|
"inflections",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -1599,7 +1599,7 @@ checksum = "1ec89e9337638ecdc08744df490b221a7399bf8d164eb52a665454e60e075ad6"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -1814,7 +1814,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-bumper"
|
name = "kcl-bumper"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"clap",
|
"clap",
|
||||||
@ -1825,26 +1825,26 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-derive-docs"
|
name = "kcl-derive-docs"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-directory-test-macro"
|
name = "kcl-directory-test-macro"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"convert_case",
|
"convert_case",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-language-server"
|
name = "kcl-language-server"
|
||||||
version = "0.2.83"
|
version = "0.2.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"clap",
|
"clap",
|
||||||
@ -1865,7 +1865,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-language-server-release"
|
name = "kcl-language-server-release"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"clap",
|
"clap",
|
||||||
@ -1885,7 +1885,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-lib"
|
name = "kcl-lib"
|
||||||
version = "0.2.83"
|
version = "0.2.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"approx 0.5.1",
|
"approx 0.5.1",
|
||||||
@ -1962,7 +1962,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-python-bindings"
|
name = "kcl-python-bindings"
|
||||||
version = "0.3.83"
|
version = "0.3.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"kcl-lib",
|
"kcl-lib",
|
||||||
@ -1977,7 +1977,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-test-server"
|
name = "kcl-test-server"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"hyper 0.14.32",
|
"hyper 0.14.32",
|
||||||
@ -1990,7 +1990,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-to-core"
|
name = "kcl-to-core"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
@ -2004,7 +2004,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kcl-wasm-lib"
|
name = "kcl-wasm-lib"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"bson",
|
"bson",
|
||||||
@ -2071,9 +2071,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "kittycad-modeling-cmds"
|
name = "kittycad-modeling-cmds"
|
||||||
version = "0.2.124"
|
version = "0.2.125"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "221aa4670a7ad7dc8f1e4e0f9990bf3cff0a64417eb76493bafe5bbbc1f8350a"
|
checksum = "cfd09d95f8bbeb090d4d1137c9bf421eb75763f7a30e4a9e8eefa249ddf20bd3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
@ -2104,7 +2104,7 @@ dependencies = [
|
|||||||
"kittycad-modeling-cmds-macros-impl",
|
"kittycad-modeling-cmds-macros-impl",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2115,7 +2115,7 @@ checksum = "fdb4ee23cc996aa2dca7584d410e8826e08161e1ac4335bb646d5ede33f37cb3"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2311,7 +2311,7 @@ checksum = "db5b29714e950dbb20d5e6f74f9dcec4edbcc1067bb7f8ed198c097b8c1a818b"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2640,7 +2640,7 @@ dependencies = [
|
|||||||
"regex",
|
"regex",
|
||||||
"regex-syntax 0.8.5",
|
"regex-syntax 0.8.5",
|
||||||
"structmeta",
|
"structmeta",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2654,7 +2654,7 @@ dependencies = [
|
|||||||
"regex",
|
"regex",
|
||||||
"regex-syntax 0.8.5",
|
"regex-syntax 0.8.5",
|
||||||
"structmeta",
|
"structmeta",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2710,7 +2710,7 @@ dependencies = [
|
|||||||
"pest_meta",
|
"pest_meta",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2754,7 +2754,7 @@ dependencies = [
|
|||||||
"phf_shared",
|
"phf_shared",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2809,7 +2809,7 @@ checksum = "6e918e4ff8c4549eb882f14b3a4bc8c8bc93de829416eacf579f1207a8fbf861"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2921,7 +2921,7 @@ dependencies = [
|
|||||||
"proc-macro-error-attr2",
|
"proc-macro-error-attr2",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2981,7 +2981,7 @@ dependencies = [
|
|||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"pyo3-macros-backend",
|
"pyo3-macros-backend",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -2994,7 +2994,7 @@ dependencies = [
|
|||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"pyo3-build-config",
|
"pyo3-build-config",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3492,7 +3492,7 @@ dependencies = [
|
|||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"serde_derive_internals",
|
"serde_derive_internals",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3556,7 +3556,7 @@ checksum = "5b0276cf7f2c73365f7157c8123c21cd9a50fbbd844757af28ca1f5925fc2a00"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3567,7 +3567,7 @@ checksum = "18d26a20a969b9e3fdf2fc2d9f21eda6c40e2de84c9408bb5d3b05d499aae711"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3591,14 +3591,14 @@ checksum = "175ee3e80ae9982737ca543e96133087cbd9a485eecc3bc4de9c1a37b47ea59c"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "serde_spanned"
|
name = "serde_spanned"
|
||||||
version = "0.6.8"
|
version = "0.6.9"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "87607cb1398ed59d48732e575a4c28a7a8ebf2454b964fe3f224f2afc07909e1"
|
checksum = "bf41e0cfaf7226dca15e8197172c295a782857fcb97fad1808a166870dee75a3"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
]
|
]
|
||||||
@ -3815,7 +3815,7 @@ dependencies = [
|
|||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"structmeta-derive",
|
"structmeta-derive",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3826,7 +3826,7 @@ checksum = "152a0b65a590ff6c3da95cabe2353ee04e6167c896b28e3b14478c2636c922fc"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3848,7 +3848,7 @@ dependencies = [
|
|||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"rustversion",
|
"rustversion",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3891,9 +3891,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "syn"
|
name = "syn"
|
||||||
version = "2.0.103"
|
version = "2.0.104"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "e4307e30089d6fd6aff212f2da3a1f9e32f3223b1f010fb09b7c95f90f3ca1e8"
|
checksum = "17b6f705963418cdb9927482fa304bc562ece2fdd4f616084c50b7023b435a40"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
@ -3917,7 +3917,7 @@ checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -3941,7 +3941,7 @@ dependencies = [
|
|||||||
"proc-macro-error2",
|
"proc-macro-error2",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -4050,7 +4050,7 @@ checksum = "4fee6c4efc90059e10f81e6d42c60a18f76588c3d74cb83a0b242a2b6c7504c1"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -4061,7 +4061,7 @@ checksum = "7f7cf42b4507d8ea322120659672cf1b9dbb93f8f2d4ecfd6e51350ff5b17a1d"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -4173,7 +4173,7 @@ checksum = "6e06d43f1345a3bcd39f6a56dbb7dcab2ba47e68e8ac134855e7e2bdbaf8cab8"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -4217,9 +4217,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "toml"
|
name = "toml"
|
||||||
version = "0.8.22"
|
version = "0.8.23"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "05ae329d1f08c4d17a59bed7ff5b5a769d062e64a62d34a3261b219e62cd5aae"
|
checksum = "dc1beb996b9d83529a9e75c17a1686767d148d70663143c7854d8b4a09ced362"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"serde",
|
"serde",
|
||||||
"serde_spanned",
|
"serde_spanned",
|
||||||
@ -4238,9 +4238,9 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "toml_edit"
|
name = "toml_edit"
|
||||||
version = "0.22.26"
|
version = "0.22.27"
|
||||||
source = "registry+https://github.com/rust-lang/crates.io-index"
|
source = "registry+https://github.com/rust-lang/crates.io-index"
|
||||||
checksum = "310068873db2c5b3e7659d2cc35d21855dbafa50d1ce336397c666e3cb08137e"
|
checksum = "41fe8c660ae4257887cf66394862d21dbca4a6ddd26f04a3560410406a2f819a"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"indexmap 2.9.0",
|
"indexmap 2.9.0",
|
||||||
"serde",
|
"serde",
|
||||||
@ -4341,7 +4341,7 @@ checksum = "84fd902d4e0b9a4b27f2f440108dc034e1758628a9b702f8ec61ad66355422fa"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -4369,7 +4369,7 @@ checksum = "395ae124c09f9e6918a2310af6038fba074bcf474ac352496d5910dd59a2226d"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -4449,7 +4449,7 @@ checksum = "e9d4ed7b4c18cc150a6a0a1e9ea1ecfa688791220781af6e119f9599a8502a0a"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
"termcolor",
|
"termcolor",
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -4635,7 +4635,7 @@ dependencies = [
|
|||||||
"proc-macro-error2",
|
"proc-macro-error2",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -4706,7 +4706,7 @@ dependencies = [
|
|||||||
"log",
|
"log",
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
"wasm-bindgen-shared",
|
"wasm-bindgen-shared",
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -4742,7 +4742,7 @@ checksum = "8ae87ea40c9f689fc23f209965b6fb8a99ad69aeeb0231408be24920604395de"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
"wasm-bindgen-backend",
|
"wasm-bindgen-backend",
|
||||||
"wasm-bindgen-shared",
|
"wasm-bindgen-shared",
|
||||||
]
|
]
|
||||||
@ -4777,7 +4777,7 @@ checksum = "17d5042cc5fa009658f9a7333ef24291b1291a25b6382dd68862a7f3b969f69b"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -5067,7 +5067,7 @@ checksum = "2380878cad4ac9aac1e2435f3eb4020e8374b5f13c296cb75b4620ff8e229154"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
"synstructure",
|
"synstructure",
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -5112,7 +5112,7 @@ checksum = "fa4f8080344d4671fb4e831a13ad1e68092748387dfc4f55e356242fae12ce3e"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -5123,7 +5123,7 @@ checksum = "6352c01d0edd5db859a63e2605f4ea3183ddbd15e2c4a9e7d32184df75e4f154"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -5143,7 +5143,7 @@ checksum = "d71e5d6e06ab090c67b5e44993ec16b72dcbaabc526db883a360057678b48502"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
"synstructure",
|
"synstructure",
|
||||||
]
|
]
|
||||||
|
|
||||||
@ -5164,7 +5164,7 @@ checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
@ -5186,7 +5186,7 @@ checksum = "6eafa6dfb17584ea3e2bd6e76e0cc15ad7af12b09abdd1ca55961bed9b1063c6"
|
|||||||
dependencies = [
|
dependencies = [
|
||||||
"proc-macro2",
|
"proc-macro2",
|
||||||
"quote",
|
"quote",
|
||||||
"syn 2.0.103",
|
"syn 2.0.104",
|
||||||
]
|
]
|
||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
|
@ -36,7 +36,7 @@ dashmap = { version = "6.1.0" }
|
|||||||
http = "1"
|
http = "1"
|
||||||
indexmap = "2.9.0"
|
indexmap = "2.9.0"
|
||||||
kittycad = { version = "0.3.37", default-features = false, features = ["js", "requests"] }
|
kittycad = { version = "0.3.37", default-features = false, features = ["js", "requests"] }
|
||||||
kittycad-modeling-cmds = { version = "0.2.124", features = ["ts-rs", "websocket"] }
|
kittycad-modeling-cmds = { version = "0.2.125", features = ["ts-rs", "websocket"] }
|
||||||
lazy_static = "1.5.0"
|
lazy_static = "1.5.0"
|
||||||
miette = "7.6.0"
|
miette = "7.6.0"
|
||||||
pyo3 = { version = "0.24.2" }
|
pyo3 = { version = "0.24.2" }
|
||||||
@ -60,6 +60,6 @@ lossy_float_literal = "warn"
|
|||||||
result_large_err = "allow"
|
result_large_err = "allow"
|
||||||
|
|
||||||
# Example: how to point modeling-app at a different repo (e.g. a branch or a local clone)
|
# Example: how to point modeling-app at a different repo (e.g. a branch or a local clone)
|
||||||
#[patch.crates-io]
|
# [patch.crates-io]
|
||||||
#kittycad-modeling-cmds = { path = "../../../modeling-api/modeling-cmds" }
|
# kittycad-modeling-cmds = { path = "../../modeling-api/modeling-cmds/" }
|
||||||
#kittycad-modeling-session = { path = "../../../modeling-api/modeling-session" }
|
# kittycad-modeling-session = { path = "../../modeling-api/modeling-session" }
|
||||||
|
@ -8,6 +8,9 @@ lint:
|
|||||||
# Ensure we can build without extra feature flags.
|
# Ensure we can build without extra feature flags.
|
||||||
cargo clippy -p kcl-lib --all-targets -- -D warnings
|
cargo clippy -p kcl-lib --all-targets -- -D warnings
|
||||||
|
|
||||||
|
lint-fix:
|
||||||
|
cargo clippy --workspace --all-targets --all-features --fix
|
||||||
|
|
||||||
# Run the stdlib docs generation
|
# Run the stdlib docs generation
|
||||||
redo-kcl-stdlib-docs-no-imgs:
|
redo-kcl-stdlib-docs-no-imgs:
|
||||||
EXPECTORATE=overwrite {{cnr}} {{kcl_lib_flags}} docs::gen_std_tests::test_generate_stdlib
|
EXPECTORATE=overwrite {{cnr}} {{kcl_lib_flags}} docs::gen_std_tests::test_generate_stdlib
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
|
|
||||||
[package]
|
[package]
|
||||||
name = "kcl-bumper"
|
name = "kcl-bumper"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
repository = "https://github.com/KittyCAD/modeling-api"
|
repository = "https://github.com/KittyCAD/modeling-api"
|
||||||
rust-version = "1.76"
|
rust-version = "1.76"
|
||||||
@ -19,7 +19,7 @@ anyhow = { workspace = true }
|
|||||||
clap = { workspace = true, features = ["derive"] }
|
clap = { workspace = true, features = ["derive"] }
|
||||||
semver = "1.0.25"
|
semver = "1.0.25"
|
||||||
serde = { workspace = true }
|
serde = { workspace = true }
|
||||||
toml_edit = "0.22.26"
|
toml_edit = "0.22.27"
|
||||||
|
|
||||||
[lints]
|
[lints]
|
||||||
workspace = true
|
workspace = true
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "kcl-derive-docs"
|
name = "kcl-derive-docs"
|
||||||
description = "A tool for generating documentation from Rust derive macros"
|
description = "A tool for generating documentation from Rust derive macros"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
repository = "https://github.com/KittyCAD/modeling-app"
|
repository = "https://github.com/KittyCAD/modeling-app"
|
||||||
@ -14,7 +14,7 @@ bench = false
|
|||||||
[dependencies]
|
[dependencies]
|
||||||
proc-macro2 = "1"
|
proc-macro2 = "1"
|
||||||
quote = "1"
|
quote = "1"
|
||||||
syn = { version = "2.0.103", features = ["full"] }
|
syn = { version = "2.0.104", features = ["full"] }
|
||||||
|
|
||||||
[lints]
|
[lints]
|
||||||
workspace = true
|
workspace = true
|
||||||
|
@ -97,8 +97,11 @@ pub const TEST_NAMES: &[&str] = &[
|
|||||||
"std-offsetPlane-2",
|
"std-offsetPlane-2",
|
||||||
"std-offsetPlane-3",
|
"std-offsetPlane-3",
|
||||||
"std-offsetPlane-4",
|
"std-offsetPlane-4",
|
||||||
|
"std-sketch-planeOf-0",
|
||||||
"std-sketch-circle-0",
|
"std-sketch-circle-0",
|
||||||
"std-sketch-circle-1",
|
"std-sketch-circle-1",
|
||||||
|
"std-sketch-rectangle-0",
|
||||||
|
"std-sketch-rectangle-1",
|
||||||
"std-sketch-patternTransform2d-0",
|
"std-sketch-patternTransform2d-0",
|
||||||
"std-sketch-revolve-0",
|
"std-sketch-revolve-0",
|
||||||
"std-sketch-revolve-1",
|
"std-sketch-revolve-1",
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "kcl-directory-test-macro"
|
name = "kcl-directory-test-macro"
|
||||||
description = "A tool for generating tests from a directory of kcl files"
|
description = "A tool for generating tests from a directory of kcl files"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
repository = "https://github.com/KittyCAD/modeling-app"
|
repository = "https://github.com/KittyCAD/modeling-app"
|
||||||
@ -14,7 +14,7 @@ bench = false
|
|||||||
convert_case = "0.8.0"
|
convert_case = "0.8.0"
|
||||||
proc-macro2 = "1"
|
proc-macro2 = "1"
|
||||||
quote = "1"
|
quote = "1"
|
||||||
syn = { version = "2.0.103", features = ["full"] }
|
syn = { version = "2.0.104", features = ["full"] }
|
||||||
|
|
||||||
[lints]
|
[lints]
|
||||||
workspace = true
|
workspace = true
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "kcl-language-server-release"
|
name = "kcl-language-server-release"
|
||||||
version = "0.1.83"
|
version = "0.1.84"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
authors = ["KittyCAD Inc <kcl@kittycad.io>"]
|
authors = ["KittyCAD Inc <kcl@kittycad.io>"]
|
||||||
publish = false
|
publish = false
|
||||||
|
@ -42,7 +42,7 @@ impl Build {
|
|||||||
.to_string();
|
.to_string();
|
||||||
|
|
||||||
if !stable {
|
if !stable {
|
||||||
version = format!("{}-nightly", version);
|
version = format!("{version}-nightly");
|
||||||
}
|
}
|
||||||
|
|
||||||
let release_tag = if stable {
|
let release_tag = if stable {
|
||||||
@ -59,10 +59,7 @@ impl Build {
|
|||||||
if stable && !release_tag.contains(&version) {
|
if stable && !release_tag.contains(&version) {
|
||||||
// bail early if the tag doesn't match the version
|
// bail early if the tag doesn't match the version
|
||||||
// TODO: error here when we use the tags with kcl
|
// TODO: error here when we use the tags with kcl
|
||||||
println!(
|
println!("Tag {release_tag} doesn't match version {version}. Did you forget to update Cargo.toml?");
|
||||||
"Tag {} doesn't match version {}. Did you forget to update Cargo.toml?",
|
|
||||||
release_tag, version
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
build_server(sh, &version, &target)?;
|
build_server(sh, &version, &target)?;
|
||||||
|
@ -95,10 +95,10 @@ async fn main() -> Result<()> {
|
|||||||
// Format fields using the provided closure.
|
// Format fields using the provided closure.
|
||||||
// We want to make this very concise otherwise the logs are not able to be read by humans.
|
// We want to make this very concise otherwise the logs are not able to be read by humans.
|
||||||
let format = tracing_subscriber::fmt::format::debug_fn(|writer, field, value| {
|
let format = tracing_subscriber::fmt::format::debug_fn(|writer, field, value| {
|
||||||
if format!("{}", field) == "message" {
|
if format!("{field}") == "message" {
|
||||||
write!(writer, "{}: {:?}", field, value)
|
write!(writer, "{field}: {value:?}")
|
||||||
} else {
|
} else {
|
||||||
write!(writer, "{}", field)
|
write!(writer, "{field}")
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
// Separate each field with a comma.
|
// Separate each field with a comma.
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
name = "kcl-language-server"
|
name = "kcl-language-server"
|
||||||
description = "A language server for KCL."
|
description = "A language server for KCL."
|
||||||
authors = ["KittyCAD Inc <kcl@kittycad.io>"]
|
authors = ["KittyCAD Inc <kcl@kittycad.io>"]
|
||||||
version = "0.2.83"
|
version = "0.2.84"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html
|
||||||
|
@ -123,7 +123,7 @@
|
|||||||
"@vscode/test-electron": "^2.4.1",
|
"@vscode/test-electron": "^2.4.1",
|
||||||
"@vscode/vsce": "^3.3.2",
|
"@vscode/vsce": "^3.3.2",
|
||||||
"cross-env": "^7.0.3",
|
"cross-env": "^7.0.3",
|
||||||
"esbuild": "^0.25.2",
|
"esbuild": "^0.25.3",
|
||||||
"glob": "^11.0.1",
|
"glob": "^11.0.1",
|
||||||
"mocha": "^11.1.0",
|
"mocha": "^11.1.0",
|
||||||
"typescript": "^5.8.3"
|
"typescript": "^5.8.3"
|
||||||
|
@ -87,10 +87,10 @@ async fn main() -> Result<()> {
|
|||||||
// Format fields using the provided closure.
|
// Format fields using the provided closure.
|
||||||
// We want to make this very concise otherwise the logs are not able to be read by humans.
|
// We want to make this very concise otherwise the logs are not able to be read by humans.
|
||||||
let format = tracing_subscriber::fmt::format::debug_fn(|writer, field, value| {
|
let format = tracing_subscriber::fmt::format::debug_fn(|writer, field, value| {
|
||||||
if format!("{}", field) == "message" {
|
if format!("{field}") == "message" {
|
||||||
write!(writer, "{}: {:?}", field, value)
|
write!(writer, "{field}: {value:?}")
|
||||||
} else {
|
} else {
|
||||||
write!(writer, "{}", field)
|
write!(writer, "{field}")
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
// Separate each field with a comma.
|
// Separate each field with a comma.
|
||||||
@ -151,7 +151,7 @@ async fn run_cmd(opts: &Opts) -> Result<()> {
|
|||||||
|
|
||||||
tokio::spawn(async move {
|
tokio::spawn(async move {
|
||||||
if let Some(sig) = signals.forever().next() {
|
if let Some(sig) = signals.forever().next() {
|
||||||
log::info!("received signal: {:?}", sig);
|
log::info!("received signal: {sig:?}");
|
||||||
log::info!("triggering cleanup...");
|
log::info!("triggering cleanup...");
|
||||||
|
|
||||||
// Exit the process.
|
// Exit the process.
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "kcl-lib"
|
name = "kcl-lib"
|
||||||
description = "KittyCAD Language implementation and tools"
|
description = "KittyCAD Language implementation and tools"
|
||||||
version = "0.2.83"
|
version = "0.2.84"
|
||||||
edition = "2021"
|
edition = "2024"
|
||||||
license = "MIT"
|
license = "MIT"
|
||||||
repository = "https://github.com/KittyCAD/modeling-app"
|
repository = "https://github.com/KittyCAD/modeling-app"
|
||||||
rust-version = "1.83"
|
rust-version = "1.88"
|
||||||
authors = ["Jess Frazelle", "Adam Chalmers", "KittyCAD, Inc"]
|
authors = ["Jess Frazelle", "Adam Chalmers", "KittyCAD, Inc"]
|
||||||
keywords = ["kcl", "KittyCAD", "CAD"]
|
keywords = ["kcl", "KittyCAD", "CAD"]
|
||||||
exclude = ["tests/*", "benches/*", "examples/*", "e2e/*", "bindings/*", "fuzz/*"]
|
exclude = ["tests/*", "benches/*", "examples/*", "e2e/*", "bindings/*", "fuzz/*"]
|
||||||
@ -74,7 +74,7 @@ sha2 = "0.10.9"
|
|||||||
tabled = { version = "0.20.0", optional = true }
|
tabled = { version = "0.20.0", optional = true }
|
||||||
tempfile = "3.20"
|
tempfile = "3.20"
|
||||||
thiserror = "2.0.0"
|
thiserror = "2.0.0"
|
||||||
toml = "0.8.22"
|
toml = "0.8.23"
|
||||||
ts-rs = { version = "11.0.1", features = [
|
ts-rs = { version = "11.0.1", features = [
|
||||||
"uuid-impl",
|
"uuid-impl",
|
||||||
"url-impl",
|
"url-impl",
|
||||||
|
@ -4,7 +4,7 @@ use std::{
|
|||||||
path::{Path, PathBuf},
|
path::{Path, PathBuf},
|
||||||
};
|
};
|
||||||
|
|
||||||
use criterion::{criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
|
||||||
const IGNORE_DIRS: [&str; 2] = ["step", "screenshots"];
|
const IGNORE_DIRS: [&str; 2] = ["step", "screenshots"];
|
||||||
|
|
||||||
@ -61,7 +61,7 @@ fn run_benchmarks(c: &mut Criterion) {
|
|||||||
|
|
||||||
// Read the file content (panic on failure)
|
// Read the file content (panic on failure)
|
||||||
let input_content = fs::read_to_string(&input_file)
|
let input_content = fs::read_to_string(&input_file)
|
||||||
.unwrap_or_else(|e| panic!("Failed to read main.kcl in directory {}: {}", dir_name, e));
|
.unwrap_or_else(|e| panic!("Failed to read main.kcl in directory {dir_name}: {e}"));
|
||||||
|
|
||||||
// Create a benchmark group for this directory
|
// Create a benchmark group for this directory
|
||||||
let mut group = c.benchmark_group(&dir_name);
|
let mut group = c.benchmark_group(&dir_name);
|
||||||
@ -72,12 +72,12 @@ fn run_benchmarks(c: &mut Criterion) {
|
|||||||
#[cfg(feature = "benchmark-execution")]
|
#[cfg(feature = "benchmark-execution")]
|
||||||
let program = kcl_lib::Program::parse_no_errs(&input_content).unwrap();
|
let program = kcl_lib::Program::parse_no_errs(&input_content).unwrap();
|
||||||
|
|
||||||
group.bench_function(format!("parse_{}", dir_name), |b| {
|
group.bench_function(format!("parse_{dir_name}"), |b| {
|
||||||
b.iter(|| kcl_lib::Program::parse_no_errs(black_box(&input_content)).unwrap())
|
b.iter(|| kcl_lib::Program::parse_no_errs(black_box(&input_content)).unwrap())
|
||||||
});
|
});
|
||||||
|
|
||||||
#[cfg(feature = "benchmark-execution")]
|
#[cfg(feature = "benchmark-execution")]
|
||||||
group.bench_function(format!("execute_{}", dir_name), |b| {
|
group.bench_function(format!("execute_{dir_name}"), |b| {
|
||||||
b.iter(|| {
|
b.iter(|| {
|
||||||
if let Err(err) = rt.block_on(async {
|
if let Err(err) = rt.block_on(async {
|
||||||
let ctx = kcl_lib::ExecutorContext::new_with_default_client().await?;
|
let ctx = kcl_lib::ExecutorContext::new_with_default_client().await?;
|
||||||
@ -86,7 +86,7 @@ fn run_benchmarks(c: &mut Criterion) {
|
|||||||
ctx.close().await;
|
ctx.close().await;
|
||||||
Ok::<(), anyhow::Error>(())
|
Ok::<(), anyhow::Error>(())
|
||||||
}) {
|
}) {
|
||||||
panic!("Failed to execute program: {}", err);
|
panic!("Failed to execute program: {err}");
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
});
|
});
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use std::hint::black_box;
|
use std::hint::black_box;
|
||||||
|
|
||||||
use criterion::{criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
|
||||||
pub fn bench_parse(c: &mut Criterion) {
|
pub fn bench_parse(c: &mut Criterion) {
|
||||||
for (name, file) in [
|
for (name, file) in [
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
use criterion::{criterion_group, criterion_main, Criterion};
|
use criterion::{Criterion, criterion_group, criterion_main};
|
||||||
|
|
||||||
pub fn bench_digest(c: &mut Criterion) {
|
pub fn bench_digest(c: &mut Criterion) {
|
||||||
for (name, file) in [
|
for (name, file) in [
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
use std::hint::black_box;
|
use std::hint::black_box;
|
||||||
|
|
||||||
use criterion::{criterion_group, criterion_main, BenchmarkId, Criterion};
|
use criterion::{BenchmarkId, Criterion, criterion_group, criterion_main};
|
||||||
use kcl_lib::kcl_lsp_server;
|
use kcl_lib::kcl_lsp_server;
|
||||||
use tokio::runtime::Runtime;
|
use tokio::runtime::Runtime;
|
||||||
use tower_lsp::LanguageServer;
|
use tower_lsp::LanguageServer;
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
//! Cache testing framework.
|
//! Cache testing framework.
|
||||||
|
|
||||||
use kcl_lib::{bust_cache, ExecError, ExecOutcome};
|
use kcl_lib::{ExecError, ExecOutcome, bust_cache};
|
||||||
#[cfg(feature = "artifact-graph")]
|
#[cfg(feature = "artifact-graph")]
|
||||||
use kcl_lib::{exec::Operation, NodePathStep};
|
use kcl_lib::{NodePathStep, exec::Operation};
|
||||||
use kcmc::{each_cmd as mcmd, ModelingCmd};
|
use kcmc::{ModelingCmd, each_cmd as mcmd};
|
||||||
use kittycad_modeling_cmds as kcmc;
|
use kittycad_modeling_cmds as kcmc;
|
||||||
use pretty_assertions::assert_eq;
|
use pretty_assertions::assert_eq;
|
||||||
|
|
||||||
@ -38,7 +38,7 @@ async fn cache_test(
|
|||||||
if !variation.other_files.is_empty() {
|
if !variation.other_files.is_empty() {
|
||||||
let tmp_dir = std::env::temp_dir();
|
let tmp_dir = std::env::temp_dir();
|
||||||
let tmp_dir = tmp_dir
|
let tmp_dir = tmp_dir
|
||||||
.join(format!("kcl_test_{}", test_name))
|
.join(format!("kcl_test_{test_name}"))
|
||||||
.join(uuid::Uuid::new_v4().to_string());
|
.join(uuid::Uuid::new_v4().to_string());
|
||||||
|
|
||||||
// Create a temporary file for each of the other files.
|
// Create a temporary file for each of the other files.
|
||||||
@ -56,7 +56,7 @@ async fn cache_test(
|
|||||||
Err(error) => {
|
Err(error) => {
|
||||||
let report = error.clone().into_miette_report_with_outputs(variation.code).unwrap();
|
let report = error.clone().into_miette_report_with_outputs(variation.code).unwrap();
|
||||||
let report = miette::Report::new(report);
|
let report = miette::Report::new(report);
|
||||||
panic!("{:?}", report);
|
panic!("{report:?}");
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -69,7 +69,7 @@ async fn cache_test(
|
|||||||
.and_then(|x| x.decode().map_err(|e| ExecError::BadPng(e.to_string())))
|
.and_then(|x| x.decode().map_err(|e| ExecError::BadPng(e.to_string())))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
// Save the snapshot.
|
// Save the snapshot.
|
||||||
let path = crate::assert_out(&format!("cache_{}_{}", test_name, index), &img);
|
let path = crate::assert_out(&format!("cache_{test_name}_{index}"), &img);
|
||||||
|
|
||||||
img_results.push((path, img, outcome));
|
img_results.push((path, img, outcome));
|
||||||
}
|
}
|
||||||
@ -337,8 +337,7 @@ extrude001 = extrude(profile001, length = 4)
|
|||||||
// 0] as a more lenient check.
|
// 0] as a more lenient check.
|
||||||
.map(|c| !c.range.is_synthetic() && c.node_path.is_empty())
|
.map(|c| !c.range.is_synthetic() && c.node_path.is_empty())
|
||||||
.unwrap_or(false),
|
.unwrap_or(false),
|
||||||
"artifact={:?}",
|
"artifact={artifact:?}"
|
||||||
artifact
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
mod cache;
|
mod cache;
|
||||||
|
|
||||||
use kcl_lib::{
|
use kcl_lib::{
|
||||||
test_server::{execute_and_export_step, execute_and_snapshot, execute_and_snapshot_no_auth},
|
|
||||||
BacktraceItem, ExecError, ModuleId, SourceRange,
|
BacktraceItem, ExecError, ModuleId, SourceRange,
|
||||||
|
test_server::{execute_and_export_step, execute_and_snapshot, execute_and_snapshot_no_auth},
|
||||||
};
|
};
|
||||||
|
|
||||||
/// The minimum permissible difference between asserted twenty-twenty images.
|
/// The minimum permissible difference between asserted twenty-twenty images.
|
||||||
@ -869,11 +869,13 @@ async fn kcl_test_revolve_bad_angle_low() {
|
|||||||
let result = execute_and_snapshot(code, None).await;
|
let result = execute_and_snapshot(code, None).await;
|
||||||
|
|
||||||
assert!(result.is_err());
|
assert!(result.is_err());
|
||||||
assert!(result
|
assert!(
|
||||||
.err()
|
result
|
||||||
.unwrap()
|
.err()
|
||||||
.to_string()
|
.unwrap()
|
||||||
.contains("Expected angle to be between -360 and 360 and not 0, found `-455`"));
|
.to_string()
|
||||||
|
.contains("Expected angle to be between -360 and 360 and not 0, found `-455`")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
@ -895,11 +897,13 @@ async fn kcl_test_revolve_bad_angle_high() {
|
|||||||
let result = execute_and_snapshot(code, None).await;
|
let result = execute_and_snapshot(code, None).await;
|
||||||
|
|
||||||
assert!(result.is_err());
|
assert!(result.is_err());
|
||||||
assert!(result
|
assert!(
|
||||||
.err()
|
result
|
||||||
.unwrap()
|
.err()
|
||||||
.to_string()
|
.unwrap()
|
||||||
.contains("Expected angle to be between -360 and 360 and not 0, found `455`"));
|
.to_string()
|
||||||
|
.contains("Expected angle to be between -360 and 360 and not 0, found `455`")
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
@ -2090,7 +2094,10 @@ async fn kcl_test_better_type_names() {
|
|||||||
},
|
},
|
||||||
None => todo!(),
|
None => todo!(),
|
||||||
};
|
};
|
||||||
assert_eq!(err, "This function expected the input argument to be one or more Solids or ImportedGeometry but it's actually of type Sketch. You can convert a sketch (2D) into a Solid (3D) by calling a function like `extrude` or `revolve`");
|
assert_eq!(
|
||||||
|
err,
|
||||||
|
"This function expected the input argument to be one or more Solids or ImportedGeometry but it's actually of type Sketch. You can convert a sketch (2D) into a Solid (3D) by calling a function like `extrude` or `revolve`"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
|
@ -10,71 +10,76 @@ DATA;
|
|||||||
NAMED_UNIT(*)
|
NAMED_UNIT(*)
|
||||||
SI_UNIT($, .METRE.)
|
SI_UNIT($, .METRE.)
|
||||||
);
|
);
|
||||||
#2 = UNCERTAINTY_MEASURE_WITH_UNIT(0.00001, #1, 'DISTANCE_ACCURACY_VALUE', $);
|
#2 = (
|
||||||
#3 = (
|
NAMED_UNIT(*)
|
||||||
|
PLANE_ANGLE_UNIT()
|
||||||
|
SI_UNIT($, .RADIAN.)
|
||||||
|
);
|
||||||
|
#3 = UNCERTAINTY_MEASURE_WITH_UNIT(0.00001, #1, 'DISTANCE_ACCURACY_VALUE', $);
|
||||||
|
#4 = (
|
||||||
GEOMETRIC_REPRESENTATION_CONTEXT(3)
|
GEOMETRIC_REPRESENTATION_CONTEXT(3)
|
||||||
GLOBAL_UNCERTAINTY_ASSIGNED_CONTEXT((#2))
|
GLOBAL_UNCERTAINTY_ASSIGNED_CONTEXT((#3))
|
||||||
GLOBAL_UNIT_ASSIGNED_CONTEXT((#1))
|
GLOBAL_UNIT_ASSIGNED_CONTEXT((#1, #2))
|
||||||
REPRESENTATION_CONTEXT('', '3D')
|
REPRESENTATION_CONTEXT('', '3D')
|
||||||
);
|
);
|
||||||
#4 = CARTESIAN_POINT('NONE', (0.015, -0.01, -0.005));
|
#5 = CARTESIAN_POINT('NONE', (0.015, -0.01, -0.005));
|
||||||
#5 = VERTEX_POINT('NONE', #4);
|
#6 = VERTEX_POINT('NONE', #5);
|
||||||
#6 = CARTESIAN_POINT('NONE', (0.015, 0, -0.005));
|
#7 = CARTESIAN_POINT('NONE', (0.015, 0, -0.005));
|
||||||
#7 = VERTEX_POINT('NONE', #6);
|
#8 = VERTEX_POINT('NONE', #7);
|
||||||
#8 = DIRECTION('NONE', (1, 0, -0));
|
#9 = DIRECTION('NONE', (1, 0, -0));
|
||||||
#9 = DIRECTION('NONE', (0, 1, 0));
|
#10 = DIRECTION('NONE', (0, 1, 0));
|
||||||
#10 = CARTESIAN_POINT('NONE', (0.005, -0.01, -0.005));
|
#11 = CARTESIAN_POINT('NONE', (0.005, -0.01, -0.005));
|
||||||
#11 = AXIS2_PLACEMENT_3D('NONE', #10, #9, #8);
|
#12 = AXIS2_PLACEMENT_3D('NONE', #11, #10, #9);
|
||||||
#12 = CIRCLE('NONE', #11, 0.01);
|
#13 = CIRCLE('NONE', #12, 0.01);
|
||||||
#13 = DIRECTION('NONE', (0, 1, 0));
|
#14 = DIRECTION('NONE', (0, 1, 0));
|
||||||
#14 = VECTOR('NONE', #13, 1);
|
#15 = VECTOR('NONE', #14, 1);
|
||||||
#15 = CARTESIAN_POINT('NONE', (0.015, -0.01, -0.005));
|
#16 = CARTESIAN_POINT('NONE', (0.015, -0.01, -0.005));
|
||||||
#16 = LINE('NONE', #15, #14);
|
#17 = LINE('NONE', #16, #15);
|
||||||
#17 = DIRECTION('NONE', (1, 0, -0));
|
#18 = DIRECTION('NONE', (1, 0, -0));
|
||||||
#18 = DIRECTION('NONE', (0, 1, 0));
|
#19 = DIRECTION('NONE', (0, 1, 0));
|
||||||
#19 = CARTESIAN_POINT('NONE', (0.005, 0, -0.005));
|
#20 = CARTESIAN_POINT('NONE', (0.005, 0, -0.005));
|
||||||
#20 = AXIS2_PLACEMENT_3D('NONE', #19, #18, #17);
|
#21 = AXIS2_PLACEMENT_3D('NONE', #20, #19, #18);
|
||||||
#21 = CIRCLE('NONE', #20, 0.01);
|
#22 = CIRCLE('NONE', #21, 0.01);
|
||||||
#22 = EDGE_CURVE('NONE', #5, #5, #12, .T.);
|
#23 = EDGE_CURVE('NONE', #6, #6, #13, .T.);
|
||||||
#23 = EDGE_CURVE('NONE', #5, #7, #16, .T.);
|
#24 = EDGE_CURVE('NONE', #6, #8, #17, .T.);
|
||||||
#24 = EDGE_CURVE('NONE', #7, #7, #21, .T.);
|
#25 = EDGE_CURVE('NONE', #8, #8, #22, .T.);
|
||||||
#25 = CARTESIAN_POINT('NONE', (0.005, -0.005, -0.005));
|
#26 = CARTESIAN_POINT('NONE', (0.005, -0.005, -0.005));
|
||||||
#26 = DIRECTION('NONE', (0, 1, 0));
|
#27 = DIRECTION('NONE', (0, 1, 0));
|
||||||
#27 = DIRECTION('NONE', (1, 0, -0));
|
#28 = DIRECTION('NONE', (1, 0, -0));
|
||||||
#28 = AXIS2_PLACEMENT_3D('NONE', #25, #26, #27);
|
#29 = AXIS2_PLACEMENT_3D('NONE', #26, #27, #28);
|
||||||
#29 = CYLINDRICAL_SURFACE('NONE', #28, 0.01);
|
#30 = CYLINDRICAL_SURFACE('NONE', #29, 0.01);
|
||||||
#30 = CARTESIAN_POINT('NONE', (0, -0.01, -0));
|
#31 = CARTESIAN_POINT('NONE', (0, -0.01, -0));
|
||||||
#31 = DIRECTION('NONE', (0, 1, 0));
|
#32 = DIRECTION('NONE', (0, 1, 0));
|
||||||
#32 = AXIS2_PLACEMENT_3D('NONE', #30, #31, $);
|
#33 = AXIS2_PLACEMENT_3D('NONE', #31, #32, $);
|
||||||
#33 = PLANE('NONE', #32);
|
#34 = PLANE('NONE', #33);
|
||||||
#34 = CARTESIAN_POINT('NONE', (0, 0, -0));
|
#35 = CARTESIAN_POINT('NONE', (0, 0, -0));
|
||||||
#35 = DIRECTION('NONE', (0, 1, 0));
|
#36 = DIRECTION('NONE', (0, 1, 0));
|
||||||
#36 = AXIS2_PLACEMENT_3D('NONE', #34, #35, $);
|
#37 = AXIS2_PLACEMENT_3D('NONE', #35, #36, $);
|
||||||
#37 = PLANE('NONE', #36);
|
#38 = PLANE('NONE', #37);
|
||||||
#38 = ORIENTED_EDGE('NONE', *, *, #22, .T.);
|
#39 = ORIENTED_EDGE('NONE', *, *, #23, .T.);
|
||||||
#39 = ORIENTED_EDGE('NONE', *, *, #24, .F.);
|
#40 = ORIENTED_EDGE('NONE', *, *, #25, .F.);
|
||||||
#40 = EDGE_LOOP('NONE', (#38));
|
#41 = EDGE_LOOP('NONE', (#39));
|
||||||
#41 = FACE_BOUND('NONE', #40, .T.);
|
#42 = FACE_BOUND('NONE', #41, .T.);
|
||||||
#42 = EDGE_LOOP('NONE', (#39));
|
#43 = EDGE_LOOP('NONE', (#40));
|
||||||
#43 = FACE_BOUND('NONE', #42, .T.);
|
#44 = FACE_BOUND('NONE', #43, .T.);
|
||||||
#44 = ADVANCED_FACE('NONE', (#41, #43), #29, .T.);
|
#45 = ADVANCED_FACE('NONE', (#42, #44), #30, .T.);
|
||||||
#45 = ORIENTED_EDGE('NONE', *, *, #22, .F.);
|
#46 = ORIENTED_EDGE('NONE', *, *, #23, .F.);
|
||||||
#46 = EDGE_LOOP('NONE', (#45));
|
#47 = EDGE_LOOP('NONE', (#46));
|
||||||
#47 = FACE_BOUND('NONE', #46, .T.);
|
#48 = FACE_BOUND('NONE', #47, .T.);
|
||||||
#48 = ADVANCED_FACE('NONE', (#47), #33, .F.);
|
#49 = ADVANCED_FACE('NONE', (#48), #34, .F.);
|
||||||
#49 = ORIENTED_EDGE('NONE', *, *, #24, .T.);
|
#50 = ORIENTED_EDGE('NONE', *, *, #25, .T.);
|
||||||
#50 = EDGE_LOOP('NONE', (#49));
|
#51 = EDGE_LOOP('NONE', (#50));
|
||||||
#51 = FACE_BOUND('NONE', #50, .T.);
|
#52 = FACE_BOUND('NONE', #51, .T.);
|
||||||
#52 = ADVANCED_FACE('NONE', (#51), #37, .T.);
|
#53 = ADVANCED_FACE('NONE', (#52), #38, .T.);
|
||||||
#53 = CLOSED_SHELL('NONE', (#44, #48, #52));
|
#54 = CLOSED_SHELL('NONE', (#45, #49, #53));
|
||||||
#54 = MANIFOLD_SOLID_BREP('NONE', #53);
|
#55 = MANIFOLD_SOLID_BREP('NONE', #54);
|
||||||
#55 = APPLICATION_CONTEXT('configuration controlled 3D design of mechanical parts and assemblies');
|
#56 = APPLICATION_CONTEXT('configuration controlled 3D design of mechanical parts and assemblies');
|
||||||
#56 = PRODUCT_DEFINITION_CONTEXT('part definition', #55, 'design');
|
#57 = PRODUCT_DEFINITION_CONTEXT('part definition', #56, 'design');
|
||||||
#57 = PRODUCT('UNIDENTIFIED_PRODUCT', 'NONE', $, ());
|
#58 = PRODUCT('UNIDENTIFIED_PRODUCT', 'NONE', $, ());
|
||||||
#58 = PRODUCT_DEFINITION_FORMATION('', $, #57);
|
#59 = PRODUCT_DEFINITION_FORMATION('', $, #58);
|
||||||
#59 = PRODUCT_DEFINITION('design', $, #58, #56);
|
#60 = PRODUCT_DEFINITION('design', $, #59, #57);
|
||||||
#60 = PRODUCT_DEFINITION_SHAPE('NONE', $, #59);
|
#61 = PRODUCT_DEFINITION_SHAPE('NONE', $, #60);
|
||||||
#61 = ADVANCED_BREP_SHAPE_REPRESENTATION('NONE', (#54), #3);
|
#62 = ADVANCED_BREP_SHAPE_REPRESENTATION('NONE', (#55), #4);
|
||||||
#62 = SHAPE_DEFINITION_REPRESENTATION(#60, #61);
|
#63 = SHAPE_DEFINITION_REPRESENTATION(#61, #62);
|
||||||
ENDSEC;
|
ENDSEC;
|
||||||
END-ISO-10303-21;
|
END-ISO-10303-21;
|
||||||
|
@ -101,7 +101,7 @@ pub trait CoreDump: Clone {
|
|||||||
.meta()
|
.meta()
|
||||||
.create_debug_uploads(vec![kittycad::types::multipart::Attachment {
|
.create_debug_uploads(vec![kittycad::types::multipart::Attachment {
|
||||||
name: "".to_string(),
|
name: "".to_string(),
|
||||||
filepath: Some(format!(r#"modeling-app/coredump-{}.json"#, coredump_id).into()),
|
filepath: Some(format!(r#"modeling-app/coredump-{coredump_id}.json"#).into()),
|
||||||
content_type: Some("application/json".to_string()),
|
content_type: Some("application/json".to_string()),
|
||||||
data,
|
data,
|
||||||
}])
|
}])
|
||||||
|
@ -189,7 +189,7 @@ fn generate_example(index: usize, src: &str, props: &ExampleProperties, file_nam
|
|||||||
index
|
index
|
||||||
);
|
);
|
||||||
let image_data =
|
let image_data =
|
||||||
std::fs::read(&image_path).unwrap_or_else(|_| panic!("Failed to read image file: {}", image_path));
|
std::fs::read(&image_path).unwrap_or_else(|_| panic!("Failed to read image file: {image_path}"));
|
||||||
base64::engine::general_purpose::STANDARD.encode(&image_data)
|
base64::engine::general_purpose::STANDARD.encode(&image_data)
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -225,7 +225,7 @@ fn generate_type_from_kcl(ty: &TyData, file_name: String, example_name: String,
|
|||||||
|
|
||||||
let output = hbs.render("kclType", &data)?;
|
let output = hbs.render("kclType", &data)?;
|
||||||
let output = cleanup_types(&output, kcl_std);
|
let output = cleanup_types(&output, kcl_std);
|
||||||
expectorate::assert_contents(format!("../../docs/kcl-std/{}.md", file_name), &output);
|
expectorate::assert_contents(format!("../../docs/kcl-std/{file_name}.md"), &output);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -267,7 +267,7 @@ fn generate_mod_from_kcl(m: &ModData, file_name: String) -> Result<()> {
|
|||||||
});
|
});
|
||||||
|
|
||||||
let output = hbs.render("module", &data)?;
|
let output = hbs.render("module", &data)?;
|
||||||
expectorate::assert_contents(format!("../../docs/kcl-std/{}.md", file_name), &output);
|
expectorate::assert_contents(format!("../../docs/kcl-std/{file_name}.md"), &output);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -334,7 +334,7 @@ fn generate_function_from_kcl(
|
|||||||
|
|
||||||
let output = hbs.render("function", &data)?;
|
let output = hbs.render("function", &data)?;
|
||||||
let output = &cleanup_types(&output, kcl_std);
|
let output = &cleanup_types(&output, kcl_std);
|
||||||
expectorate::assert_contents(format!("../../docs/kcl-std/{}.md", file_name), output);
|
expectorate::assert_contents(format!("../../docs/kcl-std/{file_name}.md"), output);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
@ -378,7 +378,7 @@ fn generate_const_from_kcl(cnst: &ConstData, file_name: String, example_name: St
|
|||||||
|
|
||||||
let output = hbs.render("const", &data)?;
|
let output = hbs.render("const", &data)?;
|
||||||
let output = cleanup_types(&output, kcl_std);
|
let output = cleanup_types(&output, kcl_std);
|
||||||
expectorate::assert_contents(format!("../../docs/kcl-std/{}.md", file_name), &output);
|
expectorate::assert_contents(format!("../../docs/kcl-std/{file_name}.md"), &output);
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
@ -8,6 +8,7 @@ use tower_lsp::lsp_types::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
ModuleId,
|
||||||
execution::annotations,
|
execution::annotations,
|
||||||
parsing::{
|
parsing::{
|
||||||
ast::types::{
|
ast::types::{
|
||||||
@ -15,7 +16,6 @@ use crate::{
|
|||||||
},
|
},
|
||||||
token::NumericSuffix,
|
token::NumericSuffix,
|
||||||
},
|
},
|
||||||
ModuleId,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
pub fn walk_prelude() -> ModData {
|
pub fn walk_prelude() -> ModData {
|
||||||
@ -97,7 +97,7 @@ fn visit_module(name: &str, preferred_prefix: &str, names: WalkForNames) -> Resu
|
|||||||
ImportSelector::None { .. } => {
|
ImportSelector::None { .. } => {
|
||||||
let name = import.module_name().unwrap();
|
let name = import.module_name().unwrap();
|
||||||
if names.contains(&name) {
|
if names.contains(&name) {
|
||||||
Some(visit_module(&path[1], &format!("{}::", name), WalkForNames::All)?)
|
Some(visit_module(&path[1], &format!("{name}::"), WalkForNames::All)?)
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
@ -451,7 +451,7 @@ impl ModData {
|
|||||||
let (name, qual_name, module_name) = if name == "prelude" {
|
let (name, qual_name, module_name) = if name == "prelude" {
|
||||||
("std", "std".to_owned(), String::new())
|
("std", "std".to_owned(), String::new())
|
||||||
} else {
|
} else {
|
||||||
(name, format!("std::{}", name), "std".to_owned())
|
(name, format!("std::{name}"), "std".to_owned())
|
||||||
};
|
};
|
||||||
Self {
|
Self {
|
||||||
preferred_name: format!("{preferred_prefix}{name}"),
|
preferred_name: format!("{preferred_prefix}{name}"),
|
||||||
@ -767,14 +767,12 @@ impl ArgData {
|
|||||||
for s in &arr.elements {
|
for s in &arr.elements {
|
||||||
let Expr::Literal(lit) = s else {
|
let Expr::Literal(lit) = s else {
|
||||||
panic!(
|
panic!(
|
||||||
"Invalid value in `snippetArray`, all items must be string literals but found {:?}",
|
"Invalid value in `snippetArray`, all items must be string literals but found {s:?}"
|
||||||
s
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
let LiteralValue::String(litstr) = &lit.inner.value else {
|
let LiteralValue::String(litstr) = &lit.inner.value else {
|
||||||
panic!(
|
panic!(
|
||||||
"Invalid value in `snippetArray`, all items must be string literals but found {:?}",
|
"Invalid value in `snippetArray`, all items must be string literals but found {s:?}"
|
||||||
s
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
items.push(litstr.to_owned());
|
items.push(litstr.to_owned());
|
||||||
@ -816,7 +814,7 @@ impl ArgData {
|
|||||||
}
|
}
|
||||||
match self.ty.as_deref() {
|
match self.ty.as_deref() {
|
||||||
Some("Sketch") if self.kind == ArgKind::Special => None,
|
Some("Sketch") if self.kind == ArgKind::Special => None,
|
||||||
Some(s) if s.starts_with("number") => Some((index, format!(r#"{label}${{{}:10}}"#, index))),
|
Some(s) if s.starts_with("number") => Some((index, format!(r#"{label}${{{index}:10}}"#))),
|
||||||
Some("Point2d") => Some((index + 1, format!(r#"{label}[${{{}:0}}, ${{{}:0}}]"#, index, index + 1))),
|
Some("Point2d") => Some((index + 1, format!(r#"{label}[${{{}:0}}, ${{{}:0}}]"#, index, index + 1))),
|
||||||
Some("Point3d") => Some((
|
Some("Point3d") => Some((
|
||||||
index + 2,
|
index + 2,
|
||||||
@ -831,7 +829,7 @@ impl ArgData {
|
|||||||
Some("Sketch") | Some("Sketch | Helix") => Some((index, format!(r#"{label}${{{index}:sketch000}}"#))),
|
Some("Sketch") | Some("Sketch | Helix") => Some((index, format!(r#"{label}${{{index}:sketch000}}"#))),
|
||||||
Some("Edge") => Some((index, format!(r#"{label}${{{index}:tag_or_edge_fn}}"#))),
|
Some("Edge") => Some((index, format!(r#"{label}${{{index}:tag_or_edge_fn}}"#))),
|
||||||
Some("[Edge; 1+]") => Some((index, format!(r#"{label}[${{{index}:tag_or_edge_fn}}]"#))),
|
Some("[Edge; 1+]") => Some((index, format!(r#"{label}[${{{index}:tag_or_edge_fn}}]"#))),
|
||||||
Some("Plane") | Some("Solid | Plane") => Some((index, format!(r#"{label}${{{}:XY}}"#, index))),
|
Some("Plane") | Some("Solid | Plane") => Some((index, format!(r#"{label}${{{index}:XY}}"#))),
|
||||||
Some("[TaggedFace; 2]") => Some((
|
Some("[TaggedFace; 2]") => Some((
|
||||||
index + 1,
|
index + 1,
|
||||||
format!(r#"{label}[${{{}:tag}}, ${{{}:tag}}]"#, index, index + 1),
|
format!(r#"{label}[${{{}:tag}}, ${{{}:tag}}]"#, index, index + 1),
|
||||||
@ -841,10 +839,10 @@ impl ArgData {
|
|||||||
if self.name == "color" {
|
if self.name == "color" {
|
||||||
Some((index, format!(r"{label}${{{}:{}}}", index, "\"#ff0000\"")))
|
Some((index, format!(r"{label}${{{}:{}}}", index, "\"#ff0000\"")))
|
||||||
} else {
|
} else {
|
||||||
Some((index, format!(r#"{label}${{{}:"string"}}"#, index)))
|
Some((index, format!(r#"{label}${{{index}:"string"}}"#)))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Some("bool") => Some((index, format!(r#"{label}${{{}:false}}"#, index))),
|
Some("bool") => Some((index, format!(r#"{label}${{{index}:false}}"#))),
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1298,7 +1296,10 @@ mod test {
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
let name = format!("{}-{i}", f.qual_name.replace("::", "-"));
|
let name = format!("{}-{i}", f.qual_name.replace("::", "-"));
|
||||||
assert!(TEST_NAMES.contains(&&*name), "Missing test for example \"{name}\", maybe need to update kcl-derive-docs/src/example_tests.rs?")
|
assert!(
|
||||||
|
TEST_NAMES.contains(&&*name),
|
||||||
|
"Missing test for example \"{name}\", maybe need to update kcl-derive-docs/src/example_tests.rs?"
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1334,7 +1335,9 @@ mod test {
|
|||||||
};
|
};
|
||||||
|
|
||||||
let Some(DocData::Fn(d)) = data.children.get(&format!("I:{qualname}")) else {
|
let Some(DocData::Fn(d)) = data.children.get(&format!("I:{qualname}")) else {
|
||||||
panic!("Could not find data for {NAME} (missing a child entry for {qualname}), maybe need to update kcl-derive-docs/src/example_tests.rs?");
|
panic!(
|
||||||
|
"Could not find data for {NAME} (missing a child entry for {qualname}), maybe need to update kcl-derive-docs/src/example_tests.rs?"
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
for (i, eg) in d.examples.iter().enumerate() {
|
for (i, eg) in d.examples.iter().enumerate() {
|
||||||
@ -1362,6 +1365,8 @@ mod test {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
panic!("Could not find data for {NAME} (no example {number}), maybe need to update kcl-derive-docs/src/example_tests.rs?");
|
panic!(
|
||||||
|
"Could not find data for {NAME} (no example {number}), maybe need to update kcl-derive-docs/src/example_tests.rs?"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,11 +2,11 @@
|
|||||||
//! tasks.
|
//! tasks.
|
||||||
|
|
||||||
use std::sync::{
|
use std::sync::{
|
||||||
atomic::{AtomicUsize, Ordering},
|
|
||||||
Arc,
|
Arc,
|
||||||
|
atomic::{AtomicUsize, Ordering},
|
||||||
};
|
};
|
||||||
|
|
||||||
use tokio::sync::{mpsc, Notify};
|
use tokio::sync::{Notify, mpsc};
|
||||||
|
|
||||||
use crate::errors::KclError;
|
use crate::errors::KclError;
|
||||||
|
|
||||||
|
@ -3,26 +3,26 @@
|
|||||||
|
|
||||||
use std::{collections::HashMap, sync::Arc};
|
use std::{collections::HashMap, sync::Arc};
|
||||||
|
|
||||||
use anyhow::{anyhow, Result};
|
use anyhow::{Result, anyhow};
|
||||||
use futures::{SinkExt, StreamExt};
|
use futures::{SinkExt, StreamExt};
|
||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
use kcmc::{
|
use kcmc::{
|
||||||
|
ModelingCmd,
|
||||||
websocket::{
|
websocket::{
|
||||||
BatchResponse, FailureWebSocketResponse, ModelingCmdReq, ModelingSessionData, OkWebSocketResponseData,
|
BatchResponse, FailureWebSocketResponse, ModelingCmdReq, ModelingSessionData, OkWebSocketResponseData,
|
||||||
SuccessWebSocketResponse, WebSocketRequest, WebSocketResponse,
|
SuccessWebSocketResponse, WebSocketRequest, WebSocketResponse,
|
||||||
},
|
},
|
||||||
ModelingCmd,
|
|
||||||
};
|
};
|
||||||
use kittycad_modeling_cmds::{self as kcmc};
|
use kittycad_modeling_cmds::{self as kcmc};
|
||||||
use tokio::sync::{mpsc, oneshot, RwLock};
|
use tokio::sync::{RwLock, mpsc, oneshot};
|
||||||
use tokio_tungstenite::tungstenite::Message as WsMsg;
|
use tokio_tungstenite::tungstenite::Message as WsMsg;
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
SourceRange,
|
||||||
engine::{AsyncTasks, EngineManager, EngineStats},
|
engine::{AsyncTasks, EngineManager, EngineStats},
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::{DefaultPlanes, IdGenerator},
|
execution::{DefaultPlanes, IdGenerator},
|
||||||
SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, PartialEq)]
|
#[derive(Debug, PartialEq)]
|
||||||
@ -85,7 +85,7 @@ impl TcpRead {
|
|||||||
let msg = match msg {
|
let msg = match msg {
|
||||||
Ok(msg) => msg,
|
Ok(msg) => msg,
|
||||||
Err(e) if matches!(e, tokio_tungstenite::tungstenite::Error::Protocol(_)) => {
|
Err(e) if matches!(e, tokio_tungstenite::tungstenite::Error::Protocol(_)) => {
|
||||||
return Err(WebSocketReadError::Read(e))
|
return Err(WebSocketReadError::Read(e));
|
||||||
}
|
}
|
||||||
Err(e) => return Err(anyhow::anyhow!("Error reading from engine's WebSocket: {e}").into()),
|
Err(e) => return Err(anyhow::anyhow!("Error reading from engine's WebSocket: {e}").into()),
|
||||||
};
|
};
|
||||||
@ -427,7 +427,7 @@ impl EngineManager for EngineConnection {
|
|||||||
request_sent: tx,
|
request_sent: tx,
|
||||||
})
|
})
|
||||||
.await
|
.await
|
||||||
.map_err(|e| KclError::new_engine(KclErrorDetails::new(format!("Failed to send debug: {}", e), vec![])))?;
|
.map_err(|e| KclError::new_engine(KclErrorDetails::new(format!("Failed to send debug: {e}"), vec![])))?;
|
||||||
|
|
||||||
let _ = rx.await;
|
let _ = rx.await;
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -463,7 +463,7 @@ impl EngineManager for EngineConnection {
|
|||||||
.await
|
.await
|
||||||
.map_err(|e| {
|
.map_err(|e| {
|
||||||
KclError::new_engine(KclErrorDetails::new(
|
KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("Failed to send modeling command: {}", e),
|
format!("Failed to send modeling command: {e}"),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
))
|
))
|
||||||
})?;
|
})?;
|
||||||
@ -533,7 +533,7 @@ impl EngineManager for EngineConnection {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Err(KclError::new_engine(KclErrorDetails::new(
|
Err(KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("Modeling command timed out `{}`", id),
|
format!("Modeling command timed out `{id}`"),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
)))
|
)))
|
||||||
}
|
}
|
||||||
|
@ -12,16 +12,16 @@ use kcmc::{
|
|||||||
WebSocketResponse,
|
WebSocketResponse,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
use kittycad_modeling_cmds::{self as kcmc, websocket::ModelingCmdReq, ImportFiles, ModelingCmd};
|
use kittycad_modeling_cmds::{self as kcmc, ImportFiles, ModelingCmd, websocket::ModelingCmdReq};
|
||||||
use tokio::sync::RwLock;
|
use tokio::sync::RwLock;
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
SourceRange,
|
||||||
engine::{AsyncTasks, EngineStats},
|
engine::{AsyncTasks, EngineStats},
|
||||||
errors::KclError,
|
errors::KclError,
|
||||||
exec::DefaultPlanes,
|
exec::DefaultPlanes,
|
||||||
execution::IdGenerator,
|
execution::IdGenerator,
|
||||||
SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
@ -11,10 +11,10 @@ use uuid::Uuid;
|
|||||||
use wasm_bindgen::prelude::*;
|
use wasm_bindgen::prelude::*;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
SourceRange,
|
||||||
engine::{AsyncTasks, EngineStats},
|
engine::{AsyncTasks, EngineStats},
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::{DefaultPlanes, IdGenerator},
|
execution::{DefaultPlanes, IdGenerator},
|
||||||
SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[wasm_bindgen(module = "/../../src/lang/std/engineConnection.ts")]
|
#[wasm_bindgen(module = "/../../src/lang/std/engineConnection.ts")]
|
||||||
|
@ -12,15 +12,15 @@ pub mod conn_wasm;
|
|||||||
use std::{
|
use std::{
|
||||||
collections::HashMap,
|
collections::HashMap,
|
||||||
sync::{
|
sync::{
|
||||||
atomic::{AtomicUsize, Ordering},
|
|
||||||
Arc,
|
Arc,
|
||||||
|
atomic::{AtomicUsize, Ordering},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
pub use async_tasks::AsyncTasks;
|
pub use async_tasks::AsyncTasks;
|
||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
use kcmc::{
|
use kcmc::{
|
||||||
each_cmd as mcmd,
|
ModelingCmd, each_cmd as mcmd,
|
||||||
length_unit::LengthUnit,
|
length_unit::LengthUnit,
|
||||||
ok_response::OkModelingCmdResponse,
|
ok_response::OkModelingCmdResponse,
|
||||||
shared::Color,
|
shared::Color,
|
||||||
@ -28,7 +28,6 @@ use kcmc::{
|
|||||||
BatchResponse, ModelingBatch, ModelingCmdReq, ModelingSessionData, OkWebSocketResponseData, WebSocketRequest,
|
BatchResponse, ModelingBatch, ModelingCmdReq, ModelingSessionData, OkWebSocketResponseData, WebSocketRequest,
|
||||||
WebSocketResponse,
|
WebSocketResponse,
|
||||||
},
|
},
|
||||||
ModelingCmd,
|
|
||||||
};
|
};
|
||||||
use kittycad_modeling_cmds as kcmc;
|
use kittycad_modeling_cmds as kcmc;
|
||||||
use parse_display::{Display, FromStr};
|
use parse_display::{Display, FromStr};
|
||||||
@ -39,9 +38,9 @@ use uuid::Uuid;
|
|||||||
use web_time::Instant;
|
use web_time::Instant;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
errors::{KclError, KclErrorDetails},
|
|
||||||
execution::{types::UnitLen, DefaultPlanes, IdGenerator, PlaneInfo, Point3d},
|
|
||||||
SourceRange,
|
SourceRange,
|
||||||
|
errors::{KclError, KclErrorDetails},
|
||||||
|
execution::{DefaultPlanes, IdGenerator, PlaneInfo, Point3d, types::UnitLen},
|
||||||
};
|
};
|
||||||
|
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
@ -291,7 +290,10 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
// the artifact graph won't care either if its gone since you can't select it
|
// the artifact graph won't care either if its gone since you can't select it
|
||||||
// anymore anyways.
|
// anymore anyways.
|
||||||
if let Err(err) = self.async_tasks().join_all().await {
|
if let Err(err) = self.async_tasks().join_all().await {
|
||||||
crate::log::logln!("Error waiting for async tasks (this is typically fine and just means that an edge became something else): {:?}", err);
|
crate::log::logln!(
|
||||||
|
"Error waiting for async tasks (this is typically fine and just means that an edge became something else): {:?}",
|
||||||
|
err
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Flush the batch to make sure nothing remains.
|
// Flush the batch to make sure nothing remains.
|
||||||
@ -499,7 +501,7 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
return Err(KclError::new_engine(KclErrorDetails::new(
|
return Err(KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("The request is not a modeling command: {:?}", req),
|
format!("The request is not a modeling command: {req:?}"),
|
||||||
vec![*range],
|
vec![*range],
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
@ -529,7 +531,7 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
} else {
|
} else {
|
||||||
// We should never get here.
|
// We should never get here.
|
||||||
Err(KclError::new_engine(KclErrorDetails::new(
|
Err(KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("Failed to get batch response: {:?}", response),
|
format!("Failed to get batch response: {response:?}"),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
)))
|
)))
|
||||||
}
|
}
|
||||||
@ -544,7 +546,7 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
// an error.
|
// an error.
|
||||||
let source_range = id_to_source_range.get(cmd_id.as_ref()).cloned().ok_or_else(|| {
|
let source_range = id_to_source_range.get(cmd_id.as_ref()).cloned().ok_or_else(|| {
|
||||||
KclError::new_engine(KclErrorDetails::new(
|
KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("Failed to get source range for command ID: {:?}", cmd_id),
|
format!("Failed to get source range for command ID: {cmd_id:?}"),
|
||||||
vec![],
|
vec![],
|
||||||
))
|
))
|
||||||
})?;
|
})?;
|
||||||
@ -554,7 +556,7 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
self.parse_websocket_response(ws_resp, source_range)
|
self.parse_websocket_response(ws_resp, source_range)
|
||||||
}
|
}
|
||||||
_ => Err(KclError::new_engine(KclErrorDetails::new(
|
_ => Err(KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("The final request is not a modeling command: {:?}", final_req),
|
format!("The final request is not a modeling command: {final_req:?}"),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
))),
|
))),
|
||||||
}
|
}
|
||||||
@ -663,7 +665,7 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
let info = DEFAULT_PLANE_INFO.get(&name).ok_or_else(|| {
|
let info = DEFAULT_PLANE_INFO.get(&name).ok_or_else(|| {
|
||||||
// We should never get here.
|
// We should never get here.
|
||||||
KclError::new_engine(KclErrorDetails::new(
|
KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("Failed to get default plane info for: {:?}", name),
|
format!("Failed to get default plane info for: {name:?}"),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
))
|
))
|
||||||
})?;
|
})?;
|
||||||
@ -739,7 +741,7 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
// Get the source range for the command.
|
// Get the source range for the command.
|
||||||
let source_range = id_to_source_range.get(cmd_id).cloned().ok_or_else(|| {
|
let source_range = id_to_source_range.get(cmd_id).cloned().ok_or_else(|| {
|
||||||
KclError::new_engine(KclErrorDetails::new(
|
KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("Failed to get source range for command ID: {:?}", cmd_id),
|
format!("Failed to get source range for command ID: {cmd_id:?}"),
|
||||||
vec![],
|
vec![],
|
||||||
))
|
))
|
||||||
})?;
|
})?;
|
||||||
@ -754,7 +756,7 @@ pub trait EngineManager: std::fmt::Debug + Send + Sync + 'static {
|
|||||||
// Return an error that we did not get an error or the response we wanted.
|
// Return an error that we did not get an error or the response we wanted.
|
||||||
// This should never happen but who knows.
|
// This should never happen but who knows.
|
||||||
Err(KclError::new_engine(KclErrorDetails::new(
|
Err(KclError::new_engine(KclErrorDetails::new(
|
||||||
format!("Failed to find response for command ID: {:?}", id),
|
format!("Failed to find response for command ID: {id:?}"),
|
||||||
vec![],
|
vec![],
|
||||||
)))
|
)))
|
||||||
}
|
}
|
||||||
|
@ -7,11 +7,11 @@ use tower_lsp::lsp_types::{Diagnostic, DiagnosticSeverity};
|
|||||||
#[cfg(feature = "artifact-graph")]
|
#[cfg(feature = "artifact-graph")]
|
||||||
use crate::execution::{ArtifactCommand, ArtifactGraph, Operation};
|
use crate::execution::{ArtifactCommand, ArtifactGraph, Operation};
|
||||||
use crate::{
|
use crate::{
|
||||||
|
ModuleId,
|
||||||
execution::DefaultPlanes,
|
execution::DefaultPlanes,
|
||||||
lsp::IntoDiagnostic,
|
lsp::IntoDiagnostic,
|
||||||
modules::{ModulePath, ModuleSource},
|
modules::{ModulePath, ModuleSource},
|
||||||
source_range::SourceRange,
|
source_range::SourceRange,
|
||||||
ModuleId,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// How did the KCL execution fail
|
/// How did the KCL execution fail
|
||||||
|
@ -2,13 +2,13 @@
|
|||||||
|
|
||||||
use std::str::FromStr;
|
use std::str::FromStr;
|
||||||
|
|
||||||
use kittycad_modeling_cmds::coord::{System, KITTYCAD, OPENGL, VULKAN};
|
use kittycad_modeling_cmds::coord::{KITTYCAD, OPENGL, System, VULKAN};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
KclError, SourceRange,
|
||||||
errors::KclErrorDetails,
|
errors::KclErrorDetails,
|
||||||
execution::types::{UnitAngle, UnitLen},
|
execution::types::{UnitAngle, UnitLen},
|
||||||
parsing::ast::types::{Annotation, Expr, LiteralValue, Node, ObjectProperty},
|
parsing::ast::types::{Annotation, Expr, LiteralValue, Node, ObjectProperty},
|
||||||
KclError, SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Annotations which should cause re-execution if they change.
|
/// Annotations which should cause re-execution if they change.
|
||||||
|
@ -1,20 +1,19 @@
|
|||||||
use fnv::FnvHashMap;
|
use fnv::FnvHashMap;
|
||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
use kittycad_modeling_cmds::{
|
use kittycad_modeling_cmds::{
|
||||||
self as kcmc,
|
self as kcmc, EnableSketchMode, ModelingCmd,
|
||||||
ok_response::OkModelingCmdResponse,
|
ok_response::OkModelingCmdResponse,
|
||||||
shared::ExtrusionFaceCapType,
|
shared::ExtrusionFaceCapType,
|
||||||
websocket::{BatchResponse, OkWebSocketResponseData, WebSocketResponse},
|
websocket::{BatchResponse, OkWebSocketResponseData, WebSocketResponse},
|
||||||
EnableSketchMode, ModelingCmd,
|
|
||||||
};
|
};
|
||||||
use serde::{ser::SerializeSeq, Serialize};
|
use serde::{Serialize, ser::SerializeSeq};
|
||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
KclError, NodePath, SourceRange,
|
||||||
errors::KclErrorDetails,
|
errors::KclErrorDetails,
|
||||||
execution::ArtifactId,
|
execution::ArtifactId,
|
||||||
parsing::ast::types::{Node, Program},
|
parsing::ast::types::{Node, Program},
|
||||||
KclError, NodePath, SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
@ -893,7 +892,10 @@ fn artifacts_to_update(
|
|||||||
),
|
),
|
||||||
};
|
};
|
||||||
if original_path_ids.len() != face_edge_infos.len() {
|
if original_path_ids.len() != face_edge_infos.len() {
|
||||||
internal_error!(range, "EntityMirror or EntityMirrorAcrossEdge response has different number face edge info than original mirrored paths: id={id:?}, cmd={cmd:?}, response={response:?}");
|
internal_error!(
|
||||||
|
range,
|
||||||
|
"EntityMirror or EntityMirrorAcrossEdge response has different number face edge info than original mirrored paths: id={id:?}, cmd={cmd:?}, response={response:?}"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
let mut return_arr = Vec::new();
|
let mut return_arr = Vec::new();
|
||||||
for (face_edge_info, original_path_id) in face_edge_infos.iter().zip(original_path_ids) {
|
for (face_edge_info, original_path_id) in face_edge_infos.iter().zip(original_path_ids) {
|
||||||
@ -909,7 +911,10 @@ fn artifacts_to_update(
|
|||||||
// of its info.
|
// of its info.
|
||||||
let Some(Artifact::Path(original_path)) = artifacts.get(&original_path_id) else {
|
let Some(Artifact::Path(original_path)) = artifacts.get(&original_path_id) else {
|
||||||
// We couldn't find the original path. This is a bug.
|
// We couldn't find the original path. This is a bug.
|
||||||
internal_error!(range, "Couldn't find original path for mirror2d: original_path_id={original_path_id:?}, cmd={cmd:?}");
|
internal_error!(
|
||||||
|
range,
|
||||||
|
"Couldn't find original path for mirror2d: original_path_id={original_path_id:?}, cmd={cmd:?}"
|
||||||
|
);
|
||||||
};
|
};
|
||||||
Path {
|
Path {
|
||||||
id: path_id,
|
id: path_id,
|
||||||
|
@ -268,7 +268,7 @@ impl ArtifactGraph {
|
|||||||
for (group_id, artifact_ids) in groups {
|
for (group_id, artifact_ids) in groups {
|
||||||
let group_id = *stable_id_map.get(&group_id).unwrap();
|
let group_id = *stable_id_map.get(&group_id).unwrap();
|
||||||
writeln!(output, "{prefix}subgraph path{group_id} [Path]")?;
|
writeln!(output, "{prefix}subgraph path{group_id} [Path]")?;
|
||||||
let indented = format!("{} ", prefix);
|
let indented = format!("{prefix} ");
|
||||||
for artifact_id in artifact_ids {
|
for artifact_id in artifact_ids {
|
||||||
let artifact = self.map.get(&artifact_id).unwrap();
|
let artifact = self.map.get(&artifact_id).unwrap();
|
||||||
let id = *stable_id_map.get(&artifact_id).unwrap();
|
let id = *stable_id_map.get(&artifact_id).unwrap();
|
||||||
@ -353,7 +353,7 @@ impl ArtifactGraph {
|
|||||||
node_path_display(output, prefix, None, &segment.code_ref)?;
|
node_path_display(output, prefix, None, &segment.code_ref)?;
|
||||||
}
|
}
|
||||||
Artifact::Solid2d(_solid2d) => {
|
Artifact::Solid2d(_solid2d) => {
|
||||||
writeln!(output, "{prefix}{}[Solid2d]", id)?;
|
writeln!(output, "{prefix}{id}[Solid2d]")?;
|
||||||
}
|
}
|
||||||
Artifact::StartSketchOnFace(StartSketchOnFace { code_ref, .. }) => {
|
Artifact::StartSketchOnFace(StartSketchOnFace { code_ref, .. }) => {
|
||||||
writeln!(
|
writeln!(
|
||||||
@ -494,24 +494,24 @@ impl ArtifactGraph {
|
|||||||
match edge.flow {
|
match edge.flow {
|
||||||
EdgeFlow::SourceToTarget => match edge.direction {
|
EdgeFlow::SourceToTarget => match edge.direction {
|
||||||
EdgeDirection::Forward => {
|
EdgeDirection::Forward => {
|
||||||
writeln!(output, "{prefix}{source_id} x{}--> {}", extra, target_id)?;
|
writeln!(output, "{prefix}{source_id} x{extra}--> {target_id}")?;
|
||||||
}
|
}
|
||||||
EdgeDirection::Backward => {
|
EdgeDirection::Backward => {
|
||||||
writeln!(output, "{prefix}{source_id} <{}--x {}", extra, target_id)?;
|
writeln!(output, "{prefix}{source_id} <{extra}--x {target_id}")?;
|
||||||
}
|
}
|
||||||
EdgeDirection::Bidirectional => {
|
EdgeDirection::Bidirectional => {
|
||||||
writeln!(output, "{prefix}{source_id} {}--- {}", extra, target_id)?;
|
writeln!(output, "{prefix}{source_id} {extra}--- {target_id}")?;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
EdgeFlow::TargetToSource => match edge.direction {
|
EdgeFlow::TargetToSource => match edge.direction {
|
||||||
EdgeDirection::Forward => {
|
EdgeDirection::Forward => {
|
||||||
writeln!(output, "{prefix}{target_id} x{}--> {}", extra, source_id)?;
|
writeln!(output, "{prefix}{target_id} x{extra}--> {source_id}")?;
|
||||||
}
|
}
|
||||||
EdgeDirection::Backward => {
|
EdgeDirection::Backward => {
|
||||||
writeln!(output, "{prefix}{target_id} <{}--x {}", extra, source_id)?;
|
writeln!(output, "{prefix}{target_id} <{extra}--x {source_id}")?;
|
||||||
}
|
}
|
||||||
EdgeDirection::Bidirectional => {
|
EdgeDirection::Bidirectional => {
|
||||||
writeln!(output, "{prefix}{target_id} {}--- {}", extra, source_id)?;
|
writeln!(output, "{prefix}{target_id} {extra}--- {source_id}")?;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -6,15 +6,14 @@ use itertools::{EitherOrBoth, Itertools};
|
|||||||
use tokio::sync::RwLock;
|
use tokio::sync::RwLock;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
ExecOutcome, ExecutorContext,
|
||||||
execution::{
|
execution::{
|
||||||
annotations,
|
EnvironmentRef, ExecutorSettings, annotations,
|
||||||
memory::Stack,
|
memory::Stack,
|
||||||
state::{self as exec_state, ModuleInfoMap},
|
state::{self as exec_state, ModuleInfoMap},
|
||||||
EnvironmentRef, ExecutorSettings,
|
|
||||||
},
|
},
|
||||||
parsing::ast::types::{Annotation, Node, Program},
|
parsing::ast::types::{Annotation, Node, Program},
|
||||||
walk::Node as WalkNode,
|
walk::Node as WalkNode,
|
||||||
ExecOutcome, ExecutorContext,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
lazy_static::lazy_static! {
|
lazy_static::lazy_static! {
|
||||||
@ -337,7 +336,7 @@ mod tests {
|
|||||||
use pretty_assertions::assert_eq;
|
use pretty_assertions::assert_eq;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::execution::{parse_execute, parse_execute_with_project_dir, ExecTestResults};
|
use crate::execution::{ExecTestResults, parse_execute, parse_execute_with_project_dir};
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
async fn test_get_changed_program_same_code() {
|
async fn test_get_changed_program_same_code() {
|
||||||
@ -755,7 +754,7 @@ extrude(profile001, length = 100)"#
|
|||||||
.await;
|
.await;
|
||||||
|
|
||||||
let CacheResult::CheckImportsOnly { reapply_settings, .. } = result else {
|
let CacheResult::CheckImportsOnly { reapply_settings, .. } = result else {
|
||||||
panic!("Expected CheckImportsOnly, got {:?}", result);
|
panic!("Expected CheckImportsOnly, got {result:?}");
|
||||||
};
|
};
|
||||||
|
|
||||||
assert_eq!(reapply_settings, false);
|
assert_eq!(reapply_settings, false);
|
||||||
@ -839,7 +838,7 @@ extrude(profile001, length = 100)
|
|||||||
.await;
|
.await;
|
||||||
|
|
||||||
let CacheResult::CheckImportsOnly { reapply_settings, .. } = result else {
|
let CacheResult::CheckImportsOnly { reapply_settings, .. } = result else {
|
||||||
panic!("Expected CheckImportsOnly, got {:?}", result);
|
panic!("Expected CheckImportsOnly, got {result:?}");
|
||||||
};
|
};
|
||||||
|
|
||||||
assert_eq!(reapply_settings, false);
|
assert_eq!(reapply_settings, false);
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use super::{types::NumericType, ArtifactId, KclValue};
|
use super::{ArtifactId, KclValue, types::NumericType};
|
||||||
#[cfg(feature = "artifact-graph")]
|
#[cfg(feature = "artifact-graph")]
|
||||||
use crate::parsing::ast::types::{Node, Program};
|
use crate::parsing::ast::types::{Node, Program};
|
||||||
use crate::{parsing::ast::types::ItemVisibility, ModuleId, NodePath, SourceRange};
|
use crate::{ModuleId, NodePath, SourceRange, parsing::ast::types::ItemVisibility};
|
||||||
|
|
||||||
/// A CAD modeling operation for display in the feature tree, AKA operations
|
/// A CAD modeling operation for display in the feature tree, AKA operations
|
||||||
/// timeline.
|
/// timeline.
|
||||||
@ -57,7 +57,7 @@ impl Operation {
|
|||||||
/// If the variant is `StdLibCall`, set the `is_error` field.
|
/// If the variant is `StdLibCall`, set the `is_error` field.
|
||||||
pub(crate) fn set_std_lib_call_is_error(&mut self, is_err: bool) {
|
pub(crate) fn set_std_lib_call_is_error(&mut self, is_err: bool) {
|
||||||
match self {
|
match self {
|
||||||
Self::StdLibCall { ref mut is_error, .. } => *is_error = is_err,
|
Self::StdLibCall { is_error, .. } => *is_error = is_err,
|
||||||
Self::VariableDeclaration { .. } | Self::GroupBegin { .. } | Self::GroupEnd => {}
|
Self::VariableDeclaration { .. } | Self::GroupBegin { .. } | Self::GroupEnd => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -226,10 +226,7 @@ impl From<&KclValue> for OpKclValue {
|
|||||||
match value {
|
match value {
|
||||||
KclValue::Uuid { value, .. } => Self::Uuid { value: *value },
|
KclValue::Uuid { value, .. } => Self::Uuid { value: *value },
|
||||||
KclValue::Bool { value, .. } => Self::Bool { value: *value },
|
KclValue::Bool { value, .. } => Self::Bool { value: *value },
|
||||||
KclValue::Number { value, ty, .. } => Self::Number {
|
KclValue::Number { value, ty, .. } => Self::Number { value: *value, ty: *ty },
|
||||||
value: *value,
|
|
||||||
ty: ty.clone(),
|
|
||||||
},
|
|
||||||
KclValue::String { value, .. } => Self::String { value: value.clone() },
|
KclValue::String { value, .. } => Self::String { value: value.clone() },
|
||||||
KclValue::Tuple { value, .. } | KclValue::HomArray { value, .. } => {
|
KclValue::Tuple { value, .. } | KclValue::HomArray { value, .. } => {
|
||||||
let value = value.iter().map(Self::from).collect();
|
let value = value.iter().map(Self::from).collect();
|
||||||
|
@ -3,17 +3,17 @@ use std::collections::HashMap;
|
|||||||
use async_recursion::async_recursion;
|
use async_recursion::async_recursion;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
CompilationError, NodePath,
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::{
|
execution::{
|
||||||
annotations,
|
BodyType, EnvironmentRef, ExecState, ExecutorContext, KclValue, Metadata, ModelingCmdMeta, ModuleArtifactState,
|
||||||
|
Operation, PlaneType, StatementKind, TagIdentifier, annotations,
|
||||||
cad_op::OpKclValue,
|
cad_op::OpKclValue,
|
||||||
fn_call::Args,
|
fn_call::Args,
|
||||||
kcl_value::{FunctionSource, TypeDef},
|
kcl_value::{FunctionSource, TypeDef},
|
||||||
memory,
|
memory,
|
||||||
state::ModuleState,
|
state::ModuleState,
|
||||||
types::{NumericType, PrimitiveType, RuntimeType},
|
types::{NumericType, PrimitiveType, RuntimeType},
|
||||||
BodyType, EnvironmentRef, ExecState, ExecutorContext, KclValue, Metadata, ModelingCmdMeta, ModuleArtifactState,
|
|
||||||
Operation, PlaneType, StatementKind, TagIdentifier,
|
|
||||||
},
|
},
|
||||||
fmt,
|
fmt,
|
||||||
modules::{ModuleId, ModulePath, ModuleRepr},
|
modules::{ModuleId, ModulePath, ModuleRepr},
|
||||||
@ -28,7 +28,6 @@ use crate::{
|
|||||||
},
|
},
|
||||||
source_range::SourceRange,
|
source_range::SourceRange,
|
||||||
std::args::TyF64,
|
std::args::TyF64,
|
||||||
CompilationError, NodePath,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
impl<'a> StatementKind<'a> {
|
impl<'a> StatementKind<'a> {
|
||||||
@ -198,19 +197,23 @@ impl ExecutorContext {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ty.is_ok() && !module_exports.contains(&ty_name) {
|
if ty.is_ok() && !module_exports.contains(&ty_name) {
|
||||||
ty = Err(KclError::new_semantic(KclErrorDetails::new(format!(
|
ty = Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
"Cannot import \"{}\" from module because it is not exported. Add \"export\" before the definition to export it.",
|
format!(
|
||||||
import_item.name.name
|
"Cannot import \"{}\" from module because it is not exported. Add \"export\" before the definition to export it.",
|
||||||
),
|
import_item.name.name
|
||||||
vec![SourceRange::from(&import_item.name)],)));
|
),
|
||||||
|
vec![SourceRange::from(&import_item.name)],
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
if mod_value.is_ok() && !module_exports.contains(&mod_name) {
|
if mod_value.is_ok() && !module_exports.contains(&mod_name) {
|
||||||
mod_value = Err(KclError::new_semantic(KclErrorDetails::new(format!(
|
mod_value = Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
"Cannot import \"{}\" from module because it is not exported. Add \"export\" before the definition to export it.",
|
format!(
|
||||||
import_item.name.name
|
"Cannot import \"{}\" from module because it is not exported. Add \"export\" before the definition to export it.",
|
||||||
),
|
import_item.name.name
|
||||||
vec![SourceRange::from(&import_item.name)],)));
|
),
|
||||||
|
vec![SourceRange::from(&import_item.name)],
|
||||||
|
)));
|
||||||
}
|
}
|
||||||
|
|
||||||
if value.is_err() && ty.is_err() && mod_value.is_err() {
|
if value.is_err() && ty.is_err() && mod_value.is_err() {
|
||||||
@ -270,7 +273,7 @@ impl ExecutorContext {
|
|||||||
.get_from(name, env_ref, source_range, 0)
|
.get_from(name, env_ref, source_range, 0)
|
||||||
.map_err(|_err| {
|
.map_err(|_err| {
|
||||||
KclError::new_internal(KclErrorDetails::new(
|
KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("{} is not defined in module (but was exported?)", name),
|
format!("{name} is not defined in module (but was exported?)"),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
))
|
))
|
||||||
})?
|
})?
|
||||||
@ -431,7 +434,7 @@ impl ExecutorContext {
|
|||||||
return Err(KclError::new_semantic(KclErrorDetails::new(
|
return Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
"User-defined types are not yet supported.".to_owned(),
|
"User-defined types are not yet supported.".to_owned(),
|
||||||
vec![metadata.source_range],
|
vec![metadata.source_range],
|
||||||
)))
|
)));
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
@ -792,11 +795,12 @@ fn var_in_own_ref_err(e: KclError, being_declared: &Option<String>) -> KclError
|
|||||||
// TODO after June 26th: replace this with a let-chain,
|
// TODO after June 26th: replace this with a let-chain,
|
||||||
// which will be available in Rust 1.88
|
// which will be available in Rust 1.88
|
||||||
// https://rust-lang.github.io/rfcs/2497-if-let-chains.html
|
// https://rust-lang.github.io/rfcs/2497-if-let-chains.html
|
||||||
match (&being_declared, &name) {
|
if let (Some(name0), Some(name1)) = (&being_declared, &name)
|
||||||
(Some(name0), Some(name1)) if name0 == name1 => {
|
&& name0 == name1
|
||||||
details.message = format!("You can't use `{name0}` because you're currently trying to define it. Use a different variable here instead.");
|
{
|
||||||
}
|
details.message = format!(
|
||||||
_ => {}
|
"You can't use `{name0}` because you're currently trying to define it. Use a different variable here instead."
|
||||||
|
);
|
||||||
}
|
}
|
||||||
KclError::UndefinedValue { details, name }
|
KclError::UndefinedValue { details, name }
|
||||||
}
|
}
|
||||||
@ -860,6 +864,9 @@ impl BinaryPart {
|
|||||||
BinaryPart::CallExpressionKw(call_expression) => call_expression.execute(exec_state, ctx).await,
|
BinaryPart::CallExpressionKw(call_expression) => call_expression.execute(exec_state, ctx).await,
|
||||||
BinaryPart::UnaryExpression(unary_expression) => unary_expression.get_result(exec_state, ctx).await,
|
BinaryPart::UnaryExpression(unary_expression) => unary_expression.get_result(exec_state, ctx).await,
|
||||||
BinaryPart::MemberExpression(member_expression) => member_expression.get_result(exec_state, ctx).await,
|
BinaryPart::MemberExpression(member_expression) => member_expression.get_result(exec_state, ctx).await,
|
||||||
|
BinaryPart::ArrayExpression(e) => e.execute(exec_state, ctx).await,
|
||||||
|
BinaryPart::ArrayRangeExpression(e) => e.execute(exec_state, ctx).await,
|
||||||
|
BinaryPart::ObjectExpression(e) => e.execute(exec_state, ctx).await,
|
||||||
BinaryPart::IfExpression(e) => e.get_result(exec_state, ctx).await,
|
BinaryPart::IfExpression(e) => e.get_result(exec_state, ctx).await,
|
||||||
BinaryPart::AscribedExpression(e) => e.get_result(exec_state, ctx).await,
|
BinaryPart::AscribedExpression(e) => e.get_result(exec_state, ctx).await,
|
||||||
}
|
}
|
||||||
@ -987,6 +994,39 @@ impl Node<MemberExpression> {
|
|||||||
|
|
||||||
// Check the property and object match -- e.g. ints for arrays, strs for objects.
|
// Check the property and object match -- e.g. ints for arrays, strs for objects.
|
||||||
match (object, property, self.computed) {
|
match (object, property, self.computed) {
|
||||||
|
(KclValue::Plane { value: plane }, Property::String(property), false) => match property.as_str() {
|
||||||
|
"yAxis" => {
|
||||||
|
let (p, u) = plane.info.y_axis.as_3_dims();
|
||||||
|
Ok(KclValue::array_from_point3d(
|
||||||
|
p,
|
||||||
|
NumericType::Known(crate::exec::UnitType::Length(u)),
|
||||||
|
vec![meta],
|
||||||
|
))
|
||||||
|
}
|
||||||
|
"xAxis" => {
|
||||||
|
let (p, u) = plane.info.x_axis.as_3_dims();
|
||||||
|
Ok(KclValue::array_from_point3d(
|
||||||
|
p,
|
||||||
|
NumericType::Known(crate::exec::UnitType::Length(u)),
|
||||||
|
vec![meta],
|
||||||
|
))
|
||||||
|
}
|
||||||
|
"origin" => {
|
||||||
|
let (p, u) = plane.info.origin.as_3_dims();
|
||||||
|
Ok(KclValue::array_from_point3d(
|
||||||
|
p,
|
||||||
|
NumericType::Known(crate::exec::UnitType::Length(u)),
|
||||||
|
vec![meta],
|
||||||
|
))
|
||||||
|
}
|
||||||
|
other => Err(KclError::new_undefined_value(
|
||||||
|
KclErrorDetails::new(
|
||||||
|
format!("Property '{other}' not found in plane"),
|
||||||
|
vec![self.clone().into()],
|
||||||
|
),
|
||||||
|
None,
|
||||||
|
)),
|
||||||
|
},
|
||||||
(KclValue::Object { value: map, meta: _ }, Property::String(property), false) => {
|
(KclValue::Object { value: map, meta: _ }, Property::String(property), false) => {
|
||||||
if let Some(value) = map.get(&property) {
|
if let Some(value) = map.get(&property) {
|
||||||
Ok(value.to_owned())
|
Ok(value.to_owned())
|
||||||
@ -1006,7 +1046,22 @@ impl Node<MemberExpression> {
|
|||||||
vec![self.clone().into()],
|
vec![self.clone().into()],
|
||||||
)))
|
)))
|
||||||
}
|
}
|
||||||
(KclValue::Object { .. }, p, _) => {
|
(KclValue::Object { value: map, .. }, p @ Property::UInt(i), _) => {
|
||||||
|
if i == 0
|
||||||
|
&& let Some(value) = map.get("x")
|
||||||
|
{
|
||||||
|
return Ok(value.to_owned());
|
||||||
|
}
|
||||||
|
if i == 1
|
||||||
|
&& let Some(value) = map.get("y")
|
||||||
|
{
|
||||||
|
return Ok(value.to_owned());
|
||||||
|
}
|
||||||
|
if i == 2
|
||||||
|
&& let Some(value) = map.get("z")
|
||||||
|
{
|
||||||
|
return Ok(value.to_owned());
|
||||||
|
}
|
||||||
let t = p.type_name();
|
let t = p.type_name();
|
||||||
let article = article_for(t);
|
let article = article_for(t);
|
||||||
Err(KclError::new_semantic(KclErrorDetails::new(
|
Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
@ -1042,6 +1097,16 @@ impl Node<MemberExpression> {
|
|||||||
(KclValue::Solid { value }, Property::String(prop), false) if prop == "sketch" => Ok(KclValue::Sketch {
|
(KclValue::Solid { value }, Property::String(prop), false) if prop == "sketch" => Ok(KclValue::Sketch {
|
||||||
value: Box::new(value.sketch),
|
value: Box::new(value.sketch),
|
||||||
}),
|
}),
|
||||||
|
(geometry @ KclValue::Solid { .. }, Property::String(prop), false) if prop == "tags" => {
|
||||||
|
// This is a common mistake.
|
||||||
|
Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
|
format!(
|
||||||
|
"Property `{prop}` not found on {}. You can get a solid's tags through its sketch, as in, `exampleSolid.sketch.tags`.",
|
||||||
|
geometry.human_friendly_type()
|
||||||
|
),
|
||||||
|
vec![self.clone().into()],
|
||||||
|
)))
|
||||||
|
}
|
||||||
(KclValue::Sketch { value: sk }, Property::String(prop), false) if prop == "tags" => Ok(KclValue::Object {
|
(KclValue::Sketch { value: sk }, Property::String(prop), false) if prop == "tags" => Ok(KclValue::Object {
|
||||||
meta: vec![Metadata {
|
meta: vec![Metadata {
|
||||||
source_range: SourceRange::from(self.clone()),
|
source_range: SourceRange::from(self.clone()),
|
||||||
@ -1052,6 +1117,12 @@ impl Node<MemberExpression> {
|
|||||||
.map(|(k, tag)| (k.to_owned(), KclValue::TagIdentifier(Box::new(tag.to_owned()))))
|
.map(|(k, tag)| (k.to_owned(), KclValue::TagIdentifier(Box::new(tag.to_owned()))))
|
||||||
.collect(),
|
.collect(),
|
||||||
}),
|
}),
|
||||||
|
(geometry @ (KclValue::Sketch { .. } | KclValue::Solid { .. }), Property::String(property), false) => {
|
||||||
|
Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
|
format!("Property `{property}` not found on {}", geometry.human_friendly_type()),
|
||||||
|
vec![self.clone().into()],
|
||||||
|
)))
|
||||||
|
}
|
||||||
(being_indexed, _, _) => Err(KclError::new_semantic(KclErrorDetails::new(
|
(being_indexed, _, _) => Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
format!(
|
format!(
|
||||||
"Only arrays can be indexed, but you're trying to index {}",
|
"Only arrays can be indexed, but you're trying to index {}",
|
||||||
@ -1077,7 +1148,7 @@ impl Node<BinaryExpression> {
|
|||||||
(&left_value, &right_value)
|
(&left_value, &right_value)
|
||||||
{
|
{
|
||||||
return Ok(KclValue::String {
|
return Ok(KclValue::String {
|
||||||
value: format!("{}{}", left, right),
|
value: format!("{left}{right}"),
|
||||||
meta,
|
meta,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -1237,7 +1308,9 @@ impl Node<BinaryExpression> {
|
|||||||
exec_state.clear_units_warnings(&sr);
|
exec_state.clear_units_warnings(&sr);
|
||||||
let mut err = CompilationError::err(
|
let mut err = CompilationError::err(
|
||||||
sr,
|
sr,
|
||||||
format!("{} numbers which have unknown or incompatible units.\nYou can probably fix this error by specifying the units using type ascription, e.g., `len: number(mm)` or `(a * b): number(deg)`.", verb),
|
format!(
|
||||||
|
"{verb} numbers which have unknown or incompatible units.\nYou can probably fix this error by specifying the units using type ascription, e.g., `len: number(mm)` or `(a * b): number(deg)`."
|
||||||
|
),
|
||||||
);
|
);
|
||||||
err.tag = crate::errors::Tag::UnknownNumericUnits;
|
err.tag = crate::errors::Tag::UnknownNumericUnits;
|
||||||
exec_state.warn(err);
|
exec_state.warn(err);
|
||||||
@ -1291,7 +1364,7 @@ impl Node<UnaryExpression> {
|
|||||||
Ok(KclValue::Number {
|
Ok(KclValue::Number {
|
||||||
value: -value,
|
value: -value,
|
||||||
meta,
|
meta,
|
||||||
ty: ty.clone(),
|
ty: *ty,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
KclValue::Plane { value } => {
|
KclValue::Plane { value } => {
|
||||||
@ -1323,7 +1396,7 @@ impl Node<UnaryExpression> {
|
|||||||
.map(|v| match v {
|
.map(|v| match v {
|
||||||
KclValue::Number { value, ty, meta } => Ok(KclValue::Number {
|
KclValue::Number { value, ty, meta } => Ok(KclValue::Number {
|
||||||
value: *value * -1.0,
|
value: *value * -1.0,
|
||||||
ty: ty.clone(),
|
ty: *ty,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
}),
|
}),
|
||||||
_ => Err(err()),
|
_ => Err(err()),
|
||||||
@ -1344,7 +1417,7 @@ impl Node<UnaryExpression> {
|
|||||||
.map(|v| match v {
|
.map(|v| match v {
|
||||||
KclValue::Number { value, ty, meta } => Ok(KclValue::Number {
|
KclValue::Number { value, ty, meta } => Ok(KclValue::Number {
|
||||||
value: *value * -1.0,
|
value: *value * -1.0,
|
||||||
ty: ty.clone(),
|
ty: *ty,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
}),
|
}),
|
||||||
_ => Err(err()),
|
_ => Err(err()),
|
||||||
@ -1417,7 +1490,7 @@ async fn inner_execute_pipe_body(
|
|||||||
for expression in body {
|
for expression in body {
|
||||||
if let Expr::TagDeclarator(_) = expression {
|
if let Expr::TagDeclarator(_) = expression {
|
||||||
return Err(KclError::new_semantic(KclErrorDetails::new(
|
return Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
format!("This cannot be in a PipeExpression: {:?}", expression),
|
format!("This cannot be in a PipeExpression: {expression:?}"),
|
||||||
vec![expression.into()],
|
vec![expression.into()],
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
@ -1538,7 +1611,7 @@ impl Node<ArrayRangeExpression> {
|
|||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|num| KclValue::Number {
|
.map(|num| KclValue::Number {
|
||||||
value: num as f64,
|
value: num as f64,
|
||||||
ty: start_ty.clone(),
|
ty: start_ty,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
})
|
})
|
||||||
.collect(),
|
.collect(),
|
||||||
@ -1699,9 +1772,15 @@ fn jvalue_to_prop(value: &KclValue, property_sr: Vec<SourceRange>, name: &str) -
|
|||||||
let make_err =
|
let make_err =
|
||||||
|message: String| Err::<Property, _>(KclError::new_semantic(KclErrorDetails::new(message, property_sr)));
|
|message: String| Err::<Property, _>(KclError::new_semantic(KclErrorDetails::new(message, property_sr)));
|
||||||
match value {
|
match value {
|
||||||
n @ KclValue::Number{value: num, ty, .. } => {
|
n @ KclValue::Number { value: num, ty, .. } => {
|
||||||
if !matches!(ty, NumericType::Known(crate::exec::UnitType::Count) | NumericType::Default { .. } | NumericType::Any ) {
|
if !matches!(
|
||||||
return make_err(format!("arrays can only be indexed by non-dimensioned numbers, found {}", n.human_friendly_type()));
|
ty,
|
||||||
|
NumericType::Known(crate::exec::UnitType::Count) | NumericType::Default { .. } | NumericType::Any
|
||||||
|
) {
|
||||||
|
return make_err(format!(
|
||||||
|
"arrays can only be indexed by non-dimensioned numbers, found {}",
|
||||||
|
n.human_friendly_type()
|
||||||
|
));
|
||||||
}
|
}
|
||||||
let num = *num;
|
let num = *num;
|
||||||
if num < 0.0 {
|
if num < 0.0 {
|
||||||
@ -1711,13 +1790,15 @@ fn jvalue_to_prop(value: &KclValue, property_sr: Vec<SourceRange>, name: &str) -
|
|||||||
if let Some(nearest_int) = nearest_int {
|
if let Some(nearest_int) = nearest_int {
|
||||||
Ok(Property::UInt(nearest_int))
|
Ok(Property::UInt(nearest_int))
|
||||||
} else {
|
} else {
|
||||||
make_err(format!("'{num}' is not an integer, so you can't index an array with it"))
|
make_err(format!(
|
||||||
|
"'{num}' is not an integer, so you can't index an array with it"
|
||||||
|
))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
KclValue::String{value: x, meta:_} => Ok(Property::String(x.to_owned())),
|
KclValue::String { value: x, meta: _ } => Ok(Property::String(x.to_owned())),
|
||||||
_ => {
|
_ => make_err(format!(
|
||||||
make_err(format!("{name} is not a valid property/index, you can only use a string to get the property of an object, or an int (>= 0) to get an item in an array"))
|
"{name} is not a valid property/index, you can only use a string to get the property of an object, or an int (>= 0) to get an item in an array"
|
||||||
}
|
)),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1745,9 +1826,9 @@ mod test {
|
|||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{
|
use crate::{
|
||||||
exec::UnitType,
|
|
||||||
execution::{parse_execute, ContextType},
|
|
||||||
ExecutorSettings, UnitLen,
|
ExecutorSettings, UnitLen,
|
||||||
|
exec::UnitType,
|
||||||
|
execution::{ContextType, parse_execute},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
@ -1777,7 +1858,7 @@ arr1 = [42]: [number(cm)]
|
|||||||
.get_from("arr1", result.mem_env, SourceRange::default(), 0)
|
.get_from("arr1", result.mem_env, SourceRange::default(), 0)
|
||||||
.unwrap();
|
.unwrap();
|
||||||
if let KclValue::HomArray { value, ty } = arr1 {
|
if let KclValue::HomArray { value, ty } = arr1 {
|
||||||
assert_eq!(value.len(), 1, "Expected Vec with specific length: found {:?}", value);
|
assert_eq!(value.len(), 1, "Expected Vec with specific length: found {value:?}");
|
||||||
assert_eq!(*ty, RuntimeType::known_length(UnitLen::Cm));
|
assert_eq!(*ty, RuntimeType::known_length(UnitLen::Cm));
|
||||||
// Compare, ignoring meta.
|
// Compare, ignoring meta.
|
||||||
if let KclValue::Number { value, ty, .. } = &value[0] {
|
if let KclValue::Number { value, ty, .. } = &value[0] {
|
||||||
@ -1946,7 +2027,7 @@ d = b + c
|
|||||||
.await
|
.await
|
||||||
.map_err(|err| {
|
.map_err(|err| {
|
||||||
KclError::new_internal(KclErrorDetails::new(
|
KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("Failed to create mock engine connection: {}", err),
|
format!("Failed to create mock engine connection: {err}"),
|
||||||
vec![SourceRange::default()],
|
vec![SourceRange::default()],
|
||||||
))
|
))
|
||||||
})
|
})
|
||||||
@ -2172,4 +2253,12 @@ y = x[0mm + 1]
|
|||||||
"#;
|
"#;
|
||||||
parse_execute(ast).await.unwrap_err();
|
parse_execute(ast).await.unwrap_err();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
|
async fn getting_property_of_plane() {
|
||||||
|
// let ast = include_str!("../../tests/inputs/planestuff.kcl");
|
||||||
|
let ast = std::fs::read_to_string("tests/inputs/planestuff.kcl").unwrap();
|
||||||
|
|
||||||
|
parse_execute(&ast).await.unwrap();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,19 +2,19 @@ use async_recursion::async_recursion;
|
|||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
CompilationError, NodePath,
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::{
|
execution::{
|
||||||
|
BodyType, EnvironmentRef, ExecState, ExecutorContext, KclValue, Metadata, StatementKind, TagEngineInfo,
|
||||||
|
TagIdentifier,
|
||||||
cad_op::{Group, OpArg, OpKclValue, Operation},
|
cad_op::{Group, OpArg, OpKclValue, Operation},
|
||||||
kcl_value::FunctionSource,
|
kcl_value::FunctionSource,
|
||||||
memory,
|
memory,
|
||||||
types::RuntimeType,
|
types::RuntimeType,
|
||||||
BodyType, EnvironmentRef, ExecState, ExecutorContext, KclValue, Metadata, StatementKind, TagEngineInfo,
|
|
||||||
TagIdentifier,
|
|
||||||
},
|
},
|
||||||
parsing::ast::types::{CallExpressionKw, DefaultParamVal, FunctionExpression, Node, Program, Type},
|
parsing::ast::types::{CallExpressionKw, DefaultParamVal, FunctionExpression, Node, Program, Type},
|
||||||
source_range::SourceRange,
|
source_range::SourceRange,
|
||||||
std::StdFn,
|
std::StdFn,
|
||||||
CompilationError, NodePath,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
@ -269,7 +269,7 @@ impl Node<CallExpressionKw> {
|
|||||||
};
|
};
|
||||||
KclError::new_undefined_value(
|
KclError::new_undefined_value(
|
||||||
KclErrorDetails::new(
|
KclErrorDetails::new(
|
||||||
format!("Result of user-defined function {} is undefined", fn_name),
|
format!("Result of user-defined function {fn_name} is undefined"),
|
||||||
source_ranges,
|
source_ranges,
|
||||||
),
|
),
|
||||||
None,
|
None,
|
||||||
@ -401,7 +401,7 @@ impl FunctionDefinition<'_> {
|
|||||||
impl FunctionBody<'_> {
|
impl FunctionBody<'_> {
|
||||||
fn prep_mem(&self, exec_state: &mut ExecState) {
|
fn prep_mem(&self, exec_state: &mut ExecState) {
|
||||||
match self {
|
match self {
|
||||||
FunctionBody::Rust(_) => exec_state.mut_stack().push_new_env_for_rust_call(),
|
FunctionBody::Rust(_) => exec_state.mut_stack().push_new_root_env(true),
|
||||||
FunctionBody::Kcl(_, memory) => exec_state.mut_stack().push_new_env_for_call(*memory),
|
FunctionBody::Kcl(_, memory) => exec_state.mut_stack().push_new_env_for_call(*memory),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -445,7 +445,7 @@ fn update_memory_for_tags_of_geometry(result: &mut KclValue, exec_state: &mut Ex
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
KclValue::Solid { ref mut value } => {
|
KclValue::Solid { value } => {
|
||||||
for v in &value.value {
|
for v in &value.value {
|
||||||
if let Some(tag) = v.get_tag() {
|
if let Some(tag) = v.get_tag() {
|
||||||
// Get the past tag and update it.
|
// Get the past tag and update it.
|
||||||
@ -555,9 +555,9 @@ fn type_err_str(expected: &Type, found: &KclValue, source_range: &SourceRange, e
|
|||||||
let found_human = found.human_friendly_type();
|
let found_human = found.human_friendly_type();
|
||||||
let found_ty = found.principal_type_string();
|
let found_ty = found.principal_type_string();
|
||||||
let found_str = if found_human == found_ty || found_human == format!("a {}", strip_backticks(&found_ty)) {
|
let found_str = if found_human == found_ty || found_human == format!("a {}", strip_backticks(&found_ty)) {
|
||||||
format!("a value with type {}", found_ty)
|
format!("a value with type {found_ty}")
|
||||||
} else {
|
} else {
|
||||||
format!("{found_human} (with type {})", found_ty)
|
format!("{found_human} (with type {found_ty})")
|
||||||
};
|
};
|
||||||
|
|
||||||
let mut result = format!("{expected_str}, but found {found_str}.");
|
let mut result = format!("{expected_str}, but found {found_str}.");
|
||||||
@ -626,7 +626,7 @@ fn type_check_params_kw(
|
|||||||
format!(
|
format!(
|
||||||
"`{label}` is not an argument of {}",
|
"`{label}` is not an argument of {}",
|
||||||
fn_name
|
fn_name
|
||||||
.map(|n| format!("`{}`", n))
|
.map(|n| format!("`{n}`"))
|
||||||
.unwrap_or_else(|| "this function".to_owned()),
|
.unwrap_or_else(|| "this function".to_owned()),
|
||||||
),
|
),
|
||||||
));
|
));
|
||||||
@ -676,7 +676,7 @@ fn type_check_params_kw(
|
|||||||
format!(
|
format!(
|
||||||
"The input argument of {} requires {}",
|
"The input argument of {} requires {}",
|
||||||
fn_name
|
fn_name
|
||||||
.map(|n| format!("`{}`", n))
|
.map(|n| format!("`{n}`"))
|
||||||
.unwrap_or_else(|| "this function".to_owned()),
|
.unwrap_or_else(|| "this function".to_owned()),
|
||||||
type_err_str(ty, &arg.1.value, &arg.1.source_range, exec_state),
|
type_err_str(ty, &arg.1.value, &arg.1.source_range, exec_state),
|
||||||
),
|
),
|
||||||
@ -691,7 +691,7 @@ fn type_check_params_kw(
|
|||||||
format!(
|
format!(
|
||||||
"{} expects an unlabeled first argument (`@{name}`), but it is labelled in the call",
|
"{} expects an unlabeled first argument (`@{name}`), but it is labelled in the call",
|
||||||
fn_name
|
fn_name
|
||||||
.map(|n| format!("The function `{}`", n))
|
.map(|n| format!("The function `{n}`"))
|
||||||
.unwrap_or_else(|| "This function".to_owned()),
|
.unwrap_or_else(|| "This function".to_owned()),
|
||||||
),
|
),
|
||||||
));
|
));
|
||||||
@ -721,7 +721,7 @@ fn assign_args_to_params_kw(
|
|||||||
)?;
|
)?;
|
||||||
}
|
}
|
||||||
None => match default {
|
None => match default {
|
||||||
Some(ref default_val) => {
|
Some(default_val) => {
|
||||||
let value = KclValue::from_default_param(default_val.clone(), exec_state);
|
let value = KclValue::from_default_param(default_val.clone(), exec_state);
|
||||||
exec_state
|
exec_state
|
||||||
.mut_stack()
|
.mut_stack()
|
||||||
@ -729,10 +729,7 @@ fn assign_args_to_params_kw(
|
|||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
return Err(KclError::new_semantic(KclErrorDetails::new(
|
return Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
format!(
|
format!("This function requires a parameter {name}, but you haven't passed it one."),
|
||||||
"This function requires a parameter {}, but you haven't passed it one.",
|
|
||||||
name
|
|
||||||
),
|
|
||||||
source_ranges,
|
source_ranges,
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
@ -746,7 +743,9 @@ fn assign_args_to_params_kw(
|
|||||||
let Some(unlabeled) = unlabelled else {
|
let Some(unlabeled) = unlabelled else {
|
||||||
return Err(if args.kw_args.labeled.contains_key(param_name) {
|
return Err(if args.kw_args.labeled.contains_key(param_name) {
|
||||||
KclError::new_semantic(KclErrorDetails::new(
|
KclError::new_semantic(KclErrorDetails::new(
|
||||||
format!("The function does declare a parameter named '{param_name}', but this parameter doesn't use a label. Try removing the `{param_name}:`"),
|
format!(
|
||||||
|
"The function does declare a parameter named '{param_name}', but this parameter doesn't use a label. Try removing the `{param_name}:`"
|
||||||
|
),
|
||||||
source_ranges,
|
source_ranges,
|
||||||
))
|
))
|
||||||
} else {
|
} else {
|
||||||
@ -799,7 +798,7 @@ mod test {
|
|||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{
|
use crate::{
|
||||||
execution::{memory::Stack, parse_execute, types::NumericType, ContextType},
|
execution::{ContextType, memory::Stack, parse_execute, types::NumericType},
|
||||||
parsing::ast::types::{DefaultParamVal, Identifier, Parameter},
|
parsing::ast::types::{DefaultParamVal, Identifier, Parameter},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -3,16 +3,16 @@ use std::ops::{Add, AddAssign, Mul};
|
|||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
use kittycad_modeling_cmds as kcmc;
|
use kittycad_modeling_cmds as kcmc;
|
||||||
use kittycad_modeling_cmds::{each_cmd as mcmd, length_unit::LengthUnit, websocket::ModelingCmdReq, ModelingCmd};
|
use kittycad_modeling_cmds::{ModelingCmd, each_cmd as mcmd, length_unit::LengthUnit, websocket::ModelingCmdReq};
|
||||||
use parse_display::{Display, FromStr};
|
use parse_display::{Display, FromStr};
|
||||||
use schemars::JsonSchema;
|
use schemars::JsonSchema;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
engine::{PlaneName, DEFAULT_PLANE_INFO},
|
engine::{DEFAULT_PLANE_INFO, PlaneName},
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::{
|
execution::{
|
||||||
types::NumericType, ArtifactId, ExecState, ExecutorContext, Metadata, TagEngineInfo, TagIdentifier, UnitLen,
|
ArtifactId, ExecState, ExecutorContext, Metadata, TagEngineInfo, TagIdentifier, UnitLen, types::NumericType,
|
||||||
},
|
},
|
||||||
parsing::ast::types::{Node, NodeRef, TagDeclarator, TagNode},
|
parsing::ast::types::{Node, NodeRef, TagDeclarator, TagNode},
|
||||||
std::{args::TyF64, sketch::PlaneData},
|
std::{args::TyF64, sketch::PlaneData},
|
||||||
@ -472,7 +472,7 @@ impl TryFrom<PlaneData> for PlaneInfo {
|
|||||||
PlaneData::Plane(_) => {
|
PlaneData::Plane(_) => {
|
||||||
// We will never get here since we already checked for PlaneData::Plane.
|
// We will never get here since we already checked for PlaneData::Plane.
|
||||||
return Err(KclError::new_internal(KclErrorDetails::new(
|
return Err(KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("PlaneData {:?} not found", value),
|
format!("PlaneData {value:?} not found"),
|
||||||
Default::default(),
|
Default::default(),
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
@ -480,7 +480,7 @@ impl TryFrom<PlaneData> for PlaneInfo {
|
|||||||
|
|
||||||
let info = DEFAULT_PLANE_INFO.get(&name).ok_or_else(|| {
|
let info = DEFAULT_PLANE_INFO.get(&name).ok_or_else(|| {
|
||||||
KclError::new_internal(KclErrorDetails::new(
|
KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("Plane {} not found", name),
|
format!("Plane {name} not found"),
|
||||||
Default::default(),
|
Default::default(),
|
||||||
))
|
))
|
||||||
})?;
|
})?;
|
||||||
@ -815,8 +815,8 @@ impl EdgeCut {
|
|||||||
|
|
||||||
pub fn set_id(&mut self, id: uuid::Uuid) {
|
pub fn set_id(&mut self, id: uuid::Uuid) {
|
||||||
match self {
|
match self {
|
||||||
EdgeCut::Fillet { id: ref mut i, .. } => *i = id,
|
EdgeCut::Fillet { id: i, .. } => *i = id,
|
||||||
EdgeCut::Chamfer { id: ref mut i, .. } => *i = id,
|
EdgeCut::Chamfer { id: i, .. } => *i = id,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -829,8 +829,8 @@ impl EdgeCut {
|
|||||||
|
|
||||||
pub fn set_edge_id(&mut self, id: uuid::Uuid) {
|
pub fn set_edge_id(&mut self, id: uuid::Uuid) {
|
||||||
match self {
|
match self {
|
||||||
EdgeCut::Fillet { edge_id: ref mut i, .. } => *i = id,
|
EdgeCut::Fillet { edge_id: i, .. } => *i = id,
|
||||||
EdgeCut::Chamfer { edge_id: ref mut i, .. } => *i = id,
|
EdgeCut::Chamfer { edge_id: i, .. } => *i = id,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -921,6 +921,12 @@ impl Point3d {
|
|||||||
units: UnitLen::Unknown,
|
units: UnitLen::Unknown,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn as_3_dims(&self) -> ([f64; 3], UnitLen) {
|
||||||
|
let p = [self.x, self.y, self.z];
|
||||||
|
let u = self.units;
|
||||||
|
(p, u)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<[TyF64; 3]> for Point3d {
|
impl From<[TyF64; 3]> for Point3d {
|
||||||
@ -939,6 +945,7 @@ impl From<Point3d> for Point3D {
|
|||||||
Self { x: p.x, y: p.y, z: p.z }
|
Self { x: p.x, y: p.y, z: p.z }
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<Point3d> for kittycad_modeling_cmds::shared::Point3d<LengthUnit> {
|
impl From<Point3d> for kittycad_modeling_cmds::shared::Point3d<LengthUnit> {
|
||||||
fn from(p: Point3d) -> Self {
|
fn from(p: Point3d) -> Self {
|
||||||
Self {
|
Self {
|
||||||
@ -1004,12 +1011,12 @@ pub struct BasePath {
|
|||||||
impl BasePath {
|
impl BasePath {
|
||||||
pub fn get_to(&self) -> [TyF64; 2] {
|
pub fn get_to(&self) -> [TyF64; 2] {
|
||||||
let ty: NumericType = self.units.into();
|
let ty: NumericType = self.units.into();
|
||||||
[TyF64::new(self.to[0], ty.clone()), TyF64::new(self.to[1], ty)]
|
[TyF64::new(self.to[0], ty), TyF64::new(self.to[1], ty)]
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_from(&self) -> [TyF64; 2] {
|
pub fn get_from(&self) -> [TyF64; 2] {
|
||||||
let ty: NumericType = self.units.into();
|
let ty: NumericType = self.units.into();
|
||||||
[TyF64::new(self.from[0], ty.clone()), TyF64::new(self.from[1], ty)]
|
[TyF64::new(self.from[0], ty), TyF64::new(self.from[1], ty)]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1030,7 +1037,7 @@ pub struct GeoMeta {
|
|||||||
#[ts(export)]
|
#[ts(export)]
|
||||||
#[serde(tag = "type")]
|
#[serde(tag = "type")]
|
||||||
pub enum Path {
|
pub enum Path {
|
||||||
/// A path that goes to a point.
|
/// A straight line which ends at the given point.
|
||||||
ToPoint {
|
ToPoint {
|
||||||
#[serde(flatten)]
|
#[serde(flatten)]
|
||||||
base: BasePath,
|
base: BasePath,
|
||||||
@ -1225,14 +1232,14 @@ impl Path {
|
|||||||
pub fn get_from(&self) -> [TyF64; 2] {
|
pub fn get_from(&self) -> [TyF64; 2] {
|
||||||
let p = &self.get_base().from;
|
let p = &self.get_base().from;
|
||||||
let ty: NumericType = self.get_base().units.into();
|
let ty: NumericType = self.get_base().units.into();
|
||||||
[TyF64::new(p[0], ty.clone()), TyF64::new(p[1], ty)]
|
[TyF64::new(p[0], ty), TyF64::new(p[1], ty)]
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Where does this path segment end?
|
/// Where does this path segment end?
|
||||||
pub fn get_to(&self) -> [TyF64; 2] {
|
pub fn get_to(&self) -> [TyF64; 2] {
|
||||||
let p = &self.get_base().to;
|
let p = &self.get_base().to;
|
||||||
let ty: NumericType = self.get_base().units.into();
|
let ty: NumericType = self.get_base().units.into();
|
||||||
[TyF64::new(p[0], ty.clone()), TyF64::new(p[1], ty)]
|
[TyF64::new(p[0], ty), TyF64::new(p[1], ty)]
|
||||||
}
|
}
|
||||||
|
|
||||||
/// The path segment start point and its type.
|
/// The path segment start point and its type.
|
||||||
|
@ -2,12 +2,12 @@ use std::str::FromStr;
|
|||||||
|
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
use kcmc::{
|
use kcmc::{
|
||||||
coord::{System, KITTYCAD},
|
ImportFile, ModelingCmd,
|
||||||
|
coord::{KITTYCAD, System},
|
||||||
each_cmd as mcmd,
|
each_cmd as mcmd,
|
||||||
format::InputFormat3d,
|
format::InputFormat3d,
|
||||||
shared::FileImportFormat,
|
shared::FileImportFormat,
|
||||||
units::UnitLength,
|
units::UnitLength,
|
||||||
ImportFile, ModelingCmd,
|
|
||||||
};
|
};
|
||||||
use kittycad_modeling_cmds as kcmc;
|
use kittycad_modeling_cmds as kcmc;
|
||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
@ -16,8 +16,8 @@ use uuid::Uuid;
|
|||||||
use crate::{
|
use crate::{
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::{
|
execution::{
|
||||||
annotations, typed_path::TypedPath, types::UnitLen, ExecState, ExecutorContext, ImportedGeometry,
|
ExecState, ExecutorContext, ImportedGeometry, ModelingCmdMeta, annotations, typed_path::TypedPath,
|
||||||
ModelingCmdMeta,
|
types::UnitLen,
|
||||||
},
|
},
|
||||||
fs::FileSystem,
|
fs::FileSystem,
|
||||||
parsing::ast::types::{Annotation, Node},
|
parsing::ast::types::{Annotation, Node},
|
||||||
@ -184,7 +184,7 @@ pub(super) fn format_from_annotations(
|
|||||||
annotations::IMPORT_LENGTH_UNIT
|
annotations::IMPORT_LENGTH_UNIT
|
||||||
),
|
),
|
||||||
vec![p.as_source_range()],
|
vec![p.as_source_range()],
|
||||||
)))
|
)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -225,7 +225,7 @@ fn set_coords(fmt: &mut InputFormat3d, coords_str: &str, source_range: SourceRan
|
|||||||
annotations::IMPORT_COORDS
|
annotations::IMPORT_COORDS
|
||||||
),
|
),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
)))
|
)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -246,7 +246,7 @@ fn set_length_unit(fmt: &mut InputFormat3d, units_str: &str, source_range: Sourc
|
|||||||
annotations::IMPORT_LENGTH_UNIT
|
annotations::IMPORT_LENGTH_UNIT
|
||||||
),
|
),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
)))
|
)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -291,7 +291,9 @@ fn get_import_format_from_extension(ext: &str) -> Result<InputFormat3d> {
|
|||||||
} else if ext == "glb" {
|
} else if ext == "glb" {
|
||||||
FileImportFormat::Gltf
|
FileImportFormat::Gltf
|
||||||
} else {
|
} else {
|
||||||
anyhow::bail!("unknown source format for file extension: {ext}. Try setting the `--src-format` flag explicitly or use a valid format.")
|
anyhow::bail!(
|
||||||
|
"unknown source format for file extension: {ext}. Try setting the `--src-format` flag explicitly or use a valid format."
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -6,12 +6,12 @@ use std::{
|
|||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
ExecState, ExecutorContext, KclError, ModuleId, SourceRange,
|
||||||
errors::KclErrorDetails,
|
errors::KclErrorDetails,
|
||||||
execution::typed_path::TypedPath,
|
execution::typed_path::TypedPath,
|
||||||
modules::{ModulePath, ModuleRepr},
|
modules::{ModulePath, ModuleRepr},
|
||||||
parsing::ast::types::{ImportPath, ImportStatement, Node as AstNode},
|
parsing::ast::types::{ImportPath, ImportStatement, Node as AstNode},
|
||||||
walk::{Node, Visitable},
|
walk::{Node, Visitable},
|
||||||
ExecState, ExecutorContext, KclError, ModuleId, SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Specific dependency between two modules. The 0th element of this info
|
/// Specific dependency between two modules. The 0th element of this info
|
||||||
@ -147,7 +147,7 @@ fn import_dependencies(
|
|||||||
ret.lock()
|
ret.lock()
|
||||||
.map_err(|err| {
|
.map_err(|err| {
|
||||||
KclError::new_internal(KclErrorDetails::new(
|
KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("Failed to lock mutex: {}", err),
|
format!("Failed to lock mutex: {err}"),
|
||||||
Default::default(),
|
Default::default(),
|
||||||
))
|
))
|
||||||
})?
|
})?
|
||||||
@ -157,7 +157,7 @@ fn import_dependencies(
|
|||||||
ret.lock()
|
ret.lock()
|
||||||
.map_err(|err| {
|
.map_err(|err| {
|
||||||
KclError::new_internal(KclErrorDetails::new(
|
KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("Failed to lock mutex: {}", err),
|
format!("Failed to lock mutex: {err}"),
|
||||||
Default::default(),
|
Default::default(),
|
||||||
))
|
))
|
||||||
})?
|
})?
|
||||||
@ -179,7 +179,7 @@ fn import_dependencies(
|
|||||||
|
|
||||||
let ret = ret.lock().map_err(|err| {
|
let ret = ret.lock().map_err(|err| {
|
||||||
KclError::new_internal(KclErrorDetails::new(
|
KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("Failed to lock mutex: {}", err),
|
format!("Failed to lock mutex: {err}"),
|
||||||
Default::default(),
|
Default::default(),
|
||||||
))
|
))
|
||||||
})?;
|
})?;
|
||||||
@ -224,7 +224,7 @@ pub(crate) async fn import_universe(
|
|||||||
let repr = {
|
let repr = {
|
||||||
let Some(module_info) = exec_state.get_module(module_id) else {
|
let Some(module_info) = exec_state.get_module(module_id) else {
|
||||||
return Err(KclError::new_internal(KclErrorDetails::new(
|
return Err(KclError::new_internal(KclErrorDetails::new(
|
||||||
format!("Module {} not found", module_id),
|
format!("Module {module_id} not found"),
|
||||||
vec![import_stmt.into()],
|
vec![import_stmt.into()],
|
||||||
)));
|
)));
|
||||||
};
|
};
|
||||||
@ -244,9 +244,7 @@ mod tests {
|
|||||||
use crate::parsing::ast::types::{ImportSelector, Program};
|
use crate::parsing::ast::types::{ImportSelector, Program};
|
||||||
|
|
||||||
macro_rules! kcl {
|
macro_rules! kcl {
|
||||||
( $kcl:expr ) => {{
|
( $kcl:expr_2021 ) => {{ $crate::parsing::top_level_parse($kcl).unwrap() }};
|
||||||
$crate::parsing::top_level_parse($kcl).unwrap()
|
|
||||||
}};
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn into_module_info(program: AstNode<Program>) -> DependencyInfo {
|
fn into_module_info(program: AstNode<Program>) -> DependencyInfo {
|
||||||
|
@ -5,18 +5,18 @@ use schemars::JsonSchema;
|
|||||||
use serde::Serialize;
|
use serde::Serialize;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
CompilationError, KclError, ModuleId, SourceRange,
|
||||||
errors::KclErrorDetails,
|
errors::KclErrorDetails,
|
||||||
execution::{
|
execution::{
|
||||||
annotations::{SETTINGS, SETTINGS_UNIT_LENGTH},
|
|
||||||
types::{NumericType, PrimitiveType, RuntimeType, UnitLen},
|
|
||||||
EnvironmentRef, ExecState, Face, Geometry, GeometryWithImportedGeometry, Helix, ImportedGeometry, MetaSettings,
|
EnvironmentRef, ExecState, Face, Geometry, GeometryWithImportedGeometry, Helix, ImportedGeometry, MetaSettings,
|
||||||
Metadata, Plane, Sketch, Solid, TagIdentifier,
|
Metadata, Plane, Sketch, Solid, TagIdentifier,
|
||||||
|
annotations::{SETTINGS, SETTINGS_UNIT_LENGTH},
|
||||||
|
types::{NumericType, PrimitiveType, RuntimeType, UnitLen},
|
||||||
},
|
},
|
||||||
parsing::ast::types::{
|
parsing::ast::types::{
|
||||||
DefaultParamVal, FunctionExpression, KclNone, Literal, LiteralValue, Node, TagDeclarator, TagNode,
|
DefaultParamVal, FunctionExpression, KclNone, Literal, LiteralValue, Node, TagDeclarator, TagNode,
|
||||||
},
|
},
|
||||||
std::{args::TyF64, StdFnProps},
|
std::{StdFnProps, args::TyF64},
|
||||||
CompilationError, KclError, ModuleId, SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
pub type KclObjectFields = HashMap<String, KclValue>;
|
pub type KclObjectFields = HashMap<String, KclValue>;
|
||||||
@ -136,9 +136,9 @@ impl JsonSchema for FunctionSource {
|
|||||||
"FunctionSource".to_owned()
|
"FunctionSource".to_owned()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn json_schema(gen: &mut schemars::gen::SchemaGenerator) -> schemars::schema::Schema {
|
fn json_schema(r#gen: &mut schemars::r#gen::SchemaGenerator) -> schemars::schema::Schema {
|
||||||
// TODO: Actually generate a reasonable schema.
|
// TODO: Actually generate a reasonable schema.
|
||||||
gen.subschema_for::<()>()
|
r#gen.subschema_for::<()>()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -415,15 +415,16 @@ impl KclValue {
|
|||||||
|
|
||||||
/// Put the point into a KCL value.
|
/// Put the point into a KCL value.
|
||||||
pub fn from_point2d(p: [f64; 2], ty: NumericType, meta: Vec<Metadata>) -> Self {
|
pub fn from_point2d(p: [f64; 2], ty: NumericType, meta: Vec<Metadata>) -> Self {
|
||||||
|
let [x, y] = p;
|
||||||
Self::Tuple {
|
Self::Tuple {
|
||||||
value: vec![
|
value: vec![
|
||||||
Self::Number {
|
Self::Number {
|
||||||
value: p[0],
|
value: x,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
ty: ty.clone(),
|
ty,
|
||||||
},
|
},
|
||||||
Self::Number {
|
Self::Number {
|
||||||
value: p[1],
|
value: y,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
ty,
|
ty,
|
||||||
},
|
},
|
||||||
@ -432,6 +433,56 @@ impl KclValue {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Put the point into a KCL value.
|
||||||
|
pub fn from_point3d(p: [f64; 3], ty: NumericType, meta: Vec<Metadata>) -> Self {
|
||||||
|
let [x, y, z] = p;
|
||||||
|
Self::Tuple {
|
||||||
|
value: vec![
|
||||||
|
Self::Number {
|
||||||
|
value: x,
|
||||||
|
meta: meta.clone(),
|
||||||
|
ty,
|
||||||
|
},
|
||||||
|
Self::Number {
|
||||||
|
value: y,
|
||||||
|
meta: meta.clone(),
|
||||||
|
ty,
|
||||||
|
},
|
||||||
|
Self::Number {
|
||||||
|
value: z,
|
||||||
|
meta: meta.clone(),
|
||||||
|
ty,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
meta,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Put the point into a KCL point.
|
||||||
|
pub fn array_from_point3d(p: [f64; 3], ty: NumericType, meta: Vec<Metadata>) -> Self {
|
||||||
|
let [x, y, z] = p;
|
||||||
|
Self::HomArray {
|
||||||
|
value: vec![
|
||||||
|
Self::Number {
|
||||||
|
value: x,
|
||||||
|
meta: meta.clone(),
|
||||||
|
ty,
|
||||||
|
},
|
||||||
|
Self::Number {
|
||||||
|
value: y,
|
||||||
|
meta: meta.clone(),
|
||||||
|
ty,
|
||||||
|
},
|
||||||
|
Self::Number {
|
||||||
|
value: z,
|
||||||
|
meta: meta.clone(),
|
||||||
|
ty,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
ty: ty.into(),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) fn as_usize(&self) -> Option<usize> {
|
pub(crate) fn as_usize(&self) -> Option<usize> {
|
||||||
match self {
|
match self {
|
||||||
KclValue::Number { value, .. } => crate::try_f64_to_usize(*value),
|
KclValue::Number { value, .. } => crate::try_f64_to_usize(*value),
|
||||||
@ -448,7 +499,7 @@ impl KclValue {
|
|||||||
|
|
||||||
pub fn as_int_with_ty(&self) -> Option<(i64, NumericType)> {
|
pub fn as_int_with_ty(&self) -> Option<(i64, NumericType)> {
|
||||||
match self {
|
match self {
|
||||||
KclValue::Number { value, ty, .. } => crate::try_f64_to_i64(*value).map(|i| (i, ty.clone())),
|
KclValue::Number { value, ty, .. } => crate::try_f64_to_i64(*value).map(|i| (i, *ty)),
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -562,7 +613,7 @@ impl KclValue {
|
|||||||
|
|
||||||
pub fn as_ty_f64(&self) -> Option<TyF64> {
|
pub fn as_ty_f64(&self) -> Option<TyF64> {
|
||||||
match self {
|
match self {
|
||||||
KclValue::Number { value, ty, .. } => Some(TyF64::new(*value, ty.clone())),
|
KclValue::Number { value, ty, .. } => Some(TyF64::new(*value, *ty)),
|
||||||
_ => None,
|
_ => None,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -587,7 +638,7 @@ impl KclValue {
|
|||||||
match self {
|
match self {
|
||||||
KclValue::TagIdentifier(t) => Ok(*t.clone()),
|
KclValue::TagIdentifier(t) => Ok(*t.clone()),
|
||||||
_ => Err(KclError::new_semantic(KclErrorDetails::new(
|
_ => Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
format!("Not a tag identifier: {:?}", self),
|
format!("Not a tag identifier: {self:?}"),
|
||||||
self.clone().into(),
|
self.clone().into(),
|
||||||
))),
|
))),
|
||||||
}
|
}
|
||||||
@ -598,7 +649,7 @@ impl KclValue {
|
|||||||
match self {
|
match self {
|
||||||
KclValue::TagDeclarator(t) => Ok((**t).clone()),
|
KclValue::TagDeclarator(t) => Ok((**t).clone()),
|
||||||
_ => Err(KclError::new_semantic(KclErrorDetails::new(
|
_ => Err(KclError::new_semantic(KclErrorDetails::new(
|
||||||
format!("Not a tag declarator: {:?}", self),
|
format!("Not a tag declarator: {self:?}"),
|
||||||
self.clone().into(),
|
self.clone().into(),
|
||||||
))),
|
))),
|
||||||
}
|
}
|
||||||
|
@ -207,8 +207,8 @@ use std::{
|
|||||||
fmt,
|
fmt,
|
||||||
pin::Pin,
|
pin::Pin,
|
||||||
sync::{
|
sync::{
|
||||||
atomic::{AtomicBool, AtomicUsize, Ordering},
|
|
||||||
Arc,
|
Arc,
|
||||||
|
atomic::{AtomicBool, AtomicUsize, Ordering},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -489,7 +489,7 @@ impl ProgramMemory {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Err(KclError::new_undefined_value(
|
Err(KclError::new_undefined_value(
|
||||||
KclErrorDetails::new(format!("`{}` is not defined", var), vec![]),
|
KclErrorDetails::new(format!("`{var}` is not defined"), vec![]),
|
||||||
Some(var.to_owned()),
|
Some(var.to_owned()),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
@ -541,22 +541,6 @@ impl Stack {
|
|||||||
self.push_new_env_for_call(snapshot);
|
self.push_new_env_for_call(snapshot);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Push a new stack frame on to the call stack for callees which should not read or write
|
|
||||||
/// from memory.
|
|
||||||
///
|
|
||||||
/// This is suitable for calling standard library functions or other functions written in Rust
|
|
||||||
/// which will use 'Rust memory' rather than KCL's memory and cannot reach into the wider
|
|
||||||
/// environment.
|
|
||||||
///
|
|
||||||
/// Trying to read or write from this environment will panic with an index out of bounds.
|
|
||||||
pub fn push_new_env_for_rust_call(&mut self) {
|
|
||||||
self.call_stack.push(self.current_env);
|
|
||||||
// Rust functions shouldn't try to set or access anything in their environment, so don't
|
|
||||||
// waste time and space on a new env. Using usize::MAX means we'll get an overflow if we
|
|
||||||
// try to access anything rather than a silent error.
|
|
||||||
self.current_env = EnvironmentRef(usize::MAX, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Push a new stack frame on to the call stack with no connection to a parent environment.
|
/// Push a new stack frame on to the call stack with no connection to a parent environment.
|
||||||
///
|
///
|
||||||
/// Suitable for executing a separate module.
|
/// Suitable for executing a separate module.
|
||||||
@ -647,7 +631,7 @@ impl Stack {
|
|||||||
let env = self.memory.get_env(self.current_env.index());
|
let env = self.memory.get_env(self.current_env.index());
|
||||||
if env.contains_key(&key) {
|
if env.contains_key(&key) {
|
||||||
return Err(KclError::new_value_already_defined(KclErrorDetails::new(
|
return Err(KclError::new_value_already_defined(KclErrorDetails::new(
|
||||||
format!("Cannot redefine `{}`", key),
|
format!("Cannot redefine `{key}`"),
|
||||||
vec![source_range],
|
vec![source_range],
|
||||||
)));
|
)));
|
||||||
}
|
}
|
||||||
@ -683,7 +667,7 @@ impl Stack {
|
|||||||
env.contains_key(var)
|
env.contains_key(var)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get a key from the first KCL (i.e., non-Rust) stack frame on the call stack.
|
/// Get a key from the first stack frame on the call stack.
|
||||||
pub fn get_from_call_stack(&self, key: &str, source_range: SourceRange) -> Result<(usize, &KclValue), KclError> {
|
pub fn get_from_call_stack(&self, key: &str, source_range: SourceRange) -> Result<(usize, &KclValue), KclError> {
|
||||||
if !self.current_env.skip_env() {
|
if !self.current_env.skip_env() {
|
||||||
return Ok((self.current_env.1, self.get(key, source_range)?));
|
return Ok((self.current_env.1, self.get(key, source_range)?));
|
||||||
@ -695,7 +679,7 @@ impl Stack {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
unreachable!("It can't be Rust frames all the way down");
|
unreachable!("No frames on the stack?");
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Iterate over all keys in the current environment which satisfy the provided predicate.
|
/// Iterate over all keys in the current environment which satisfy the provided predicate.
|
||||||
@ -1047,7 +1031,7 @@ mod env {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Take all bindings from the environment.
|
/// Take all bindings from the environment.
|
||||||
pub(super) fn take_bindings(self: Pin<&mut Self>) -> impl Iterator<Item = (String, (usize, KclValue))> {
|
pub(super) fn take_bindings(self: Pin<&mut Self>) -> impl Iterator<Item = (String, (usize, KclValue))> + use<> {
|
||||||
// SAFETY: caller must have unique access since self is mut. We're not moving or invalidating `self`.
|
// SAFETY: caller must have unique access since self is mut. We're not moving or invalidating `self`.
|
||||||
let bindings = std::mem::take(unsafe { self.bindings.get().as_mut().unwrap() });
|
let bindings = std::mem::take(unsafe { self.bindings.get().as_mut().unwrap() });
|
||||||
bindings.into_iter()
|
bindings.into_iter()
|
||||||
@ -1217,24 +1201,6 @@ mod test {
|
|||||||
assert_get_from(mem, "c", 5, callee);
|
assert_get_from(mem, "c", 5, callee);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn rust_env() {
|
|
||||||
let mem = &mut Stack::new_for_tests();
|
|
||||||
mem.add("a".to_owned(), val(1), sr()).unwrap();
|
|
||||||
mem.add("b".to_owned(), val(3), sr()).unwrap();
|
|
||||||
let sn = mem.snapshot();
|
|
||||||
|
|
||||||
mem.push_new_env_for_rust_call();
|
|
||||||
mem.push_new_env_for_call(sn);
|
|
||||||
assert_get(mem, "b", 3);
|
|
||||||
mem.add("b".to_owned(), val(4), sr()).unwrap();
|
|
||||||
assert_get(mem, "b", 4);
|
|
||||||
|
|
||||||
mem.pop_env();
|
|
||||||
mem.pop_env();
|
|
||||||
assert_get(mem, "b", 3);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn deep_call_env() {
|
fn deep_call_env() {
|
||||||
let mem = &mut Stack::new_for_tests();
|
let mem = &mut Stack::new_for_tests();
|
||||||
|
@ -16,10 +16,9 @@ pub(crate) use import::PreImportedGeometry;
|
|||||||
use indexmap::IndexMap;
|
use indexmap::IndexMap;
|
||||||
pub use kcl_value::{KclObjectFields, KclValue};
|
pub use kcl_value::{KclObjectFields, KclValue};
|
||||||
use kcmc::{
|
use kcmc::{
|
||||||
each_cmd as mcmd,
|
ImageFormat, ModelingCmd, each_cmd as mcmd,
|
||||||
ok_response::{output::TakeSnapshot, OkModelingCmdResponse},
|
ok_response::{OkModelingCmdResponse, output::TakeSnapshot},
|
||||||
websocket::{ModelingSessionData, OkWebSocketResponseData},
|
websocket::{ModelingSessionData, OkWebSocketResponseData},
|
||||||
ImageFormat, ModelingCmd,
|
|
||||||
};
|
};
|
||||||
use kittycad_modeling_cmds::{self as kcmc, id::ModelingCmdId};
|
use kittycad_modeling_cmds::{self as kcmc, id::ModelingCmdId};
|
||||||
pub use memory::EnvironmentRef;
|
pub use memory::EnvironmentRef;
|
||||||
@ -31,6 +30,7 @@ pub use state::{ExecState, MetaSettings};
|
|||||||
use uuid::Uuid;
|
use uuid::Uuid;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
CompilationError, ExecError, KclErrorWithOutputs,
|
||||||
engine::{EngineManager, GridScaleBehavior},
|
engine::{EngineManager, GridScaleBehavior},
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::{
|
execution::{
|
||||||
@ -43,7 +43,6 @@ use crate::{
|
|||||||
modules::{ModuleId, ModulePath, ModuleRepr},
|
modules::{ModuleId, ModulePath, ModuleRepr},
|
||||||
parsing::ast::types::{Expr, ImportPath, NodeRef},
|
parsing::ast::types::{Expr, ImportPath, NodeRef},
|
||||||
source_range::SourceRange,
|
source_range::SourceRange,
|
||||||
CompilationError, ExecError, KclErrorWithOutputs,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
pub(crate) mod annotations;
|
pub(crate) mod annotations;
|
||||||
@ -1329,7 +1328,7 @@ impl ExecutorContext {
|
|||||||
created: if deterministic_time {
|
created: if deterministic_time {
|
||||||
Some("2021-01-01T00:00:00Z".parse().map_err(|e| {
|
Some("2021-01-01T00:00:00Z".parse().map_err(|e| {
|
||||||
KclError::new_internal(crate::errors::KclErrorDetails::new(
|
KclError::new_internal(crate::errors::KclErrorDetails::new(
|
||||||
format!("Failed to parse date: {}", e),
|
format!("Failed to parse date: {e}"),
|
||||||
vec![SourceRange::default()],
|
vec![SourceRange::default()],
|
||||||
))
|
))
|
||||||
})?)
|
})?)
|
||||||
@ -1409,7 +1408,7 @@ pub(crate) async fn parse_execute_with_project_dir(
|
|||||||
engine: Arc::new(Box::new(
|
engine: Arc::new(Box::new(
|
||||||
crate::engine::conn_mock::EngineConnection::new().await.map_err(|err| {
|
crate::engine::conn_mock::EngineConnection::new().await.map_err(|err| {
|
||||||
KclError::new_internal(crate::errors::KclErrorDetails::new(
|
KclError::new_internal(crate::errors::KclErrorDetails::new(
|
||||||
format!("Failed to create mock engine connection: {}", err),
|
format!("Failed to create mock engine connection: {err}"),
|
||||||
vec![SourceRange::default()],
|
vec![SourceRange::default()],
|
||||||
))
|
))
|
||||||
})?,
|
})?,
|
||||||
@ -1446,7 +1445,7 @@ mod tests {
|
|||||||
use pretty_assertions::assert_eq;
|
use pretty_assertions::assert_eq;
|
||||||
|
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{errors::KclErrorDetails, execution::memory::Stack, ModuleId};
|
use crate::{ModuleId, errors::KclErrorDetails, execution::memory::Stack};
|
||||||
|
|
||||||
/// Convenience function to get a JSON value from memory and unwrap.
|
/// Convenience function to get a JSON value from memory and unwrap.
|
||||||
#[track_caller]
|
#[track_caller]
|
||||||
@ -1921,6 +1920,22 @@ shape = layer() |> patternTransform(instances = 10, transform = transform)
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
|
async fn pass_std_to_std() {
|
||||||
|
let ast = r#"sketch001 = startSketchOn(XY)
|
||||||
|
profile001 = circle(sketch001, center = [0, 0], radius = 2)
|
||||||
|
extrude001 = extrude(profile001, length = 5)
|
||||||
|
extrudes = patternLinear3d(
|
||||||
|
extrude001,
|
||||||
|
instances = 3,
|
||||||
|
distance = 5,
|
||||||
|
axis = [1, 1, 0],
|
||||||
|
)
|
||||||
|
clone001 = map(extrudes, f = clone)
|
||||||
|
"#;
|
||||||
|
parse_execute(ast).await.unwrap();
|
||||||
|
}
|
||||||
|
|
||||||
#[tokio::test(flavor = "multi_thread")]
|
#[tokio::test(flavor = "multi_thread")]
|
||||||
async fn test_zero_param_fn() {
|
async fn test_zero_param_fn() {
|
||||||
let ast = r#"sigmaAllow = 35000 // psi
|
let ast = r#"sigmaAllow = 35000 // psi
|
||||||
@ -2045,8 +2060,7 @@ notFunction = !x";
|
|||||||
fn_err
|
fn_err
|
||||||
.message()
|
.message()
|
||||||
.starts_with("Cannot apply unary operator ! to non-boolean value: "),
|
.starts_with("Cannot apply unary operator ! to non-boolean value: "),
|
||||||
"Actual error: {:?}",
|
"Actual error: {fn_err:?}"
|
||||||
fn_err
|
|
||||||
);
|
);
|
||||||
|
|
||||||
let code8 = "
|
let code8 = "
|
||||||
@ -2059,8 +2073,7 @@ notTagDeclarator = !myTagDeclarator";
|
|||||||
tag_declarator_err
|
tag_declarator_err
|
||||||
.message()
|
.message()
|
||||||
.starts_with("Cannot apply unary operator ! to non-boolean value: a tag declarator"),
|
.starts_with("Cannot apply unary operator ! to non-boolean value: a tag declarator"),
|
||||||
"Actual error: {:?}",
|
"Actual error: {tag_declarator_err:?}"
|
||||||
tag_declarator_err
|
|
||||||
);
|
);
|
||||||
|
|
||||||
let code9 = "
|
let code9 = "
|
||||||
@ -2073,8 +2086,7 @@ notTagIdentifier = !myTag";
|
|||||||
tag_identifier_err
|
tag_identifier_err
|
||||||
.message()
|
.message()
|
||||||
.starts_with("Cannot apply unary operator ! to non-boolean value: a tag identifier"),
|
.starts_with("Cannot apply unary operator ! to non-boolean value: a tag identifier"),
|
||||||
"Actual error: {:?}",
|
"Actual error: {tag_identifier_err:?}"
|
||||||
tag_identifier_err
|
|
||||||
);
|
);
|
||||||
|
|
||||||
let code10 = "notPipe = !(1 |> 2)";
|
let code10 = "notPipe = !(1 |> 2)";
|
||||||
@ -2226,7 +2238,7 @@ w = f() + f()
|
|||||||
if let Err(err) = ctx.run_with_caching(old_program).await {
|
if let Err(err) = ctx.run_with_caching(old_program).await {
|
||||||
let report = err.into_miette_report_with_outputs(code).unwrap();
|
let report = err.into_miette_report_with_outputs(code).unwrap();
|
||||||
let report = miette::Report::new(report);
|
let report = miette::Report::new(report);
|
||||||
panic!("Error executing program: {:?}", report);
|
panic!("Error executing program: {report:?}");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the id_generator from the first execution.
|
// Get the id_generator from the first execution.
|
||||||
|
@ -8,10 +8,10 @@ use uuid::Uuid;
|
|||||||
#[cfg(feature = "artifact-graph")]
|
#[cfg(feature = "artifact-graph")]
|
||||||
use crate::exec::ArtifactCommand;
|
use crate::exec::ArtifactCommand;
|
||||||
use crate::{
|
use crate::{
|
||||||
|
ExecState, ExecutorContext, KclError, SourceRange,
|
||||||
exec::{IdGenerator, KclValue},
|
exec::{IdGenerator, KclValue},
|
||||||
execution::Solid,
|
execution::Solid,
|
||||||
std::Args,
|
std::Args,
|
||||||
ExecState, ExecutorContext, KclError, SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Context and metadata needed to send a single modeling command.
|
/// Context and metadata needed to send a single modeling command.
|
||||||
|
@ -9,20 +9,19 @@ use uuid::Uuid;
|
|||||||
#[cfg(feature = "artifact-graph")]
|
#[cfg(feature = "artifact-graph")]
|
||||||
use crate::execution::{Artifact, ArtifactCommand, ArtifactGraph, ArtifactId};
|
use crate::execution::{Artifact, ArtifactCommand, ArtifactGraph, ArtifactId};
|
||||||
use crate::{
|
use crate::{
|
||||||
|
CompilationError, EngineManager, ExecutorContext, KclErrorWithOutputs,
|
||||||
errors::{KclError, KclErrorDetails, Severity},
|
errors::{KclError, KclErrorDetails, Severity},
|
||||||
exec::DefaultPlanes,
|
exec::DefaultPlanes,
|
||||||
execution::{
|
execution::{
|
||||||
annotations,
|
EnvironmentRef, ExecOutcome, ExecutorSettings, KclValue, UnitAngle, UnitLen, annotations,
|
||||||
cad_op::Operation,
|
cad_op::Operation,
|
||||||
id_generator::IdGenerator,
|
id_generator::IdGenerator,
|
||||||
memory::{ProgramMemory, Stack},
|
memory::{ProgramMemory, Stack},
|
||||||
types::{self, NumericType},
|
types::{self, NumericType},
|
||||||
EnvironmentRef, ExecOutcome, ExecutorSettings, KclValue, UnitAngle, UnitLen,
|
|
||||||
},
|
},
|
||||||
modules::{ModuleId, ModuleInfo, ModuleLoader, ModulePath, ModuleRepr, ModuleSource},
|
modules::{ModuleId, ModuleInfo, ModuleLoader, ModulePath, ModuleRepr, ModuleSource},
|
||||||
parsing::ast::types::{Annotation, NodeRef},
|
parsing::ast::types::{Annotation, NodeRef},
|
||||||
source_range::SourceRange,
|
source_range::SourceRange,
|
||||||
CompilationError, EngineManager, ExecutorContext, KclErrorWithOutputs,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/// State for executing a program.
|
/// State for executing a program.
|
||||||
@ -555,7 +554,7 @@ impl MetaSettings {
|
|||||||
annotations::SETTINGS_UNIT_ANGLE
|
annotations::SETTINGS_UNIT_ANGLE
|
||||||
),
|
),
|
||||||
vec![annotation.as_source_range()],
|
vec![annotation.as_source_range()],
|
||||||
)))
|
)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -220,9 +220,9 @@ impl schemars::JsonSchema for TypedPath {
|
|||||||
"TypedPath".to_owned()
|
"TypedPath".to_owned()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn json_schema(gen: &mut schemars::gen::SchemaGenerator) -> schemars::schema::Schema {
|
fn json_schema(r#gen: &mut schemars::r#gen::SchemaGenerator) -> schemars::schema::Schema {
|
||||||
// TODO: Actually generate a reasonable schema.
|
// TODO: Actually generate a reasonable schema.
|
||||||
gen.subschema_for::<std::path::PathBuf>()
|
r#gen.subschema_for::<std::path::PathBuf>()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -5,17 +5,17 @@ use schemars::JsonSchema;
|
|||||||
use serde::{Deserialize, Serialize};
|
use serde::{Deserialize, Serialize};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
CompilationError, SourceRange,
|
||||||
execution::{
|
execution::{
|
||||||
|
ExecState, Plane, PlaneInfo, Point3d,
|
||||||
kcl_value::{KclValue, TypeDef},
|
kcl_value::{KclValue, TypeDef},
|
||||||
memory::{self},
|
memory::{self},
|
||||||
ExecState, Plane, PlaneInfo, Point3d,
|
|
||||||
},
|
},
|
||||||
parsing::{
|
parsing::{
|
||||||
ast::types::{PrimitiveType as AstPrimitiveType, Type},
|
ast::types::{PrimitiveType as AstPrimitiveType, Type},
|
||||||
token::NumericSuffix,
|
token::NumericSuffix,
|
||||||
},
|
},
|
||||||
std::args::{FromKclValue, TyF64},
|
std::args::{FromKclValue, TyF64},
|
||||||
CompilationError, SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq)]
|
#[derive(Debug, Clone, PartialEq)]
|
||||||
@ -210,7 +210,7 @@ impl RuntimeType {
|
|||||||
let ty_val = exec_state
|
let ty_val = exec_state
|
||||||
.stack()
|
.stack()
|
||||||
.get(&format!("{}{}", memory::TYPE_PREFIX, alias), source_range)
|
.get(&format!("{}{}", memory::TYPE_PREFIX, alias), source_range)
|
||||||
.map_err(|_| CompilationError::err(source_range, format!("Unknown type: {}", alias)))?;
|
.map_err(|_| CompilationError::err(source_range, format!("Unknown type: {alias}")))?;
|
||||||
|
|
||||||
Ok(match ty_val {
|
Ok(match ty_val {
|
||||||
KclValue::Type { value, .. } => match value {
|
KclValue::Type { value, .. } => match value {
|
||||||
@ -241,7 +241,7 @@ impl RuntimeType {
|
|||||||
"a tuple with values of types ({})",
|
"a tuple with values of types ({})",
|
||||||
tys.iter().map(Self::human_friendly_type).collect::<Vec<_>>().join(", ")
|
tys.iter().map(Self::human_friendly_type).collect::<Vec<_>>().join(", ")
|
||||||
),
|
),
|
||||||
RuntimeType::Object(_) => format!("an object with fields {}", self),
|
RuntimeType::Object(_) => format!("an object with fields {self}"),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -460,7 +460,7 @@ impl fmt::Display for PrimitiveType {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone, Serialize, Deserialize, PartialEq, ts_rs::TS, JsonSchema)]
|
#[derive(Debug, Clone, Copy, Serialize, Deserialize, PartialEq, ts_rs::TS, JsonSchema)]
|
||||||
#[ts(export)]
|
#[ts(export)]
|
||||||
#[serde(tag = "type")]
|
#[serde(tag = "type")]
|
||||||
pub enum NumericType {
|
pub enum NumericType {
|
||||||
@ -575,7 +575,7 @@ impl NumericType {
|
|||||||
match (&ty, &i.ty) {
|
match (&ty, &i.ty) {
|
||||||
(Any, Default { .. }) if i.n == 0.0 => {}
|
(Any, Default { .. }) if i.n == 0.0 => {}
|
||||||
(Any, t) => {
|
(Any, t) => {
|
||||||
ty = t.clone();
|
ty = *t;
|
||||||
}
|
}
|
||||||
(_, Unknown) | (Default { .. }, Default { .. }) => return (result, Unknown),
|
(_, Unknown) | (Default { .. }, Default { .. }) => return (result, Unknown),
|
||||||
|
|
||||||
@ -598,7 +598,7 @@ impl NumericType {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if ty == Any && !input.is_empty() {
|
if ty == Any && !input.is_empty() {
|
||||||
ty = input[0].ty.clone();
|
ty = input[0].ty;
|
||||||
}
|
}
|
||||||
|
|
||||||
(result, ty)
|
(result, ty)
|
||||||
@ -722,7 +722,7 @@ impl NumericType {
|
|||||||
if ty.subtype(self) {
|
if ty.subtype(self) {
|
||||||
return Ok(KclValue::Number {
|
return Ok(KclValue::Number {
|
||||||
value: *value,
|
value: *value,
|
||||||
ty: ty.clone(),
|
ty: *ty,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@ -736,7 +736,7 @@ impl NumericType {
|
|||||||
|
|
||||||
(Any, _) => Ok(KclValue::Number {
|
(Any, _) => Ok(KclValue::Number {
|
||||||
value: *value,
|
value: *value,
|
||||||
ty: self.clone(),
|
ty: *self,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
}),
|
}),
|
||||||
|
|
||||||
@ -744,7 +744,7 @@ impl NumericType {
|
|||||||
// means accept any number rather than force the current default.
|
// means accept any number rather than force the current default.
|
||||||
(_, Default { .. }) => Ok(KclValue::Number {
|
(_, Default { .. }) => Ok(KclValue::Number {
|
||||||
value: *value,
|
value: *value,
|
||||||
ty: ty.clone(),
|
ty: *ty,
|
||||||
meta: meta.clone(),
|
meta: meta.clone(),
|
||||||
}),
|
}),
|
||||||
|
|
||||||
@ -840,6 +840,18 @@ pub enum UnitType {
|
|||||||
Angle(UnitAngle),
|
Angle(UnitAngle),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl UnitType {
|
||||||
|
pub(crate) fn to_suffix(self) -> Option<String> {
|
||||||
|
match self {
|
||||||
|
UnitType::Count => Some("_".to_owned()),
|
||||||
|
UnitType::Length(UnitLen::Unknown) => None,
|
||||||
|
UnitType::Angle(UnitAngle::Unknown) => None,
|
||||||
|
UnitType::Length(l) => Some(l.to_string()),
|
||||||
|
UnitType::Angle(a) => Some(a.to_string()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl std::fmt::Display for UnitType {
|
impl std::fmt::Display for UnitType {
|
||||||
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
||||||
match self {
|
match self {
|
||||||
@ -1479,7 +1491,7 @@ impl KclValue {
|
|||||||
pub fn principal_type(&self) -> Option<RuntimeType> {
|
pub fn principal_type(&self) -> Option<RuntimeType> {
|
||||||
match self {
|
match self {
|
||||||
KclValue::Bool { .. } => Some(RuntimeType::Primitive(PrimitiveType::Boolean)),
|
KclValue::Bool { .. } => Some(RuntimeType::Primitive(PrimitiveType::Boolean)),
|
||||||
KclValue::Number { ty, .. } => Some(RuntimeType::Primitive(PrimitiveType::Number(ty.clone()))),
|
KclValue::Number { ty, .. } => Some(RuntimeType::Primitive(PrimitiveType::Number(*ty))),
|
||||||
KclValue::String { .. } => Some(RuntimeType::Primitive(PrimitiveType::String)),
|
KclValue::String { .. } => Some(RuntimeType::Primitive(PrimitiveType::String)),
|
||||||
KclValue::Object { value, .. } => {
|
KclValue::Object { value, .. } => {
|
||||||
let properties = value
|
let properties = value
|
||||||
@ -1529,7 +1541,7 @@ impl KclValue {
|
|||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod test {
|
mod test {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::execution::{parse_execute, ExecTestResults};
|
use crate::execution::{ExecTestResults, parse_execute};
|
||||||
|
|
||||||
fn values(exec_state: &mut ExecState) -> Vec<KclValue> {
|
fn values(exec_state: &mut ExecState) -> Vec<KclValue> {
|
||||||
vec![
|
vec![
|
||||||
@ -1975,14 +1987,16 @@ mod test {
|
|||||||
])
|
])
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
assert!(RuntimeType::Union(vec![
|
assert!(
|
||||||
RuntimeType::Primitive(PrimitiveType::Number(NumericType::Any)),
|
RuntimeType::Union(vec![
|
||||||
RuntimeType::Primitive(PrimitiveType::Boolean)
|
RuntimeType::Primitive(PrimitiveType::Number(NumericType::Any)),
|
||||||
])
|
RuntimeType::Primitive(PrimitiveType::Boolean)
|
||||||
.subtype(&RuntimeType::Union(vec![
|
])
|
||||||
RuntimeType::Primitive(PrimitiveType::Number(NumericType::Any)),
|
.subtype(&RuntimeType::Union(vec![
|
||||||
RuntimeType::Primitive(PrimitiveType::Boolean)
|
RuntimeType::Primitive(PrimitiveType::Number(NumericType::Any)),
|
||||||
])));
|
RuntimeType::Primitive(PrimitiveType::Boolean)
|
||||||
|
]))
|
||||||
|
);
|
||||||
|
|
||||||
// Covariance
|
// Covariance
|
||||||
let count = KclValue::Number {
|
let count = KclValue::Number {
|
||||||
|
@ -45,6 +45,31 @@ pub fn format_number_literal(value: f64, suffix: NumericSuffix) -> Result<String
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Serialize, thiserror::Error)]
|
||||||
|
#[serde(tag = "type")]
|
||||||
|
pub enum FormatNumericTypeError {
|
||||||
|
#[error("Invalid numeric type: {0:?}")]
|
||||||
|
Invalid(NumericType),
|
||||||
|
}
|
||||||
|
|
||||||
|
/// For UI code generation, format a number value with a suffix such that the
|
||||||
|
/// result can parse as a literal. If it can't be done, returns an error.
|
||||||
|
///
|
||||||
|
/// This is used by TS.
|
||||||
|
pub fn format_number_value(value: f64, ty: NumericType) -> Result<String, FormatNumericTypeError> {
|
||||||
|
match ty {
|
||||||
|
NumericType::Default { .. } => Ok(value.to_string()),
|
||||||
|
// There isn't a syntactic suffix for these. For unknown, we don't want
|
||||||
|
// to ever generate the unknown suffix. We currently warn on it, and we
|
||||||
|
// may remove it in the future.
|
||||||
|
NumericType::Unknown | NumericType::Any => Err(FormatNumericTypeError::Invalid(ty)),
|
||||||
|
NumericType::Known(unit_type) => unit_type
|
||||||
|
.to_suffix()
|
||||||
|
.map(|suffix| format!("{value}{suffix}"))
|
||||||
|
.ok_or(FormatNumericTypeError::Invalid(ty)),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
#[cfg(test)]
|
||||||
mod tests {
|
mod tests {
|
||||||
use pretty_assertions::assert_eq;
|
use pretty_assertions::assert_eq;
|
||||||
@ -134,4 +159,74 @@ mod tests {
|
|||||||
Err(FormatNumericSuffixError::Invalid(NumericSuffix::Unknown))
|
Err(FormatNumericSuffixError::Invalid(NumericSuffix::Unknown))
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_format_number_value() {
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(
|
||||||
|
1.0,
|
||||||
|
NumericType::Default {
|
||||||
|
len: Default::default(),
|
||||||
|
angle: Default::default()
|
||||||
|
}
|
||||||
|
),
|
||||||
|
Ok("1".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Length(UnitLen::Unknown))),
|
||||||
|
Err(FormatNumericTypeError::Invalid(NumericType::Known(UnitType::Length(
|
||||||
|
UnitLen::Unknown
|
||||||
|
))))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Angle(UnitAngle::Unknown))),
|
||||||
|
Err(FormatNumericTypeError::Invalid(NumericType::Known(UnitType::Angle(
|
||||||
|
UnitAngle::Unknown
|
||||||
|
))))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Count)),
|
||||||
|
Ok("1_".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Length(UnitLen::Mm))),
|
||||||
|
Ok("1mm".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Length(UnitLen::Cm))),
|
||||||
|
Ok("1cm".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Length(UnitLen::M))),
|
||||||
|
Ok("1m".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Length(UnitLen::Inches))),
|
||||||
|
Ok("1in".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Length(UnitLen::Feet))),
|
||||||
|
Ok("1ft".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Length(UnitLen::Yards))),
|
||||||
|
Ok("1yd".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Angle(UnitAngle::Degrees))),
|
||||||
|
Ok("1deg".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Known(UnitType::Angle(UnitAngle::Radians))),
|
||||||
|
Ok("1rad".to_owned())
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Unknown),
|
||||||
|
Err(FormatNumericTypeError::Invalid(NumericType::Unknown))
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
format_number_value(1.0, NumericType::Any),
|
||||||
|
Err(FormatNumericTypeError::Invalid(NumericType::Any))
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -3,10 +3,10 @@
|
|||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
SourceRange,
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::typed_path::TypedPath,
|
execution::typed_path::TypedPath,
|
||||||
fs::FileSystem,
|
fs::FileSystem,
|
||||||
SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
@ -2,7 +2,7 @@
|
|||||||
|
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
use crate::{execution::typed_path::TypedPath, SourceRange};
|
use crate::{SourceRange, execution::typed_path::TypedPath};
|
||||||
|
|
||||||
#[cfg(not(target_arch = "wasm32"))]
|
#[cfg(not(target_arch = "wasm32"))]
|
||||||
pub mod local;
|
pub mod local;
|
||||||
|
@ -4,11 +4,11 @@ use anyhow::Result;
|
|||||||
use wasm_bindgen::prelude::wasm_bindgen;
|
use wasm_bindgen::prelude::wasm_bindgen;
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
|
SourceRange,
|
||||||
errors::{KclError, KclErrorDetails},
|
errors::{KclError, KclErrorDetails},
|
||||||
execution::typed_path::TypedPath,
|
execution::typed_path::TypedPath,
|
||||||
fs::FileSystem,
|
fs::FileSystem,
|
||||||
wasm::JsFuture,
|
wasm::JsFuture,
|
||||||
SourceRange,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
#[wasm_bindgen(module = "/../../src/lang/std/fileSystemManager.ts")]
|
#[wasm_bindgen(module = "/../../src/lang/std/fileSystemManager.ts")]
|
||||||
|