diff --git a/.env.development b/.env.development index 09f2b8385..20c146ae2 100644 --- a/.env.development +++ b/.env.development @@ -2,7 +2,9 @@ NODE_ENV=development DEV=true VITE_KC_API_WS_MODELING_URL=wss://api.dev.zoo.dev/ws/modeling/commands VITE_KC_API_BASE_URL=https://api.dev.zoo.dev +BASE_URL=https://api.dev.zoo.dev VITE_KC_SITE_BASE_URL=https://dev.zoo.dev VITE_KC_SKIP_AUTH=false VITE_KC_CONNECTION_TIMEOUT_MS=5000 -VITE_KC_DEV_TOKEN="your token from dev.zoo.dev should go in .env.development.local" +# ONLY add your token in .env.development.local if you want to skip auth, otherwise this token takes precedence! +#VITE_KC_DEV_TOKEN="your token from dev.zoo.dev should go in .env.development.local" diff --git a/.github/workflows/build-test-publish-apps.yml b/.github/workflows/build-test-publish-apps.yml index f7bcf1e35..c8afd3353 100644 --- a/.github/workflows/build-test-publish-apps.yml +++ b/.github/workflows/build-test-publish-apps.yml @@ -1,4 +1,4 @@ -name: build-test-publish-apps +name: build-publish-apps on: pull_request: @@ -21,7 +21,7 @@ concurrency: cancel-in-progress: true jobs: - prepare-json-files: + prepare-files: runs-on: ubuntu-22.04 # seperate job on Ubuntu for easy string manipulations (compared to Windows) outputs: version: ${{ steps.export_version.outputs.version }} @@ -33,6 +33,19 @@ jobs: node-version-file: '.nvmrc' cache: 'yarn' + - run: yarn install + + - name: Setup Rust + uses: dtolnay/rust-toolchain@stable + + - uses: Swatinem/rust-cache@v2 + with: + workspaces: './src/wasm-lib' + + # TODO: see if we can fetch from main instead if no diff at src/wasm-lib + - name: Run build:wasm + run: "yarn build:wasm" + - name: Set nightly version if: github.event_name == 'schedule' run: | @@ -42,36 +55,48 @@ jobs: # TODO: see if we ned to add updater test URL here https://dl.zoo.dev/releases/modeling-app/updater-test/last_update.json - uses: actions/upload-artifact@v3 - if: ${{ github.event_name == 'schedule' || env.CUT_RELEASE_PR == 'true' }} with: + name: prepared-files path: | package.json + src/wasm-lib/pkg/wasm_lib* - id: export_version run: echo "version=`cat package.json | jq -r '.version'`" >> "$GITHUB_OUTPUT" - build-test-app-macos: - needs: [prepare-json-files] - runs-on: macos-14 + build-apps: + needs: [prepare-files] + strategy: + fail-fast: false + matrix: + os: [macos-14, windows-2022, ubuntu-22.04] + runs-on: ${{ matrix.os }} env: APPLE_ID: ${{ secrets.APPLE_ID }} APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} + APPLE_APP_SPECIFIC_PASSWORD: ${{ secrets.APPLE_PASSWORD }} APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} - APPLE_CERTIFICATE: ${{ secrets.APPLE_CERTIFICATE }} - APPLE_CERTIFICATE_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} - APPLE_SIGNING_IDENTITY: ${{ secrets.APPLE_SIGNING_IDENTITY }} + CSC_LINK: ${{ secrets.APPLE_CERTIFICATE }} + CSC_KEY_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} + CSC_KEYCHAIN: ${{ secrets.APPLE_SIGNING_IDENTITY }} + CSC_FOR_PULL_REQUEST: true + VERSION: ${{ github.event_name == 'schedule' && needs.prepare-files.outputs.version || format('v{0}', needs.prepare-files.outputs.version) }} + VERSION_NO_V: ${{ needs.prepare-files.outputs.version }} + WINDOWS_CERTIFICATE_THUMBPRINT: F4C9A52FF7BC26EE5E054946F6B11DEEA94C748D steps: - uses: actions/checkout@v4 - uses: actions/download-artifact@v3 - if: github.event_name == 'schedule' + name: prepared-files - - name: Copy updated .json files - if: github.event_name == 'schedule' + - name: Copy prepared files run: | - ls -l artifact - cp artifact/package.json package.json + ls -R prepared-files + cp prepared-files/package.json package.json + cp prepared-files/src/wasm-lib/pkg/wasm_lib_bg.wasm public + mkdir src/wasm-lib/pkg + cp prepared-files/src/wasm-lib/pkg/wasm_lib* src/wasm-lib/pkg - name: Sync node version and setup cache uses: actions/setup-node@v4 @@ -81,79 +106,10 @@ jobs: - run: yarn install - - name: Setup Rust - uses: dtolnay/rust-toolchain@stable - - - uses: Swatinem/rust-cache@v2 - with: - workspaces: './src/wasm-lib' - - - name: Run build:wasm - run: "yarn build:wasm${{ env.BUILD_RELEASE == 'true' && '-dev' || ''}}" - - # TODO: sign the app (and updater bundle potentially) - - name: Add signing certificate - if: ${{ env.BUILD_RELEASE == 'true' }} - run: chmod +x add-osx-cert.sh && ./add-osx-cert.sh - - - name: Build the app for arm64 - run: "yarn electron-forge make" - - - name: Build the app for x64 - run: "yarn electron-forge make --arch x64" - - - name: List artifacts - run: "ls -R out/make" - - # TODO: add the 'Build for Mac TestFlight (nightly)' stage back - - - uses: actions/upload-artifact@v3 - with: - path: "out/make/*/*/*/*" - - - build-test-app-windows: - needs: [prepare-json-files] - runs-on: windows-2022 - steps: - - uses: actions/checkout@v4 - - - uses: actions/download-artifact@v3 - - - name: Copy updated .json files - if: github.event_name == 'schedule' - run: | - ls -l artifact - cp artifact/package.json package.json - - - name: Sync node version and setup cache - uses: actions/setup-node@v4 - with: - node-version-file: '.nvmrc' - cache: 'yarn' # Set this to npm, yarn or pnpm. - - - run: yarn install - - - name: Setup Rust - uses: dtolnay/rust-toolchain@stable - - - uses: Swatinem/rust-cache@v2 - with: - workspaces: './src/wasm-lib' - - - name: Run build:wasm manually - shell: bash - env: - MODE: ${{ env.BUILD_RELEASE == 'true' && '--release' || '--debug' }} - run: | - mkdir src/wasm-lib/pkg; cd src/wasm-lib - echo "building with ${{ env.MODE }}" - npx wasm-pack build --target web --out-dir pkg ${{ env.MODE }} - cd ../../ - cp src/wasm-lib/pkg/wasm_lib_bg.wasm public + - run: yarn tronb:vite - name: Prepare certificate and variables (Windows only) - if: ${{ env.BUILD_RELEASE == 'true' }} + if: ${{ env.BUILD_RELEASE == 'true' && matrix.os == 'windows-2022' }} run: | echo "${{secrets.SM_CLIENT_CERT_FILE_B64 }}" | base64 --decode > /d/Certificate_pkcs12.p12 cat /d/Certificate_pkcs12.p12 @@ -168,7 +124,7 @@ jobs: shell: bash - name: Setup certicate with SSM KSP (Windows only) - if: ${{ env.BUILD_RELEASE == 'true' }} + if: ${{ env.BUILD_RELEASE == 'true' && matrix.os == 'windows-2022' }} run: | curl -X GET https://one.digicert.com/signingmanager/api-ui/v1/releases/smtools-windows-x64.msi/download -H "x-api-key:%SM_API_KEY%" -o smtools-windows-x64.msi msiexec /i smtools-windows-x64.msi /quiet /qn @@ -178,83 +134,22 @@ jobs: smksp_cert_sync.exe shell: cmd - - name: Build the app for x64 - run: "yarn electron-forge make --arch x64" + - name: Build the app + run: yarn electron-builder --config ${{ env.BUILD_RELEASE && '--publish always' || '' }} - - name: Build the app for arm64 - run: "yarn electron-forge make --arch arm64" - - - name: List artifacts - run: "ls -R out/make" - - - name: Sign using Signtool - if: ${{ env.BUILD_RELEASE == 'true' }} - env: - THUMBPRINT: "F4C9A52FF7BC26EE5E054946F6B11DEEA94C748D" - X64_FILE: "D:\\a\\modeling-app\\modeling-app\\out\\make\\squirrel.windows\\x64\\Zoo Modeling App-*Setup.exe" - ARM64_FILE: "D:\\a\\modeling-app\\modeling-app\\out\\make\\squirrel.windows\\arm64\\Zoo Modeling App-*Setup.exe" - run: | - signtool.exe sign /sha1 ${{ env.THUMBPRINT }} /tr http://timestamp.digicert.com /td SHA256 /fd SHA256 "${{ env.X64_FILE }}" - signtool.exe verify /v /pa "${{ env.X64_FILE }}" - signtool.exe sign /sha1 ${{ env.THUMBPRINT }} /tr http://timestamp.digicert.com /td SHA256 /fd SHA256 "${{ env.ARM64_FILE }}" - signtool.exe verify /v /pa "${{ env.ARM64_FILE }}" + - name: List artifacts in out/ + run: ls -R out - uses: actions/upload-artifact@v3 with: - path: "out/make/*/*/*" - - # TODO: Run e2e tests - - - build-test-app-ubuntu: - needs: [prepare-json-files] - runs-on: ubuntu-22.04 - steps: - - uses: actions/checkout@v4 - - - uses: actions/download-artifact@v3 - if: github.event_name == 'schedule' - - - name: Copy updated .json files - if: github.event_name == 'schedule' - run: | - ls -l artifact - cp artifact/package.json package.json - - - name: Sync node version and setup cache - uses: actions/setup-node@v4 - with: - node-version-file: '.nvmrc' - cache: 'yarn' # Set this to npm, yarn or pnpm. - - - run: yarn install - - - name: Setup Rust - uses: dtolnay/rust-toolchain@stable - - - uses: Swatinem/rust-cache@v2 - with: - workspaces: './src/wasm-lib' - - - name: Run build:wasm - run: "yarn build:wasm${{ env.BUILD_RELEASE == 'true' && '-dev' || ''}}" - - - name: Build the app for arm64 - run: "yarn electron-forge make --arch arm64" - - - name: Build the app for x64 - run: "yarn electron-forge make --arch x64" - - - name: List artifacts - run: "ls -R out/make" + name: out-${{ matrix.os }} + path: | + out/Zoo*.* + out/latest*.yml # TODO: add the 'Build for Mac TestFlight (nightly)' stage back - # TODO: sign the app (and updater bundle potentially) - - - uses: actions/upload-artifact@v3 - with: - path: "out/make/*/*/*" + # TODO: add the updater tests back publish-apps-release: @@ -262,88 +157,76 @@ jobs: permissions: contents: write if: ${{ github.event_name == 'release' || github.event_name == 'schedule' }} - needs: [prepare-json-files, build-test-app-macos, build-test-app-windows, build-test-app-ubuntu] + needs: [prepare-files, build-apps] env: - VERSION_NO_V: ${{ needs.prepare-json-files.outputs.version }} - VERSION: ${{ github.event_name == 'release' && format('v{0}', needs.prepare-json-files.outputs.version) || needs.prepare-json-files.outputs.version }} + VERSION_NO_V: ${{ needs.prepare-files.outputs.version }} + VERSION: ${{ github.event_name == 'schedule' && needs.prepare-files.outputs.version || format('v{0}', needs.prepare-files.outputs.version) }} PUB_DATE: ${{ github.event_name == 'release' && github.event.release.created_at || github.event.repository.updated_at }} - NOTES: ${{ github.event_name == 'release' && github.event.release.body || format('Nightly build, commit {0}', github.sha) }} - BUCKET_DIR: ${{ github.event_name == 'release' && 'dl.kittycad.io/releases/modeling-app' || 'dl.kittycad.io/releases/modeling-app/nightly' }} - WEBSITE_DIR: ${{ github.event_name == 'release' && 'dl.zoo.dev/releases/modeling-app' || 'dl.zoo.dev/releases/modeling-app/nightly' }} + NOTES: ${{ github.event_name == 'release' && github.event.release.body || format('Non-release build, commit {0}', github.sha) }} + BUCKET_DIR: ${{ github.event_name == 'schedule' && 'dl.kittycad.io/releases/modeling-app/nightly' || 'dl.kittycad.io/releases/modeling-app' }} + WEBSITE_DIR: ${{ github.event_name == 'schedule' && 'dl.zoo.dev/releases/modeling-app/nightly' || 'dl.zoo.dev/releases/modeling-app' }} URL_CODED_NAME: ${{ github.event_name == 'schedule' && 'Zoo%20Modeling%20App%20%28Nightly%29' || 'Zoo%20Modeling%20App' }} steps: - - uses: actions/download-artifact@v3 + - uses: actions/checkout@v4 - - name: Generate the update static endpoint - run: | - ls -l artifact/*/*oo* - DARWIN_SIG=`cat artifact/macos/*.app.tar.gz.sig` - WINDOWS_X86_64_SIG=`cat artifact/msi/*x64*.msi.zip.sig` - WINDOWS_AARCH64_SIG=`cat artifact/msi/*arm64*.msi.zip.sig` - RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} - jq --null-input \ - --arg version "${VERSION}" \ - --arg pub_date "${PUB_DATE}" \ - --arg notes "${NOTES}" \ - --arg darwin_sig "$DARWIN_SIG" \ - --arg darwin_url "$RELEASE_DIR/macos/${{ env.URL_CODED_NAME }}.app.tar.gz" \ - --arg windows_x86_64_sig "$WINDOWS_X86_64_SIG" \ - --arg windows_x86_64_url "$RELEASE_DIR/msi/${{ env.URL_CODED_NAME }}_${VERSION_NO_V}_x64_en-US.msi.zip" \ - --arg windows_aarch64_sig "$WINDOWS_AARCH64_SIG" \ - --arg windows_aarch64_url "$RELEASE_DIR/msi/${{ env.URL_CODED_NAME }}_${VERSION_NO_V}_arm64_en-US.msi.zip" \ - '{ - "version": $version, - "pub_date": $pub_date, - "notes": $notes, - "platforms": { - "darwin-x86_64": { - "signature": $darwin_sig, - "url": $darwin_url - }, - "darwin-aarch64": { - "signature": $darwin_sig, - "url": $darwin_url - }, - "windows-x86_64": { - "signature": $windows_x86_64_sig, - "url": $windows_x86_64_url - }, - "windows-aarch64": { - "signature": $windows_aarch64_sig, - "url": $windows_aarch64_url - } - } - }' > last_update.json - cat last_update.json + - uses: actions/download-artifact@v3 + with: + name: out-windows-2022 + path: out + + - uses: actions/download-artifact@v3 + with: + name: out-macos-14 + path: out + + - uses: actions/download-artifact@v3 + with: + name: out-ubuntu-22.04 + path: out - name: Generate the download static endpoint run: | - RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} + RELEASE_DIR=https://${WEBSITE_DIR} jq --null-input \ --arg version "${VERSION}" \ --arg pub_date "${PUB_DATE}" \ --arg notes "${NOTES}" \ - --arg darwin_url "$RELEASE_DIR/dmg/${{ env.URL_CODED_NAME }}_${VERSION_NO_V}_universal.dmg" \ - --arg windows_x86_64_url "$RELEASE_DIR/msi/${{ env.URL_CODED_NAME }}_${VERSION_NO_V}_x64_en-US.msi" \ - --arg windows_aarch64_url "$RELEASE_DIR/msi/${{ env.URL_CODED_NAME }}_${VERSION_NO_V}_arm64_en-US.msi" \ + --arg mac_arm64_url "$RELEASE_DIR/${{ env.URL_CODED_NAME }}-${VERSION_NO_V}-arm64-mac.dmg" \ + --arg mac_x64_url "$RELEASE_DIR/${{ env.URL_CODED_NAME }}-${VERSION_NO_V}-x64-mac.dmg" \ + --arg windows_arm64_url "$RELEASE_DIR/${{ env.URL_CODED_NAME }}-${VERSION_NO_V}-arm64-win.msi" \ + --arg windows_x64_url "$RELEASE_DIR/${{ env.URL_CODED_NAME }}-${VERSION_NO_V}-x64-win.msi" \ + --arg linux_arm64_url "$RELEASE_DIR/${{ env.URL_CODED_NAME }}-${VERSION_NO_V}-arm64-linux.AppImage" \ + --arg linux_x64_url "$RELEASE_DIR/${{ env.URL_CODED_NAME }}-${VERSION_NO_V}-x86_64-linux.AppImage" \ '{ "version": $version, "pub_date": $pub_date, "notes": $notes, "platforms": { - "dmg-universal": { - "url": $darwin_url + "dmg-arm64": { + "url": $mac_arm64_url }, - "msi-x86_64": { - "url": $windows_x86_64_url + "dmg-x64": { + "url": $mac_x64_url }, - "msi-aarch64": { - "url": $windows_aarch64_url + "msi-arm64": { + "url": $windows_arm64_url + }, + "msi-x64": { + "url": $windows_x64_url + }, + "appimage-arm64": { + "url": $linux_arm64_url + }, + "appimage-x64": { + "url": $linux_x64_url } } }' > last_download.json cat last_download.json + - name: List artifacts + run: "ls -R out" + - name: Authenticate to Google Cloud uses: 'google-github-actions/auth@v2.1.5' with: @@ -352,24 +235,44 @@ jobs: - name: Set up Google Cloud SDK uses: google-github-actions/setup-gcloud@v2.1.0 with: - project_id: kittycadapi + project_id: ${{ env.GOOGLE_CLOUD_PROJECT_ID }} - name: Upload release files to public bucket - uses: google-github-actions/upload-cloud-storage@v2.1.3 + uses: google-github-actions/upload-cloud-storage@v2.2.0 with: - path: artifact - glob: '*/Zoo*' + path: out + glob: 'Zoo*' parent: false - destination: ${{ env.BUCKET_DIR }}/${{ env.VERSION }} - - - name: Upload update endpoint to public bucket - uses: google-github-actions/upload-cloud-storage@v2.1.3 - with: - path: last_update.json destination: ${{ env.BUCKET_DIR }} + # TODO: remove workaround introduced in https://github.com/KittyCAD/modeling-app/issues/3817 + - name: Upload release files to public bucket (test/electron-builder workaround) + uses: google-github-actions/upload-cloud-storage@v2.2.0 + with: + path: out + glob: 'Zoo*' + parent: false + destination: '${{ env.BUCKET_DIR }}/test/electron-builder' + + - name: Upload update endpoint to public bucket + uses: google-github-actions/upload-cloud-storage@v2.2.0 + with: + path: out + glob: 'latest*' + parent: false + destination: ${{ env.BUCKET_DIR }} + + # TODO: remove workaround introduced in https://github.com/KittyCAD/modeling-app/issues/3817 + - name: Upload update endpoint to public bucket (test/electron-builder workaround) + uses: google-github-actions/upload-cloud-storage@v2.2.0 + with: + path: out + glob: 'latest*' + parent: false + destination: '${{ env.BUCKET_DIR }}/test/electron-builder' + - name: Upload download endpoint to public bucket - uses: google-github-actions/upload-cloud-storage@v2.1.3 + uses: google-github-actions/upload-cloud-storage@v2.2.0 with: path: last_download.json destination: ${{ env.BUCKET_DIR }} @@ -378,7 +281,9 @@ jobs: if: ${{ github.event_name == 'release' }} uses: softprops/action-gh-release@v2 with: - files: 'artifact/*/Zoo*' + files: 'out/Zoo*' + + # TODO: Add GitHub publisher announce_release: needs: [publish-apps-release] diff --git a/.github/workflows/cargo-clippy.yml b/.github/workflows/cargo-clippy.yml index b50a496b9..84100f24f 100644 --- a/.github/workflows/cargo-clippy.yml +++ b/.github/workflows/cargo-clippy.yml @@ -28,6 +28,7 @@ jobs: dir: ['src/wasm-lib'] steps: - uses: actions/checkout@v4 + - uses: taiki-e/install-action@just - name: Install latest rust uses: actions-rs/toolchain@v1 with: @@ -41,7 +42,7 @@ jobs: - name: Run clippy run: | cd "${{ matrix.dir }}" - cargo clippy --all --tests --benches -- -D warnings + just lint # If this fails, run "cargo check" to update Cargo.lock, # then add Cargo.lock to the PR. - name: Check Cargo.lock doesn't need updating diff --git a/.github/workflows/playwright.yml b/.github/workflows/playwright.yml index 45ffa8421..21cfac5ad 100644 --- a/.github/workflows/playwright.yml +++ b/.github/workflows/playwright.yml @@ -262,7 +262,7 @@ jobs: fail-fast: false matrix: os: [ubuntu-latest, windows-latest, macos-14] - timeout-minutes: 30 + timeout-minutes: 40 runs-on: ${{ matrix.os }} needs: check-rust-changes steps: @@ -381,7 +381,7 @@ jobs: echo "retried=true" >>$GITHUB_OUTPUT echo "run playwright with last failed tests and retry $retry" if [[ "$IS_UBUNTU" == "true" ]]; then - xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn playwright test --config=playwright.electron.config.ts --grep=@electron || true + xvfb-run --auto-servernum --server-args="-screen 0 1280x960x24" -- yarn playwright test --config=playwright.electron.config.ts --last-failed --grep=@electron || true else yarn playwright test --config=playwright.electron.config.ts --grep=@electron || true fi diff --git a/.gitignore b/.gitignore index f88189f84..baa59a3d4 100644 --- a/.gitignore +++ b/.gitignore @@ -59,6 +59,7 @@ src/wasm-lib/grackle/stdlib_cube_partial.json Mac_App_Distribution.provisionprofile *.tsbuildinfo +src/wasm-lib/pkg venv .vite/ diff --git a/Makefile b/Makefile index 20d11334f..5b01a2652 100644 --- a/Makefile +++ b/Makefile @@ -7,6 +7,14 @@ XSTATE_TYPEGENS := $(wildcard src/machines/*.typegen.ts) dev: node_modules public/wasm_lib_bg.wasm $(XSTATE_TYPEGENS) yarn start +# I'm sorry this is so specific to my setup you may as well ignore this. +# This is so you don't have to deal with electron windows popping up constantly. +# It should work for you other Linux users. +lee-electron-test: + Xephyr -br -ac -noreset -screen 1200x500 :2 & + DISPLAY=:2 NODE_ENV=development PW_TEST_CONNECT_WS_ENDPOINT=ws://127.0.0.1:4444/ yarn tron:test -g "when using the file tree" + killall Xephyr + $(XSTATE_TYPEGENS): $(TS_SRC) yarn xstate typegen 'src/**/*.ts?(x)' diff --git a/README.md b/README.md index d232650b6..d682bdbbe 100644 --- a/README.md +++ b/README.md @@ -351,25 +351,6 @@ PS: for the debug panel, the following JSON is useful for snapping the camera -### Tauri e2e tests - -#### Windows (local only until the CI edge version mismatch is fixed) - -``` -yarn install -yarn build:wasm-dev -cp src/wasm-lib/pkg/wasm_lib_bg.wasm public -yarn vite build --mode development -yarn tauri build --debug -b -$env:KITTYCAD_API_TOKEN="" -$env:VITE_KC_API_BASE_URL="https://api.dev.zoo.dev" -$env:E2E_TAURI_ENABLED="true" -$env:TS_NODE_COMPILER_OPTIONS='{"module": "commonjs"}' -$env:E2E_APPLICATION=".\src-tauri\target\debug\Zoo Modeling App.exe" -Stop-Process -Name msedgedriver -yarn wdio run wdio.conf.ts -``` - ## KCL For how to contribute to KCL, [see our KCL README](https://github.com/KittyCAD/modeling-app/tree/main/src/wasm-lib/kcl). diff --git a/add-osx-cert.sh b/add-osx-cert.sh deleted file mode 100644 index 173322565..000000000 --- a/add-osx-cert.sh +++ /dev/null @@ -1,24 +0,0 @@ -#!/usr/bin/env sh -# From https://dev.to/rwwagner90/signing-electron-apps-with-github-actions-4cof - -KEY_CHAIN=build.keychain -CERTIFICATE_P12=certificate.p12 - -# Recreate the certificate from the secure environment variable -echo $APPLE_CERTIFICATE | base64 --decode > $CERTIFICATE_P12 - -#create a keychain -security create-keychain -p actions $KEY_CHAIN - -# Make the keychain the default so identities are found -security default-keychain -s $KEY_CHAIN - -# Unlock the keychain -security unlock-keychain -p actions $KEY_CHAIN - -security import $CERTIFICATE_P12 -k $KEY_CHAIN -P $APPLE_CERTIFICATE_PASSWORD -T /usr/bin/codesign; - -security set-key-partition-list -S apple-tool:,apple: -s -k actions $KEY_CHAIN - -# remove certs -rm -fr *.p12 \ No newline at end of file diff --git a/docs/kcl/index.md b/docs/kcl/index.md index ab59cfc83..e11e2cd85 100644 --- a/docs/kcl/index.md +++ b/docs/kcl/index.md @@ -56,6 +56,7 @@ layout: manual * [`line`](kcl/line) * [`lineTo`](kcl/lineTo) * [`ln`](kcl/ln) +* [`loft`](kcl/loft) * [`log`](kcl/log) * [`log10`](kcl/log10) * [`log2`](kcl/log2) @@ -63,6 +64,7 @@ layout: manual * [`max`](kcl/max) * [`min`](kcl/min) * [`mm`](kcl/mm) +* [`offsetPlane`](kcl/offsetPlane) * [`patternCircular2d`](kcl/patternCircular2d) * [`patternCircular3d`](kcl/patternCircular3d) * [`patternLinear2d`](kcl/patternLinear2d) diff --git a/docs/kcl/loft.md b/docs/kcl/loft.md new file mode 100644 index 000000000..80f4e1767 --- /dev/null +++ b/docs/kcl/loft.md @@ -0,0 +1,516 @@ +--- +title: "loft" +excerpt: "Create a 3D surface or solid by interpolating between two or more sketches." +layout: manual +--- + +Create a 3D surface or solid by interpolating between two or more sketches. + +The sketches need to closed and on the same plane. + +```js +loft(sketch_groups: [SketchGroup], data?: LoftData) -> ExtrudeGroup +``` + +### Examples + +```js +// Loft a square and a triangle. +const squareSketch = startSketchOn('XY') + |> startProfileAt([-100, 200], %) + |> line([200, 0], %) + |> line([0, -200], %) + |> line([-200, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +const triangleSketch = startSketchOn(offsetPlane('XY', 75)) + |> startProfileAt([0, 125], %) + |> line([-15, -30], %) + |> line([30, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +loft([squareSketch, triangleSketch]) +``` + +![Rendered example of loft 0]() + +```js +// Loft a square, a circle, and another circle. +const squareSketch = startSketchOn('XY') + |> startProfileAt([-100, 200], %) + |> line([200, 0], %) + |> line([0, -200], %) + |> line([-200, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +const circleSketch0 = startSketchOn(offsetPlane('XY', 75)) + |> circle([0, 100], 50, %) + +const circleSketch1 = startSketchOn(offsetPlane('XY', 150)) + |> circle([0, 100], 20, %) + +loft([ + squareSketch, + circleSketch0, + circleSketch1 +]) +``` + +![Rendered example of loft 1]() + +```js +// Loft a square, a circle, and another circle with options. +const squareSketch = startSketchOn('XY') + |> startProfileAt([-100, 200], %) + |> line([200, 0], %) + |> line([0, -200], %) + |> line([-200, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +const circleSketch0 = startSketchOn(offsetPlane('XY', 75)) + |> circle([0, 100], 50, %) + +const circleSketch1 = startSketchOn(offsetPlane('XY', 150)) + |> circle([0, 100], 20, %) + +loft([ + squareSketch, + circleSketch0, + circleSketch1 +], { + // This can be set to override the automatically determined + // topological base curve, which is usually the first section encountered. + baseCurveIndex: 0, + // Attempt to approximate rational curves (such as arcs) using a bezier. + // This will remove banding around interpolations between arcs and non-arcs. + // It may produce errors in other scenarios Over time, this field won't be necessary. + bezApproximateRational: false, + // Tolerance for the loft operation. + tolerance: 0.000001, + // Degree of the interpolation. Must be greater than zero. + // For example, use 2 for quadratic, or 3 for cubic interpolation in + // the V direction. This defaults to 2, if not specified. + vDegree: 2 +}) +``` + +![Rendered example of loft 2]() + +### Arguments + +* `sketch_groups`: `[SketchGroup]` (REQUIRED) +* `data`: `LoftData` - Data for a loft. (OPTIONAL) +```js +{ + // This can be set to override the automatically determined topological base curve, which is usually the first section encountered. + baseCurveIndex: number, + // Attempt to approximate rational curves (such as arcs) using a bezier. This will remove banding around interpolations between arcs and non-arcs. It may produce errors in other scenarios Over time, this field won't be necessary. + bezApproximateRational: bool, + // Tolerance for the loft operation. + tolerance: number, + // Degree of the interpolation. Must be greater than zero. For example, use 2 for quadratic, or 3 for cubic interpolation in the V direction. This defaults to 2, if not specified. + vDegree: number, +} +``` + +### Returns + +`ExtrudeGroup` - An extrude group is a collection of extrude surfaces. +```js +{ + // Chamfers or fillets on this extrude group. + edgeCuts: [{ + // The engine id of the edge to fillet. + edgeId: uuid, + // The id of the engine command that called this fillet. + id: uuid, + radius: number, + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "fillet", +} | +{ + // The engine id of the edge to chamfer. + edgeId: uuid, + // The id of the engine command that called this chamfer. + id: uuid, + length: number, + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "chamfer", +}], + // The id of the extrusion end cap + endCapId: uuid, + // The height of the extrude group. + height: number, + // The id of the extrude group. + id: uuid, + // The sketch group. + sketchGroup: { + // The id of the sketch group (this will change when the engine's reference to it changes. + id: uuid, + // What the sketch is on (can be a plane or a face). + on: { + // The id of the plane. + id: uuid, + // Origin of the plane. + origin: { + x: number, + y: number, + z: number, +}, + type: "plane", + // Type for a plane. + value: "XY" | "XZ" | "YZ" | "Custom", + // What should the plane’s X axis be? + xAxis: { + x: number, + y: number, + z: number, +}, + // What should the plane’s Y axis be? + yAxis: { + x: number, + y: number, + z: number, +}, + // The z-axis (normal). + zAxis: { + x: number, + y: number, + z: number, +}, +} | +{ + // The extrude group the face is on. + extrudeGroup: { + // Chamfers or fillets on this extrude group. + edgeCuts: [{ + // The engine id of the edge to fillet. + edgeId: uuid, + // The id of the engine command that called this fillet. + id: uuid, + radius: number, + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "fillet", +} | +{ + // The engine id of the edge to chamfer. + edgeId: uuid, + // The id of the engine command that called this chamfer. + id: uuid, + length: number, + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "chamfer", +}], + // The id of the extrusion end cap + endCapId: uuid, + // The height of the extrude group. + height: number, + // The id of the extrude group. + id: uuid, + // The sketch group. + sketchGroup: SketchGroup, + // The id of the extrusion start cap + startCapId: uuid, + // The extrude surfaces. + value: [{ + // The face id for the extrude plane. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "extrudePlane", +} | +{ + // The face id for the extrude plane. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "extrudeArc", +} | +{ + // The id for the chamfer surface. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "chamfer", +} | +{ + // The id for the fillet surface. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "fillet", +}], +}, + // The id of the face. + id: uuid, + type: "face", + // The tag of the face. + value: string, + // What should the face’s X axis be? + xAxis: { + x: number, + y: number, + z: number, +}, + // What should the face’s Y axis be? + yAxis: { + x: number, + y: number, + z: number, +}, + // The z-axis (normal). + zAxis: { + x: number, + y: number, + z: number, +}, +}, + // The starting path. + start: { + // The from point. + from: [number, number], + // The tag of the path. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + // The to point. + to: [number, number], +}, + // Tag identifiers that have been declared in this sketch group. + tags: { +}, + // The paths in the sketch group. + value: [{ + // The from point. + from: [number, number], + // The tag of the path. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + // The to point. + to: [number, number], + type: "ToPoint", +} | +{ + // arc's direction + ccw: bool, + // the arc's center + center: [number, number], + // The from point. + from: [number, number], + // The tag of the path. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + // The to point. + to: [number, number], + type: "TangentialArcTo", +} | +{ + // arc's direction + ccw: bool, + // the arc's center + center: [number, number], + // The from point. + from: [number, number], + // The tag of the path. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + // The to point. + to: [number, number], + type: "TangentialArc", +} | +{ + // The from point. + from: [number, number], + // The tag of the path. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + // The to point. + to: [number, number], + type: "Horizontal", + // The x coordinate. + x: number, +} | +{ + // The from point. + from: [number, number], + // The tag of the path. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + // The to point. + to: [number, number], + type: "AngledLineTo", + // The x coordinate. + x: number, + // The y coordinate. + y: number, +} | +{ + // The from point. + from: [number, number], + // The tag of the path. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + // The to point. + to: [number, number], + type: "Base", +}], +}, + // The id of the extrusion start cap + startCapId: uuid, + // The extrude surfaces. + value: [{ + // The face id for the extrude plane. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "extrudePlane", +} | +{ + // The face id for the extrude plane. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "extrudeArc", +} | +{ + // The id for the chamfer surface. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "chamfer", +} | +{ + // The id for the fillet surface. + faceId: uuid, + // The id of the geometry. + id: uuid, + // The source range. + sourceRange: [number, number], + // The tag. + tag: { + digest: [number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number, number], + end: number, + start: number, + value: string, +}, + type: "fillet", +}], +} +``` + + + diff --git a/docs/kcl/offsetPlane.md b/docs/kcl/offsetPlane.md new file mode 100644 index 000000000..329a03a37 --- /dev/null +++ b/docs/kcl/offsetPlane.md @@ -0,0 +1,138 @@ +--- +title: "offsetPlane" +excerpt: "Offset a plane by a distance along its normal." +layout: manual +--- + +Offset a plane by a distance along its normal. + +For example, if you offset the 'XZ' plane by 10, the new plane will be parallel to the 'XZ' plane and 10 units away from it. + +```js +offsetPlane(std_plane: StandardPlane, offset: number) -> PlaneData +``` + +### Examples + +```js +// Loft a square and a circle on the `XY` plane using offset. +const squareSketch = startSketchOn('XY') + |> startProfileAt([-100, 200], %) + |> line([200, 0], %) + |> line([0, -200], %) + |> line([-200, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +const circleSketch = startSketchOn(offsetPlane('XY', 150)) + |> circle([0, 100], 50, %) + +loft([squareSketch, circleSketch]) +``` + +![Rendered example of offsetPlane 0]() + +```js +// Loft a square and a circle on the `XZ` plane using offset. +const squareSketch = startSketchOn('XZ') + |> startProfileAt([-100, 200], %) + |> line([200, 0], %) + |> line([0, -200], %) + |> line([-200, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +const circleSketch = startSketchOn(offsetPlane('XZ', 150)) + |> circle([0, 100], 50, %) + +loft([squareSketch, circleSketch]) +``` + +![Rendered example of offsetPlane 1]() + +```js +// Loft a square and a circle on the `YZ` plane using offset. +const squareSketch = startSketchOn('YZ') + |> startProfileAt([-100, 200], %) + |> line([200, 0], %) + |> line([0, -200], %) + |> line([-200, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +const circleSketch = startSketchOn(offsetPlane('YZ', 150)) + |> circle([0, 100], 50, %) + +loft([squareSketch, circleSketch]) +``` + +![Rendered example of offsetPlane 2]() + +```js +// Loft a square and a circle on the `-XZ` plane using offset. +const squareSketch = startSketchOn('-XZ') + |> startProfileAt([-100, 200], %) + |> line([200, 0], %) + |> line([0, -200], %) + |> line([-200, 0], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) + +const circleSketch = startSketchOn(offsetPlane('-XZ', -150)) + |> circle([0, 100], 50, %) + +loft([squareSketch, circleSketch]) +``` + +![Rendered example of offsetPlane 3]() + +### Arguments + +* `std_plane`: `StandardPlane` - One of the standard planes. (REQUIRED) +```js +"XY" | "-XY" | "XZ" | "-XZ" | "YZ" | "-YZ" +``` +* `offset`: `number` (REQUIRED) + +### Returns + +`PlaneData` - Data for a plane. +```js +"XY" | +"-XY" | +"XZ" | +"-XZ" | +"YZ" | +"-YZ" | +{ + plane: { + // Origin of the plane. + origin: { + x: number, + y: number, + z: number, +}, + // What should the plane’s X axis be? + xAxis: { + x: number, + y: number, + z: number, +}, + // What should the plane’s Y axis be? + yAxis: { + x: number, + y: number, + z: number, +}, + // The z-axis (normal). + zAxis: { + x: number, + y: number, + z: number, +}, +}, +} +``` + + + diff --git a/docs/kcl/std.json b/docs/kcl/std.json index 95a2e0b1f..389a6ea6c 100644 --- a/docs/kcl/std.json +++ b/docs/kcl/std.json @@ -143020,6 +143020,4703 @@ "const exampleSketch = startSketchOn(\"XZ\")\n |> startProfileAt([0, 0], %)\n |> line([ln(100), 15], %)\n |> line([5, -6], %)\n |> line([-10, -10], %)\n |> close(%)\n\nconst example = extrude(5, exampleSketch)" ] }, + { + "name": "loft", + "summary": "Create a 3D surface or solid by interpolating between two or more sketches.", + "description": "The sketches need to closed and on the same plane.", + "tags": [], + "args": [ + { + "name": "sketch_groups", + "type": "[SketchGroup]", + "schema": { + "type": "array", + "items": { + "description": "A sketch group is a collection of paths.", + "type": "object", + "required": [ + "__meta", + "id", + "on", + "start", + "value" + ], + "properties": { + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "id": { + "description": "The id of the sketch group (this will change when the engine's reference to it changes.", + "type": "string", + "format": "uuid" + }, + "on": { + "description": "What the sketch is on (can be a plane or a face).", + "oneOf": [ + { + "description": "A plane.", + "type": "object", + "required": [ + "__meta", + "id", + "origin", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "__meta": { + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "id": { + "description": "The id of the plane.", + "type": "string", + "format": "uuid" + }, + "origin": { + "description": "Origin of the plane.", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "type": { + "type": "string", + "enum": [ + "plane" + ] + }, + "value": { + "description": "Type for a plane.", + "oneOf": [ + { + "type": "string", + "enum": [ + "XY", + "XZ", + "YZ" + ] + }, + { + "description": "A custom plane.", + "type": "string", + "enum": [ + "Custom" + ] + } + ] + }, + "xAxis": { + "description": "What should the plane’s X axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "yAxis": { + "description": "What should the plane’s Y axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "zAxis": { + "description": "The z-axis (normal).", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + } + } + }, + { + "description": "A face.", + "type": "object", + "required": [ + "__meta", + "extrudeGroup", + "id", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "__meta": { + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "extrudeGroup": { + "description": "The extrude group the face is on.", + "type": "object", + "required": [ + "__meta", + "height", + "id", + "sketchGroup", + "value" + ], + "properties": { + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "edgeCuts": { + "description": "Chamfers or fillets on this extrude group.", + "type": "array", + "items": { + "description": "A fillet or a chamfer.", + "oneOf": [ + { + "description": "A fillet.", + "type": "object", + "required": [ + "edgeId", + "id", + "radius", + "type" + ], + "properties": { + "edgeId": { + "description": "The engine id of the edge to fillet.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the engine command that called this fillet.", + "type": "string", + "format": "uuid" + }, + "radius": { + "type": "number", + "format": "double" + }, + "tag": { + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + }, + { + "description": "A chamfer.", + "type": "object", + "required": [ + "edgeId", + "id", + "length", + "type" + ], + "properties": { + "edgeId": { + "description": "The engine id of the edge to chamfer.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the engine command that called this chamfer.", + "type": "string", + "format": "uuid" + }, + "length": { + "type": "number", + "format": "double" + }, + "tag": { + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + } + ] + } + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "height": { + "description": "The height of the extrude group.", + "type": "number", + "format": "double" + }, + "id": { + "description": "The id of the extrude group.", + "type": "string", + "format": "uuid" + }, + "sketchGroup": { + "description": "The sketch group.", + "$ref": "#/components/schemas/SketchGroup" + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + } + ] + } + } + } + }, + "id": { + "description": "The id of the face.", + "type": "string", + "format": "uuid" + }, + "type": { + "type": "string", + "enum": [ + "face" + ] + }, + "value": { + "description": "The tag of the face.", + "type": "string" + }, + "xAxis": { + "description": "What should the face’s X axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "yAxis": { + "description": "What should the face’s Y axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "zAxis": { + "description": "The z-axis (normal).", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + } + } + } + ] + }, + "start": { + "description": "The starting path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "tags": { + "description": "Tag identifiers that have been declared in this sketch group.", + "type": "object", + "additionalProperties": { + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "__meta": { + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "info": { + "description": "Engine information for a tag.", + "type": "object", + "required": [ + "id", + "sketchGroup" + ], + "properties": { + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" + }, + "path": { + "description": "The path the tag is on.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + } + }, + "nullable": true + }, + "sketchGroup": { + "description": "The sketch group the tag is on.", + "type": "string", + "format": "uuid" + }, + "surface": { + "description": "The surface information for the tag.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + } + ], + "nullable": true + } + }, + "nullable": true + }, + "value": { + "type": "string" + } + } + } + }, + "value": { + "description": "The paths in the sketch group.", + "type": "array", + "items": { + "description": "A path.", + "oneOf": [ + { + "description": "A path that goes to a point.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "ToPoint" + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment that goes to a point", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "TangentialArcTo" + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "TangentialArc" + ] + } + } + }, + { + "description": "A path that is horizontal.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type", + "x" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "Horizontal" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + } + } + }, + { + "description": "An angled line to.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "AngledLineTo" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + } + } + }, + { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "Base" + ] + } + } + } + ] + } + } + } + } + }, + "required": true + }, + { + "name": "data", + "type": "LoftData", + "schema": { + "description": "Data for a loft.", + "type": "object", + "properties": { + "baseCurveIndex": { + "description": "This can be set to override the automatically determined topological base curve, which is usually the first section encountered.", + "default": null, + "type": "integer", + "format": "uint32", + "minimum": 0.0, + "nullable": true + }, + "bezApproximateRational": { + "description": "Attempt to approximate rational curves (such as arcs) using a bezier. This will remove banding around interpolations between arcs and non-arcs. It may produce errors in other scenarios Over time, this field won't be necessary.", + "default": null, + "type": "boolean", + "nullable": true + }, + "tolerance": { + "description": "Tolerance for the loft operation.", + "default": null, + "type": "number", + "format": "double", + "nullable": true + }, + "vDegree": { + "description": "Degree of the interpolation. Must be greater than zero. For example, use 2 for quadratic, or 3 for cubic interpolation in the V direction. This defaults to 2, if not specified.", + "type": "integer", + "format": "uint32", + "minimum": 1.0, + "nullable": true + } + }, + "nullable": true + }, + "required": false + } + ], + "returnValue": { + "name": "", + "type": "ExtrudeGroup", + "schema": { + "description": "An extrude group is a collection of extrude surfaces.", + "type": "object", + "required": [ + "__meta", + "height", + "id", + "sketchGroup", + "value" + ], + "properties": { + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "edgeCuts": { + "description": "Chamfers or fillets on this extrude group.", + "type": "array", + "items": { + "description": "A fillet or a chamfer.", + "oneOf": [ + { + "description": "A fillet.", + "type": "object", + "required": [ + "edgeId", + "id", + "radius", + "type" + ], + "properties": { + "edgeId": { + "description": "The engine id of the edge to fillet.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the engine command that called this fillet.", + "type": "string", + "format": "uuid" + }, + "radius": { + "type": "number", + "format": "double" + }, + "tag": { + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + }, + { + "description": "A chamfer.", + "type": "object", + "required": [ + "edgeId", + "id", + "length", + "type" + ], + "properties": { + "edgeId": { + "description": "The engine id of the edge to chamfer.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the engine command that called this chamfer.", + "type": "string", + "format": "uuid" + }, + "length": { + "type": "number", + "format": "double" + }, + "tag": { + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + } + ] + } + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "height": { + "description": "The height of the extrude group.", + "type": "number", + "format": "double" + }, + "id": { + "description": "The id of the extrude group.", + "type": "string", + "format": "uuid" + }, + "sketchGroup": { + "description": "The sketch group.", + "type": "object", + "required": [ + "__meta", + "id", + "on", + "start", + "value" + ], + "properties": { + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "id": { + "description": "The id of the sketch group (this will change when the engine's reference to it changes.", + "type": "string", + "format": "uuid" + }, + "on": { + "description": "What the sketch is on (can be a plane or a face).", + "oneOf": [ + { + "description": "A plane.", + "type": "object", + "required": [ + "__meta", + "id", + "origin", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "__meta": { + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "id": { + "description": "The id of the plane.", + "type": "string", + "format": "uuid" + }, + "origin": { + "description": "Origin of the plane.", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "type": { + "type": "string", + "enum": [ + "plane" + ] + }, + "value": { + "description": "Type for a plane.", + "oneOf": [ + { + "type": "string", + "enum": [ + "XY", + "XZ", + "YZ" + ] + }, + { + "description": "A custom plane.", + "type": "string", + "enum": [ + "Custom" + ] + } + ] + }, + "xAxis": { + "description": "What should the plane’s X axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "yAxis": { + "description": "What should the plane’s Y axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "zAxis": { + "description": "The z-axis (normal).", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + } + } + }, + { + "description": "A face.", + "type": "object", + "required": [ + "__meta", + "extrudeGroup", + "id", + "type", + "value", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "__meta": { + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "extrudeGroup": { + "description": "The extrude group the face is on.", + "type": "object", + "required": [ + "__meta", + "height", + "id", + "sketchGroup", + "value" + ], + "properties": { + "__meta": { + "description": "Metadata.", + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "edgeCuts": { + "description": "Chamfers or fillets on this extrude group.", + "type": "array", + "items": { + "description": "A fillet or a chamfer.", + "oneOf": [ + { + "description": "A fillet.", + "type": "object", + "required": [ + "edgeId", + "id", + "radius", + "type" + ], + "properties": { + "edgeId": { + "description": "The engine id of the edge to fillet.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the engine command that called this fillet.", + "type": "string", + "format": "uuid" + }, + "radius": { + "type": "number", + "format": "double" + }, + "tag": { + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + }, + { + "description": "A chamfer.", + "type": "object", + "required": [ + "edgeId", + "id", + "length", + "type" + ], + "properties": { + "edgeId": { + "description": "The engine id of the edge to chamfer.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the engine command that called this chamfer.", + "type": "string", + "format": "uuid" + }, + "length": { + "type": "number", + "format": "double" + }, + "tag": { + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + } + ] + } + }, + "endCapId": { + "description": "The id of the extrusion end cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "height": { + "description": "The height of the extrude group.", + "type": "number", + "format": "double" + }, + "id": { + "description": "The id of the extrude group.", + "type": "string", + "format": "uuid" + }, + "sketchGroup": { + "description": "The sketch group.", + "$ref": "#/components/schemas/SketchGroup" + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + } + ] + } + } + } + }, + "id": { + "description": "The id of the face.", + "type": "string", + "format": "uuid" + }, + "type": { + "type": "string", + "enum": [ + "face" + ] + }, + "value": { + "description": "The tag of the face.", + "type": "string" + }, + "xAxis": { + "description": "What should the face’s X axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "yAxis": { + "description": "What should the face’s Y axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "zAxis": { + "description": "The z-axis (normal).", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + } + } + } + ] + }, + "start": { + "description": "The starting path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "tags": { + "description": "Tag identifiers that have been declared in this sketch group.", + "type": "object", + "additionalProperties": { + "type": "object", + "required": [ + "__meta", + "value" + ], + "properties": { + "__meta": { + "type": "array", + "items": { + "description": "Metadata.", + "type": "object", + "required": [ + "sourceRange" + ], + "properties": { + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + } + }, + "info": { + "description": "Engine information for a tag.", + "type": "object", + "required": [ + "id", + "sketchGroup" + ], + "properties": { + "id": { + "description": "The id of the tagged object.", + "type": "string", + "format": "uuid" + }, + "path": { + "description": "The path the tag is on.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + } + }, + "nullable": true + }, + "sketchGroup": { + "description": "The sketch group the tag is on.", + "type": "string", + "format": "uuid" + }, + "surface": { + "description": "The surface information for the tag.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + } + ], + "nullable": true + } + }, + "nullable": true + }, + "value": { + "type": "string" + } + } + } + }, + "value": { + "description": "The paths in the sketch group.", + "type": "array", + "items": { + "description": "A path.", + "oneOf": [ + { + "description": "A path that goes to a point.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "ToPoint" + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment that goes to a point", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "TangentialArcTo" + ] + } + } + }, + { + "description": "A arc that is tangential to the last path segment", + "type": "object", + "required": [ + "__geoMeta", + "ccw", + "center", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "ccw": { + "description": "arc's direction", + "type": "boolean" + }, + "center": { + "description": "the arc's center", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "TangentialArc" + ] + } + } + }, + { + "description": "A path that is horizontal.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type", + "x" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "Horizontal" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double" + } + } + }, + { + "description": "An angled line to.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "AngledLineTo" + ] + }, + "x": { + "description": "The x coordinate.", + "type": "number", + "format": "double", + "nullable": true + }, + "y": { + "description": "The y coordinate.", + "type": "number", + "format": "double", + "nullable": true + } + } + }, + { + "description": "A base path.", + "type": "object", + "required": [ + "__geoMeta", + "from", + "to", + "type" + ], + "properties": { + "__geoMeta": { + "description": "Metadata.", + "type": "object", + "required": [ + "id", + "sourceRange" + ], + "properties": { + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + } + } + }, + "from": { + "description": "The from point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag of the path.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "to": { + "description": "The to point.", + "type": "array", + "items": { + "type": "number", + "format": "double" + }, + "maxItems": 2, + "minItems": 2 + }, + "type": { + "type": "string", + "enum": [ + "Base" + ] + } + } + } + ] + } + } + } + }, + "startCapId": { + "description": "The id of the extrusion start cap", + "type": "string", + "format": "uuid", + "nullable": true + }, + "value": { + "description": "The extrude surfaces.", + "type": "array", + "items": { + "description": "An extrude surface.", + "oneOf": [ + { + "description": "An extrude plane.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudePlane" + ] + } + } + }, + { + "description": "An extruded arc.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The face id for the extrude plane.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "extrudeArc" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the chamfer surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "chamfer" + ] + } + } + }, + { + "description": "Geometry metadata.", + "type": "object", + "required": [ + "faceId", + "id", + "sourceRange", + "type" + ], + "properties": { + "faceId": { + "description": "The id for the fillet surface.", + "type": "string", + "format": "uuid" + }, + "id": { + "description": "The id of the geometry.", + "type": "string", + "format": "uuid" + }, + "sourceRange": { + "description": "The source range.", + "type": "array", + "items": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "maxItems": 2, + "minItems": 2 + }, + "tag": { + "description": "The tag.", + "type": "object", + "required": [ + "end", + "start", + "value" + ], + "properties": { + "digest": { + "type": "array", + "items": { + "type": "integer", + "format": "uint8", + "minimum": 0.0 + }, + "maxItems": 32, + "minItems": 32, + "nullable": true + }, + "end": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "start": { + "type": "integer", + "format": "uint", + "minimum": 0.0 + }, + "value": { + "type": "string" + } + }, + "nullable": true + }, + "type": { + "type": "string", + "enum": [ + "fillet" + ] + } + } + } + ] + } + } + } + }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "// Loft a square and a triangle.\nconst squareSketch = startSketchOn('XY')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nconst triangleSketch = startSketchOn(offsetPlane('XY', 75))\n |> startProfileAt([0, 125], %)\n |> line([-15, -30], %)\n |> line([30, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nloft([squareSketch, triangleSketch])", + "// Loft a square, a circle, and another circle.\nconst squareSketch = startSketchOn('XY')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nconst circleSketch0 = startSketchOn(offsetPlane('XY', 75))\n |> circle([0, 100], 50, %)\n\nconst circleSketch1 = startSketchOn(offsetPlane('XY', 150))\n |> circle([0, 100], 20, %)\n\nloft([\n squareSketch,\n circleSketch0,\n circleSketch1\n])", + "// Loft a square, a circle, and another circle with options.\nconst squareSketch = startSketchOn('XY')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nconst circleSketch0 = startSketchOn(offsetPlane('XY', 75))\n |> circle([0, 100], 50, %)\n\nconst circleSketch1 = startSketchOn(offsetPlane('XY', 150))\n |> circle([0, 100], 20, %)\n\nloft([\n squareSketch,\n circleSketch0,\n circleSketch1\n], {\n // This can be set to override the automatically determined\n // topological base curve, which is usually the first section encountered.\n baseCurveIndex: 0,\n // Attempt to approximate rational curves (such as arcs) using a bezier.\n // This will remove banding around interpolations between arcs and non-arcs.\n // It may produce errors in other scenarios Over time, this field won't be necessary.\n bezApproximateRational: false,\n // Tolerance for the loft operation.\n tolerance: 0.000001,\n // Degree of the interpolation. Must be greater than zero.\n // For example, use 2 for quadratic, or 3 for cubic interpolation in\n // the V direction. This defaults to 2, if not specified.\n vDegree: 2\n})" + ] + }, { "name": "log", "summary": "Compute the logarithm of the number with respect to an arbitrary base.", @@ -143246,6 +147943,248 @@ "const totalWidth = 10 * mm()" ] }, + { + "name": "offsetPlane", + "summary": "Offset a plane by a distance along its normal.", + "description": "For example, if you offset the 'XZ' plane by 10, the new plane will be parallel to the 'XZ' plane and 10 units away from it.", + "tags": [], + "args": [ + { + "name": "std_plane", + "type": "StandardPlane", + "schema": { + "description": "One of the standard planes.", + "oneOf": [ + { + "description": "The XY plane.", + "type": "string", + "enum": [ + "XY" + ] + }, + { + "description": "The opposite side of the XY plane.", + "type": "string", + "enum": [ + "-XY" + ] + }, + { + "description": "The XZ plane.", + "type": "string", + "enum": [ + "XZ" + ] + }, + { + "description": "The opposite side of the XZ plane.", + "type": "string", + "enum": [ + "-XZ" + ] + }, + { + "description": "The YZ plane.", + "type": "string", + "enum": [ + "YZ" + ] + }, + { + "description": "The opposite side of the YZ plane.", + "type": "string", + "enum": [ + "-YZ" + ] + } + ] + }, + "required": true + }, + { + "name": "offset", + "type": "number", + "schema": { + "type": "number", + "format": "double" + }, + "required": true + } + ], + "returnValue": { + "name": "", + "type": "PlaneData", + "schema": { + "description": "Data for a plane.", + "oneOf": [ + { + "description": "The XY plane.", + "type": "string", + "enum": [ + "XY" + ] + }, + { + "description": "The opposite side of the XY plane.", + "type": "string", + "enum": [ + "-XY" + ] + }, + { + "description": "The XZ plane.", + "type": "string", + "enum": [ + "XZ" + ] + }, + { + "description": "The opposite side of the XZ plane.", + "type": "string", + "enum": [ + "-XZ" + ] + }, + { + "description": "The YZ plane.", + "type": "string", + "enum": [ + "YZ" + ] + }, + { + "description": "The opposite side of the YZ plane.", + "type": "string", + "enum": [ + "-YZ" + ] + }, + { + "description": "A defined plane.", + "type": "object", + "required": [ + "plane" + ], + "properties": { + "plane": { + "type": "object", + "required": [ + "origin", + "xAxis", + "yAxis", + "zAxis" + ], + "properties": { + "origin": { + "description": "Origin of the plane.", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "xAxis": { + "description": "What should the plane’s X axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "yAxis": { + "description": "What should the plane’s Y axis be?", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + }, + "zAxis": { + "description": "The z-axis (normal).", + "type": "object", + "required": [ + "x", + "y", + "z" + ], + "properties": { + "x": { + "type": "number", + "format": "double" + }, + "y": { + "type": "number", + "format": "double" + }, + "z": { + "type": "number", + "format": "double" + } + } + } + } + } + }, + "additionalProperties": false + } + ] + }, + "required": true + }, + "unpublished": false, + "deprecated": false, + "examples": [ + "// Loft a square and a circle on the `XY` plane using offset.\nconst squareSketch = startSketchOn('XY')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nconst circleSketch = startSketchOn(offsetPlane('XY', 150))\n |> circle([0, 100], 50, %)\n\nloft([squareSketch, circleSketch])", + "// Loft a square and a circle on the `XZ` plane using offset.\nconst squareSketch = startSketchOn('XZ')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nconst circleSketch = startSketchOn(offsetPlane('XZ', 150))\n |> circle([0, 100], 50, %)\n\nloft([squareSketch, circleSketch])", + "// Loft a square and a circle on the `YZ` plane using offset.\nconst squareSketch = startSketchOn('YZ')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nconst circleSketch = startSketchOn(offsetPlane('YZ', 150))\n |> circle([0, 100], 50, %)\n\nloft([squareSketch, circleSketch])", + "// Loft a square and a circle on the `-XZ` plane using offset.\nconst squareSketch = startSketchOn('-XZ')\n |> startProfileAt([-100, 200], %)\n |> line([200, 0], %)\n |> line([0, -200], %)\n |> line([-200, 0], %)\n |> lineTo([profileStartX(%), profileStartY(%)], %)\n |> close(%)\n\nconst circleSketch = startSketchOn(offsetPlane('-XZ', -150))\n |> circle([0, 100], 50, %)\n\nloft([squareSketch, circleSketch])" + ] + }, { "name": "patternCircular2d", "summary": "Repeat a 2-dimensional sketch some number of times along a partial or", diff --git a/e2e/playwright/basic-sketch.spec.ts b/e2e/playwright/basic-sketch.spec.ts index edb937b08..1463bee7f 100644 --- a/e2e/playwright/basic-sketch.spec.ts +++ b/e2e/playwright/basic-sketch.spec.ts @@ -96,33 +96,49 @@ async function doBasicSketch(page: Page, openPanes: string[]) { } // deselect line tool - await page.getByTestId('line').click() + const btnLine = page.getByTestId('line') + const btnLineAriaPressed = await btnLine.getAttribute('aria-pressed') + if (btnLineAriaPressed === 'true') { + await btnLine.click() + } + + await page.waitForTimeout(100) const line1 = await u.getSegmentBodyCoords(`[data-overlay-index="${0}"]`, 0) if (openPanes.includes('code')) { await expect .poll(async () => u.getGreatestPixDiff(line1, TEST_COLORS.WHITE)) .toBeLessThan(3) + await page.waitForTimeout(100) await expect - .poll(() => u.getGreatestPixDiff(line1, [249, 249, 249])) + .poll(async () => u.getGreatestPixDiff(line1, [249, 249, 249])) .toBeLessThan(3) + await page.waitForTimeout(100) } + // click between first two clicks to get center of the line await page.mouse.click(startXPx + PUR * 15, 500 - PUR * 10) await page.waitForTimeout(100) + if (openPanes.includes('code')) { - expect(await u.getGreatestPixDiff(line1, TEST_COLORS.BLUE)).toBeLessThan(3) + await expect( + await u.getGreatestPixDiff(line1, TEST_COLORS.BLUE) + ).toBeLessThan(3) await expect(await u.getGreatestPixDiff(line1, [0, 0, 255])).toBeLessThan(3) } // hold down shift await page.keyboard.down('Shift') + await page.waitForTimeout(100) + // click between the latest two clicks to get center of the line await page.mouse.click(startXPx + PUR * 10, 500 - PUR * 20) + await page.waitForTimeout(100) // selected two lines therefore there should be two cursors if (openPanes.includes('code')) { await expect(page.locator('.cm-cursor')).toHaveCount(2) + await page.waitForTimeout(100) } await page.getByRole('button', { name: 'Length: open menu' }).click() diff --git a/e2e/playwright/code-pane-and-errors.spec.ts b/e2e/playwright/code-pane-and-errors.spec.ts index 32f7d6341..de08da81e 100644 --- a/e2e/playwright/code-pane-and-errors.spec.ts +++ b/e2e/playwright/code-pane-and-errors.spec.ts @@ -27,9 +27,19 @@ test.describe('Code pane and errors', () => { const u = await getUtils(page) // Load the app with the working starter code - await page.addInitScript((code) => { - localStorage.setItem('persistCode', code) - }, bracket) + await page.addInitScript(() => { + localStorage.setItem( + 'persistCode', + `// Extruded Triangle +const sketch001 = startSketchOn('XZ') + |> startProfileAt([0, 0], %) + |> line([10, 0], %) + |> line([-5, 10], %) + |> lineTo([profileStartX(%), profileStartY(%)], %) + |> close(%) +const extrude001 = extrude(5, sketch001)` + ) + }) await page.setViewportSize({ width: 1200, height: 500 }) await u.waitForAuthSkipAppStart() @@ -261,10 +271,7 @@ test( await page.getByText('bracket').click() - await expect(page.getByTestId('loading')).toBeAttached() - await expect(page.getByTestId('loading')).not.toBeAttached({ - timeout: 20_000, - }) + await u.waitForPageLoad() }) // If they're open by default, we're not actually testing anything. @@ -292,16 +299,7 @@ test( await page.getByText('router-template-slate').click() - await expect(page.getByTestId('loading')).toBeAttached() - await expect(page.getByTestId('loading')).not.toBeAttached({ - timeout: 20_000, - }) - - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).toBeEnabled({ - timeout: 20_000, - }) + await u.waitForPageLoad() }) await test.step('All panes opened before should be visible', async () => { diff --git a/e2e/playwright/desktop-export.spec.ts b/e2e/playwright/desktop-export.spec.ts index 713060b43..6dafd041f 100644 --- a/e2e/playwright/desktop-export.spec.ts +++ b/e2e/playwright/desktop-export.spec.ts @@ -43,12 +43,6 @@ test( // open the project await page.getByText(`bracket`).click() - // wait for the project to load - await expect(page.getByTestId('loading')).toBeAttached() - await expect(page.getByTestId('loading')).not.toBeAttached({ - timeout: 20_000, - }) - // expect zero errors in guter await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() @@ -56,6 +50,12 @@ test( const exportButton = page.getByTestId('export-pane-button') await expect(exportButton).toBeVisible() + // Wait for the model to finish loading + const modelStateIndicator = page.getByTestId( + 'model-state-indicator-execution-done' + ) + await expect(modelStateIndicator).toBeVisible({ timeout: 60000 }) + const gltfOption = page.getByText('glTF') const submitButton = page.getByText('Confirm Export') const exportingToastMessage = page.getByText(`Exporting...`) @@ -104,7 +104,7 @@ test( }, { timeout: 15_000 } ) - .toBe(477327) + .toBe(477481) // clean up output.gltf await fsp.rm('output.gltf') diff --git a/e2e/playwright/file-tree.spec.ts b/e2e/playwright/file-tree.spec.ts index c6eb0e6b7..a33013f16 100644 --- a/e2e/playwright/file-tree.spec.ts +++ b/e2e/playwright/file-tree.spec.ts @@ -112,7 +112,8 @@ test.describe('when using the file tree to', () => { }) const { - panesOpen, + openKclCodePanel, + openFilePanel, createAndSelectProject, pasteCodeInEditor, createNewFileAndSelect, @@ -124,9 +125,9 @@ test.describe('when using the file tree to', () => { await page.setViewportSize({ width: 1200, height: 500 }) page.on('console', console.log) - await panesOpen(['files', 'code']) - await createAndSelectProject('project-000') + await openKclCodePanel() + await openFilePanel() // File the main.kcl with contents const kclCube = await fsp.readFile( 'src/wasm-lib/tests/executor/inputs/cube.kcl', @@ -201,4 +202,78 @@ test.describe('when using the file tree to', () => { await electronApp.close() } ) + + test( + 'loading small file, then large, then back to small', + { + tag: '@electron', + }, + async ({ browser: _ }, testInfo) => { + const { page } = await setupElectron({ + testInfo, + }) + + const { + panesOpen, + createAndSelectProject, + pasteCodeInEditor, + createNewFile, + openDebugPanel, + closeDebugPanel, + expectCmdLog, + } = await getUtils(page, test) + + await page.setViewportSize({ width: 1200, height: 500 }) + page.on('console', console.log) + + await panesOpen(['files', 'code']) + await createAndSelectProject('project-000') + + // Create a small file + const kclCube = await fsp.readFile( + 'src/wasm-lib/tests/executor/inputs/cube.kcl', + 'utf-8' + ) + // pasted into main.kcl + await pasteCodeInEditor(kclCube) + + // Create a large lego file + await createNewFile('lego') + const legoFile = page.getByRole('listitem').filter({ + has: page.getByRole('button', { name: 'lego.kcl' }), + }) + await expect(legoFile).toBeVisible({ timeout: 60_000 }) + await legoFile.click() + const kclLego = await fsp.readFile( + 'src/wasm-lib/tests/executor/inputs/lego.kcl', + 'utf-8' + ) + await pasteCodeInEditor(kclLego) + const mainFile = page.getByRole('listitem').filter({ + has: page.getByRole('button', { name: 'main.kcl' }), + }) + + // Open settings and enable the debug panel + await page + .getByRole('link', { + name: 'settings Settings', + }) + .click() + await page.locator('#showDebugPanel').getByText('OffOn').click() + await page.getByTestId('settings-close-button').click() + + await test.step('swap between small and large files', async () => { + await openDebugPanel() + // Previously created a file so we need to start back at main.kcl + await mainFile.click() + await expectCmdLog('[data-message-type="execution-done"]', 60_000) + // Click the large file + await legoFile.click() + // Once it is building, click back to the smaller file + await mainFile.click() + await expectCmdLog('[data-message-type="execution-done"]', 60_000) + await closeDebugPanel() + }) + } + ) }) diff --git a/e2e/playwright/projects.spec.ts b/e2e/playwright/projects.spec.ts index fc662f40a..6aba2a6bd 100644 --- a/e2e/playwright/projects.spec.ts +++ b/e2e/playwright/projects.spec.ts @@ -147,9 +147,6 @@ test.describe('Can export from electron app', () => { const u = await getUtils(page) page.on('console', console.log) - await electronApp.context().addInitScript(async () => { - ;(window as any).playwrightSkipFilePicker = true - }) const pointOnModel = { x: 630, y: 280 } @@ -173,10 +170,10 @@ test.describe('Can export from electron app', () => { // gray at this pixel means the stream has loaded in the most // user way we can verify it (pixel color) await expect - .poll(() => u.getGreatestPixDiff(pointOnModel, [75, 75, 75]), { + .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { timeout: 10_000, }) - .toBeLessThan(10) + .toBeLessThan(15) }) const exportLocations: Array = [] @@ -207,7 +204,7 @@ test.describe('Can export from electron app', () => { }, { timeout: 15_000 } ) - .toBe(477327) + .toBe(477481) // clean up output.gltf await fsp.rm('output.gltf') @@ -495,10 +492,6 @@ test( await file.click() - await expect(page.getByTestId('loading')).toBeAttached() - await expect(page.getByTestId('loading')).not.toBeAttached({ - timeout: 20_000, - }) await expect(u.codeLocator).toContainText( 'A mounting bracket for the Focusrite Scarlett Solo audio interface' ) @@ -856,10 +849,10 @@ const extrude001 = extrude(200, sketch001)`) // gray at this pixel means the stream has loaded in the most // user way we can verify it (pixel color) await expect - .poll(() => u.getGreatestPixDiff(pointOnModel, [132, 132, 132]), { + .poll(() => u.getGreatestPixDiff(pointOnModel, [143, 143, 143]), { timeout: 10_000, }) - .toBeLessThan(10) + .toBeLessThan(15) await expect(async () => { await page.mouse.move(0, 0, { steps: 5 }) @@ -867,8 +860,8 @@ const extrude001 = extrude(200, sketch001)`) await page.mouse.click(pointOnModel.x, pointOnModel.y) // check user can interact with model by checking it turns yellow await expect - .poll(() => u.getGreatestPixDiff(pointOnModel, [176, 180, 132])) - .toBeLessThan(10) + .poll(() => u.getGreatestPixDiff(pointOnModel, [180, 180, 137])) + .toBeLessThan(15) }).toPass({ timeout: 40_000, intervals: [1_000] }) await page.getByTestId('app-logo').click() @@ -942,24 +935,15 @@ test( await page.getByText('bracket').click() - await expect(page.getByTestId('loading')).toBeAttached() - await expect(page.getByTestId('loading')).not.toBeAttached({ - timeout: 20_000, - }) - - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).toBeEnabled({ - timeout: 20_000, - }) + await u.waitForPageLoad() // gray at this pixel means the stream has loaded in the most // user way we can verify it (pixel color) await expect - .poll(() => u.getGreatestPixDiff(pointOnModel, [75, 75, 75]), { + .poll(() => u.getGreatestPixDiff(pointOnModel, [85, 85, 85]), { timeout: 10_000, }) - .toBeLessThan(10) + .toBeLessThan(15) }) await test.step('Clicking the logo takes us back to the projects page / home', async () => { @@ -976,24 +960,15 @@ test( await page.getByText('router-template-slate').click() - await expect(page.getByTestId('loading')).toBeAttached() - await expect(page.getByTestId('loading')).not.toBeAttached({ - timeout: 20_000, - }) - - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).toBeEnabled({ - timeout: 20_000, - }) + await u.waitForPageLoad() // gray at this pixel means the stream has loaded in the most // user way we can verify it (pixel color) await expect - .poll(() => u.getGreatestPixDiff(pointOnModel, [132, 132, 132]), { + .poll(() => u.getGreatestPixDiff(pointOnModel, [143, 143, 143]), { timeout: 10_000, }) - .toBeLessThan(10) + .toBeLessThan(15) }) await test.step('Opening the router-template project should load the stream', async () => { @@ -1744,7 +1719,7 @@ test.describe('Renaming in the file tree', () => { }) await test.step('Rename the folder', async () => { - await page.waitForTimeout(60000) + await page.waitForTimeout(2000) await folderToRename.click({ button: 'right' }) await expect(renameMenuItem).toBeVisible() await renameMenuItem.click() diff --git a/e2e/playwright/regression-tests.spec.ts b/e2e/playwright/regression-tests.spec.ts index c8cdc8b7d..00f4c693b 100644 --- a/e2e/playwright/regression-tests.spec.ts +++ b/e2e/playwright/regression-tests.spec.ts @@ -358,6 +358,7 @@ const sketch001 = startSketchAt([-0, -0]) await page.addInitScript( async ({ code }) => { localStorage.setItem('persistCode', code) + ;(window as any).playwrightSkipFilePicker = true }, { code: bracket, @@ -393,20 +394,22 @@ const sketch001 = startSketchAt([-0, -0]) await test.step('The second export is blocked', async () => { // Find the toast. // Look out for the toast message - await expect(exportingToastMessage).toBeVisible() - await expect(alreadyExportingToastMessage).toBeVisible() - - await page.waitForTimeout(1000) + await Promise.all([ + expect(exportingToastMessage.first()).toBeVisible(), + expect(alreadyExportingToastMessage).toBeVisible(), + ]) }) await test.step('The first export still succeeds', async () => { - await expect(exportingToastMessage).not.toBeVisible() - await expect(errorToastMessage).not.toBeVisible() - await expect(engineErrorToastMessage).not.toBeVisible() - - await expect(successToastMessage).toBeVisible() - - await expect(alreadyExportingToastMessage).not.toBeVisible() + await Promise.all([ + expect(exportingToastMessage).not.toBeVisible({ timeout: 15_000 }), + expect(errorToastMessage).not.toBeVisible(), + expect(engineErrorToastMessage).not.toBeVisible(), + expect(successToastMessage).toBeVisible({ timeout: 15_000 }), + expect(alreadyExportingToastMessage).not.toBeVisible({ + timeout: 15_000, + }), + ]) }) }) @@ -419,10 +422,12 @@ const sketch001 = startSketchAt([-0, -0]) await expect(exportingToastMessage).toBeVisible() // Expect it to succeed. - await expect(exportingToastMessage).not.toBeVisible() - await expect(errorToastMessage).not.toBeVisible() - await expect(engineErrorToastMessage).not.toBeVisible() - await expect(alreadyExportingToastMessage).not.toBeVisible() + await Promise.all([ + expect(exportingToastMessage).not.toBeVisible(), + expect(errorToastMessage).not.toBeVisible(), + expect(engineErrorToastMessage).not.toBeVisible(), + expect(alreadyExportingToastMessage).not.toBeVisible(), + ]) await expect(successToastMessage).toBeVisible() }) diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png index fff84998f..b211e1dad 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png index 77835d7ab..e5fce91ff 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Client-side-scene-scale-should-match-engine-scale-Millimeter-scale-2-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png index ce68eceaa..a07a0c17b 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-linux.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png index 0df1fe5b0..4678294e6 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Draft-rectangles-should-look-right-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png index db07581cf..07ec6a70c 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/Zoom-to-fit-on-load---solid-3d-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png index c096d52ba..e29c712a3 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable--XY-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png index 5472ba67c..a1e82414d 100644 Binary files a/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png and b/e2e/playwright/snapshot-tests.spec.ts-snapshots/extrude-on-default-planes-should-be-stable-XY-1-Google-Chrome-win32.png differ diff --git a/e2e/playwright/test-utils.ts b/e2e/playwright/test-utils.ts index 8ec9b4d10..5923598f5 100644 --- a/e2e/playwright/test-utils.ts +++ b/e2e/playwright/test-utils.ts @@ -548,13 +548,16 @@ export async function getUtils(page: Page, test_?: typeof test) { createNewFileAndSelect: async (name: string) => { return test?.step(`Create a file named ${name}, select it`, async () => { + await openFilePanel(page) await page.getByTestId('create-file-button').click() await page.getByTestId('file-rename-field').fill(name) await page.keyboard.press('Enter') - await page + const newFile = page .locator('[data-testid="file-pane-scroll-container"] button') .filter({ hasText: name }) - .click() + + await expect(newFile).toBeVisible() + await newFile.click() }) }, @@ -585,6 +588,15 @@ export async function getUtils(page: Page, test_?: typeof test) { }) }, + /** + * @deprecated Sorry I don't have time to fix this right now, but runs like + * the one linked below show me that setting the open panes in this manner is not reliable. + * You can either set `openPanes` as a part of the same initScript we run in setupElectron/setup, + * or you can imperatively open the panes with functions like {openKclCodePanel} + * (or we can make a general openPane function that takes a paneId)., + * but having a separate initScript does not seem to work reliably. + * @link https://github.com/KittyCAD/modeling-app/actions/runs/10731890169/job/29762700806?pr=3807#step:20:19553 + */ panesOpen: async (paneIds: PaneId[]) => { return test?.step(`Setting ${paneIds} panes to be open`, async () => { await page.addInitScript( @@ -852,10 +864,12 @@ export async function setupElectron({ testInfo, folderSetupFn, cleanProjectDir = true, + appSettings, }: { testInfo: TestInfo folderSetupFn?: (projectDirName: string) => Promise cleanProjectDir?: boolean + appSettings?: Partial }) { // create or otherwise clear the folder const projectDirName = testInfo.outputPath('electron-test-projects-dir') @@ -889,15 +903,19 @@ export async function setupElectron({ if (cleanProjectDir) { const tempSettingsFilePath = join(projectDirName, SETTINGS_FILE_NAME) - const settingsOverrides = TOML.stringify({ - ...TEST_SETTINGS, - settings: { - app: { - ...TEST_SETTINGS.app, - projectDirectory: projectDirName, - }, - }, - }) + const settingsOverrides = TOML.stringify( + appSettings + ? { settings: appSettings } + : { + ...TEST_SETTINGS, + settings: { + app: { + ...TEST_SETTINGS.app, + projectDirectory: projectDirName, + }, + }, + } + ) await fsp.writeFile(tempSettingsFilePath, settingsOverrides) } diff --git a/e2e/playwright/testing-selections.spec.ts b/e2e/playwright/testing-selections.spec.ts index a8e7ca6f9..452992647 100644 --- a/e2e/playwright/testing-selections.spec.ts +++ b/e2e/playwright/testing-selections.spec.ts @@ -773,9 +773,9 @@ const extrude001 = extrude(50, sketch001) await page.waitForTimeout(1000) - let noHoverColor: [number, number, number] = [82, 82, 82] - let hoverColor: [number, number, number] = [116, 116, 116] - let selectColor: [number, number, number] = [144, 148, 97] + let noHoverColor: [number, number, number] = [92, 92, 92] + let hoverColor: [number, number, number] = [127, 127, 127] + let selectColor: [number, number, number] = [155, 155, 105] const extrudeWall = { x: 670, y: 275 } const extrudeText = `line([170.36, -121.61], %, $seg01)` @@ -787,7 +787,7 @@ const extrude001 = extrude(50, sketch001) await expect .poll(() => u.getGreatestPixDiff(extrudeWall, noHoverColor)) - .toBeLessThan(5) + .toBeLessThan(15) await page.mouse.move(nothing.x, nothing.y) await page.waitForTimeout(100) await page.mouse.move(extrudeWall.x, extrudeWall.y) @@ -798,43 +798,43 @@ const extrude001 = extrude(50, sketch001) await page.waitForTimeout(200) await expect( await u.getGreatestPixDiff(extrudeWall, hoverColor) - ).toBeLessThan(6) + ).toBeLessThan(15) await page.mouse.click(extrudeWall.x, extrudeWall.y) await expect(page.locator('.cm-activeLine')).toHaveText(`|> ${extrudeText}`) await page.waitForTimeout(200) await expect( await u.getGreatestPixDiff(extrudeWall, selectColor) - ).toBeLessThan(6) + ).toBeLessThan(15) await page.waitForTimeout(1000) // check color stays there, i.e. not overridden (this was a bug previously) await expect( await u.getGreatestPixDiff(extrudeWall, selectColor) - ).toBeLessThan(6) + ).toBeLessThan(15) await page.mouse.move(nothing.x, nothing.y) await page.waitForTimeout(300) await expect(page.getByTestId('hover-highlight')).not.toBeVisible() // because of shading, color is not exact everywhere on the face - noHoverColor = [104, 104, 104] - hoverColor = [134, 134, 134] - selectColor = [158, 162, 110] + noHoverColor = [115, 115, 115] + hoverColor = [145, 145, 145] + selectColor = [168, 168, 120] - await expect(await u.getGreatestPixDiff(cap, noHoverColor)).toBeLessThan(6) + await expect(await u.getGreatestPixDiff(cap, noHoverColor)).toBeLessThan(15) await page.mouse.move(cap.x, cap.y) await expect(page.getByTestId('hover-highlight').first()).toBeVisible() await expect(page.getByTestId('hover-highlight').first()).toContainText( removeAfterFirstParenthesis(capText) ) await page.waitForTimeout(200) - await expect(await u.getGreatestPixDiff(cap, hoverColor)).toBeLessThan(6) + await expect(await u.getGreatestPixDiff(cap, hoverColor)).toBeLessThan(15) await page.mouse.click(cap.x, cap.y) await expect(page.locator('.cm-activeLine')).toHaveText(`|> ${capText}`) await page.waitForTimeout(200) - await expect(await u.getGreatestPixDiff(cap, selectColor)).toBeLessThan(6) + await expect(await u.getGreatestPixDiff(cap, selectColor)).toBeLessThan(15) await page.waitForTimeout(1000) // check color stays there, i.e. not overridden (this was a bug previously) - await expect(await u.getGreatestPixDiff(cap, selectColor)).toBeLessThan(6) + await expect(await u.getGreatestPixDiff(cap, selectColor)).toBeLessThan(15) }) test("Various pipe expressions should and shouldn't allow edit and or extrude", async ({ page, diff --git a/e2e/playwright/testing-settings.spec.ts b/e2e/playwright/testing-settings.spec.ts index 6ee51d362..8a6e7b245 100644 --- a/e2e/playwright/testing-settings.spec.ts +++ b/e2e/playwright/testing-settings.spec.ts @@ -288,7 +288,7 @@ test.describe('Testing settings', () => { }) await test.step('Refresh the application and see project setting applied', async () => { - await page.reload() + await page.reload({ waitUntil: 'domcontentloaded' }) await expect(logoLink).toHaveCSS('--primary-hue', projectThemeColor) await settingsCloseButton.click() @@ -303,53 +303,109 @@ test.describe('Testing settings', () => { } ) + test( + `Load desktop app with no settings file`, + { tag: '@electron' }, + async ({ browser: _ }, testInfo) => { + const { electronApp, page } = await setupElectron({ + // This is what makes no settings file get created + cleanProjectDir: false, + testInfo, + }) + + await page.setViewportSize({ width: 1200, height: 500 }) + + // Selectors and constants + const errorHeading = page.getByRole('heading', { + name: 'An unextected error occurred', + }) + const projectDirLink = page.getByText('Loaded from') + + // If the app loads without exploding we're in the clear + await expect(errorHeading).not.toBeVisible() + await expect(projectDirLink).toBeVisible() + + await electronApp.close() + } + ) + + test( + `Load desktop app with a settings file, but no project directory setting`, + { tag: '@electron' }, + async ({ browser: _ }, testInfo) => { + const { electronApp, page } = await setupElectron({ + testInfo, + appSettings: { + app: { + themeColor: '259', + }, + }, + }) + + await page.setViewportSize({ width: 1200, height: 500 }) + + // Selectors and constants + const errorHeading = page.getByRole('heading', { + name: 'An unextected error occurred', + }) + const projectDirLink = page.getByText('Loaded from') + + // If the app loads without exploding we're in the clear + await expect(errorHeading).not.toBeVisible() + await expect(projectDirLink).toBeVisible() + + await electronApp.close() + } + ) + test( `Closing settings modal should go back to the original file being viewed`, { tag: '@electron' }, async ({ browser: _ }, testInfo) => { const { electronApp, page } = await setupElectron({ testInfo, - folderSetupFn: async () => {}, + folderSetupFn: async (dir) => { + const bracketDir = join(dir, 'project-000') + await fsp.mkdir(bracketDir, { recursive: true }) + await fsp.copyFile( + executorInputPath('cube.kcl'), + join(bracketDir, 'main.kcl') + ) + await fsp.copyFile( + executorInputPath('cylinder.kcl'), + join(bracketDir, '2.kcl') + ) + }, }) + const kclCube = await fsp.readFile(executorInputPath('cube.kcl'), 'utf-8') + const kclCylinder = await fsp.readFile( + executorInputPath('cylinder.kcl'), + 'utf8' + ) const { - panesOpen, - createAndSelectProject, - pasteCodeInEditor, - clickPane, - createNewFileAndSelect, + openKclCodePanel, + openFilePanel, + waitForPageLoad, + selectFile, editorTextMatches, } = await getUtils(page, test) await page.setViewportSize({ width: 1200, height: 500 }) page.on('console', console.log) - await panesOpen([]) - - await test.step('Precondition: No projects exist', async () => { + await test.step('Precondition: Open to second project file', async () => { await expect(page.getByTestId('home-section')).toBeVisible() - const projectLinksPre = page.getByTestId('project-link') - await expect(projectLinksPre).toHaveCount(0) + await page.getByText('project-000').click() + await waitForPageLoad() + await openKclCodePanel() + await openFilePanel() + await editorTextMatches(kclCube) + + await selectFile('2.kcl') + await editorTextMatches(kclCylinder) }) - await createAndSelectProject('project-000') - - await clickPane('code') - const kclCube = await fsp.readFile( - 'src/wasm-lib/tests/executor/inputs/cube.kcl', - 'utf-8' - ) - await pasteCodeInEditor(kclCube) - - await clickPane('files') - await createNewFileAndSelect('2.kcl') - - const kclCylinder = await fsp.readFile( - 'src/wasm-lib/tests/executor/inputs/cylinder.kcl', - 'utf-8' - ) - await pasteCodeInEditor(kclCylinder) - const settingsOpenButton = page.getByRole('link', { name: 'settings Settings', }) @@ -357,6 +413,9 @@ test.describe('Testing settings', () => { await test.step('Open and close settings', async () => { await settingsOpenButton.click() + await expect( + page.getByRole('heading', { name: 'Settings', exact: true }) + ).toBeVisible() await settingsCloseButton.click() }) diff --git a/e2e/playwright/text-to-cad-tests.spec.ts b/e2e/playwright/text-to-cad-tests.spec.ts index 8dc2b6d7b..2ad968668 100644 --- a/e2e/playwright/text-to-cad-tests.spec.ts +++ b/e2e/playwright/text-to-cad-tests.spec.ts @@ -534,7 +534,7 @@ test.describe('Text-to-CAD tests', () => { // Ensure the final toast remains. await expect(page.getByText(`a 2x10 lego`)).not.toBeVisible() - await expect(page.getByText(`a 2x8 lego`)).not.toBeVisible() + await expect(page.getByText(`Prompt: "a 2x8 lego`)).not.toBeVisible() await expect(page.getByText(`a 2x4 lego`)).toBeVisible() // Ensure you can copy the code for the final model. @@ -690,40 +690,53 @@ test( 'Text-to-CAD functionality', { tag: '@electron' }, async ({ browserName }, testInfo) => { + const projectName = 'project-000' + const prompt = 'lego 2x4' + const textToCadFileName = 'lego-2x4.kcl' + const { electronApp, page, dir } = await setupElectron({ testInfo }) const fileExists = () => - fs.existsSync(join(dir, 'project-000', 'lego-2x4.kcl')) + fs.existsSync(join(dir, projectName, textToCadFileName)) - const { createAndSelectProject, panesOpen } = await getUtils(page, test) + const { + createAndSelectProject, + openFilePanel, + openKclCodePanel, + waitForPageLoad, + } = await getUtils(page, test) await page.setViewportSize({ width: 1200, height: 500 }) - await panesOpen(['code', 'files']) + // Locators + const projectMenuButton = page.getByRole('button', { name: projectName }) + const textToCadFileButton = page.getByRole('listitem').filter({ + has: page.getByRole('button', { name: textToCadFileName }), + }) + const textToCadComment = page.getByText( + `// Generated by Text-to-CAD: ${prompt}` + ) // Create and navigate to the project await createAndSelectProject('project-000') // Wait for Start Sketch otherwise you will not have access Text-to-CAD command - await expect( - page.getByRole('button', { name: 'Start Sketch' }) - ).toBeEnabled({ - timeout: 20_000, - }) + await waitForPageLoad() + await openFilePanel() + await openKclCodePanel() await test.step(`Test file creation`, async () => { - await sendPromptFromCommandBar(page, 'lego 2x4') + await sendPromptFromCommandBar(page, prompt) // File is considered created if it shows up in the Project Files pane - const file = page.getByRole('button', { name: 'lego-2x4.kcl' }) - await expect(file).toBeVisible({ timeout: 20_000 }) + await expect(textToCadFileButton).toBeVisible({ timeout: 20_000 }) expect(fileExists()).toBeTruthy() }) await test.step(`Test file navigation`, async () => { - const file = page.getByRole('button', { name: 'lego-2x4.kcl' }) - await file.click() - const kclComment = page.getByText('Lego 2x4 Brick') + await expect(projectMenuButton).toContainText('main.kcl') + await textToCadFileButton.click() // File can be navigated and loaded assuming a specific KCL comment is loaded into the KCL code pane - await expect(kclComment).toBeVisible({ timeout: 20_000 }) + await expect(textToCadComment).toBeVisible({ timeout: 20_000 }) + await expect(projectMenuButton).toContainText(textToCadFileName) }) await test.step(`Test file deletion on rejection`, async () => { @@ -737,6 +750,8 @@ test( ) await expect(submittingToastMessage).toBeVisible() expect(fileExists()).toBeFalsy() + // Confirm we've navigated back to the main.kcl file after deletion + await expect(projectMenuButton).toContainText('main.kcl') }) await electronApp.close() diff --git a/electron-builder.yml b/electron-builder.yml new file mode 100644 index 000000000..9453b2b28 --- /dev/null +++ b/electron-builder.yml @@ -0,0 +1,83 @@ +appId: dev.zoo.modeling-app + +directories: + output: out + buildResources: assets + +files: + - .vite/** + +mac: + category: public.app-category.developer-tools + artifactName: "${productName}-${version}-${arch}-${os}.${ext}" + target: + - target: dmg + arch: + - x64 + - arm64 + - target: zip + arch: + - x64 + - arm64 + notarize: + teamId: 92H8YB3B95 + fileAssociations: + - ext: kcl + name: kcl + mimeType: text/vnd.zoo.kcl + description: Zoo KCL File + role: Editor + rank: Owner + +win: + artifactName: "${productName}-${version}-${arch}-${os}.${ext}" + target: + - target: nsis + arch: + - x64 + - arm64 + - target: msi + arch: + - x64 + - arm64 + signingHashAlgorithms: + - sha256 + sign: "./sign-win.js" + publisherName: "KittyCAD Inc" # needs to be exactly like on Digicert + icon: "assets/icon.ico" + fileAssociations: + - ext: kcl + name: kcl + mimeType: text/vnd.zoo.kcl + description: Zoo KCL File + role: Editor + +msi: + oneClick: false + perMachine: true + +nsis: + oneClick: false + perMachine: true + allowElevation: true + installerIcon: "assets/icon.ico" + include: "./installer.nsh" + +linux: + artifactName: "${productName}-${version}-${arch}-${os}.${ext}" + target: + - target: appImage + arch: + - x64 + - arm64 + fileAssociations: + - ext: kcl + name: kcl + mimeType: text/vnd.zoo.kcl + description: Zoo KCL File + role: Editor + +publish: + - provider: generic + url: https://dl.zoo.dev/releases/modeling-app + channel: latest diff --git a/installer.nsh b/installer.nsh new file mode 100644 index 000000000..d6e78fb2e --- /dev/null +++ b/installer.nsh @@ -0,0 +1,8 @@ +!macro preInit + SetRegView 64 + WriteRegExpandStr HKLM "${INSTALL_REGISTRY_KEY}" InstallLocation "C:\Program Files\Zoo Modeling App" + WriteRegExpandStr HKCU "${INSTALL_REGISTRY_KEY}" InstallLocation "C:\Program Files\Zoo Modeling App" + SetRegView 32 + WriteRegExpandStr HKLM "${INSTALL_REGISTRY_KEY}" InstallLocation "C:\Program Files\Zoo Modeling App" + WriteRegExpandStr HKCU "${INSTALL_REGISTRY_KEY}" InstallLocation "C:\Program Files\Zoo Modeling App" +!macroend \ No newline at end of file diff --git a/interface.d.ts b/interface.d.ts index 37eee378e..faf9ce274 100644 --- a/interface.d.ts +++ b/interface.d.ts @@ -30,8 +30,6 @@ export interface IElectronAPI { join: typeof path.join sep: typeof path.sep rename: (prev: string, next: string) => typeof fs.rename - setBaseUrl: (value: string) => void - loadProjectAtStartup: () => Promise packageJson: { name: string } diff --git a/package.json b/package.json index e32316ae4..1bdf21f2b 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "zoo-modeling-app", - "version": "0.24.12", + "version": "0.25.0", "private": true, "productName": "Zoo Modeling App", "author": { @@ -39,6 +39,7 @@ "codemirror": "^6.0.1", "decamelize": "^6.0.0", "electron-squirrel-startup": "^1.0.1", + "electron-updater": "^6.3.0", "fuse.js": "^7.0.0", "html2canvas-pro": "^1.5.8", "isomorphic-fetch": "^3.0.0", @@ -50,7 +51,7 @@ "react": "^18.3.1", "react-dom": "^18.2.0", "react-hot-toast": "^2.4.1", - "react-hotkeys-hook": "^4.5.0", + "react-hotkeys-hook": "^4.5.1", "react-json-view": "^1.21.3", "react-modal": "^3.16.1", "react-modal-promise": "^1.0.2", @@ -97,7 +98,9 @@ "tron:package": "electron-forge package", "tron:make": "electron-forge make", "tron:publish": "electron-forge publish", - "tron:test": "NODE_ENV=development yarn playwright test --config=playwright.electron.config.ts --grep=@electron" + "tron:test": "NODE_ENV=development yarn playwright test --config=playwright.electron.config.ts --grep=@electron", + "tronb:vite": "vite build -c vite.main.config.ts && vite build -c vite.preload.config.ts && vite build -c vite.renderer.config.ts", + "tronb:package": "electron-builder --config electron-builder.yml" }, "prettier": { "trailingComma": "es5", @@ -150,7 +153,6 @@ "@types/three": "^0.163.0", "@types/ua-parser-js": "^0.7.39", "@types/uuid": "^9.0.8", - "@types/wait-on": "^5.3.4", "@types/wicg-file-system-access": "^2023.10.5", "@types/ws": "^8.5.10", "@typescript-eslint/eslint-plugin": "^5.0.0", @@ -161,10 +163,12 @@ "autoprefixer": "^10.4.19", "d3-force": "^3.0.0", "electron": "^32.0.1", + "electron-builder": "^24.13.3", + "electron-notarize": "^1.2.2", "eslint": "^8.0.1", "eslint-config-react-app": "^7.0.1", "eslint-plugin-css-modules": "^2.12.0", - "eslint-plugin-import": "^2.25.0", + "eslint-plugin-import": "^2.30.0", "eslint-plugin-suggest-no-throw": "^1.0.0", "happy-dom": "^14.3.10", "http-server": "^14.1.1", @@ -172,7 +176,7 @@ "node-fetch": "^3.3.2", "pixelmatch": "^5.3.0", "pngjs": "^7.0.0", - "postcss": "^8.4.31", + "postcss": "^8.4.43", "postinstall-postinstall": "^2.1.0", "prettier": "^2.8.8", "setimmediate": "^1.0.5", @@ -185,7 +189,6 @@ "vite-tsconfig-paths": "^4.3.2", "vitest": "^1.6.0", "vitest-webgl-canvas-mock": "^1.1.0", - "wait-on": "^7.2.0", "wasm-pack": "^0.13.0", "ws": "^8.17.0", "yarn": "^1.22.22" diff --git a/sign-win.js b/sign-win.js new file mode 100644 index 000000000..b7280ed03 --- /dev/null +++ b/sign-win.js @@ -0,0 +1,38 @@ +// From https://github.com/OpenBuilds/OpenBuilds-CONTROL/blob/4800540ffaa517925fc2cff26670809efa341ffe/signWin.js +const { execSync } = require('node:child_process') + +exports.default = async (configuration) => { + if (!process.env.SM_API_KEY) { + console.error( + 'Signing using signWin.js script: failed: SM_API_KEY ENV VAR NOT FOUND' + ) + return + } + + if (!process.env.WINDOWS_CERTIFICATE_THUMBPRINT) { + console.error( + 'Signing using signWin.js script: failed: FINGERPRINT ENV VAR NOT FOUND' + ) + return + } + + if (!configuration.path) { + throw new Error( + `Signing using signWin.js script: failed: TARGET PATH NOT FOUND` + ) + } + + try { + execSync( + `smctl sign --fingerprint="${ + process.env.WINDOWS_CERTIFICATE_THUMBPRINT + }" --input "${String(configuration.path)}"`, + { + stdio: 'inherit', + } + ) + console.log('Signing using signWin.js script: successful') + } catch (error) { + console.error('Signing using signWin.js script: failed:', error) + } +} diff --git a/src/App.tsx b/src/App.tsx index 1450a5f67..8be525016 100644 --- a/src/App.tsx +++ b/src/App.tsx @@ -122,11 +122,11 @@ export function App() { // Override the electron window draggable region behavior as well // when the button is down in the stream style={ - { - '-webkit-app-region': context.store?.buttonDownInStream - ? 'no-drag' - : '', - } as React.CSSProperties + isDesktop() && context.store?.buttonDownInStream + ? ({ + '-webkit-app-region': 'no-drag', + } as React.CSSProperties) + : {} } project={{ project, file }} enableMenu={true} diff --git a/src/Router.tsx b/src/Router.tsx index 4c9c4d228..b80b4c789 100644 --- a/src/Router.tsx +++ b/src/Router.tsx @@ -69,19 +69,6 @@ const router = createRouter([ path: PATHS.INDEX, loader: async () => { const onDesktop = isDesktop() - if (onDesktop) { - const projectStartupFile = - await window.electron.loadProjectAtStartup() - if (projectStartupFile !== null) { - // Redirect to the file if we have a file path. - if (projectStartupFile.length > 0) { - return redirect( - PATHS.FILE + '/' + encodeURIComponent(projectStartupFile) - ) - } - } - } - return onDesktop ? redirect(PATHS.HOME) : redirect(PATHS.FILE + '/%2F' + BROWSER_PROJECT_NAME) diff --git a/src/Toolbar.tsx b/src/Toolbar.tsx index 7791368ef..8f0c0fb91 100644 --- a/src/Toolbar.tsx +++ b/src/Toolbar.tsx @@ -20,6 +20,8 @@ import { ToolbarItemResolved, ToolbarModeName, } from 'lib/toolbar' +import { isDesktop } from 'lib/isDesktop' +import { openExternalBrowserIfDesktop } from 'lib/openWindow' export function Toolbar({ className = '', @@ -288,6 +290,11 @@ const ToolbarItemTooltip = memo(function ToolbarItemContents({ return ( void] { +export function useEngineCommands(): [CommandLog[], () => void] { const [engineCommands, setEngineCommands] = useState( engineCommandManager.commandLogs ) diff --git a/src/components/FileTree.tsx b/src/components/FileTree.tsx index 1b9b608c8..8ecf4764e 100644 --- a/src/components/FileTree.tsx +++ b/src/components/FileTree.tsx @@ -179,10 +179,7 @@ const FileTreeItem = ({ codeManager.writeToFile() // Prevent seeing the model built one piece at a time when changing files - kclManager.isFirstRender = true - kclManager.executeCode(true).then(() => { - kclManager.isFirstRender = false - }) + kclManager.executeCode(true) } else { // Let the lsp servers know we closed a file. onFileClose(currentFile?.path || null, project?.path || null) diff --git a/src/components/Loading.tsx b/src/components/Loading.tsx index 5135245e2..a097956f6 100644 --- a/src/components/Loading.tsx +++ b/src/components/Loading.tsx @@ -11,6 +11,8 @@ import { import { engineCommandManager } from '../lib/singletons' +import { Spinner } from './Spinner' + const Loading = ({ children }: React.PropsWithChildren) => { const [error, setError] = useState(ConnectionError.Unset) @@ -65,17 +67,7 @@ const Loading = ({ children }: React.PropsWithChildren) => { className="body-bg flex flex-col items-center justify-center h-screen" data-testid="loading" > - - - +

{children || 'Loading'}

{children}

+ {!location.pathname.startsWith(PATHS.HOME) && } { + const [commands] = useEngineCommands() + + const lastCommandType = commands[commands.length - 1]?.type + + let className = 'w-6 h-6 ' + let icon = + let dataTestId = 'model-state-indicator' + + if (lastCommandType === 'receive-reliable') { + className += + 'bg-chalkboard-20 dark:bg-chalkboard-80 !group-disabled:bg-chalkboard-30 !dark:group-disabled:bg-chalkboard-80 rounded-sm bg-succeed-10/30 dark:bg-succeed' + icon = ( + + ) + } else if (lastCommandType === 'execution-done') { + className += + 'border-6 border border-solid border-chalkboard-60 dark:border-chalkboard-80 bg-chalkboard-20 dark:bg-chalkboard-80 !group-disabled:bg-chalkboard-30 !dark:group-disabled:bg-chalkboard-80 rounded-sm bg-succeed-10/30 dark:bg-succeed' + icon = ( + + ) + } else if (lastCommandType === 'export-done') { + className += + 'border-6 border border-solid border-chalkboard-60 dark:border-chalkboard-80 bg-chalkboard-20 dark:bg-chalkboard-80 !group-disabled:bg-chalkboard-30 !dark:group-disabled:bg-chalkboard-80 rounded-sm bg-succeed-10/30 dark:bg-succeed' + icon = ( + + ) + } + + return ( +
+ {icon} +
+ ) +} diff --git a/src/components/ModelingMachineProvider.tsx b/src/components/ModelingMachineProvider.tsx index c1a4e532f..71b5941d4 100644 --- a/src/components/ModelingMachineProvider.tsx +++ b/src/components/ModelingMachineProvider.tsx @@ -67,7 +67,6 @@ import { hasExtrudableGeometry, isSingleCursorInPipe, } from 'lang/queryAst' -import { TEST } from 'env' import { exportFromEngine } from 'lib/exportFromEngine' import { Models } from '@kittycad/lib/dist/types/src' import toast from 'react-hot-toast' @@ -162,9 +161,7 @@ export const ModelingMachineProvider = ({ store.videoElement?.pause() - kclManager.isFirstRender = true kclManager.executeCode().then(() => { - kclManager.isFirstRender = false if (engineCommandManager.engineConnection?.idleMode) return store.videoElement?.play().catch((e) => { @@ -366,7 +363,7 @@ export const ModelingMachineProvider = ({ return {} }), Make: async (_, event) => { - if (event.type !== 'Make' || TEST) return + if (event.type !== 'Make') return // Check if we already have an export intent. if (engineCommandManager.exportIntent) { toast.error('Already exporting') @@ -410,7 +407,7 @@ export const ModelingMachineProvider = ({ ) }, 'Engine export': async (_, event) => { - if (event.type !== 'Export' || TEST) return + if (event.type !== 'Export') return if (engineCommandManager.exportIntent) { toast.error('Already exporting') return diff --git a/src/components/SettingsAuthProvider.tsx b/src/components/SettingsAuthProvider.tsx index 83ea0f692..54ad27929 100644 --- a/src/components/SettingsAuthProvider.tsx +++ b/src/components/SettingsAuthProvider.tsx @@ -193,10 +193,7 @@ export const SettingsAuthProviderBase = ({ resetSettingsIncludesUnitChange ) { // Unit changes requires a re-exec of code - kclManager.isFirstRender = true - kclManager.executeCode(true).then(() => { - kclManager.isFirstRender = false - }) + kclManager.executeCode(true) } else { // For any future logging we'd like to do // console.log( diff --git a/src/components/Spinner.tsx b/src/components/Spinner.tsx new file mode 100644 index 000000000..dea0c20fd --- /dev/null +++ b/src/components/Spinner.tsx @@ -0,0 +1,17 @@ +import { SVGProps } from 'react' + +export const Spinner = (props: SVGProps) => { + return ( + + + + ) +} diff --git a/src/components/Stream.tsx b/src/components/Stream.tsx index 32ecf1aa1..eb712f1e9 100644 --- a/src/components/Stream.tsx +++ b/src/components/Stream.tsx @@ -54,12 +54,10 @@ export const Stream = () => { * central place, we can move this code there. */ async function executeCodeAndPlayStream() { - kclManager.isFirstRender = true kclManager.executeCode(true).then(() => { videoRef.current?.play().catch((e) => { console.warn('Video playing was prevented', e, videoRef.current) }) - kclManager.isFirstRender = false setStreamState(StreamState.Playing) }) } @@ -219,7 +217,7 @@ export const Stream = () => { * Play the vid */ useEffect(() => { - if (!kclManager.isFirstRender) { + if (!kclManager.isExecuting) { setTimeout(() => // execute in the next event loop videoRef.current?.play().catch((e) => { @@ -227,7 +225,7 @@ export const Stream = () => { }) ) } - }, [kclManager.isFirstRender]) + }, [kclManager.isExecuting]) useEffect(() => { if ( @@ -382,15 +380,15 @@ export const Stream = () => { )} - {(!isNetworkOkay || isLoading || kclManager.isFirstRender) && ( + {(!isNetworkOkay || isLoading) && (
- {!isNetworkOkay && !isLoading && !kclManager.isFirstRender ? ( + {!isNetworkOkay && !isLoading ? ( Stream disconnected... - ) : !isLoading && kclManager.isFirstRender ? ( - Building scene... ) : ( - Loading stream... + !isLoading && ( + Loading stream... + ) )}
diff --git a/src/components/Tooltip.tsx b/src/components/Tooltip.tsx index 8e0167c08..cab9b1a0e 100644 --- a/src/components/Tooltip.tsx +++ b/src/components/Tooltip.tsx @@ -12,6 +12,7 @@ interface TooltipProps extends React.PropsWithChildren { position?: TooltipPosition wrapperClassName?: string contentClassName?: string + wrapperStyle?: React.CSSProperties delay?: number hoverOnly?: boolean inert?: boolean @@ -22,6 +23,7 @@ export default function Tooltip({ position = 'top', wrapperClassName: className, contentClassName, + wrapperStyle = {}, delay = 200, hoverOnly = false, inert = true, @@ -36,7 +38,10 @@ export default function Tooltip({ } ${styles.tooltipWrapper} ${hoverOnly ? '' : styles.withFocus} ${ styles[position] } ${className}`} - style={{ '--_delay': delay + 'ms' } as React.CSSProperties} + style={Object.assign( + { '--_delay': delay + 'ms' } as React.CSSProperties, + wrapperStyle + )} >
{children} diff --git a/src/hooks/useToolbarGuards.ts b/src/hooks/useToolbarGuards.ts index ff31a5141..a8a7cb082 100644 --- a/src/hooks/useToolbarGuards.ts +++ b/src/hooks/useToolbarGuards.ts @@ -8,7 +8,7 @@ import { moveValueIntoNewVariable } from 'lang/modifyAst' import { isNodeSafeToReplace } from 'lang/queryAst' import { useEffect, useState } from 'react' import { useModelingContext } from './useModelingContext' -import { PathToNode, SourceRange, parse, recast } from 'lang/wasm' +import { PathToNode, SourceRange } from 'lang/wasm' import { useKclContext } from 'lang/KclProvider' export const getVarNameModal = createSetVarNameModal(SetVarNameModal) @@ -23,8 +23,7 @@ export function useConvertToVariable(range?: SourceRange) { }, [enable]) useEffect(() => { - const parsed = parse(recast(ast)) - if (trap(parsed)) return + const parsed = ast const meta = isNodeSafeToReplace( parsed, diff --git a/src/index.css b/src/index.css index 8a0a993cf..f1221c546 100644 --- a/src/index.css +++ b/src/index.css @@ -50,6 +50,14 @@ body.dark { @apply text-chalkboard-10; } +@media (prefers-color-scheme: dark) { + body, + .body-bg, + .dark .body-bg { + @apply bg-chalkboard-100; + } +} + select { @apply bg-chalkboard-20; } @@ -287,32 +295,11 @@ code { } @layer utilities { - /* Modified from the very helpful https://www.transition.style/#in:circle:hesitate */ - @keyframes circle-in-hesitate { - 0% { - clip-path: circle( - var(--circle-size-start, 0%) at var(--circle-x, 50%) - var(--circle-y, 50%) - ); - } - 40% { - clip-path: circle( - var(--circle-size-mid, 40%) at var(--circle-x, 50%) var(--circle-y, 50%) - ); - } - 100% { - clip-path: circle( - var(--circle-size-end, 125%) at var(--circle-x, 50%) - var(--circle-y, 50%) - ); - } - } - - .in-circle-hesitate { - animation: var(--circle-duration, 2.5s) - var(--circle-timing, cubic-bezier(0.25, 1, 0.3, 1)) circle-in-hesitate - both; - } + /* + This is where your own custom Tailwind utility classes can go, + which lets you use them with @apply in your CSS, and get + autocomplete in classNames in your JSX. + */ } #code-mirror-override .cm-scroller, diff --git a/src/lang/KclSingleton.ts b/src/lang/KclSingleton.ts index 7e16c6ed7..c9bbcf7cf 100644 --- a/src/lang/KclSingleton.ts +++ b/src/lang/KclSingleton.ts @@ -60,8 +60,6 @@ export class KclManager { private _wasmInitFailedCallback: (arg: boolean) => void = () => {} private _executeCallback: () => void = () => {} - isFirstRender = true - get ast() { return this._ast } diff --git a/src/lang/std/artifactGraph.ts b/src/lang/std/artifactGraph.ts index 25b3584e6..cb688c4f2 100644 --- a/src/lang/std/artifactGraph.ts +++ b/src/lang/std/artifactGraph.ts @@ -3,6 +3,8 @@ import { Models } from '@kittycad/lib' import { getNodePathFromSourceRange } from 'lang/queryAst' import { err } from 'lib/trap' +export type ArtifactId = string + interface CommonCommandProperties { range: SourceRange pathToNode: PathToNode @@ -10,7 +12,7 @@ interface CommonCommandProperties { export interface PlaneArtifact { type: 'plane' - pathIds: Array + pathIds: Array codeRef: CommonCommandProperties } export interface PlaneArtifactRich { @@ -21,16 +23,16 @@ export interface PlaneArtifactRich { export interface PathArtifact { type: 'path' - planeId: string - segIds: Array - extrusionId: string - solid2dId?: string + planeId: ArtifactId + segIds: Array + extrusionId: ArtifactId + solid2dId?: ArtifactId codeRef: CommonCommandProperties } interface solid2D { type: 'solid2D' - pathId: string + pathId: ArtifactId } export interface PathArtifactRich { type: 'path' @@ -42,10 +44,10 @@ export interface PathArtifactRich { interface SegmentArtifact { type: 'segment' - pathId: string - surfaceId: string - edgeIds: Array - edgeCutId?: string + pathId: ArtifactId + surfaceId: ArtifactId + edgeIds: Array + edgeCutId?: ArtifactId codeRef: CommonCommandProperties } interface SegmentArtifactRich { @@ -59,9 +61,9 @@ interface SegmentArtifactRich { interface ExtrusionArtifact { type: 'extrusion' - pathId: string - surfaceIds: Array - edgeIds: Array + pathId: ArtifactId + surfaceIds: Array + edgeIds: Array codeRef: CommonCommandProperties } interface ExtrusionArtifactRich { @@ -74,23 +76,23 @@ interface ExtrusionArtifactRich { interface WallArtifact { type: 'wall' - segId: string - edgeCutEdgeIds: Array - extrusionId: string - pathIds: Array + segId: ArtifactId + edgeCutEdgeIds: Array + extrusionId: ArtifactId + pathIds: Array } interface CapArtifact { type: 'cap' subType: 'start' | 'end' - edgeCutEdgeIds: Array - extrusionId: string - pathIds: Array + edgeCutEdgeIds: Array + extrusionId: ArtifactId + pathIds: Array } interface ExtrudeEdge { type: 'extrudeEdge' - segId: string - extrusionId: string + segId: ArtifactId + extrusionId: ArtifactId subType: 'opposite' | 'adjacent' } @@ -98,16 +100,16 @@ interface ExtrudeEdge { interface EdgeCut { type: 'edgeCut' subType: 'fillet' | 'chamfer' - consumedEdgeId: string - edgeIds: Array - surfaceId: string + consumedEdgeId: ArtifactId + edgeIds: Array + surfaceId: ArtifactId codeRef: CommonCommandProperties } interface EdgeCutEdge { type: 'edgeCutEdge' - edgeCutId: string - surfaceId: string + edgeCutId: ArtifactId + surfaceId: ArtifactId } export type Artifact = @@ -122,7 +124,7 @@ export type Artifact = | EdgeCutEdge | solid2D -export type ArtifactGraph = Map +export type ArtifactGraph = Map export type EngineCommand = Models['WebSocketRequest_type'] @@ -149,7 +151,7 @@ export function createArtifactGraph({ responseMap: ResponseMap ast: Program }) { - const myMap = new Map() + const myMap = new Map() /** see docstring for {@link getArtifactsToUpdate} as to why this is needed */ let currentPlaneId = '' @@ -166,7 +168,7 @@ export function createArtifactGraph({ const artifactsToUpdate = getArtifactsToUpdate({ orderedCommand, responseMap, - getArtifact: (id: string) => myMap.get(id), + getArtifact: (id: ArtifactId) => myMap.get(id), currentPlaneId, ast, }) @@ -224,11 +226,11 @@ export function getArtifactsToUpdate({ orderedCommand: OrderedCommand responseMap: ResponseMap /** Passing in a getter because we don't wan this function to update the map directly */ - getArtifact: (id: string) => Artifact | undefined - currentPlaneId: string + getArtifact: (id: ArtifactId) => Artifact | undefined + currentPlaneId: ArtifactId ast: Program }): Array<{ - id: string + id: ArtifactId artifact: Artifact }> { const pathToNode = getNodePathFromSourceRange(ast, range) @@ -514,7 +516,7 @@ export function filterArtifacts( (!predicate || predicate(value as Extract)) ) - ) as Map> + ) as Map> } export function getArtifactsOfTypes( @@ -528,7 +530,7 @@ export function getArtifactsOfTypes( predicate?: (value: Extract) => boolean }, map: ArtifactGraph -): Map> { +): Map> { return new Map( [...map].filter( ([key, value]) => @@ -537,7 +539,7 @@ export function getArtifactsOfTypes( (!predicate || predicate(value as Extract)) ) - ) as Map> + ) as Map> } export function getArtifactOfTypes( @@ -545,7 +547,7 @@ export function getArtifactOfTypes( key, types, }: { - key: string + key: ArtifactId types: T }, map: ArtifactGraph @@ -718,7 +720,7 @@ export function getExtrudeEdgeCodeRef( } export function getExtrusionFromSuspectedExtrudeSurface( - id: string, + id: ArtifactId, artifactGraph: ArtifactGraph ): ExtrusionArtifact | Error { const artifact = getArtifactOfTypes( @@ -733,7 +735,7 @@ export function getExtrusionFromSuspectedExtrudeSurface( } export function getExtrusionFromSuspectedPath( - id: string, + id: ArtifactId, artifactGraph: ArtifactGraph ): ExtrusionArtifact | Error { const path = getArtifactOfTypes({ key: id, types: ['path'] }, artifactGraph) diff --git a/src/lang/std/engineConnection.ts b/src/lang/std/engineConnection.ts index 169b198d4..078962a24 100644 --- a/src/lang/std/engineConnection.ts +++ b/src/lang/std/engineConnection.ts @@ -1252,6 +1252,10 @@ export type CommandLog = type: 'execution-done' data: null } + | { + type: 'export-done' + data: null + } export enum EngineCommandManagerEvents { // engineConnection is available but scene setup may not have run @@ -1918,7 +1922,13 @@ export class EngineCommandManager extends EventTarget { } else if (cmd.type === 'export') { const promise = new Promise((resolve, reject) => { this.pendingExport = { - resolve, + resolve: (passThrough) => { + this.addCommandLog({ + type: 'export-done', + data: null, + }) + resolve(passThrough) + }, reject: (reason: string) => { this.exportIntent = null reject(reason) diff --git a/src/lang/wasm.ts b/src/lang/wasm.ts index 5ca4ea5bd..18ed50f4e 100644 --- a/src/lang/wasm.ts +++ b/src/lang/wasm.ts @@ -95,8 +95,6 @@ export const wasmUrl = () => { document.location.pathname.split('/').slice(0, -1).join('/') + '/wasm_lib_bg.wasm' - console.log(`Full URL for WASM: ${fullUrl}`) - return fullUrl } diff --git a/src/lib/desktop.ts b/src/lib/desktop.ts index e38113438..7cdf6fe31 100644 --- a/src/lib/desktop.ts +++ b/src/lib/desktop.ts @@ -8,7 +8,6 @@ import { parseProjectSettings, } from 'lang/wasm' import { - DEFAULT_HOST, PROJECT_ENTRYPOINT, PROJECT_FOLDER, PROJECT_SETTINGS_FILE_NAME, @@ -462,29 +461,60 @@ export const readProjectSettingsFile = async ( */ export const readAppSettingsFile = async () => { let settingsPath = await getAppSettingsFilePath() + const initialProjectDirConfig: DeepPartial< + Configuration['settings']['project'] + > = { directory: await getInitialDefaultDir() } // The file exists, read it and parse it. if (window.electron.exists(settingsPath)) { const configToml = await window.electron.readFile(settingsPath) - const configObj = parseAppSettings(configToml) - if (err(configObj)) { - return Promise.reject(configObj) + const parsedAppConfig = parseAppSettings(configToml) + if (err(parsedAppConfig)) { + return Promise.reject(parsedAppConfig) } - return configObj + const hasProjectDirectorySetting = + parsedAppConfig.settings?.project?.directory || + parsedAppConfig.settings?.app?.project_directory + + if (hasProjectDirectorySetting) { + return parsedAppConfig + } else { + // inject the default project directory setting + const mergedConfig: DeepPartial = { + ...parsedAppConfig, + settings: { + ...parsedAppConfig.settings, + project: Object.assign( + {}, + parsedAppConfig.settings?.project, + initialProjectDirConfig + ), + }, + } + return mergedConfig + } } // The file doesn't exist, create a new one. - // This defaultAppConfig is truly an empty object every time. const defaultAppConfig = defaultAppSettings() if (err(defaultAppConfig)) { return Promise.reject(defaultAppConfig) } - const initialDirConfig: DeepPartial = { - settings: { project: { directory: await getInitialDefaultDir() } }, + + // inject the default project directory setting + const mergedDefaultConfig: DeepPartial = { + ...defaultAppConfig, + settings: { + ...defaultAppConfig.settings, + project: Object.assign( + {}, + defaultAppConfig.settings?.project, + initialProjectDirConfig + ), + }, } - const config = Object.assign(defaultAppConfig, initialDirConfig) - return config + return mergedDefaultConfig } export const writeAppSettingsFile = async (tomlStr: string) => { @@ -525,28 +555,6 @@ export const getUser = async ( token: string, hostname: string ): Promise => { - // Use the host passed in if it's set. - // Otherwise, use the default host. - const host = !hostname ? DEFAULT_HOST : hostname - - // Change the baseURL to the one we want. - let baseurl = host - if (!(host.indexOf('http://') === 0) && !(host.indexOf('https://') === 0)) { - baseurl = `https://${host}` - if (host.indexOf('localhost') === 0) { - baseurl = `http://${host}` - } - } - - // Use kittycad library to fetch the user info from /user/me - if (baseurl !== DEFAULT_HOST) { - // The TypeScript generated library uses environment variables for this - // because it was intended for NodeJS. - // Needs to stay like this because window.electron.kittycad needs it - // internally. - window.electron.setBaseUrl(baseurl) - } - try { const user = await window.electron.kittycad('users.get_user_self', { client: { token }, diff --git a/src/lib/exampleKcl.ts b/src/lib/exampleKcl.ts index 335efcb72..9a59d2b25 100644 --- a/src/lib/exampleKcl.ts +++ b/src/lib/exampleKcl.ts @@ -31,11 +31,11 @@ const bracket = startSketchOn('XY') |> extrude(width, %) |> fillet({ radius: filletR, - tags: [getPreviousAdjacentEdge(innerEdge)] + tags: [getNextAdjacentEdge(innerEdge)] }, %) |> fillet({ radius: filletR + thickness, - tags: [getPreviousAdjacentEdge(outerEdge)] + tags: [getNextAdjacentEdge(outerEdge)] }, %)` /** diff --git a/src/lib/exportSave.ts b/src/lib/exportSave.ts index f21d94140..8d7701ef7 100644 --- a/src/lib/exportSave.ts +++ b/src/lib/exportSave.ts @@ -14,7 +14,7 @@ const save_ = async (file: ModelingAppFile) => { extensions.push(extension) } - if (!(window as any).playwrightSkipFilePicker) { + if (window.electron.process.env.IS_PLAYWRIGHT) { // skip file picker, save to default location await window.electron.writeFile( file.name, diff --git a/src/lib/machineManager.ts b/src/lib/machineManager.ts index d514b8dd9..128a59c45 100644 --- a/src/lib/machineManager.ts +++ b/src/lib/machineManager.ts @@ -81,7 +81,6 @@ export class MachineManager { } this._machines = await window.electron.listMachines() - console.log('Machines:', this._machines) } private async updateMachineApiIp(): Promise { diff --git a/src/lib/selections.ts b/src/lib/selections.ts index bc9626b92..a984d3625 100644 --- a/src/lib/selections.ts +++ b/src/lib/selections.ts @@ -5,7 +5,7 @@ import { kclManager, sceneEntitiesManager, } from 'lib/singletons' -import { CallExpression, SourceRange, Expr, parse, recast } from 'lang/wasm' +import { CallExpression, SourceRange, Expr, parse } from 'lang/wasm' import { ModelingMachineEvent } from 'machines/modelingMachine' import { uuidv4 } from 'lib/utils' import { EditorSelection, SelectionRange } from '@codemirror/state' @@ -302,8 +302,7 @@ export function processCodeMirrorRanges({ } function updateSceneObjectColors(codeBasedSelections: Selection[]) { - const updated = parse(recast(kclManager.ast)) - if (err(updated)) return + const updated = kclManager.ast Object.values(sceneEntitiesManager.activeSegments).forEach((segmentGroup) => { if ( diff --git a/src/lib/settings/settingsUtils.ts b/src/lib/settings/settingsUtils.ts index 6f5c05951..bf5fa84c4 100644 --- a/src/lib/settings/settingsUtils.ts +++ b/src/lib/settings/settingsUtils.ts @@ -14,6 +14,7 @@ import { Configuration } from 'wasm-lib/kcl/bindings/Configuration' import { mouseControlsToCameraSystem } from 'lib/cameraControls' import { appThemeToTheme } from 'lib/theme' import { + getInitialDefaultDir, readAppSettingsFile, readProjectSettingsFile, writeAppSettingsFile, @@ -176,6 +177,11 @@ export async function loadAndValidateSettings( if (err(appSettingsPayload)) return Promise.reject(appSettingsPayload) const settings = createSettings() + // Because getting the default directory is async, we need to set it after + if (onDesktop) { + settings.app.projectDirectory.default = await getInitialDefaultDir() + } + setSettingsAtLevel( settings, 'user', diff --git a/src/lib/singletons.ts b/src/lib/singletons.ts index 2e6df784e..092350120 100644 --- a/src/lib/singletons.ts +++ b/src/lib/singletons.ts @@ -16,7 +16,6 @@ window.tearDown = engineCommandManager.tearDown // This needs to be after codeManager is created. export const kclManager = new KclManager(engineCommandManager) -kclManager.isFirstRender = true engineCommandManager.kclManager = kclManager engineCommandManager.getAstCb = () => kclManager.ast diff --git a/src/lib/toolbar.ts b/src/lib/toolbar.ts index cf178e061..cb3a24463 100644 --- a/src/lib/toolbar.ts +++ b/src/lib/toolbar.ts @@ -129,12 +129,16 @@ export const toolbarConfig: Record = { id: 'loft', onClick: () => console.error('Loft not yet implemented'), icon: 'loft', - status: 'unavailable', + status: 'kcl-only', title: 'Loft', hotkey: 'L', description: 'Create a 3D body by blending between two or more sketches.', links: [ + { + label: 'KCL docs', + url: 'https://zoo.dev/docs/kcl/loft', + }, { label: 'GitHub discussion', url: 'https://github.com/KittyCAD/modeling-app/discussions/613', diff --git a/src/main.ts b/src/main.ts index 4ecd576e7..8c68a5da4 100644 --- a/src/main.ts +++ b/src/main.ts @@ -2,14 +2,23 @@ // template that ElectronJS provides. import dotenv from 'dotenv' -import { app, BrowserWindow, ipcMain, dialog, shell } from 'electron' +import { + app, + BrowserWindow, + ipcMain, + dialog, + shell, + nativeTheme, +} from 'electron' import path from 'path' import { Issuer } from 'openid-client' import { Bonjour, Service } from 'bonjour-service' // @ts-ignore: TS1343 import * as kittycad from '@kittycad/lib/import' +import electronUpdater, { type AppUpdater } from 'electron-updater' import minimist from 'minimist' import getCurrentProjectFile from 'lib/getCurrentProjectFile' +import os from 'node:os' let mainWindow: BrowserWindow | null = null @@ -22,8 +31,20 @@ if (!process.env.NODE_ENV) console.warn( '*FOX SCREAM* process.env.NODE_ENV is not explicitly set!, defaulting to production' ) +// Default prod values + +// dotenv override when present dotenv.config({ path: [`.env.${NODE_ENV}.local`, `.env.${NODE_ENV}`] }) +console.log(process.env) + +process.env.VITE_KC_API_WS_MODELING_URL ??= + 'wss://api.zoo.dev/ws/modeling/commands' +process.env.VITE_KC_API_BASE_URL ??= 'https://api.zoo.dev' +process.env.VITE_KC_SITE_BASE_URL ??= 'https://zoo.dev' +process.env.VITE_KC_SKIP_AUTH ??= 'false' +process.env.VITE_KC_CONNECTION_TIMEOUT_MS ??= '15000' + // Handle creating/removing shortcuts on Windows when installing/uninstalling. if (require('electron-squirrel-startup')) { app.quit() @@ -46,7 +67,7 @@ if (process.defaultApp) { // Must be done before ready event. registerStartupListeners() -const createWindow = (): BrowserWindow => { +const createWindow = (filePath?: string): BrowserWindow => { const newWindow = new BrowserWindow({ autoHideMenuBar: true, show: false, @@ -59,17 +80,35 @@ const createWindow = (): BrowserWindow => { preload: path.join(__dirname, './preload.js'), }, icon: path.resolve(process.cwd(), 'assets', 'icon.png'), - frame: false, + frame: os.platform() !== 'darwin', titleBarStyle: 'hiddenInset', + backgroundColor: nativeTheme.shouldUseDarkColors ? '#1C1C1C' : '#FCFCFC', }) // and load the index.html of the app. if (MAIN_WINDOW_VITE_DEV_SERVER_URL) { newWindow.loadURL(MAIN_WINDOW_VITE_DEV_SERVER_URL) } else { - newWindow.loadFile( - path.join(__dirname, `../renderer/${MAIN_WINDOW_VITE_NAME}/index.html`) - ) + getProjectPathAtStartup(filePath).then((projectPath) => { + const startIndex = path.join( + __dirname, + `../renderer/${MAIN_WINDOW_VITE_NAME}/index.html` + ) + + if (projectPath === null) { + newWindow.loadFile(startIndex) + return + } + + console.log('Loading file', projectPath) + + const fullUrl = `/file/${encodeURIComponent(projectPath)}` + console.log('Full URL', fullUrl) + + newWindow.loadFile(startIndex, { + hash: fullUrl, + }) + }) } // Open the DevTools. @@ -80,13 +119,11 @@ const createWindow = (): BrowserWindow => { return newWindow } -// Quit when all windows are closed, except on macOS. There, it's common +// Quit when all windows are closed, even on macOS. There, it's common // for applications and their menu bar to stay active until the user quits -// explicitly with Cmd + Q. +// explicitly with Cmd + Q, but it is a really weird behavior with our app. app.on('window-all-closed', () => { - if (process.platform !== 'darwin') { - app.quit() - } + app.quit() }) // This method will be called when Electron has finished @@ -191,7 +228,39 @@ ipcMain.handle('find_machine_api', () => { }) }) -ipcMain.handle('loadProjectAtStartup', async () => { +export function getAutoUpdater(): AppUpdater { + // Using destructuring to access autoUpdater due to the CommonJS module of 'electron-updater'. + // It is a workaround for ESM compatibility issues, see https://github.com/electron-userland/electron-builder/issues/7976. + const { autoUpdater } = electronUpdater + return autoUpdater +} + +export async function checkForUpdates(autoUpdater: AppUpdater) { + // TODO: figure out how to get the update modal back + const result = await autoUpdater.checkForUpdatesAndNotify() + console.log(result) +} + +app.on('ready', async () => { + const autoUpdater = getAutoUpdater() + checkForUpdates(autoUpdater) + const fifteenMinutes = 15 * 60 * 1000 + setInterval(() => { + checkForUpdates(autoUpdater) + }, fifteenMinutes) + + autoUpdater.on('update-available', (info) => { + console.log('update-available', info) + }) + + autoUpdater.on('update-downloaded', (info) => { + console.log('update-downloaded', info) + }) +}) + +const getProjectPathAtStartup = async ( + filePath?: string +): Promise => { // If we are in development mode, we don't want to load a project at // startup. // Since the args passed are always '.' @@ -199,52 +268,54 @@ ipcMain.handle('loadProjectAtStartup', async () => { return null } - let projectPath: string | null = null - // macOS: open-file events that were received before the app is ready - const macOpenFiles: string[] = (global as any).macOpenFiles - if (macOpenFiles && macOpenFiles && macOpenFiles.length > 0) { - projectPath = macOpenFiles[0] // We only do one project at a time - } - // Reset this so we don't accidentally use it again. - const macOpenFilesEmpty: string[] = [] - // @ts-ignore - global['macOpenFiles'] = macOpenFilesEmpty + let projectPath: string | null = filePath || null + if (projectPath === null) { + // macOS: open-file events that were received before the app is ready + const macOpenFiles: string[] = (global as any).macOpenFiles + if (macOpenFiles && macOpenFiles && macOpenFiles.length > 0) { + projectPath = macOpenFiles[0] // We only do one project at a time + } + // Reset this so we don't accidentally use it again. + const macOpenFilesEmpty: string[] = [] + // @ts-ignore + global['macOpenFiles'] = macOpenFilesEmpty - // macOS: open-url events that were received before the app is ready - const getOpenUrls: string[] = (global as any).getOpenUrls - if (getOpenUrls && getOpenUrls.length > 0) { - projectPath = getOpenUrls[0] // We only do one project at a - } - // Reset this so we don't accidentally use it again. - // @ts-ignore - global['getOpenUrls'] = [] + // macOS: open-url events that were received before the app is ready + const getOpenUrls: string[] = (global as any).getOpenUrls + if (getOpenUrls && getOpenUrls.length > 0) { + projectPath = getOpenUrls[0] // We only do one project at a + } + // Reset this so we don't accidentally use it again. + // @ts-ignore + global['getOpenUrls'] = [] - // Check if we have a project path in the command line arguments - // If we do, we will load the project at that path - if (args._.length > 1) { - if (args._[1].length > 0) { - projectPath = args._[1] - // Reset all this value so we don't accidentally use it again. - args._[1] = '' + // Check if we have a project path in the command line arguments + // If we do, we will load the project at that path + if (args._.length > 1) { + if (args._[1].length > 0) { + projectPath = args._[1] + // Reset all this value so we don't accidentally use it again. + args._[1] = '' + } } } if (projectPath) { // We have a project path, load the project information. console.log(`Loading project at startup: ${projectPath}`) - try { - const currentFile = await getCurrentProjectFile(projectPath) - console.log(`Project loaded: ${currentFile}`) - return currentFile - } catch (e) { - console.error(e) + const currentFile = await getCurrentProjectFile(projectPath) + + if (currentFile instanceof Error) { + console.error(currentFile) + return null } - return null + console.log(`Project loaded: ${currentFile}`) + return currentFile } return null -}) +} function parseCLIArgs(): minimist.ParsedArgs { return minimist(process.argv, {}) @@ -261,10 +332,11 @@ function registerStartupListeners() { app.on('open-file', function (event, path) { event.preventDefault() - macOpenFiles.push(path) // If we have a mainWindow, lets open another window. if (mainWindow) { - createWindow() + createWindow(path) + } else { + macOpenFiles.push(path) } }) @@ -280,10 +352,11 @@ function registerStartupListeners() { ) { event.preventDefault() - openUrls.push(url) // If we have a mainWindow, lets open another window. if (mainWindow) { - createWindow() + createWindow(url) + } else { + openUrls.push(url) } } diff --git a/src/preload.ts b/src/preload.ts index 5f55f145a..b75575e6c 100644 --- a/src/preload.ts +++ b/src/preload.ts @@ -60,9 +60,6 @@ const listMachines = async (): Promise => { const getMachineApiIp = async (): Promise => ipcRenderer.invoke('find_machine_api') -const loadProjectAtStartup = async (): Promise => - ipcRenderer.invoke('loadProjectAtStartup') - contextBridge.exposeInMainWorld('electron', { login, // Passing fs directly is not recommended since it gives a lot of power @@ -96,10 +93,6 @@ contextBridge.exposeInMainWorld('electron', { isWindows, isLinux, }, - loadProjectAtStartup, - // IMPORTANT NOTE: kittycad.ts reads process.env.BASE_URL. But there is - // no way to set it across the bridge boundary. We need to make it a command. - setBaseUrl: (value: string) => (process.env.BASE_URL = value), process: { // Setter/getter has to be created because // these are read-only over the boundary. diff --git a/src/routes/Onboarding/Introduction.tsx b/src/routes/Onboarding/Introduction.tsx index e06c8e87f..29ed153d3 100644 --- a/src/routes/Onboarding/Introduction.tsx +++ b/src/routes/Onboarding/Introduction.tsx @@ -107,10 +107,7 @@ function OnboardingWarningWeb(props: OnboardingResetWarningProps) { codeManager.updateCodeStateEditor(bracket) await codeManager.writeToFile() - kclManager.isFirstRender = true - await kclManager.executeCode(true).then(() => { - kclManager.isFirstRender = false - }) + await kclManager.executeCode(true) props.setShouldShowWarning(false) }} nextText="Overwrite code and continue" diff --git a/src/routes/Onboarding/Sketching.tsx b/src/routes/Onboarding/Sketching.tsx index 1bb1f7e01..983d00a20 100644 --- a/src/routes/Onboarding/Sketching.tsx +++ b/src/routes/Onboarding/Sketching.tsx @@ -13,10 +13,7 @@ export default function Sketching() { async function clearEditor() { // We do want to update both the state and editor here. codeManager.updateCodeStateEditor('') - kclManager.isFirstRender = true - await kclManager.executeCode(true).then(() => { - kclManager.isFirstRender = false - }) + await kclManager.executeCode(true) } clearEditor() diff --git a/src/routes/Onboarding/index.tsx b/src/routes/Onboarding/index.tsx index 5be7929cf..46f2d880c 100644 --- a/src/routes/Onboarding/index.tsx +++ b/src/routes/Onboarding/index.tsx @@ -82,10 +82,7 @@ export function useDemoCode() { if (!editorManager.editorView || codeManager.code === bracket) return setTimeout(async () => { codeManager.updateCodeStateEditor(bracket) - kclManager.isFirstRender = true - await kclManager.executeCode(true).then(() => { - kclManager.isFirstRender = false - }) + await kclManager.executeCode(true) await codeManager.writeToFile() }) }, [editorManager.editorView]) diff --git a/src/routes/SignIn.tsx b/src/routes/SignIn.tsx index cd7c3cdfe..7e5e3f983 100644 --- a/src/routes/SignIn.tsx +++ b/src/routes/SignIn.tsx @@ -58,19 +58,23 @@ const SignIn = () => { } return ( -
+
@@ -194,7 +198,7 @@ const SignIn = () => {
diff --git a/src/wasm-lib/Cargo.lock b/src/wasm-lib/Cargo.lock index 1f4f884cb..117e4904c 100644 --- a/src/wasm-lib/Cargo.lock +++ b/src/wasm-lib/Cargo.lock @@ -127,18 +127,18 @@ checksum = "3b43422f69d8ff38f95f1b2bb76517c91589a924d1559a0e935d7c8ce0274c11" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] name = "async-trait" -version = "0.1.81" +version = "0.1.82" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6e0c28dcc82d7c8ead5cb13beb15405b57b8546e93215673ff8ca0349a028107" +checksum = "a27b8a3a6e1a44fa4c8baf1f653e4172e81486d4941f2237e20dc2d0cf4ddff1" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -149,7 +149,7 @@ checksum = "3c87f3f15e7794432337fc718554eaa4dc8f04c9677a950ffe366f20a162ae42" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -370,9 +370,9 @@ dependencies = [ [[package]] name = "clap" -version = "4.5.16" +version = "4.5.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ed6719fffa43d0d87e5fd8caeab59be1554fb028cd30edc88fc4369b17971019" +checksum = "3e5a21b8495e732f1b3c364c9949b201ca7bae518c502c80256c96ad79eaf6ac" dependencies = [ "clap_builder", "clap_derive", @@ -380,9 +380,9 @@ dependencies = [ [[package]] name = "clap_builder" -version = "4.5.15" +version = "4.5.17" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "216aec2b177652e3846684cbfe25c9964d18ec45234f0f5da5157b207ed1aab6" +checksum = "8cf2dd12af7a047ad9d6da2b6b249759a22a7abc0f474c1dae1777afa4b21a73" dependencies = [ "anstyle", "clap_lex", @@ -397,7 +397,7 @@ dependencies = [ "heck 0.5.0", "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -591,7 +591,7 @@ dependencies = [ "proc-macro2", "quote", "strsim", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -602,7 +602,7 @@ checksum = "a668eda54683121533a393014d8692171709ff57a7d61f187b6e782719f8933f" dependencies = [ "darling_core", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -620,9 +620,9 @@ dependencies = [ [[package]] name = "dashmap" -version = "6.0.1" +version = "6.1.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "804c8821570c3f8b70230c2ba75ffa5c0f9a4189b9a432b6656c536712acae28" +checksum = "5041cc499144891f3790297212f32a74fb938e5136a14943f338ef9e0ae276cf" dependencies = [ "cfg-if", "crossbeam-utils", @@ -657,7 +657,7 @@ checksum = "4078275de501a61ceb9e759d37bdd3d7210e654dbc167ac1a3678ef4435ed57b" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", "synstructure", ] @@ -672,7 +672,7 @@ dependencies = [ [[package]] name = "derive-docs" -version = "0.1.25" +version = "0.1.26" dependencies = [ "Inflector", "anyhow", @@ -686,7 +686,7 @@ dependencies = [ "rustfmt-wrapper", "serde", "serde_tokenstream", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -697,7 +697,7 @@ checksum = "67e77553c4162a157adbf834ebae5b415acbecbeafc7a74b0e886657506a7611" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -724,7 +724,7 @@ checksum = "487585f4d0c6655fe74905e2504d8ad6908e4db67f744eb140876906c2f3175d" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -896,7 +896,7 @@ checksum = "87750cf4b7a4c0625b1529e4c543c2182106e4dedc60a2a6455e00d212c489ac" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -986,7 +986,7 @@ dependencies = [ "inflections", "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -1345,7 +1345,7 @@ dependencies = [ [[package]] name = "kcl-lib" -version = "0.2.11" +version = "0.2.14" dependencies = [ "anyhow", "approx", @@ -1357,7 +1357,7 @@ dependencies = [ "clap", "convert_case", "criterion", - "dashmap 6.0.1", + "dashmap 6.1.0", "databake", "derive-docs", "expectorate", @@ -1399,7 +1399,7 @@ dependencies = [ "wasm-bindgen", "wasm-bindgen-futures", "web-sys", - "winnow 0.5.40", + "winnow", "zip", ] @@ -1412,12 +1412,12 @@ dependencies = [ "pretty_assertions", "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] name = "kcl-test-server" -version = "0.1.9" +version = "0.1.10" dependencies = [ "anyhow", "hyper", @@ -1430,9 +1430,9 @@ dependencies = [ [[package]] name = "kittycad" -version = "0.3.17" +version = "0.3.18" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "fbb7c076d64ad00a29ae900108707d1bbb583944d4b2d005e1eca9914a18c7c2" +checksum = "94feea5b1cf851b33dd108aa35aa01bde99772aa74d2ba1590295aac0b7ca33e" dependencies = [ "anyhow", "async-trait", @@ -1799,7 +1799,7 @@ dependencies = [ "regex", "regex-syntax 0.8.3", "structmeta", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -1852,7 +1852,7 @@ checksum = "2f38a4412a78282e09a2cf38d195ea5420d15ba0602cb375210efbc877243965" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2016,7 +2016,7 @@ dependencies = [ "proc-macro2", "pyo3-macros-backend", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2029,7 +2029,7 @@ dependencies = [ "proc-macro2", "pyo3-build-config", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2491,7 +2491,7 @@ dependencies = [ "proc-macro2", "quote", "serde_derive_internals", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2565,7 +2565,7 @@ checksum = "a5831b979fd7b5439637af1752d535ff49f4860c0f341d1baeb6faf0f4242170" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2576,14 +2576,14 @@ checksum = "330f01ce65a3a5fe59a60c82f3c9a024b573b8a6e875bd233fe5f934e71d54e3" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] name = "serde_json" -version = "1.0.127" +version = "1.0.128" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8043c06d9f82bd7271361ed64f415fe5e12a77fdb52e573e7f06a516dea329ad" +checksum = "6ff5456707a1de34e7e37f2a6fd3d3f808c318259cbd01ab6377795054b483d8" dependencies = [ "indexmap 2.2.5", "itoa", @@ -2600,7 +2600,7 @@ checksum = "0b2e6b945e9d3df726b65d6ee24060aff8e3533d431f677a9695db04eff9dfdb" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2621,7 +2621,7 @@ dependencies = [ "proc-macro2", "quote", "serde", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2752,7 +2752,7 @@ dependencies = [ "proc-macro2", "quote", "structmeta-derive", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2763,7 +2763,7 @@ checksum = "152a0b65a590ff6c3da95cabe2353ee04e6167c896b28e3b14478c2636c922fc" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2807,9 +2807,9 @@ dependencies = [ [[package]] name = "syn" -version = "2.0.76" +version = "2.0.77" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "578e081a14e0cefc3279b0472138c513f37b41a08d5a3cca9b6e4e8ceb6cd525" +checksum = "9f35bcdf61fd8e7be6caf75f429fdca8beb3ed76584befb503b1569faee373ed" dependencies = [ "proc-macro2", "quote", @@ -2830,7 +2830,7 @@ checksum = "c8af7666ab7b6390ab78131fb5b0fce11d6b7a6951602017c35fa82800708971" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -2937,7 +2937,7 @@ checksum = "a4558b58466b9ad7ca0f102865eccc95938dca1a74a856f2b57b6629050da261" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -3008,9 +3008,9 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20" [[package]] name = "tokio" -version = "1.39.3" +version = "1.40.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9babc99b9923bfa4804bd74722ff02c0381021eafa4db9949217e3be8e84fff5" +checksum = "e2b070231665d27ad9ec9b8df639893f46727666c6767db40317fbe920a5d998" dependencies = [ "backtrace", "bytes", @@ -3032,7 +3032,7 @@ checksum = "693d596312e88961bc67d7f1f97af8a70227d9f90c31bba5806eec004978d752" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -3117,7 +3117,7 @@ dependencies = [ "serde", "serde_spanned", "toml_datetime", - "winnow 0.6.18", + "winnow", ] [[package]] @@ -3185,7 +3185,7 @@ checksum = "84fd902d4e0b9a4b27f2f440108dc034e1758628a9b702f8ec61ad66355422fa" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -3213,7 +3213,7 @@ checksum = "34704c8d6ebcbc939824180af020566b01a7c01f80641264eba0999f6c2b6be7" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -3290,7 +3290,7 @@ checksum = "c88cc88fd23b5a04528f3a8436024f20010a16ec18eb23c164b1242f65860130" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", "termcolor", ] @@ -3448,7 +3448,7 @@ dependencies = [ "proc-macro-error", "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] @@ -3509,7 +3509,7 @@ dependencies = [ "once_cell", "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", "wasm-bindgen-shared", ] @@ -3544,7 +3544,7 @@ checksum = "e94f17b526d0a461a191c78ea52bbce64071ed5c04c9ffe424dcb38f74171bb7" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", "wasm-bindgen-backend", "wasm-bindgen-shared", ] @@ -3800,15 +3800,6 @@ version = "0.52.4" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "32b752e52a2da0ddfbdbcc6fceadfeede4c939ed16d13e648833a61dfb611ed8" -[[package]] -name = "winnow" -version = "0.5.40" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f593a95398737aeed53e489c785df13f3618e41dbcd6718c6addbf1395aa6876" -dependencies = [ - "memchr", -] - [[package]] name = "winnow" version = "0.6.18" @@ -3869,7 +3860,7 @@ checksum = "9ce1b18ccd8e73a9321186f97e46f9f04b778851177567b1975109d26a08d2a6" dependencies = [ "proc-macro2", "quote", - "syn 2.0.76", + "syn 2.0.77", ] [[package]] diff --git a/src/wasm-lib/Cargo.toml b/src/wasm-lib/Cargo.toml index b51720012..3c4e95deb 100644 --- a/src/wasm-lib/Cargo.toml +++ b/src/wasm-lib/Cargo.toml @@ -15,8 +15,8 @@ data-encoding = "2.6.0" gloo-utils = "0.2.0" kcl-lib = { path = "kcl" } kittycad.workspace = true -serde_json = "1.0.127" -tokio = { version = "1.39.3", features = ["sync"] } +serde_json = "1.0.128" +tokio = { version = "1.40.0", features = ["sync"] } toml = "0.8.19" uuid = { version = "1.10.0", features = ["v4", "js", "serde"] } wasm-bindgen = "0.2.91" @@ -29,7 +29,7 @@ image = { version = "0.25.1", default-features = false, features = ["png"] } kittycad = { workspace = true, default-features = true } pretty_assertions = "1.4.0" reqwest = { version = "0.11.26", default-features = false } -tokio = { version = "1.39.3", features = ["rt-multi-thread", "macros", "time"] } +tokio = { version = "1.40.0", features = ["rt-multi-thread", "macros", "time"] } twenty-twenty = "0.8" uuid = { version = "1.10.0", features = ["v4", "js", "serde"] } @@ -70,7 +70,7 @@ members = [ [workspace.dependencies] http = "0.2.12" -kittycad = { version = "0.3.17", default-features = false, features = ["js", "requests"] } +kittycad = { version = "0.3.18", default-features = false, features = ["js", "requests"] } kittycad-modeling-session = "0.1.4" [[test]] diff --git a/src/wasm-lib/derive-docs/Cargo.toml b/src/wasm-lib/derive-docs/Cargo.toml index eb614abfe..f761ce383 100644 --- a/src/wasm-lib/derive-docs/Cargo.toml +++ b/src/wasm-lib/derive-docs/Cargo.toml @@ -1,7 +1,7 @@ [package] name = "derive-docs" description = "A tool for generating documentation from Rust derive macros" -version = "0.1.25" +version = "0.1.26" edition = "2021" license = "MIT" repository = "https://github.com/KittyCAD/modeling-app" @@ -20,7 +20,7 @@ quote = "1" regex = "1.10" serde = { version = "1.0.209", features = ["derive"] } serde_tokenstream = "0.2" -syn = { version = "2.0.76", features = ["full"] } +syn = { version = "2.0.77", features = ["full"] } [dev-dependencies] anyhow = "1.0.86" diff --git a/src/wasm-lib/justfile b/src/wasm-lib/justfile index d4367c4ee..181182515 100644 --- a/src/wasm-lib/justfile +++ b/src/wasm-lib/justfile @@ -2,3 +2,6 @@ new-test name: echo "kcl_test!(\"{{name}}\", {{name}});" >> tests/executor/visuals.rs TWENTY_TWENTY=overwrite cargo nextest run --test executor -E 'test(=visuals::{{name}})' + +lint: + cargo clippy --all --tests --benches -- -D warnings diff --git a/src/wasm-lib/kcl-macros/Cargo.toml b/src/wasm-lib/kcl-macros/Cargo.toml index 128f992a2..9612501fe 100644 --- a/src/wasm-lib/kcl-macros/Cargo.toml +++ b/src/wasm-lib/kcl-macros/Cargo.toml @@ -15,7 +15,7 @@ databake = "0.1.8" kcl-lib = { path = "../kcl" } proc-macro2 = "1" quote = "1" -syn = { version = "2.0.76", features = ["full"] } +syn = { version = "2.0.77", features = ["full"] } [dev-dependencies] pretty_assertions = "1.4.0" diff --git a/src/wasm-lib/kcl-test-server/Cargo.toml b/src/wasm-lib/kcl-test-server/Cargo.toml index 7f96197d4..403b09dd1 100644 --- a/src/wasm-lib/kcl-test-server/Cargo.toml +++ b/src/wasm-lib/kcl-test-server/Cargo.toml @@ -1,7 +1,7 @@ [package] name = "kcl-test-server" description = "A test server for KCL" -version = "0.1.9" +version = "0.1.10" edition = "2021" license = "MIT" @@ -11,5 +11,5 @@ hyper = { version = "0.14.29", features = ["server"] } kcl-lib = { version = "0.2", path = "../kcl" } pico-args = "0.5.0" serde = { version = "1.0.209", features = ["derive"] } -serde_json = "1.0.127" -tokio = { version = "1.39.3", features = ["macros", "rt-multi-thread"] } +serde_json = "1.0.128" +tokio = { version = "1.40.0", features = ["macros", "rt-multi-thread"] } diff --git a/src/wasm-lib/kcl/Cargo.toml b/src/wasm-lib/kcl/Cargo.toml index e1943de93..33ed416db 100644 --- a/src/wasm-lib/kcl/Cargo.toml +++ b/src/wasm-lib/kcl/Cargo.toml @@ -1,7 +1,7 @@ [package] name = "kcl-lib" description = "KittyCAD Language implementation and tools" -version = "0.2.11" +version = "0.2.14" edition = "2021" license = "MIT" repository = "https://github.com/KittyCAD/modeling-app" @@ -13,14 +13,14 @@ keywords = ["kcl", "KittyCAD", "CAD"] [dependencies] anyhow = { version = "1.0.86", features = ["backtrace"] } async-recursion = "1.1.1" -async-trait = "0.1.81" +async-trait = "0.1.82" base64 = "0.22.1" chrono = "0.4.38" -clap = { version = "4.5.16", default-features = false, optional = true, features = ["std", "derive"] } +clap = { version = "4.5.17", default-features = false, optional = true, features = ["std", "derive"] } convert_case = "0.6.0" -dashmap = "6.0.1" +dashmap = "6.1.0" databake = { version = "0.1.8", features = ["derive"] } -derive-docs = { version = "0.1.24", path = "../derive-docs" } +derive-docs = { version = "0.1.26", path = "../derive-docs" } form_urlencoded = "1.2.1" futures = { version = "0.3.30" } git_rev = "0.1.0" @@ -37,7 +37,7 @@ reqwest = { version = "0.11.26", default-features = false, features = ["stream", ropey = "1.6.1" schemars = { version = "0.8.17", features = ["impl_json_schema", "url", "uuid1"] } serde = { version = "1.0.209", features = ["derive"] } -serde_json = "1.0.127" +serde_json = "1.0.128" sha2 = "0.10.8" tabled = { version = "0.15.0", optional = true } thiserror = "1.0.63" @@ -47,12 +47,12 @@ url = { version = "2.5.2", features = ["serde"] } urlencoding = "2.1.3" uuid = { version = "1.10.0", features = ["v4", "js", "serde"] } validator = { version = "0.18.1", features = ["derive"] } -winnow = "0.5.40" +winnow = "0.6.18" zip = { version = "2.0.0", default-features = false } [target.'cfg(target_arch = "wasm32")'.dependencies] js-sys = { version = "0.3.69" } -tokio = { version = "1.39.3", features = ["sync", "time"] } +tokio = { version = "1.40.0", features = ["sync", "time"] } tower-lsp = { version = "0.20.0", default-features = false, features = ["runtime-agnostic"] } wasm-bindgen = "0.2.91" wasm-bindgen-futures = "0.4.42" @@ -94,7 +94,7 @@ image = { version = "0.25.1", default-features = false, features = ["png"] } insta = { version = "1.38.0", features = ["json"] } itertools = "0.13.0" pretty_assertions = "1.4.0" -tokio = { version = "1.39.2", features = ["rt-multi-thread", "macros", "time"] } +tokio = { version = "1.40.0", features = ["rt-multi-thread", "macros", "time"] } twenty-twenty = "0.8.0" [[bench]] diff --git a/src/wasm-lib/kcl/benches/executor_benchmark_criterion.rs b/src/wasm-lib/kcl/benches/executor_benchmark_criterion.rs index 90f74a52f..613c16436 100644 --- a/src/wasm-lib/kcl/benches/executor_benchmark_criterion.rs +++ b/src/wasm-lib/kcl/benches/executor_benchmark_criterion.rs @@ -1,5 +1,5 @@ use criterion::{criterion_group, criterion_main, BenchmarkId, Criterion}; -use kcl_lib::test_server; +use kcl_lib::{settings::types::UnitLength::Mm, test_server}; use tokio::runtime::Runtime; pub fn bench_execute(c: &mut Criterion) { @@ -13,26 +13,42 @@ pub fn bench_execute(c: &mut Criterion) { // Configure Criterion.rs to detect smaller differences and increase sample size to improve // precision and counteract the resulting noise. group.sample_size(10); - group.bench_with_input(BenchmarkId::new("execute_", name), &code, |b, &s| { + group.bench_with_input(BenchmarkId::new("execute", name), &code, |b, &s| { let rt = Runtime::new().unwrap(); - // Spawn a future onto the runtime b.iter(|| { - rt.block_on(test_server::execute_and_snapshot( - s, - kcl_lib::settings::types::UnitLength::Mm, - )) - .unwrap(); + rt.block_on(test_server::execute_and_snapshot(s, Mm)).unwrap(); }); }); group.finish(); } } -criterion_group!(benches, bench_execute); +pub fn bench_lego(c: &mut Criterion) { + let mut group = c.benchmark_group("executor_lego_pattern"); + // Configure Criterion.rs to detect smaller differences and increase sample size to improve + // precision and counteract the resulting noise. + group.sample_size(10); + // Create lego bricks with N x 10 bumps, where N is each element of `sizes`. + let sizes = vec![1, 2, 4]; + for size in sizes { + group.bench_with_input(BenchmarkId::from_parameter(size), &size, |b, &size| { + let rt = Runtime::new().unwrap(); + let code = LEGO_PROGRAM.replace("{{N}}", &size.to_string()); + // Spawn a future onto the runtime + b.iter(|| { + rt.block_on(test_server::execute_and_snapshot(&code, Mm)).unwrap(); + }); + }); + } + group.finish(); +} + +criterion_group!(benches, bench_lego, bench_execute); criterion_main!(benches); const KITT_PROGRAM: &str = include_str!("../../tests/executor/inputs/kittycad_svg.kcl"); const CUBE_PROGRAM: &str = include_str!("../../tests/executor/inputs/cube.kcl"); const SERVER_RACK_HEAVY_PROGRAM: &str = include_str!("../../tests/executor/inputs/server-rack-heavy.kcl"); const SERVER_RACK_LITE_PROGRAM: &str = include_str!("../../tests/executor/inputs/server-rack-lite.kcl"); +const LEGO_PROGRAM: &str = include_str!("../../tests/executor/inputs/slow_lego.kcl.tmpl"); diff --git a/src/wasm-lib/kcl/src/executor.rs b/src/wasm-lib/kcl/src/executor.rs index 356925ccf..187703888 100644 --- a/src/wasm-lib/kcl/src/executor.rs +++ b/src/wasm-lib/kcl/src/executor.rs @@ -995,20 +995,20 @@ impl SketchSurface { } pub(crate) fn x_axis(&self) -> Point3d { match self { - SketchSurface::Plane(plane) => plane.x_axis.clone(), - SketchSurface::Face(face) => face.x_axis.clone(), + SketchSurface::Plane(plane) => plane.x_axis, + SketchSurface::Face(face) => face.x_axis, } } pub(crate) fn y_axis(&self) -> Point3d { match self { - SketchSurface::Plane(plane) => plane.y_axis.clone(), - SketchSurface::Face(face) => face.y_axis.clone(), + SketchSurface::Plane(plane) => plane.y_axis, + SketchSurface::Face(face) => face.y_axis, } } pub(crate) fn z_axis(&self) -> Point3d { match self { - SketchSurface::Plane(plane) => plane.z_axis.clone(), - SketchSurface::Face(face) => face.z_axis.clone(), + SketchSurface::Plane(plane) => plane.z_axis, + SketchSurface::Face(face) => face.z_axis, } } } @@ -1304,7 +1304,7 @@ impl Point2d { } } -#[derive(Debug, Deserialize, Serialize, PartialEq, Clone, ts_rs::TS, JsonSchema, Default)] +#[derive(Debug, Deserialize, Serialize, PartialEq, Clone, Copy, ts_rs::TS, JsonSchema, Default)] #[ts(export)] pub struct Point3d { pub x: f64, @@ -1313,6 +1313,7 @@ pub struct Point3d { } impl Point3d { + pub const ZERO: Self = Self { x: 0.0, y: 0.0, z: 0.0 }; pub fn new(x: f64, y: f64, z: f64) -> Self { Self { x, y, z } } diff --git a/src/wasm-lib/kcl/src/parser/parser_impl.rs b/src/wasm-lib/kcl/src/parser/parser_impl.rs index adb690c68..8312c8d69 100644 --- a/src/wasm-lib/kcl/src/parser/parser_impl.rs +++ b/src/wasm-lib/kcl/src/parser/parser_impl.rs @@ -927,7 +927,7 @@ pub fn function_body(i: TokenSlice) -> PResult { match body_items_within_function.parse_next(i) { Err(ErrMode::Backtrack(_)) => { - i.reset(start); + i.reset(&start); break; } Err(e) => return Err(e), @@ -937,7 +937,7 @@ pub fn function_body(i: TokenSlice) -> PResult { } } (Err(ErrMode::Backtrack(_)), _) => { - i.reset(start); + i.reset(&start); break; } (Err(e), _) => return Err(e), @@ -1276,7 +1276,7 @@ fn unary_expression(i: TokenSlice) -> PResult { /// Consume tokens that make up a binary expression, but don't actually return them. /// Why not? -/// Because this is designed to be used with .recognize() within the `binary_expression` parser. +/// Because this is designed to be used with .take() within the `binary_expression` parser. fn binary_expression_tokens(i: TokenSlice) -> PResult> { let first = operand.parse_next(i).map(BinaryExpressionToken::from)?; let remaining: Vec<_> = repeat( @@ -1308,7 +1308,7 @@ fn binary_expression(i: TokenSlice) -> PResult { } fn binary_expr_in_parens(i: TokenSlice) -> PResult { - let span_with_brackets = bracketed_section.recognize().parse_next(i)?; + let span_with_brackets = bracketed_section.take().parse_next(i)?; let n = span_with_brackets.len(); let mut span_no_brackets = &span_with_brackets[1..n - 1]; let expr = binary_expression.parse_next(&mut span_no_brackets)?; diff --git a/src/wasm-lib/kcl/src/parser/parser_impl/error.rs b/src/wasm-lib/kcl/src/parser/parser_impl/error.rs index 827735013..11dea7c3f 100644 --- a/src/wasm-lib/kcl/src/parser/parser_impl/error.rs +++ b/src/wasm-lib/kcl/src/parser/parser_impl/error.rs @@ -1,5 +1,6 @@ use winnow::{ error::{ErrorKind, ParseError, StrContext}, + stream::Stream, Located, }; @@ -102,14 +103,17 @@ impl std::default::Default for ContextError { } } -impl winnow::error::ParserError for ContextError { +impl winnow::error::ParserError for ContextError +where + I: Stream, +{ #[inline] fn from_error_kind(_input: &I, _kind: ErrorKind) -> Self { Self::default() } #[inline] - fn append(self, _input: &I, _kind: ErrorKind) -> Self { + fn append(self, _input: &I, _input_checkpoint: &::Checkpoint, _kind: ErrorKind) -> Self { self } @@ -119,9 +123,12 @@ impl winnow::error::ParserError for ContextError { } } -impl winnow::error::AddContext for ContextError { +impl winnow::error::AddContext for ContextError +where + I: Stream, +{ #[inline] - fn add_context(mut self, _input: &I, ctx: C) -> Self { + fn add_context(mut self, _input: &I, _input_checkpoint: &::Checkpoint, ctx: C) -> Self { self.context.push(ctx); self } diff --git a/src/wasm-lib/kcl/src/std/args.rs b/src/wasm-lib/kcl/src/std/args.rs index 42080dff1..94899417b 100644 --- a/src/wasm-lib/kcl/src/std/args.rs +++ b/src/wasm-lib/kcl/src/std/args.rs @@ -295,6 +295,13 @@ impl Args { FromArgs::from_args(self, 0) } + pub(crate) fn get_sketch_groups_and_data<'a, T>(&'a self) -> Result<(Vec, Option), KclError> + where + T: FromArgs<'a> + serde::de::DeserializeOwned + FromKclValue<'a> + Sized, + { + FromArgs::from_args(self, 0) + } + pub(crate) fn get_data_and_optional_tag<'a, T>(&'a self) -> Result<(T, Option), KclError> where T: serde::de::DeserializeOwned + FromKclValue<'a> + Sized, @@ -361,6 +368,13 @@ impl Args { FromArgs::from_args(self, 0) } + pub(crate) fn get_data_and_float<'a, T>(&'a self) -> Result<(T, f64), KclError> + where + T: serde::de::DeserializeOwned + FromKclValue<'a> + Sized, + { + FromArgs::from_args(self, 0) + } + pub(crate) fn get_number_sketch_group_set(&self) -> Result<(f64, SketchGroupSet), KclError> { FromArgs::from_args(self, 0) } @@ -622,6 +636,8 @@ impl_from_arg_via_json!(super::revolve::RevolveData); impl_from_arg_via_json!(super::sketch::SketchData); impl_from_arg_via_json!(crate::std::import::ImportFormat); impl_from_arg_via_json!(crate::std::polar::PolarCoordsData); +impl_from_arg_via_json!(crate::std::loft::LoftData); +impl_from_arg_via_json!(crate::std::planes::StandardPlane); impl_from_arg_via_json!(SketchGroup); impl_from_arg_via_json!(FaceTag); impl_from_arg_via_json!(String); @@ -692,3 +708,13 @@ impl<'a> FromKclValue<'a> for SketchSurface { } } } + +impl<'a> FromKclValue<'a> for Vec { + fn from_mem_item(arg: &'a KclValue) -> Option { + let KclValue::UserVal(uv) = arg else { + return None; + }; + + uv.get::>().map(|x| x.0) + } +} diff --git a/src/wasm-lib/kcl/src/std/extrude.rs b/src/wasm-lib/kcl/src/std/extrude.rs index e772366dc..b5a43760d 100644 --- a/src/wasm-lib/kcl/src/std/extrude.rs +++ b/src/wasm-lib/kcl/src/std/extrude.rs @@ -1,7 +1,10 @@ //! Functions related to extruding. +use std::collections::HashMap; + use anyhow::Result; use derive_docs::stdlib; +use kittycad::types::{ExtrusionFaceCapType, ExtrusionFaceInfo}; use schemars::JsonSchema; use uuid::Uuid; @@ -90,7 +93,7 @@ async fn inner_extrude(length: f64, sketch_group_set: SketchGroupSet, args: Args adjust_camera: false, planar_normal: if let SketchSurface::Plane(plane) = &sketch_group.on { // We pass in the normal for the plane here. - Some(plane.z_axis.clone().into()) + Some(plane.z_axis.into()) } else { None }, @@ -98,7 +101,7 @@ async fn inner_extrude(length: f64, sketch_group_set: SketchGroupSet, args: Args ) .await?; - args.send_modeling_cmd( + args.batch_modeling_cmd( id, kittycad::types::ModelingCmd::Extrude { target: sketch_group.id, @@ -111,7 +114,7 @@ async fn inner_extrude(length: f64, sketch_group_set: SketchGroupSet, args: Args // Disable the sketch mode. args.batch_modeling_cmd(uuid::Uuid::new_v4(), kittycad::types::ModelingCmd::SketchModeDisable {}) .await?; - extrude_groups.push(do_post_extrude(sketch_group.clone(), length, id, args.clone()).await?); + extrude_groups.push(do_post_extrude(sketch_group.clone(), length, args.clone()).await?); } Ok(extrude_groups.into()) @@ -120,7 +123,6 @@ async fn inner_extrude(length: f64, sketch_group_set: SketchGroupSet, args: Args pub(crate) async fn do_post_extrude( sketch_group: SketchGroup, length: f64, - id: Uuid, args: Args, ) -> Result, KclError> { // Bring the object to the front of the scene. @@ -164,7 +166,7 @@ pub(crate) async fn do_post_extrude( let solid3d_info = args .send_modeling_cmd( - id, + uuid::Uuid::new_v4(), kittycad::types::ModelingCmd::Solid3DGetExtrusionFaceInfo { edge_id, object_id: sketch_group.id, @@ -181,91 +183,95 @@ pub(crate) async fn do_post_extrude( vec![] }; - for face_info in face_infos.iter() { - if face_info.cap == kittycad::types::ExtrusionFaceCapType::None { + for (curve_id, face_id) in face_infos + .iter() + .filter(|face_info| face_info.cap == ExtrusionFaceCapType::None) + .filter_map(|face_info| { if let (Some(curve_id), Some(face_id)) = (face_info.curve_id, face_info.face_id) { - args.batch_modeling_cmd( - uuid::Uuid::new_v4(), - kittycad::types::ModelingCmd::Solid3DGetOppositeEdge { - edge_id: curve_id, - object_id: sketch_group.id, - face_id, - }, - ) - .await?; - - args.batch_modeling_cmd( - uuid::Uuid::new_v4(), - kittycad::types::ModelingCmd::Solid3DGetPrevAdjacentEdge { - edge_id: curve_id, - object_id: sketch_group.id, - face_id, - }, - ) - .await?; + Some((curve_id, face_id)) + } else { + None } - } - } - - // Create a hashmap for quick id lookup - let mut face_id_map = std::collections::HashMap::new(); - // creating fake ids for start and end caps is to make extrudes mock-execute safe - let mut start_cap_id = if args.ctx.is_mock { Some(Uuid::new_v4()) } else { None }; - let mut end_cap_id = if args.ctx.is_mock { Some(Uuid::new_v4()) } else { None }; - - for face_info in face_infos { - match face_info.cap { - kittycad::types::ExtrusionFaceCapType::Bottom => start_cap_id = face_info.face_id, - kittycad::types::ExtrusionFaceCapType::Top => end_cap_id = face_info.face_id, - _ => { - if let Some(curve_id) = face_info.curve_id { - face_id_map.insert(curve_id, face_info.face_id); - } - } - } + }) + { + // Batch these commands, because the Rust code doesn't actually care about the outcome. + // So, there's no need to await them. + // Instead, the Typescript codebases (which handles WebSocket sends when compiled via Wasm) + // uses this to build the artifact graph, which the UI needs. + args.batch_modeling_cmd( + uuid::Uuid::new_v4(), + kittycad::types::ModelingCmd::Solid3DGetOppositeEdge { + edge_id: curve_id, + object_id: sketch_group.id, + face_id, + }, + ) + .await?; + + args.batch_modeling_cmd( + uuid::Uuid::new_v4(), + kittycad::types::ModelingCmd::Solid3DGetPrevAdjacentEdge { + edge_id: curve_id, + object_id: sketch_group.id, + face_id, + }, + ) + .await?; } + let Faces { + sides: face_id_map, + start_cap_id, + end_cap_id, + } = analyze_faces(&args, face_infos); // Iterate over the sketch_group.value array and add face_id to GeoMeta - let mut new_value: Vec = Vec::new(); - for path in sketch_group.value.iter() { - if let Some(Some(actual_face_id)) = face_id_map.get(&path.get_base().geo_meta.id) { - match path { - Path::TangentialArc { .. } | Path::TangentialArcTo { .. } | Path::Circle { .. } => { - let extrude_surface = ExtrudeSurface::ExtrudeArc(crate::executor::ExtrudeArc { - face_id: *actual_face_id, - tag: path.get_base().tag.clone(), - geo_meta: GeoMeta { - id: path.get_base().geo_meta.id, - metadata: path.get_base().geo_meta.metadata.clone(), - }, - }); - new_value.push(extrude_surface); - } - Path::Base { .. } | Path::ToPoint { .. } | Path::Horizontal { .. } | Path::AngledLineTo { .. } => { - let extrude_surface = ExtrudeSurface::ExtrudePlane(crate::executor::ExtrudePlane { - face_id: *actual_face_id, - tag: path.get_base().tag.clone(), - geo_meta: GeoMeta { - id: path.get_base().geo_meta.id, - metadata: path.get_base().geo_meta.metadata.clone(), - }, - }); - new_value.push(extrude_surface); + let new_value = sketch_group + .value + .iter() + .flat_map(|path| { + if let Some(Some(actual_face_id)) = face_id_map.get(&path.get_base().geo_meta.id) { + match path { + Path::TangentialArc { .. } | Path::TangentialArcTo { .. } | Path::Circle { .. } => { + let extrude_surface = ExtrudeSurface::ExtrudeArc(crate::executor::ExtrudeArc { + face_id: *actual_face_id, + tag: path.get_base().tag.clone(), + geo_meta: GeoMeta { + id: path.get_base().geo_meta.id, + metadata: path.get_base().geo_meta.metadata.clone(), + }, + }); + Some(extrude_surface) + } + Path::Base { .. } | Path::ToPoint { .. } | Path::Horizontal { .. } | Path::AngledLineTo { .. } => { + let extrude_surface = ExtrudeSurface::ExtrudePlane(crate::executor::ExtrudePlane { + face_id: *actual_face_id, + tag: path.get_base().tag.clone(), + geo_meta: GeoMeta { + id: path.get_base().geo_meta.id, + metadata: path.get_base().geo_meta.metadata.clone(), + }, + }); + Some(extrude_surface) + } } + } else if args.ctx.is_mock { + // Only pre-populate the extrude surface if we are in mock mode. + + let extrude_surface = ExtrudeSurface::ExtrudePlane(crate::executor::ExtrudePlane { + // pushing this values with a fake face_id to make extrudes mock-execute safe + face_id: Uuid::new_v4(), + tag: path.get_base().tag.clone(), + geo_meta: GeoMeta { + id: path.get_base().geo_meta.id, + metadata: path.get_base().geo_meta.metadata.clone(), + }, + }); + Some(extrude_surface) + } else { + None } - } else if args.ctx.is_mock { - // Only pre-populate the extrude surface if we are in mock mode. - new_value.push(ExtrudeSurface::ExtrudePlane(crate::executor::ExtrudePlane { - // pushing this values with a fake face_id to make extrudes mock-execute safe - face_id: Uuid::new_v4(), - tag: path.get_base().tag.clone(), - geo_meta: GeoMeta { - id: path.get_base().geo_meta.id, - metadata: path.get_base().geo_meta.metadata.clone(), - }, - })); - } - } + }) + .collect(); Ok(Box::new(ExtrudeGroup { // Ok so you would think that the id would be the id of the extrude group, @@ -273,11 +279,45 @@ pub(crate) async fn do_post_extrude( // sketch group. id: sketch_group.id, value: new_value, - sketch_group: sketch_group.clone(), + meta: sketch_group.meta.clone(), + sketch_group, height: length, start_cap_id, end_cap_id, edge_cuts: vec![], - meta: sketch_group.meta, })) } + +#[derive(Default)] +struct Faces { + /// Maps curve ID to face ID for each side. + sides: HashMap>, + /// Top face ID. + end_cap_id: Option, + /// Bottom face ID. + start_cap_id: Option, +} + +fn analyze_faces(args: &Args, face_infos: Vec) -> Faces { + let mut faces = Faces { + sides: HashMap::with_capacity(face_infos.len()), + ..Default::default() + }; + if args.ctx.is_mock { + // Create fake IDs for start and end caps, to make extrudes mock-execute safe + faces.start_cap_id = Some(Uuid::new_v4()); + faces.end_cap_id = Some(Uuid::new_v4()); + } + for face_info in face_infos { + match face_info.cap { + ExtrusionFaceCapType::Bottom => faces.start_cap_id = face_info.face_id, + ExtrusionFaceCapType::Top => faces.end_cap_id = face_info.face_id, + ExtrusionFaceCapType::None => { + if let Some(curve_id) = face_info.curve_id { + faces.sides.insert(curve_id, face_info.face_id); + } + } + } + } + faces +} diff --git a/src/wasm-lib/kcl/src/std/fillet.rs b/src/wasm-lib/kcl/src/std/fillet.rs index e99513b57..d17f9ef47 100644 --- a/src/wasm-lib/kcl/src/std/fillet.rs +++ b/src/wasm-lib/kcl/src/std/fillet.rs @@ -304,7 +304,7 @@ async fn inner_get_next_adjacent_edge(tag: TagIdentifier, args: Args) -> Result< let resp = args .send_modeling_cmd( uuid::Uuid::new_v4(), - ModelingCmd::Solid3DGetPrevAdjacentEdge { + ModelingCmd::Solid3DGetNextAdjacentEdge { edge_id: tagged_path.id, object_id: tagged_path.sketch_group, face_id, @@ -312,7 +312,7 @@ async fn inner_get_next_adjacent_edge(tag: TagIdentifier, args: Args) -> Result< ) .await?; let kittycad::types::OkWebSocketResponseData::Modeling { - modeling_response: kittycad::types::OkModelingCmdResponse::Solid3DGetPrevAdjacentEdge { data: ajacent_edge }, + modeling_response: kittycad::types::OkModelingCmdResponse::Solid3DGetNextAdjacentEdge { data: ajacent_edge }, } = &resp else { return Err(KclError::Engine(KclErrorDetails { @@ -386,7 +386,7 @@ async fn inner_get_previous_adjacent_edge(tag: TagIdentifier, args: Args) -> Res let resp = args .send_modeling_cmd( uuid::Uuid::new_v4(), - ModelingCmd::Solid3DGetNextAdjacentEdge { + ModelingCmd::Solid3DGetPrevAdjacentEdge { edge_id: tagged_path.id, object_id: tagged_path.sketch_group, face_id, @@ -394,7 +394,7 @@ async fn inner_get_previous_adjacent_edge(tag: TagIdentifier, args: Args) -> Res ) .await?; let kittycad::types::OkWebSocketResponseData::Modeling { - modeling_response: kittycad::types::OkModelingCmdResponse::Solid3DGetNextAdjacentEdge { data: ajacent_edge }, + modeling_response: kittycad::types::OkModelingCmdResponse::Solid3DGetPrevAdjacentEdge { data: ajacent_edge }, } = &resp else { return Err(KclError::Engine(KclErrorDetails { diff --git a/src/wasm-lib/kcl/src/std/loft.rs b/src/wasm-lib/kcl/src/std/loft.rs new file mode 100644 index 000000000..57cf8f16d --- /dev/null +++ b/src/wasm-lib/kcl/src/std/loft.rs @@ -0,0 +1,174 @@ +//! Standard library lofts. + +use anyhow::Result; +use derive_docs::stdlib; +use kittycad::types::ModelingCmd; +use schemars::JsonSchema; +use serde::{Deserialize, Serialize}; + +use crate::{ + errors::{KclError, KclErrorDetails}, + executor::{ExtrudeGroup, KclValue, SketchGroup}, + std::{extrude::do_post_extrude, fillet::default_tolerance, Args}, +}; + +const DEFAULT_V_DEGREE: u32 = 2; + +/// Data for a loft. +#[derive(Debug, Clone, Deserialize, Serialize, PartialEq, ts_rs::TS, JsonSchema)] +#[ts(export)] +#[serde(rename_all = "camelCase")] +pub struct LoftData { + /// Degree of the interpolation. Must be greater than zero. + /// For example, use 2 for quadratic, or 3 for cubic interpolation in the V direction. + /// This defaults to 2, if not specified. + pub v_degree: Option, + /// Attempt to approximate rational curves (such as arcs) using a bezier. + /// This will remove banding around interpolations between arcs and non-arcs. It may produce errors in other scenarios + /// Over time, this field won't be necessary. + #[serde(default)] + pub bez_approximate_rational: Option, + /// This can be set to override the automatically determined topological base curve, which is usually the first section encountered. + #[serde(default)] + pub base_curve_index: Option, + /// Tolerance for the loft operation. + #[serde(default)] + pub tolerance: Option, +} + +impl Default for LoftData { + fn default() -> Self { + Self { + // This unwrap is safe because the default value is always greater than zero. + v_degree: Some(std::num::NonZeroU32::new(DEFAULT_V_DEGREE).unwrap()), + bez_approximate_rational: None, + base_curve_index: None, + tolerance: None, + } + } +} + +/// Create a 3D surface or solid by interpolating between two or more sketches. +pub async fn loft(args: Args) -> Result { + let (sketch_groups, data): (Vec, Option) = args.get_sketch_groups_and_data()?; + + let extrude_group = inner_loft(sketch_groups, data, args).await?; + Ok(KclValue::ExtrudeGroup(extrude_group)) +} + +/// Create a 3D surface or solid by interpolating between two or more sketches. +/// +/// The sketches need to closed and on the same plane. +/// +/// ```no_run +/// // Loft a square and a triangle. +/// const squareSketch = startSketchOn('XY') +/// |> startProfileAt([-100, 200], %) +/// |> line([200, 0], %) +/// |> line([0, -200], %) +/// |> line([-200, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// const triangleSketch = startSketchOn(offsetPlane('XY', 75)) +/// |> startProfileAt([0, 125], %) +/// |> line([-15, -30], %) +/// |> line([30, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// loft([squareSketch, triangleSketch]) +/// ``` +/// +/// ```no_run +/// // Loft a square, a circle, and another circle. +/// const squareSketch = startSketchOn('XY') +/// |> startProfileAt([-100, 200], %) +/// |> line([200, 0], %) +/// |> line([0, -200], %) +/// |> line([-200, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// const circleSketch0 = startSketchOn(offsetPlane('XY', 75)) +/// |> circle([0, 100], 50, %) +/// +/// const circleSketch1 = startSketchOn(offsetPlane('XY', 150)) +/// |> circle([0, 100], 20, %) +/// +/// loft([squareSketch, circleSketch0, circleSketch1]) +/// ``` +/// +/// ```no_run +/// // Loft a square, a circle, and another circle with options. +/// const squareSketch = startSketchOn('XY') +/// |> startProfileAt([-100, 200], %) +/// |> line([200, 0], %) +/// |> line([0, -200], %) +/// |> line([-200, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// const circleSketch0 = startSketchOn(offsetPlane('XY', 75)) +/// |> circle([0, 100], 50, %) +/// +/// const circleSketch1 = startSketchOn(offsetPlane('XY', 150)) +/// |> circle([0, 100], 20, %) +/// +/// loft([squareSketch, circleSketch0, circleSketch1], { +/// // This can be set to override the automatically determined +/// // topological base curve, which is usually the first section encountered. +/// baseCurveIndex: 0, +/// // Attempt to approximate rational curves (such as arcs) using a bezier. +/// // This will remove banding around interpolations between arcs and non-arcs. +/// // It may produce errors in other scenarios Over time, this field won't be necessary. +/// bezApproximateRational: false, +/// // Tolerance for the loft operation. +/// tolerance: 0.000001, +/// // Degree of the interpolation. Must be greater than zero. +/// // For example, use 2 for quadratic, or 3 for cubic interpolation in +/// // the V direction. This defaults to 2, if not specified. +/// vDegree: 2, +/// }) +/// ``` +#[stdlib { + name = "loft", +}] +async fn inner_loft( + sketch_groups: Vec, + data: Option, + args: Args, +) -> Result, KclError> { + // Make sure we have at least two sketches. + if sketch_groups.len() < 2 { + return Err(KclError::Semantic(KclErrorDetails { + message: format!( + "Loft requires at least two sketches, but only {} were provided.", + sketch_groups.len() + ), + source_ranges: vec![args.source_range], + })); + } + + // Get the loft data. + let data = data.unwrap_or_default(); + + let id = uuid::Uuid::new_v4(); + args.batch_modeling_cmd( + id, + ModelingCmd::Loft { + section_ids: sketch_groups.iter().map(|group| group.id).collect(), + base_curve_index: data.base_curve_index, + bez_approximate_rational: data.bez_approximate_rational.unwrap_or(false), + tolerance: data.tolerance.unwrap_or(default_tolerance(&args.ctx.settings.units)), + v_degree: data + .v_degree + .unwrap_or_else(|| std::num::NonZeroU32::new(DEFAULT_V_DEGREE).unwrap()) + .into(), + }, + ) + .await?; + + // Using the first sketch as the base curve, idk we might want to change this later. + do_post_extrude(sketch_groups[0].clone(), 0.0, args).await +} diff --git a/src/wasm-lib/kcl/src/std/mod.rs b/src/wasm-lib/kcl/src/std/mod.rs index 3317e003a..97e6f3c28 100644 --- a/src/wasm-lib/kcl/src/std/mod.rs +++ b/src/wasm-lib/kcl/src/std/mod.rs @@ -9,8 +9,10 @@ pub mod fillet; pub mod helix; pub mod import; pub mod kcl_stdlib; +pub mod loft; pub mod math; pub mod patterns; +pub mod planes; pub mod polar; pub mod revolve; pub mod segment; @@ -98,6 +100,8 @@ lazy_static! { Box::new(crate::std::shell::Shell), Box::new(crate::std::shell::Hollow), Box::new(crate::std::revolve::Revolve), + Box::new(crate::std::loft::Loft), + Box::new(crate::std::planes::OffsetPlane), Box::new(crate::std::import::Import), Box::new(crate::std::math::Cos), Box::new(crate::std::math::Sin), @@ -484,7 +488,7 @@ layout: manual buf.push_str(&fn_docs); // Write the file. - expectorate::assert_contents(&format!("../../../docs/kcl/{}.md", internal_fn.name()), &buf); + expectorate::assert_contents(format!("../../../docs/kcl/{}.md", internal_fn.name()), &buf); } } diff --git a/src/wasm-lib/kcl/src/std/planes.rs b/src/wasm-lib/kcl/src/std/planes.rs new file mode 100644 index 000000000..47ad5c379 --- /dev/null +++ b/src/wasm-lib/kcl/src/std/planes.rs @@ -0,0 +1,168 @@ +//! Standard library plane helpers. + +use derive_docs::stdlib; +use schemars::JsonSchema; +use serde::{Deserialize, Serialize}; + +use crate::{ + errors::KclError, + executor::{KclValue, Metadata, Plane, UserVal}, + std::{sketch::PlaneData, Args}, +}; + +/// One of the standard planes. +#[derive(Debug, Clone, Copy, Deserialize, Serialize, PartialEq, JsonSchema)] +#[serde(rename_all = "camelCase")] +pub enum StandardPlane { + /// The XY plane. + #[serde(rename = "XY", alias = "xy")] + XY, + /// The opposite side of the XY plane. + #[serde(rename = "-XY", alias = "-xy")] + NegXY, + /// The XZ plane. + #[serde(rename = "XZ", alias = "xz")] + XZ, + /// The opposite side of the XZ plane. + #[serde(rename = "-XZ", alias = "-xz")] + NegXZ, + /// The YZ plane. + #[serde(rename = "YZ", alias = "yz")] + YZ, + /// The opposite side of the YZ plane. + #[serde(rename = "-YZ", alias = "-yz")] + NegYZ, +} + +impl From for PlaneData { + fn from(value: StandardPlane) -> Self { + match value { + StandardPlane::XY => PlaneData::XY, + StandardPlane::NegXY => PlaneData::NegXY, + StandardPlane::XZ => PlaneData::XZ, + StandardPlane::NegXZ => PlaneData::NegXZ, + StandardPlane::YZ => PlaneData::YZ, + StandardPlane::NegYZ => PlaneData::NegYZ, + } + } +} + +/// Offset a plane by a distance along its normal. +pub async fn offset_plane(args: Args) -> Result { + let (std_plane, offset): (StandardPlane, f64) = args.get_data_and_float()?; + + let plane = inner_offset_plane(std_plane, offset).await?; + + Ok(KclValue::UserVal(UserVal::set( + vec![Metadata { + source_range: args.source_range, + }], + plane, + ))) +} + +/// Offset a plane by a distance along its normal. +/// +/// For example, if you offset the 'XZ' plane by 10, the new plane will be parallel to the 'XZ' +/// plane and 10 units away from it. +/// +/// ```no_run +/// // Loft a square and a circle on the `XY` plane using offset. +/// const squareSketch = startSketchOn('XY') +/// |> startProfileAt([-100, 200], %) +/// |> line([200, 0], %) +/// |> line([0, -200], %) +/// |> line([-200, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// const circleSketch = startSketchOn(offsetPlane('XY', 150)) +/// |> circle([0, 100], 50, %) +/// +/// loft([squareSketch, circleSketch]) +/// ``` +/// +/// ```no_run +/// // Loft a square and a circle on the `XZ` plane using offset. +/// const squareSketch = startSketchOn('XZ') +/// |> startProfileAt([-100, 200], %) +/// |> line([200, 0], %) +/// |> line([0, -200], %) +/// |> line([-200, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// const circleSketch = startSketchOn(offsetPlane('XZ', 150)) +/// |> circle([0, 100], 50, %) +/// +/// loft([squareSketch, circleSketch]) +/// ``` +/// +/// ```no_run +/// // Loft a square and a circle on the `YZ` plane using offset. +/// const squareSketch = startSketchOn('YZ') +/// |> startProfileAt([-100, 200], %) +/// |> line([200, 0], %) +/// |> line([0, -200], %) +/// |> line([-200, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// const circleSketch = startSketchOn(offsetPlane('YZ', 150)) +/// |> circle([0, 100], 50, %) +/// +/// loft([squareSketch, circleSketch]) +/// ``` +/// +/// ```no_run +/// // Loft a square and a circle on the `-XZ` plane using offset. +/// const squareSketch = startSketchOn('-XZ') +/// |> startProfileAt([-100, 200], %) +/// |> line([200, 0], %) +/// |> line([0, -200], %) +/// |> line([-200, 0], %) +/// |> lineTo([profileStartX(%), profileStartY(%)], %) +/// |> close(%) +/// +/// const circleSketch = startSketchOn(offsetPlane('-XZ', -150)) +/// |> circle([0, 100], 50, %) +/// +/// loft([squareSketch, circleSketch]) +/// ``` +#[stdlib { + name = "offsetPlane", +}] +async fn inner_offset_plane(std_plane: StandardPlane, offset: f64) -> Result { + // Convert to the plane type. + let plane_data: PlaneData = std_plane.into(); + // Convert to a plane. + let mut plane = Plane::from(plane_data); + + match std_plane { + StandardPlane::XY => { + plane.origin.z += offset; + } + StandardPlane::XZ => { + plane.origin.y -= offset; + } + StandardPlane::YZ => { + plane.origin.x += offset; + } + StandardPlane::NegXY => { + plane.origin.z -= offset; + } + StandardPlane::NegXZ => { + plane.origin.y += offset; + } + StandardPlane::NegYZ => { + plane.origin.x -= offset; + } + } + + Ok(PlaneData::Plane { + origin: Box::new(plane.origin), + x_axis: Box::new(plane.x_axis), + y_axis: Box::new(plane.y_axis), + z_axis: Box::new(plane.z_axis), + }) +} diff --git a/src/wasm-lib/kcl/src/std/revolve.rs b/src/wasm-lib/kcl/src/std/revolve.rs index b91940a0f..ffdffc7a4 100644 --- a/src/wasm-lib/kcl/src/std/revolve.rs +++ b/src/wasm-lib/kcl/src/std/revolve.rs @@ -299,7 +299,7 @@ async fn inner_revolve( } } - do_post_extrude(sketch_group, 0.0, id, args).await + do_post_extrude(sketch_group, 0.0, args).await } #[cfg(test)] diff --git a/src/wasm-lib/kcl/src/std/sketch.rs b/src/wasm-lib/kcl/src/std/sketch.rs index 87d99fdcf..e62cc719e 100644 --- a/src/wasm-lib/kcl/src/std/sketch.rs +++ b/src/wasm-lib/kcl/src/std/sketch.rs @@ -1269,7 +1269,7 @@ pub(crate) async fn inner_start_profile_at( adjust_camera: false, planar_normal: if let SketchSurface::Plane(plane) = &sketch_surface { // We pass in the normal for the plane here. - Some(plane.z_axis.clone().into()) + Some(plane.z_axis.into()) } else { None }, diff --git a/src/wasm-lib/kcl/src/token/tokeniser.rs b/src/wasm-lib/kcl/src/token/tokeniser.rs index 60f229a2e..f90e413c7 100644 --- a/src/wasm-lib/kcl/src/token/tokeniser.rs +++ b/src/wasm-lib/kcl/src/token/tokeniser.rs @@ -50,13 +50,13 @@ pub fn token(i: &mut Located<&str>) -> PResult { } fn block_comment(i: &mut Located<&str>) -> PResult { - let inner = ("/*", take_until(0.., "*/"), "*/").recognize(); + let inner = ("/*", take_until(0.., "*/"), "*/").take(); let (value, range) = inner.with_span().parse_next(i)?; Ok(Token::from_range(range, TokenType::BlockComment, value.to_string())) } fn line_comment(i: &mut Located<&str>) -> PResult { - let inner = (r#"//"#, take_till(0.., ['\n', '\r'])).recognize(); + let inner = (r#"//"#, take_till(0.., ['\n', '\r'])).take(); let (value, range) = inner.with_span().parse_next(i)?; Ok(Token::from_range(range, TokenType::LineComment, value.to_string())) } @@ -68,7 +68,7 @@ fn number(i: &mut Located<&str>) -> PResult { // No digits before the decimal point. ('.', digit1).map(|_| ()), )); - let (value, range) = number_parser.recognize().with_span().parse_next(i)?; + let (value, range) = number_parser.take().with_span().parse_next(i)?; Ok(Token::from_range(range, TokenType::Number, value.to_string())) } @@ -79,12 +79,12 @@ fn whitespace(i: &mut Located<&str>) -> PResult { fn inner_word(i: &mut Located<&str>) -> PResult<()> { one_of(('a'..='z', 'A'..='Z', '_')).parse_next(i)?; - repeat(0.., one_of(('a'..='z', 'A'..='Z', '0'..='9', '_'))).parse_next(i)?; + repeat::<_, _, (), _, _>(0.., one_of(('a'..='z', 'A'..='Z', '0'..='9', '_'))).parse_next(i)?; Ok(()) } fn word(i: &mut Located<&str>) -> PResult { - let (value, range) = inner_word.recognize().with_span().parse_next(i)?; + let (value, range) = inner_word.take().with_span().parse_next(i)?; Ok(Token::from_range(range, TokenType::Word, value.to_string())) } @@ -162,9 +162,9 @@ fn inner_single_quote(i: &mut Located<&str>) -> PResult<()> { } fn string(i: &mut Located<&str>) -> PResult { - let single_quoted_string = ('\'', inner_single_quote.recognize(), '\''); - let double_quoted_string = ('"', inner_double_quote.recognize(), '"'); - let either_quoted_string = alt((single_quoted_string.recognize(), double_quoted_string.recognize())); + let single_quoted_string = ('\'', inner_single_quote.take(), '\''); + let double_quoted_string = ('"', inner_double_quote.take(), '"'); + let either_quoted_string = alt((single_quoted_string.take(), double_quoted_string.take())); let (value, range): (&str, _) = either_quoted_string.with_span().parse_next(i)?; Ok(Token::from_range(range, TokenType::String, value.to_string())) } diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft0.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft0.png new file mode 100644 index 000000000..fe4cd85c5 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft0.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft1.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft1.png new file mode 100644 index 000000000..a44145b36 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft1.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft2.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft2.png new file mode 100644 index 000000000..a44145b36 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_loft2.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane0.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane0.png new file mode 100644 index 000000000..35e17abd9 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane0.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane1.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane1.png new file mode 100644 index 000000000..0f07c1225 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane1.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane2.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane2.png new file mode 100644 index 000000000..a7f52a56e Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane2.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane3.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane3.png new file mode 100644 index 000000000..c3efc7840 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane3.png differ diff --git a/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane4.png b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane4.png new file mode 100644 index 000000000..701b46c98 Binary files /dev/null and b/src/wasm-lib/kcl/tests/outputs/serial_test_example_offset_plane4.png differ diff --git a/src/wasm-lib/rust-toolchain.toml b/src/wasm-lib/rust-toolchain.toml index 4ec1fbb28..251f956ff 100644 --- a/src/wasm-lib/rust-toolchain.toml +++ b/src/wasm-lib/rust-toolchain.toml @@ -1,3 +1,3 @@ [toolchain] -channel = "1.80.1" +channel = "1.81.0" components = ["clippy", "rustfmt"] diff --git a/src/wasm-lib/tests/executor/inputs/extrude-inside-fn-with-tags.kcl b/src/wasm-lib/tests/executor/inputs/extrude-inside-fn-with-tags.kcl index 3a84036a8..91e3a8633 100644 --- a/src/wasm-lib/tests/executor/inputs/extrude-inside-fn-with-tags.kcl +++ b/src/wasm-lib/tests/executor/inputs/extrude-inside-fn-with-tags.kcl @@ -56,10 +56,10 @@ const bracketBody = bs |> fillet({ radius: radius, tags: [ - getNextAdjacentEdge(bs.sketchGroup.tags.edge7), - getNextAdjacentEdge(bs.sketchGroup.tags.edge2), - getNextAdjacentEdge(bs.sketchGroup.tags.edge3), - getNextAdjacentEdge(bs.sketchGroup.tags.edge6) + getPreviousAdjacentEdge(bs.sketchGroup.tags.edge7), + getPreviousAdjacentEdge(bs.sketchGroup.tags.edge2), + getPreviousAdjacentEdge(bs.sketchGroup.tags.edge3), + getPreviousAdjacentEdge(bs.sketchGroup.tags.edge6) ] }, %) diff --git a/src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl b/src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl index bdd48897a..c6d2b9a0c 100644 --- a/src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl +++ b/src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl @@ -55,10 +55,10 @@ const bracketBody = bs |> fillet({ radius: radius, tags: [ - getNextAdjacentEdge(bs.tags.edge7), - getNextAdjacentEdge(bs.tags.edge2), - getNextAdjacentEdge(bs.tags.edge3), - getNextAdjacentEdge(bs.tags.edge6) + getPreviousAdjacentEdge(bs.tags.edge7), + getPreviousAdjacentEdge(bs.tags.edge2), + getPreviousAdjacentEdge(bs.tags.edge3), + getPreviousAdjacentEdge(bs.tags.edge6) ] }, %) diff --git a/src/wasm-lib/tests/executor/inputs/global-tags.kcl b/src/wasm-lib/tests/executor/inputs/global-tags.kcl index 8f7902a68..978183170 100644 --- a/src/wasm-lib/tests/executor/inputs/global-tags.kcl +++ b/src/wasm-lib/tests/executor/inputs/global-tags.kcl @@ -56,10 +56,10 @@ const bracketBody = bs |> fillet({ radius: radius, tags: [ - getNextAdjacentEdge(bs.tags.edge7), - getNextAdjacentEdge(bs.tags.edge2), - getNextAdjacentEdge(bs.tags.edge3), - getNextAdjacentEdge(bs.tags.edge6) + getPreviousAdjacentEdge(bs.tags.edge7), + getPreviousAdjacentEdge(bs.tags.edge2), + getPreviousAdjacentEdge(bs.tags.edge3), + getPreviousAdjacentEdge(bs.tags.edge6) ] }, %) @@ -89,7 +89,7 @@ const tabsR = startSketchOn(tabPlane) radius: holeDiam / 2, tags: [ getNextAdjacentEdge(edge12), - getNextAdjacentEdge(edge13) + getNextAdjacentEdge(edge11) ] }, %) |> patternLinear3d({ diff --git a/src/wasm-lib/tests/executor/inputs/slow_lego.kcl.tmpl b/src/wasm-lib/tests/executor/inputs/slow_lego.kcl.tmpl new file mode 100644 index 000000000..a8aae57f2 --- /dev/null +++ b/src/wasm-lib/tests/executor/inputs/slow_lego.kcl.tmpl @@ -0,0 +1,81 @@ +// 2x8 Lego Brick +// A standard Lego brick with 2 bumps wide and 8 bumps long. +// Define constants +const lbumps = 10 // number of bumps long +const wbumps = {{N}} // number of bumps wide +const pitch = 8.0 +const clearance = 0.1 +const bumpDiam = 4.8 +const bumpHeight = 1.8 +const height = 9.6 +const t = (pitch - (2 * clearance) - bumpDiam) / 2.0 +const totalLength = lbumps * pitch - (2.0 * clearance) +const totalWidth = wbumps * pitch - (2.0 * clearance) +// Create the plane for the pegs. This is a hack so that the pegs can be patterned along the face of the lego base. +const pegFace = { + plane: { + origin: { x: 0, y: 0, z: height }, + xAxis: { x: 1, y: 0, z: 0 }, + yAxis: { x: 0, y: 1, z: 0 }, + zAxis: { x: 0, y: 0, z: 1 } + } +} +// Create the plane for the tubes underneath the lego. This is a hack so that the tubes can be patterned underneath the lego. +const tubeFace = { + plane: { + origin: { x: 0, y: 0, z: height - t }, + xAxis: { x: 1, y: 0, z: 0 }, + yAxis: { x: 0, y: 1, z: 0 }, + zAxis: { x: 0, y: 0, z: 1 } + } +} +// Make the base +const s = startSketchOn('XY') + |> startProfileAt([-totalWidth / 2, -totalLength / 2], %) + |> line([totalWidth, 0], %) + |> line([0, totalLength], %) + |> line([-totalWidth, 0], %) + |> close(%) + |> extrude(height, %) + +// Sketch and extrude a rectangular shape to create the shell underneath the lego. This is a hack until we have a shell function. +const shellExtrude = startSketchOn(s, "start") + |> startProfileAt([ + -(totalWidth / 2 - t), + -(totalLength / 2 - t) + ], %) + |> line([totalWidth - (2 * t), 0], %) + |> line([0, totalLength - (2 * t)], %) + |> line([-(totalWidth - (2 * t)), 0], %) + |> close(%) + |> extrude(-(height - t), %) + +fn tr = (i) => { + let j = i + 1 + let x = (j/wbumps) * pitch + let y = (j % wbumps) * pitch + return { + translate: [x, y, 0], + } +} + +// Create the pegs on the top of the base +const totalBumps = (wbumps * lbumps)-1 +const peg = startSketchOn(s, 'end') + |> circle([ + -(pitch*(wbumps-1)/2), + -(pitch*(lbumps-1)/2) + ], bumpDiam / 2, %) + |> patternLinear2d({ + axis: [1, 0], + repetitions: wbumps-1, + distance: pitch + }, %) + |> patternLinear2d({ + axis: [0, 1], + repetitions: lbumps-1, + distance: pitch + }, %) + |> extrude(bumpHeight, %) + // |> patternTransform(int(totalBumps-1), tr, %) + diff --git a/src/wasm-lib/tests/executor/main.rs b/src/wasm-lib/tests/executor/main.rs index 43e7edb40..66f3c18e4 100644 --- a/src/wasm-lib/tests/executor/main.rs +++ b/src/wasm-lib/tests/executor/main.rs @@ -1287,7 +1287,7 @@ capScrew([0, 0.5, 0], 50, 37.5, 50, 25) } #[tokio::test(flavor = "multi_thread")] -async fn kcl_test_bracket_with_fillets_ensure_fail_on_flush_source_ranges() { +async fn kcl_test_bracket_with_fillets() { let code = r#"// Shelf Bracket // This is a shelf bracket made out of 6061-T6 aluminum sheet metal. The required thickness is calculated based on a point load of 300 lbs applied to the end of the shelf. There are two brackets holding up the shelf, so the moment experienced is divided by 2. The shelf is 1 foot long from the wall. @@ -1328,11 +1328,7 @@ const bracket = startSketchOn('XY') "#; let result = execute_and_snapshot(code, UnitLength::Mm).await; - assert!(result.is_err()); - assert_eq!( - result.err().unwrap().to_string(), - r#"engine: KclErrorDetails { source_ranges: [SourceRange([1329, 1430])], message: "Modeling command failed: [ApiError { error_code: BadRequest, message: \"Fillet failed\" }]" }"# - ); + assert!(result.is_ok()); } #[tokio::test(flavor = "multi_thread")] @@ -1536,13 +1532,13 @@ const bracket = startSketchOn('XY') |> fillet({ radius: filletR, tags: [ - getPreviousAdjacentEdge(innerEdge) + getNextAdjacentEdge(innerEdge) ] }, %) |> fillet({ radius: filletR + thickness, tags: [ - getPreviousAdjacentEdge(outerEdge) + getNextAdjacentEdge(outerEdge) ] }, %) diff --git a/vite.base.config.ts b/vite.base.config.ts index 624b808df..fd213f2df 100644 --- a/vite.base.config.ts +++ b/vite.base.config.ts @@ -49,7 +49,13 @@ export function getDefineKeys(names: string[]) { export function getBuildDefine(env: ConfigEnv<'build'>) { const { command, forgeConfig } = env - const names = forgeConfig.renderer + const renderer = (forgeConfig && forgeConfig.renderer) ?? [ + { + name: 'main_window', + config: 'vite.renderer.config.ts', + }, + ] + const names = renderer .filter(({ name }) => name != null) .map(({ name }) => name!) const defineKeys = getDefineKeys(names) diff --git a/vite.main.config.ts b/vite.main.config.ts index ccbbdd3d4..ed4e6203a 100644 --- a/vite.main.config.ts +++ b/vite.main.config.ts @@ -67,7 +67,7 @@ export default defineConfig((env) => { }, build: { lib: { - entry: forgeConfigSelf.entry!, + entry: forgeConfigSelf?.entry ?? 'src/main.ts', fileName: () => '[name].js', formats: ['cjs'], }, diff --git a/vite.preload.config.ts b/vite.preload.config.ts index 21305601b..c3a78a01a 100644 --- a/vite.preload.config.ts +++ b/vite.preload.config.ts @@ -20,7 +20,7 @@ export default defineConfig((env) => { base: './', build: { lib: { - entry: forgeConfigSelf.entry!, + entry: forgeConfigSelf?.entry ?? 'src/preload.ts', fileName: () => '[name].js', formats: ['cjs'], }, diff --git a/vite.renderer.config.ts b/vite.renderer.config.ts index 90f156cd8..95086356c 100644 --- a/vite.renderer.config.ts +++ b/vite.renderer.config.ts @@ -9,7 +9,7 @@ import { lezer } from '@lezer/generator/rollup' export default defineConfig((env) => { const forgeEnv = env as ConfigEnv<'renderer'> const { root, mode, forgeConfigSelf } = forgeEnv - const name = forgeConfigSelf.name ?? '' + const name = forgeConfigSelf?.name ?? 'main_window' return { root, diff --git a/yarn.lock b/yarn.lock index 566711336..25eee8acd 100644 --- a/yarn.lock +++ b/yarn.lock @@ -2,6 +2,11 @@ # yarn lockfile v1 +"7zip-bin@~5.2.0": + version "5.2.0" + resolved "https://registry.yarnpkg.com/7zip-bin/-/7zip-bin-5.2.0.tgz#7a03314684dd6572b7dfa89e68ce31d60286854d" + integrity sha512-ukTPVhqG4jNzMro2qA9HSCSSVJN3aN7tlb+hfqYCt3ER0yWroeA2VR38MNrOHLQ/cVj+DaIMad0kFCtWWowh/A== + "@adobe/css-tools@^4.0.1": version "4.4.0" resolved "https://registry.yarnpkg.com/@adobe/css-tools/-/css-tools-4.4.0.tgz#728c484f4e10df03d5a3acd0d8adcbbebff8ad63" @@ -1336,6 +1341,14 @@ resolved "https://registry.yarnpkg.com/@csstools/utilities/-/utilities-2.0.0.tgz#f7ff0fee38c9ffb5646d47b6906e0bc8868bde60" integrity sha512-5VdOr0Z71u+Yp3ozOx8T11N703wIFGVRgOWbOZMKgglPJsWA54MRIoMNVMa7shUToIhx5J8vX4sOZgD2XiihiQ== +"@develar/schema-utils@~2.6.5": + version "2.6.5" + resolved "https://registry.yarnpkg.com/@develar/schema-utils/-/schema-utils-2.6.5.tgz#3ece22c5838402419a6e0425f85742b961d9b6c6" + integrity sha512-0cp4PsWQ/9avqTVMCtZ+GirikIA36ikvjtHweU4/j8yLtgObI0+JUPhYFScgwlteveGB1rt3Cm8UhN04XayDig== + dependencies: + ajv "^6.12.0" + ajv-keywords "^3.4.1" + "@electron-forge/cli@^7.4.0": version "7.4.0" resolved "https://registry.yarnpkg.com/@electron-forge/cli/-/cli-7.4.0.tgz#db16f4bc678d1f6cec8890cdf86041e9c8336350" @@ -1638,6 +1651,15 @@ optionalDependencies: global-agent "^3.0.0" +"@electron/notarize@2.2.1": + version "2.2.1" + resolved "https://registry.yarnpkg.com/@electron/notarize/-/notarize-2.2.1.tgz#d0aa6bc43cba830c41bfd840b85dbe0e273f59fe" + integrity sha512-aL+bFMIkpR0cmmj5Zgy0LMKEpgy43/hw5zadEArgmAMWWlKc5buwFvFT9G/o/YJkvXAJm5q3iuTuLaiaXW39sg== + dependencies: + debug "^4.1.1" + fs-extra "^9.0.1" + promise-retry "^2.0.1" + "@electron/notarize@^2.1.0": version "2.3.2" resolved "https://registry.yarnpkg.com/@electron/notarize/-/notarize-2.3.2.tgz#20a52a961747be8542a35003380988a0d3fe15e6" @@ -1647,6 +1669,18 @@ fs-extra "^9.0.1" promise-retry "^2.0.1" +"@electron/osx-sign@1.0.5": + version "1.0.5" + resolved "https://registry.yarnpkg.com/@electron/osx-sign/-/osx-sign-1.0.5.tgz#0af7149f2fce44d1a8215660fd25a9fb610454d8" + integrity sha512-k9ZzUQtamSoweGQDV2jILiRIHUu7lYlJ3c6IEmjv1hC17rclE+eb9U+f6UFlOOETo0JzY1HNlXy4YOlCvl+Lww== + dependencies: + compare-version "^0.1.2" + debug "^4.3.4" + fs-extra "^10.0.0" + isbinaryfile "^4.0.8" + minimist "^1.2.6" + plist "^3.0.5" + "@electron/osx-sign@^1.0.5": version "1.3.1" resolved "https://registry.yarnpkg.com/@electron/osx-sign/-/osx-sign-1.3.1.tgz#faf7eeca7ca004a6be541dc4cf7a1bd59ec59b1c" @@ -1704,6 +1738,19 @@ tar "^6.0.5" yargs "^17.0.1" +"@electron/universal@1.5.1": + version "1.5.1" + resolved "https://registry.yarnpkg.com/@electron/universal/-/universal-1.5.1.tgz#f338bc5bcefef88573cf0ab1d5920fac10d06ee5" + integrity sha512-kbgXxyEauPJiQQUNG2VgUeyfQNFk6hBF11ISN2PNI6agUgPl55pv4eQmaqHzTAzchBvqZ2tQuRVaPStGf0mxGw== + dependencies: + "@electron/asar" "^3.2.1" + "@malept/cross-spawn-promise" "^1.1.0" + debug "^4.3.1" + dir-compare "^3.0.0" + fs-extra "^9.0.1" + minimatch "^3.0.4" + plist "^3.0.4" + "@electron/universal@^2.0.1": version "2.0.1" resolved "https://registry.yarnpkg.com/@electron/universal/-/universal-2.0.1.tgz#7b070ab355e02957388f3dbd68e2c3cd08c448ae" @@ -1913,18 +1960,6 @@ resolved "https://registry.yarnpkg.com/@gar/promisify/-/promisify-1.1.3.tgz#555193ab2e3bb3b6adc3d551c9c030d9e860daf6" integrity sha512-k2Ty1JcVojjJFwrg/ThKi2ujJ7XNLYaFGNB/bWT9wGR+oSMJHMa5w+CUq6p/pVrKeNNgA7pCqEcjSnHVoqJQFw== -"@hapi/hoek@^9.0.0", "@hapi/hoek@^9.3.0": - version "9.3.0" - resolved "https://registry.yarnpkg.com/@hapi/hoek/-/hoek-9.3.0.tgz#8368869dcb735be2e7f5cb7647de78e167a251fb" - integrity sha512-/c6rf4UJlmHlC9b5BaNvzAcFv7HZ2QHaV0D4/HNlBdvFnvQq8RI4kYdhyPCl7Xj+oWvTWQ8ujhqS53LIgAe6KQ== - -"@hapi/topo@^5.1.0": - version "5.1.0" - resolved "https://registry.yarnpkg.com/@hapi/topo/-/topo-5.1.0.tgz#dc448e332c6c6e37a4dc02fd84ba8d44b9afb012" - integrity sha512-foQZKJig7Ob0BMAYBfcJk8d77QtOe7Wo4ox7ff1lQYoNNAb6jwcY1ncdoy2e9wQZzvNy7ODZCYJkK8kzmcAnAg== - dependencies: - "@hapi/hoek" "^9.0.0" - "@headlessui/react@^1.7.19": version "1.7.19" resolved "https://registry.yarnpkg.com/@headlessui/react/-/react-1.7.19.tgz#91c78cf5fcb254f4a0ebe96936d48421caf75f40" @@ -2095,6 +2130,16 @@ dependencies: cross-spawn "^7.0.1" +"@malept/flatpak-bundler@^0.4.0": + version "0.4.0" + resolved "https://registry.yarnpkg.com/@malept/flatpak-bundler/-/flatpak-bundler-0.4.0.tgz#e8a32c30a95d20c2b1bb635cc580981a06389858" + integrity sha512-9QOtNffcOF/c1seMCDnjckb3R9WHcG34tky+FHpNKKCW0wc/scYLwMtO+ptyGUfMW0/b/n4qRiALlaFHc9Oj7Q== + dependencies: + debug "^4.1.1" + fs-extra "^9.0.0" + lodash "^4.17.15" + tmp-promise "^3.0.2" + "@nicolo-ribaudo/eslint-scope-5-internals@5.1.1-v1": version "5.1.1-v1" resolved "https://registry.yarnpkg.com/@nicolo-ribaudo/eslint-scope-5-internals/-/eslint-scope-5-internals-5.1.1-v1.tgz#dbf733a965ca47b1973177dc0bb6c889edcfb129" @@ -2269,28 +2314,16 @@ resolved "https://registry.yarnpkg.com/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.21.0.tgz#20c09cf44dcb082140cc7f439dd679fe4bba3375" integrity sha512-2jsCDZwtQvRhejHLfZ1JY6w6kEuEtfF9nzYsZxzSlNVKDX+DpsDJ+Rbjkm74nvg2rdx0gwBS+IMdvwJuq3S9pQ== +"@rtsao/scc@^1.1.0": + version "1.1.0" + resolved "https://registry.yarnpkg.com/@rtsao/scc/-/scc-1.1.0.tgz#927dd2fae9bc3361403ac2c7a00c32ddce9ad7e8" + integrity sha512-zt6OdqaDoOnJ1ZYsCYGt9YmWzDXl4vQdKTyJev62gFhRGKdx7mcT54V9KIjg+d2wi9EXsPvAPKe7i7WjfVWB8g== + "@rushstack/eslint-patch@^1.1.0": version "1.10.4" resolved "https://registry.yarnpkg.com/@rushstack/eslint-patch/-/eslint-patch-1.10.4.tgz#427d5549943a9c6fce808e39ea64dbe60d4047f1" integrity sha512-WJgX9nzTqknM393q1QJDJmoW28kUfEnybeTfVNcNAPnIx210RXm2DiXiHzfNPJNIUUb1tJnz/l4QGtJ30PgWmA== -"@sideway/address@^4.1.5": - version "4.1.5" - resolved "https://registry.yarnpkg.com/@sideway/address/-/address-4.1.5.tgz#4bc149a0076623ced99ca8208ba780d65a99b9d5" - integrity sha512-IqO/DUQHUkPeixNQ8n0JA6102hT9CmaljNTPmQ1u8MEhBo/R4Q8eKLN/vGZxuebwOroDB4cbpjheD4+/sKFK4Q== - dependencies: - "@hapi/hoek" "^9.0.0" - -"@sideway/formula@^3.0.1": - version "3.0.1" - resolved "https://registry.yarnpkg.com/@sideway/formula/-/formula-3.0.1.tgz#80fcbcbaf7ce031e0ef2dd29b1bfc7c3f583611f" - integrity sha512-/poHZJJVjx3L+zVD6g9KgHfYnb443oi7wLu/XKojDviHy6HOEOA6z1Trk5aR1dGcmPenJEgb2sK2I80LeS3MIg== - -"@sideway/pinpoint@^2.0.0": - version "2.0.0" - resolved "https://registry.yarnpkg.com/@sideway/pinpoint/-/pinpoint-2.0.0.tgz#cff8ffadc372ad29fd3f78277aeb29e632cc70df" - integrity sha512-RNiOoTPkptFtSVzQevY/yWtZwf/RxyVnPy/OcA9HBM3MlGDnBEYL5B41H0MTn0Uec8Hi+2qUtTfG2WWZBmMejQ== - "@sinclair/typebox@^0.27.8": version "0.27.8" resolved "https://registry.yarnpkg.com/@sinclair/typebox/-/typebox-0.27.8.tgz#6667fac16c436b5434a387a34dedb013198f6e6e" @@ -2448,6 +2481,13 @@ resolved "https://registry.yarnpkg.com/@types/d3-force/-/d3-force-3.0.10.tgz#6dc8fc6e1f35704f3b057090beeeb7ac674bff1a" integrity sha512-ZYeSaCF3p73RdOKcjj+swRlZfnYpK1EbaDiYICEEp5Q6sUiqFaFQ9qgoshp5CzIyyb/yD09kD9o2zEltCexlgw== +"@types/debug@^4.1.6": + version "4.1.12" + resolved "https://registry.yarnpkg.com/@types/debug/-/debug-4.1.12.tgz#a155f21690871953410df4b6b6f53187f0500917" + integrity sha512-vIChWdVG3LG1SMxEvI/AK+FWJthlrqlTu7fbrlywTkkaONwk/UAGaULXRlf8vkzFBLVm0zkMdCquhL5aOjhXPQ== + dependencies: + "@types/ms" "*" + "@types/electron@^1.6.10": version "1.6.10" resolved "https://registry.yarnpkg.com/@types/electron/-/electron-1.6.10.tgz#7e87888ed3888767cca68e92772c2c8ea46bc873" @@ -2468,7 +2508,7 @@ resolved "https://registry.yarnpkg.com/@types/estree/-/estree-1.0.5.tgz#a6ce3e556e00fd9895dd872dd172ad0d4bd687f4" integrity sha512-/kYRxGDLWzHOB7q+wtSUQlFrtcdUccpfy+X+9iMBpHK8QLLhx2wIPYuS5DYtR9Wa/YlZAbIovy7qVdB1Aq6Lyw== -"@types/fs-extra@^9.0.1": +"@types/fs-extra@9.0.13", "@types/fs-extra@^9.0.1", "@types/fs-extra@^9.0.11": version "9.0.13" resolved "https://registry.yarnpkg.com/@types/fs-extra/-/fs-extra-9.0.13.tgz#7594fbae04fe7f1918ce8b3d213f74ff44ac1f45" integrity sha512-nEnwB++1u5lVDM2UI4c1+5R+FYaKfaAzS4OococimjVm3nQw3TuzH5UNsocrcTBbhnerblyHj4A49qXbIiZdpA== @@ -2552,6 +2592,11 @@ resolved "https://registry.yarnpkg.com/@types/mocha/-/mocha-10.0.7.tgz#4c620090f28ca7f905a94b706f74dc5b57b44f2f" integrity sha512-GN8yJ1mNTcFcah/wKEFIJckJx9iJLoMSzWcfRRuxz/Jk+U6KQNnml+etbtxFK8lPjzOw3zp4Ha/kjSst9fsHYw== +"@types/ms@*": + version "0.7.34" + resolved "https://registry.yarnpkg.com/@types/ms/-/ms-0.7.34.tgz#10964ba0dee6ac4cd462e2795b6bebd407303433" + integrity sha512-nG96G3Wp6acyAgJqGasjODb+acrI7KltPiRxzHPXnP3NgI28bpQDRv53olbqGXbfcgF5aiiHmO3xpwEpS5Ld9g== + "@types/node@*", "@types/node@^22.5.0": version "22.5.0" resolved "https://registry.yarnpkg.com/@types/node/-/node-22.5.0.tgz#10f01fe9465166b4cab72e75f60d8b99d019f958" @@ -2578,6 +2623,14 @@ dependencies: "@types/node" "*" +"@types/plist@^3.0.1": + version "3.0.5" + resolved "https://registry.yarnpkg.com/@types/plist/-/plist-3.0.5.tgz#9a0c49c0f9886c8c8696a7904dd703f6284036e0" + integrity sha512-E6OCaRmAe4WDmWNsL/9RMqdkkzDCY1etutkflWk4c+AcjDU07Pcz1fQwTX0TQz+Pxqn9i4L1TU3UFpjnrcDgxA== + dependencies: + "@types/node" "*" + xmlbuilder ">=11.0.1" + "@types/pngjs@^6.0.4": version "6.0.5" resolved "https://registry.yarnpkg.com/@types/pngjs/-/pngjs-6.0.5.tgz#6dec2f7eb8284543ca4e423f3c09b119fa939ea3" @@ -2662,12 +2715,10 @@ resolved "https://registry.yarnpkg.com/@types/uuid/-/uuid-9.0.8.tgz#7545ba4fc3c003d6c756f651f3bf163d8f0f29ba" integrity sha512-jg+97EGIcY9AGHJJRaaPVgetKDsrTgbRjQ5Msgjh/DQKEFl0DtyRr/VCOyD1T2R1MNeWPK/u7JoGhlDZnKBAfA== -"@types/wait-on@^5.3.4": - version "5.3.4" - resolved "https://registry.yarnpkg.com/@types/wait-on/-/wait-on-5.3.4.tgz#5ee270b3e073fb01073f9f044922c6893de8c4d2" - integrity sha512-EBsPjFMrFlMbbUFf9D1Fp+PAB2TwmUn7a3YtHyD9RLuTIk1jDd8SxXVAoez2Ciy+8Jsceo2MYEYZzJ/DvorOKw== - dependencies: - "@types/node" "*" +"@types/verror@^1.10.3": + version "1.10.10" + resolved "https://registry.yarnpkg.com/@types/verror/-/verror-1.10.10.tgz#d5a4b56abac169bfbc8b23d291363a682e6fa087" + integrity sha512-l4MM0Jppn18hb9xmM6wwD1uTdShpf9Pn80aXTStnK1C94gtPvJcV2FrDmbOQUAQfJ1cKZHktkQUDwEqaAKXMMg== "@types/webxr@*": version "0.5.19" @@ -2968,7 +3019,12 @@ aggregate-error@^3.0.0: clean-stack "^2.0.0" indent-string "^4.0.0" -ajv@^6.12.4: +ajv-keywords@^3.4.1: + version "3.5.2" + resolved "https://registry.yarnpkg.com/ajv-keywords/-/ajv-keywords-3.5.2.tgz#31f29da5ab6e00d1c2d329acf7b5929614d5014d" + integrity sha512-5p6WTN0DdTGVQk6VjcEju19IgaHudalcfabD7yhDGeA6bcQnmL+CpveLJq/3hvfwd1aof6L386Ougkx6RfyMIQ== + +ajv@^6.10.0, ajv@^6.12.0, ajv@^6.12.4: version "6.12.6" resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== @@ -3032,6 +3088,44 @@ anymatch@~3.1.2: normalize-path "^3.0.0" picomatch "^2.0.4" +app-builder-bin@4.0.0: + version "4.0.0" + resolved "https://registry.yarnpkg.com/app-builder-bin/-/app-builder-bin-4.0.0.tgz#1df8e654bd1395e4a319d82545c98667d7eed2f0" + integrity sha512-xwdG0FJPQMe0M0UA4Tz0zEB8rBJTRA5a476ZawAqiBkMv16GRK5xpXThOjMaEOFnZ6zabejjG4J3da0SXG63KA== + +app-builder-lib@24.13.3: + version "24.13.3" + resolved "https://registry.yarnpkg.com/app-builder-lib/-/app-builder-lib-24.13.3.tgz#36e47b65fecb8780bb73bff0fee4e0480c28274b" + integrity sha512-FAzX6IBit2POXYGnTCT8YHFO/lr5AapAII6zzhQO3Rw4cEDOgK+t1xhLc5tNcKlicTHlo9zxIwnYCX9X2DLkig== + dependencies: + "@develar/schema-utils" "~2.6.5" + "@electron/notarize" "2.2.1" + "@electron/osx-sign" "1.0.5" + "@electron/universal" "1.5.1" + "@malept/flatpak-bundler" "^0.4.0" + "@types/fs-extra" "9.0.13" + async-exit-hook "^2.0.1" + bluebird-lst "^1.0.9" + builder-util "24.13.1" + builder-util-runtime "9.2.4" + chromium-pickle-js "^0.2.0" + debug "^4.3.4" + ejs "^3.1.8" + electron-publish "24.13.1" + form-data "^4.0.0" + fs-extra "^10.1.0" + hosted-git-info "^4.1.0" + is-ci "^3.0.0" + isbinaryfile "^5.0.0" + js-yaml "^4.1.0" + lazy-val "^1.0.5" + minimatch "^5.1.1" + read-config-file "6.3.2" + sanitize-filename "^1.6.3" + semver "^7.3.8" + tar "^6.1.12" + temp-file "^3.4.0" + "aproba@^1.0.3 || ^2.0.0": version "2.0.0" resolved "https://registry.yarnpkg.com/aproba/-/aproba-2.0.0.tgz#52520b8ae5b569215b354efc0caa3fe1e45a8adc" @@ -3087,7 +3181,7 @@ array-flatten@1.1.1: resolved "https://registry.yarnpkg.com/array-flatten/-/array-flatten-1.1.1.tgz#9a5f699051b1e7073328f2a008968b64ea2955d2" integrity sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg== -array-includes@^3.1.6, array-includes@^3.1.7, array-includes@^3.1.8: +array-includes@^3.1.6, array-includes@^3.1.8: version "3.1.8" resolved "https://registry.yarnpkg.com/array-includes/-/array-includes-3.1.8.tgz#5e370cbe172fdd5dd6530c1d4aadda25281ba97d" integrity sha512-itaWrbYbqpGXkGhZPGUulwnhVf5Hpy1xiCFsGqyIGglbBxmG5vSjxQen3/WGOjPpNEv1RtBLKxbmVXm8HpJStQ== @@ -3116,7 +3210,7 @@ array.prototype.findlast@^1.2.5: es-object-atoms "^1.0.0" es-shim-unscopables "^1.0.2" -array.prototype.findlastindex@^1.2.3: +array.prototype.findlastindex@^1.2.5: version "1.2.5" resolved "https://registry.yarnpkg.com/array.prototype.findlastindex/-/array.prototype.findlastindex-1.2.5.tgz#8c35a755c72908719453f87145ca011e39334d0d" integrity sha512-zfETvRFA8o7EiNn++N5f/kaCw221hrpGsDmcpndVupkPzEc1Wuf3VgC0qby1BbHs7f5DVYjgtEU2LLh5bqeGfQ== @@ -3190,6 +3284,11 @@ asar@^3.0.0: optionalDependencies: "@types/glob" "^7.1.1" +assert-plus@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/assert-plus/-/assert-plus-1.0.0.tgz#f12e0f3c5d77b0b1cdd9146942e4e96c1e4dd525" + integrity sha512-NfJ4UzBCcQGLDlQq7nHxH+tv3kyZ0hHQqF5BO6J7tNJeP5do1llPr8dZ8zHonfhAu0PHAdMkSo+8o0wxg9lZWw== + assertion-error@^1.1.0: version "1.1.0" resolved "https://registry.yarnpkg.com/assertion-error/-/assertion-error-1.1.0.tgz#e60b6b0e8f301bd97e5375215bda406c85118c0b" @@ -3207,6 +3306,16 @@ ast-types@^0.16.1: dependencies: tslib "^2.0.1" +astral-regex@^2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31" + integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ== + +async-exit-hook@^2.0.1: + version "2.0.1" + resolved "https://registry.yarnpkg.com/async-exit-hook/-/async-exit-hook-2.0.1.tgz#8bd8b024b0ec9b1c01cccb9af9db29bd717dfaf3" + integrity sha512-NW2cX8m1Q7KPA7a5M2ULQeZ2wR5qI5PAbw5L0UOMxdioVk9PMZ0h1TmyZEkPYrCvYjDlFICusOu1dlEKAAeXBw== + async@^2.6.4: version "2.6.4" resolved "https://registry.yarnpkg.com/async/-/async-2.6.4.tgz#706b7ff6084664cd7eae713f6f965433b5504221" @@ -3214,6 +3323,11 @@ async@^2.6.4: dependencies: lodash "^4.17.14" +async@^3.2.3: + version "3.2.6" + resolved "https://registry.yarnpkg.com/async/-/async-3.2.6.tgz#1b0728e14929d51b85b449b7f06e27c1145e38ce" + integrity sha512-htCUDlxyyCLMgaM3xXg0C0LW2xqfuQ6p05pCEIsXuyQ+a1koYKTuBMzRNwmybfLgvJDMd0r1LTn4+E0Ti6C2AA== + asynckit@^0.4.0: version "0.4.0" resolved "https://registry.yarnpkg.com/asynckit/-/asynckit-0.4.0.tgz#c79ed97f7f34cb8f2ba1bc9790bcc366474b4b79" @@ -3260,15 +3374,6 @@ axios@^0.26.1: dependencies: follow-redirects "^1.14.8" -axios@^1.6.1: - version "1.7.3" - resolved "https://registry.yarnpkg.com/axios/-/axios-1.7.3.tgz#a1125f2faf702bc8e8f2104ec3a76fab40257d85" - integrity sha512-Ar7ND9pU99eJ9GpoGQKhKf58GpUOgnzuaB7ueNQ5BMi0p+LZ5oaEnfF999fAArcTIBwXTCHAmGcHOZJaWPq9Nw== - dependencies: - follow-redirects "^1.15.6" - form-data "^4.0.0" - proxy-from-env "^1.1.0" - axobject-query@~3.1.1: version "3.1.1" resolved "https://registry.yarnpkg.com/axobject-query/-/axobject-query-3.1.1.tgz#3b6e5c6d4e43ca7ba51c5babf99d22a9c68485e1" @@ -3386,7 +3491,14 @@ bl@^4.1.0: inherits "^2.0.4" readable-stream "^3.4.0" -bluebird@^3.1.1: +bluebird-lst@^1.0.9: + version "1.0.9" + resolved "https://registry.yarnpkg.com/bluebird-lst/-/bluebird-lst-1.0.9.tgz#a64a0e4365658b9ab5fe875eb9dfb694189bb41c" + integrity sha512-7B1Rtx82hjnSD4PGLAjVWeYH3tHAcVUmChh85a3lltKQm6FresXh9ErQo6oAv6CqxttczC3/kEg8SY5NluPuUw== + dependencies: + bluebird "^3.5.5" + +bluebird@^3.1.1, bluebird@^3.5.5: version "3.7.2" resolved "https://registry.yarnpkg.com/bluebird/-/bluebird-3.7.2.tgz#9f229c15be272454ffa973ace0dbee79a1b0c36f" integrity sha512-XpNj6GDQzdfW+r2Wnn7xiSAd7TM3jzkxGXBGTtWKuSXv1xUV+azxAm8jdWZN06QTQk+2N2XB9jRDkvbmQmcRtg== @@ -3469,12 +3581,17 @@ buffer-crc32@~0.2.3: resolved "https://registry.yarnpkg.com/buffer-crc32/-/buffer-crc32-0.2.13.tgz#0d333e3f00eac50aa1454abd30ef8c2a5d9a7242" integrity sha512-VO9Ht/+p3SN7SKWqcrgEzjGbRSJYTx+Q1pTQC0wrWqHx0vpJraQ6GtHx8tvcg1rlK1byhU5gccxgOgj7B0TDkQ== +buffer-equal@^1.0.0: + version "1.0.1" + resolved "https://registry.yarnpkg.com/buffer-equal/-/buffer-equal-1.0.1.tgz#2f7651be5b1b3f057fcd6e7ee16cf34767077d90" + integrity sha512-QoV3ptgEaQpvVwbXdSO39iqPQTCxSF7A5U99AxbHYqUdCizL/lH2Z0A2y6nbZucxMEOtNyZfG2s6gsVugGpKkg== + buffer-from@^1.0.0: version "1.1.2" resolved "https://registry.yarnpkg.com/buffer-from/-/buffer-from-1.1.2.tgz#2b146a6fd72e80b4f55d255f35ed59a3a9a41bd5" integrity sha512-E+XQCRwSbaaiChtv6k6Dwgc+bx+Bs6vuKJHHl5kox/BaKbhiXzqQOwK4cO22yElGp2OCmjwVhT3HmxgyPGnJfQ== -buffer@^5.5.0: +buffer@^5.1.0, buffer@^5.5.0: version "5.7.1" resolved "https://registry.yarnpkg.com/buffer/-/buffer-5.7.1.tgz#ba62e7c13133053582197160851a8f648e99eed0" integrity sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ== @@ -3482,6 +3599,44 @@ buffer@^5.5.0: base64-js "^1.3.1" ieee754 "^1.1.13" +builder-util-runtime@9.2.4: + version "9.2.4" + resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-9.2.4.tgz#13cd1763da621e53458739a1e63f7fcba673c42a" + integrity sha512-upp+biKpN/XZMLim7aguUyW8s0FUpDvOtK6sbanMFDAMBzpHDqdhgVYm6zc9HJ6nWo7u2Lxk60i2M6Jd3aiNrA== + dependencies: + debug "^4.3.4" + sax "^1.2.4" + +builder-util-runtime@9.2.5: + version "9.2.5" + resolved "https://registry.yarnpkg.com/builder-util-runtime/-/builder-util-runtime-9.2.5.tgz#0afdffa0adb5c84c14926c7dd2cf3c6e96e9be83" + integrity sha512-HjIDfhvqx/8B3TDN4GbABQcgpewTU4LMRTQPkVpKYV3lsuxEJoIfvg09GyWTNmfVNSUAYf+fbTN//JX4TH20pg== + dependencies: + debug "^4.3.4" + sax "^1.2.4" + +builder-util@24.13.1: + version "24.13.1" + resolved "https://registry.yarnpkg.com/builder-util/-/builder-util-24.13.1.tgz#4a4c4f9466b016b85c6990a0ea15aa14edec6816" + integrity sha512-NhbCSIntruNDTOVI9fdXz0dihaqX2YuE1D6zZMrwiErzH4ELZHE6mdiB40wEgZNprDia+FghRFgKoAqMZRRjSA== + dependencies: + "7zip-bin" "~5.2.0" + "@types/debug" "^4.1.6" + app-builder-bin "4.0.0" + bluebird-lst "^1.0.9" + builder-util-runtime "9.2.4" + chalk "^4.1.2" + cross-spawn "^7.0.3" + debug "^4.3.4" + fs-extra "^10.1.0" + http-proxy-agent "^5.0.0" + https-proxy-agent "^5.0.1" + is-ci "^3.0.0" + js-yaml "^4.1.0" + source-map-support "^0.5.19" + stat-mode "^1.0.0" + temp-file "^3.4.0" + bytes@3.1.2: version "3.1.2" resolved "https://registry.yarnpkg.com/bytes/-/bytes-3.1.2.tgz#8b0beeb98605adf1b128fa4386403c009e0221a5" @@ -3595,7 +3750,7 @@ chalk@^3.0.0: ansi-styles "^4.1.0" supports-color "^7.1.0" -chalk@^4.0.0, chalk@^4.1.0, chalk@^4.1.1, chalk@^4.1.2: +chalk@^4.0.0, chalk@^4.0.2, chalk@^4.1.0, chalk@^4.1.1, chalk@^4.1.2: version "4.1.2" resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== @@ -3669,6 +3824,14 @@ cli-spinners@^2.5.0: resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-2.9.2.tgz#1773a8f4b9c4d6ac31563df53b3fc1d79462fe41" integrity sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg== +cli-truncate@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-2.1.0.tgz#c39e28bf05edcde5be3b98992a22deed5a2b93c7" + integrity sha512-n8fOixwDD6b/ObinzTrp1ZKFzbgvKZvuz/TvejnLn1aQfC6r52XEx85FmuC+3HI+JM7coBRXUvNqEU2PHVrHpg== + dependencies: + slice-ansi "^3.0.0" + string-width "^4.2.0" + cli-truncate@^3.1.0: version "3.1.0" resolved "https://registry.yarnpkg.com/cli-truncate/-/cli-truncate-3.1.0.tgz#3f23ab12535e3d73e839bb43e73c9de487db1389" @@ -3806,6 +3969,14 @@ confbox@^0.1.7: resolved "https://registry.yarnpkg.com/confbox/-/confbox-0.1.7.tgz#ccfc0a2bcae36a84838e83a3b7f770fb17d6c579" integrity sha512-uJcB/FKZtBMCJpK8MQji6bJHgu1tixKPxRLeGkNzBoOZzpnZUJm0jm2/sBDWcuBx1dYgxV4JU+g5hmNxCyAmdA== +config-file-ts@^0.2.4: + version "0.2.6" + resolved "https://registry.yarnpkg.com/config-file-ts/-/config-file-ts-0.2.6.tgz#b424ff74612fb37f626d6528f08f92ddf5d22027" + integrity sha512-6boGVaglwblBgJqGyxm4+xCmEGcWgnWHSWHY5jad58awQhB6gftq0G8HbzU39YqCIYHMLAiL1yjwiZ36m/CL8w== + dependencies: + glob "^10.3.10" + typescript "^5.3.3" + confusing-browser-globals@^1.0.11: version "1.0.11" resolved "https://registry.yarnpkg.com/confusing-browser-globals/-/confusing-browser-globals-1.0.11.tgz#ae40e9b57cdd3915408a2805ebd3a5585608dc81" @@ -3850,6 +4021,11 @@ core-js-compat@^3.37.1, core-js-compat@^3.38.0: dependencies: browserslist "^4.23.3" +core-util-is@1.0.2: + version "1.0.2" + resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.2.tgz#b5fd54220aa2bc5ab57aab7140c940754503c1a7" + integrity sha512-3lqz5YjWTYnW6dlDa5TLaTCcShfar1e40rmcJVwCBJC6mWlFuj0eCHIElmG1g5kyuJ/GD+8Wn4FFCcz4gJPfaQ== + core-util-is@~1.0.0: version "1.0.3" resolved "https://registry.yarnpkg.com/core-util-is/-/core-util-is-1.0.3.tgz#a6042d3634c2b27e9328f837b965fac83808db85" @@ -3871,6 +4047,13 @@ cosmiconfig@^7.0.0: path-type "^4.0.0" yaml "^1.10.0" +crc@^3.8.0: + version "3.8.0" + resolved "https://registry.yarnpkg.com/crc/-/crc-3.8.0.tgz#ad60269c2c856f8c299e2c4cc0de4556914056c6" + integrity sha512-iX3mfgcTMIq3ZKLIsVFAbv7+Mc10kxabAGQb8HvjA1o3T1PIYprbakQ65d3I+2HGHt6nSKkM9PYjgoJO2KcFBQ== + dependencies: + buffer "^5.1.0" + create-require@^1.1.0: version "1.1.1" resolved "https://registry.yarnpkg.com/create-require/-/create-require-1.1.1.tgz#c1d7e8f1e5f6cfc9ff65f9cd352d37348756c333" @@ -4164,6 +4347,14 @@ diff@^4.0.1: resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d" integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== +dir-compare@^3.0.0: + version "3.3.0" + resolved "https://registry.yarnpkg.com/dir-compare/-/dir-compare-3.3.0.tgz#2c749f973b5c4b5d087f11edaae730db31788416" + integrity sha512-J7/et3WlGUCxjdnD3HAAzQ6nsnc0WL6DD7WcwJb7c39iH1+AWfg+9OqzJNaI6PkBwBvm1mhZNL9iY/nRiZXlPg== + dependencies: + buffer-equal "^1.0.0" + minimatch "^3.0.4" + dir-compare@^4.2.0: version "4.2.0" resolved "https://registry.yarnpkg.com/dir-compare/-/dir-compare-4.2.0.tgz#d1d4999c14fbf55281071fdae4293b3b9ce86f19" @@ -4184,6 +4375,34 @@ dlv@^1.1.3: resolved "https://registry.yarnpkg.com/dlv/-/dlv-1.1.3.tgz#5c198a8a11453596e751494d49874bc7732f2e79" integrity sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA== +dmg-builder@24.13.3: + version "24.13.3" + resolved "https://registry.yarnpkg.com/dmg-builder/-/dmg-builder-24.13.3.tgz#95d5b99c587c592f90d168a616d7ec55907c7e55" + integrity sha512-rcJUkMfnJpfCboZoOOPf4L29TRtEieHNOeAbYPWPxlaBw/Z1RKrRA86dOI9rwaI4tQSc/RD82zTNHprfUHXsoQ== + dependencies: + app-builder-lib "24.13.3" + builder-util "24.13.1" + builder-util-runtime "9.2.4" + fs-extra "^10.1.0" + iconv-lite "^0.6.2" + js-yaml "^4.1.0" + optionalDependencies: + dmg-license "^1.0.11" + +dmg-license@^1.0.11: + version "1.0.11" + resolved "https://registry.yarnpkg.com/dmg-license/-/dmg-license-1.0.11.tgz#7b3bc3745d1b52be7506b4ee80cb61df6e4cd79a" + integrity sha512-ZdzmqwKmECOWJpqefloC5OJy1+WZBBse5+MR88z9g9Zn4VY+WYUkAyojmhzJckH5YbbZGcYIuGAkY5/Ys5OM2Q== + dependencies: + "@types/plist" "^3.0.1" + "@types/verror" "^1.10.3" + ajv "^6.10.0" + crc "^3.8.0" + iconv-corefoundation "^1.1.7" + plist "^3.0.4" + smart-buffer "^4.0.2" + verror "^1.10.0" + dns-packet@^5.2.2: version "5.6.1" resolved "https://registry.yarnpkg.com/dns-packet/-/dns-packet-5.6.1.tgz#ae888ad425a9d1478a0674256ab866de1012cf2f" @@ -4210,11 +4429,21 @@ dom-accessibility-api@^0.5.6, dom-accessibility-api@^0.5.9: resolved "https://registry.yarnpkg.com/dom-accessibility-api/-/dom-accessibility-api-0.5.16.tgz#5a7429e6066eb3664d911e33fb0e45de8eb08453" integrity sha512-X7BJ2yElsnOJ30pZF4uIIDfBEVgF4XEBxL9Bxhy6dnrm5hkzqmsWHGTiHqRiITNhMyFLyAiWndIJP7Z1NTteDg== +dotenv-expand@^5.1.0: + version "5.1.0" + resolved "https://registry.yarnpkg.com/dotenv-expand/-/dotenv-expand-5.1.0.tgz#3fbaf020bfd794884072ea26b1e9791d45a629f0" + integrity sha512-YXQl1DSa4/PQyRfgrv6aoNjhasp/p4qs9FjJ4q4cQk+8m4r6k4ZSiEyytKG8f8W9gi8WsQtIObNmKd+tMzNTmA== + dotenv@^16.0.3: version "16.4.5" resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-16.4.5.tgz#cdd3b3b604cb327e286b4762e13502f717cb099f" integrity sha512-ZmdL2rui+eB2YwhsWzjInR8LldtZHGDoQ1ugH85ppHKwpUHL7j7rN0Ti9NCnGiQbhaZ11FpR+7ao1dNsmduNUg== +dotenv@^9.0.2: + version "9.0.2" + resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-9.0.2.tgz#dacc20160935a37dea6364aa1bef819fb9b6ab05" + integrity sha512-I9OvvrHp4pIARv4+x9iuewrWycX6CcZtoAu1XrzPxc5UygMJXJZYmBsynku8IkrJwgypE5DGNjDPmPRhDCptUg== + eastasianwidth@^0.2.0: version "0.2.0" resolved "https://registry.yarnpkg.com/eastasianwidth/-/eastasianwidth-0.2.0.tgz#696ce2ec0aa0e6ea93a397ffcf24aa7840c827cb" @@ -4225,6 +4454,30 @@ ee-first@1.1.1: resolved "https://registry.yarnpkg.com/ee-first/-/ee-first-1.1.1.tgz#590c61156b0ae2f4f0255732a158b266bc56b21d" integrity sha512-WMwm9LhRUo+WUaRN+vRuETqG89IgZphVSNkdFgeb6sS/E4OrDIN7t48CAewSHXc6C8lefD8KKfr5vY61brQlow== +ejs@^3.1.8: + version "3.1.10" + resolved "https://registry.yarnpkg.com/ejs/-/ejs-3.1.10.tgz#69ab8358b14e896f80cc39e62087b88500c3ac3b" + integrity sha512-UeJmFfOrAQS8OJWPZ4qtgHyWExa088/MtK5UEyoJGFH67cDEXkZSviOiKRCZ4Xij0zxI3JECgYs3oKx+AizQBA== + dependencies: + jake "^10.8.5" + +electron-builder@^24.13.3: + version "24.13.3" + resolved "https://registry.yarnpkg.com/electron-builder/-/electron-builder-24.13.3.tgz#c506dfebd36d9a50a83ee8aa32d803d83dbe4616" + integrity sha512-yZSgVHft5dNVlo31qmJAe4BVKQfFdwpRw7sFp1iQglDRCDD6r22zfRJuZlhtB5gp9FHUxCMEoWGq10SkCnMAIg== + dependencies: + app-builder-lib "24.13.3" + builder-util "24.13.1" + builder-util-runtime "9.2.4" + chalk "^4.1.2" + dmg-builder "24.13.3" + fs-extra "^10.1.0" + is-ci "^3.0.0" + lazy-val "^1.0.5" + read-config-file "6.3.2" + simple-update-notifier "2.0.0" + yargs "^17.6.2" + electron-installer-common@^0.10.2: version "0.10.3" resolved "https://registry.yarnpkg.com/electron-installer-common/-/electron-installer-common-0.10.3.tgz#40f9db644ca60eb28673d545b67ee0113aef4444" @@ -4269,6 +4522,27 @@ electron-installer-redhat@^3.2.0: word-wrap "^1.2.3" yargs "^16.0.2" +electron-notarize@^1.2.2: + version "1.2.2" + resolved "https://registry.yarnpkg.com/electron-notarize/-/electron-notarize-1.2.2.tgz#ebf2b258e8e08c1c9f8ff61dc53d5b16b439daf4" + integrity sha512-ZStVWYcWI7g87/PgjPJSIIhwQXOaw4/XeXU+pWqMMktSLHaGMLHdyPPN7Cmao7+Cr7fYufA16npdtMndYciHNw== + dependencies: + debug "^4.1.1" + fs-extra "^9.0.1" + +electron-publish@24.13.1: + version "24.13.1" + resolved "https://registry.yarnpkg.com/electron-publish/-/electron-publish-24.13.1.tgz#57289b2f7af18737dc2ad134668cdd4a1b574a0c" + integrity sha512-2ZgdEqJ8e9D17Hwp5LEq5mLQPjqU3lv/IALvgp+4W8VeNhryfGhYEQC/PgDPMrnWUp+l60Ou5SJLsu+k4mhQ8A== + dependencies: + "@types/fs-extra" "^9.0.11" + builder-util "24.13.1" + builder-util-runtime "9.2.4" + chalk "^4.1.2" + fs-extra "^10.1.0" + lazy-val "^1.0.5" + mime "^2.5.2" + electron-squirrel-startup@^1.0.1: version "1.0.1" resolved "https://registry.yarnpkg.com/electron-squirrel-startup/-/electron-squirrel-startup-1.0.1.tgz#c9171568d724884c7a2b03760bfeedcf921c63ab" @@ -4286,6 +4560,20 @@ electron-to-chromium@^1.5.4: resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.5.13.tgz#1abf0410c5344b2b829b7247e031f02810d442e6" integrity sha512-lbBcvtIJ4J6sS4tb5TLp1b4LyfCdMkwStzXPyAgVgTRAsep4bvrAGaBOP7ZJtQMNJpSQ9SqG4brWOroNaQtm7Q== +electron-updater@^6.3.0: + version "6.3.0" + resolved "https://registry.yarnpkg.com/electron-updater/-/electron-updater-6.3.0.tgz#13a5c3c3f0b2b114fe33181e24a8270096734b3e" + integrity sha512-3Xlezhk+dKaSQrOnkQNqCGiuGSSUPO9BV9TQZ4Iig6AyTJ4FzJONE5gFFc382sY53Sh9dwJfzKsA3DxRHt2btw== + dependencies: + builder-util-runtime "9.2.5" + fs-extra "^10.1.0" + js-yaml "^4.1.0" + lazy-val "^1.0.5" + lodash.escaperegexp "^4.1.2" + lodash.isequal "^4.5.0" + semver "^7.3.8" + tiny-typed-emitter "^2.1.0" + electron-winstaller@^5.3.0: version "5.4.0" resolved "https://registry.yarnpkg.com/electron-winstaller/-/electron-winstaller-5.4.0.tgz#f0660d476d5c4f579fdf7edd2f0cf01d54c4d0b2" @@ -4594,10 +4882,10 @@ eslint-import-resolver-node@^0.3.9: is-core-module "^2.13.0" resolve "^1.22.4" -eslint-module-utils@^2.8.0: - version "2.8.1" - resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.8.1.tgz#52f2404300c3bd33deece9d7372fb337cc1d7c34" - integrity sha512-rXDXR3h7cs7dy9RNpUlQf80nX31XWJEyGq1tRMo+6GsO5VmTe4UTwtmonAD4ZkAsrfMVDA2wlGJ3790Ys+D49Q== +eslint-module-utils@^2.9.0: + version "2.9.0" + resolved "https://registry.yarnpkg.com/eslint-module-utils/-/eslint-module-utils-2.9.0.tgz#95d4ac038a68cd3f63482659dffe0883900eb342" + integrity sha512-McVbYmwA3NEKwRQY5g4aWMdcZE5xZxV8i8l7CqJSrameuGSQJtSWaL/LxTEzSKKaCcOhlpDR8XEfYXWPrdo/ZQ== dependencies: debug "^3.2.7" @@ -4617,26 +4905,27 @@ eslint-plugin-flowtype@^8.0.3: lodash "^4.17.21" string-natural-compare "^3.0.1" -eslint-plugin-import@^2.25.0, eslint-plugin-import@^2.25.3: - version "2.29.1" - resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.29.1.tgz#d45b37b5ef5901d639c15270d74d46d161150643" - integrity sha512-BbPC0cuExzhiMo4Ff1BTVwHpjjv28C5R+btTOGaCRC7UEz801up0JadwkeSk5Ued6TG34uaczuVuH6qyy5YUxw== +eslint-plugin-import@^2.25.3, eslint-plugin-import@^2.30.0: + version "2.30.0" + resolved "https://registry.yarnpkg.com/eslint-plugin-import/-/eslint-plugin-import-2.30.0.tgz#21ceea0fc462657195989dd780e50c92fe95f449" + integrity sha512-/mHNE9jINJfiD2EKkg1BKyPyUk4zdnT54YgbOgfjSakWT5oyX/qQLVNTkehyfpcMxZXMy1zyonZ2v7hZTX43Yw== dependencies: - array-includes "^3.1.7" - array.prototype.findlastindex "^1.2.3" + "@rtsao/scc" "^1.1.0" + array-includes "^3.1.8" + array.prototype.findlastindex "^1.2.5" array.prototype.flat "^1.3.2" array.prototype.flatmap "^1.3.2" debug "^3.2.7" doctrine "^2.1.0" eslint-import-resolver-node "^0.3.9" - eslint-module-utils "^2.8.0" - hasown "^2.0.0" - is-core-module "^2.13.1" + eslint-module-utils "^2.9.0" + hasown "^2.0.2" + is-core-module "^2.15.1" is-glob "^4.0.3" minimatch "^3.1.2" - object.fromentries "^2.0.7" - object.groupby "^1.0.1" - object.values "^1.1.7" + object.fromentries "^2.0.8" + object.groupby "^1.0.3" + object.values "^1.2.0" semver "^6.3.1" tsconfig-paths "^3.15.0" @@ -4961,6 +5250,11 @@ extract-zip@^2.0.0, extract-zip@^2.0.1: optionalDependencies: "@types/yauzl" "^2.9.1" +extsprintf@^1.2.0: + version "1.4.1" + resolved "https://registry.yarnpkg.com/extsprintf/-/extsprintf-1.4.1.tgz#8d172c064867f235c0c84a596806d279bf4bcc07" + integrity sha512-Wrk35e8ydCKDj/ArClo1VrPVmN8zph5V4AtHwIuHhvMXsKf73UT3BOD+azBIW+3wOJ4FhEH7zyaJCFvChjYvMA== + fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: version "3.1.3" resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" @@ -5051,6 +5345,13 @@ file-entry-cache@^6.0.1: dependencies: flat-cache "^3.0.4" +filelist@^1.0.4: + version "1.0.4" + resolved "https://registry.yarnpkg.com/filelist/-/filelist-1.0.4.tgz#f78978a1e944775ff9e62e744424f215e58352b5" + integrity sha512-w1cEuf3S+DrLCQL7ET6kz+gmlJdbq9J7yXCSjK/OZCPA+qEN1WyF4ZAf0YYJa4/shHJra2t/d/r8SV4Ji+x+8Q== + dependencies: + minimatch "^5.0.1" + filename-reserved-regex@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/filename-reserved-regex/-/filename-reserved-regex-2.0.0.tgz#abf73dfab735d045440abfea2d91f389ebbfa229" @@ -5138,7 +5439,7 @@ flux@^4.0.1: fbemitter "^3.0.0" fbjs "^3.0.1" -follow-redirects@^1.0.0, follow-redirects@^1.14.8, follow-redirects@^1.15.6: +follow-redirects@^1.0.0, follow-redirects@^1.14.8: version "1.15.6" resolved "https://registry.yarnpkg.com/follow-redirects/-/follow-redirects-1.15.6.tgz#7f815c0cda4249c74ff09e95ef97c23b5fd0399b" integrity sha512-wWN62YITEaOpSK584EZXJafH1AGpO8RVgElfkuXbTOrPX4fIfOyEpW/CsiNd8JdYrAoOvafRTOEnvsO++qCqFA== @@ -5630,6 +5931,13 @@ hosted-git-info@^2.1.4: resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-2.8.9.tgz#dffc0bf9a21c02209090f2aa69429e1414daf3f9" integrity sha512-mxIDAb9Lsm6DoOJ7xH+5+X4y1LU/4Hi50L9C5sIswK3JzULS4bwk1FvjdBgvYR4bzT4tuUQiC15FE2f5HbLvYw== +hosted-git-info@^4.1.0: + version "4.1.0" + resolved "https://registry.yarnpkg.com/hosted-git-info/-/hosted-git-info-4.1.0.tgz#827b82867e9ff1c8d0c4d9d53880397d2c86d224" + integrity sha512-kyCuEOWjJqZuDbRHzL8V93NzQhwIB71oFWSyzVo+KPZI+pnQPPxucdkrOZvkLRnrf5URsQM+IJ09Dw29cRALIA== + dependencies: + lru-cache "^6.0.0" + html-encoding-sniffer@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/html-encoding-sniffer/-/html-encoding-sniffer-3.0.0.tgz#2cb1a8cf0db52414776e5b2a7a04d5dd98158de9" @@ -5706,7 +6014,7 @@ http2-wrapper@^1.0.0-beta.5.2: quick-lru "^5.1.1" resolve-alpn "^1.0.0" -https-proxy-agent@^5.0.0: +https-proxy-agent@^5.0.0, https-proxy-agent@^5.0.1: version "5.0.1" resolved "https://registry.yarnpkg.com/https-proxy-agent/-/https-proxy-agent-5.0.1.tgz#c59ef224a04fe8b754f3db0063a25ea30d0005d6" integrity sha512-dFcAjpTQFgoLMzC2VwU+C/CbS7uRL0lWmxDITmqm7C+7F0Odmj6s9l6alZc6AELXhrnggM2CeWSXHGOdX2YtwA== @@ -5731,6 +6039,14 @@ husky@^9.1.5: resolved "https://registry.yarnpkg.com/husky/-/husky-9.1.5.tgz#2b6edede53ee1adbbd3a3da490628a23f5243b83" integrity sha512-rowAVRUBfI0b4+niA4SJMhfQwc107VLkBUgEYYAOQAbqDCnra1nYh83hF/MDmhYs9t9n1E3DuKOrs2LYNC+0Ag== +iconv-corefoundation@^1.1.7: + version "1.1.7" + resolved "https://registry.yarnpkg.com/iconv-corefoundation/-/iconv-corefoundation-1.1.7.tgz#31065e6ab2c9272154c8b0821151e2c88f1b002a" + integrity sha512-T10qvkw0zz4wnm560lOEg0PovVqUXuOFhhHAkixw8/sycy7TJt7v/RrkEKEQnAw2viPSJu6iAkErxnzR0g8PpQ== + dependencies: + cli-truncate "^2.1.0" + node-addon-api "^1.6.3" + iconv-lite@0.4.24: version "0.4.24" resolved "https://registry.yarnpkg.com/iconv-lite/-/iconv-lite-0.4.24.tgz#2022b4b25fbddc21d2f524974a474aafe733908b" @@ -5883,10 +6199,17 @@ is-callable@^1.1.3, is-callable@^1.1.4, is-callable@^1.2.7: resolved "https://registry.yarnpkg.com/is-callable/-/is-callable-1.2.7.tgz#3bc2a85ea742d9e36205dcacdd72ca1fdc51b055" integrity sha512-1BC0BVFhS/p0qtw6enp8e+8OD0UrK0oFLztSjNzhcKA3WDuJxxAPXzPuPtKkjEY9UUoEWlX/8fgKeu2S8i9JTA== -is-core-module@^2.13.0, is-core-module@^2.13.1: - version "2.15.0" - resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.15.0.tgz#71c72ec5442ace7e76b306e9d48db361f22699ea" - integrity sha512-Dd+Lb2/zvk9SKy1TGCt1wFJFo/MWBPMX5x7KcvLajWTGuomczdQX61PvY5yK6SVACwpoexWo81IfFyoKY2QnTA== +is-ci@^3.0.0: + version "3.0.1" + resolved "https://registry.yarnpkg.com/is-ci/-/is-ci-3.0.1.tgz#db6ecbed1bd659c43dac0f45661e7674103d1867" + integrity sha512-ZYvCgrefwqoQ6yTyYUbQu64HsITZ3NfKX1lzaEYdkTDcfKzzCI/wthRRYKkdjHKFVgNiXKAKm65Zo1pk2as/QQ== + dependencies: + ci-info "^3.2.0" + +is-core-module@^2.13.0, is-core-module@^2.15.1: + version "2.15.1" + resolved "https://registry.yarnpkg.com/is-core-module/-/is-core-module-2.15.1.tgz#a7363a25bee942fefab0de13bf6aa372c82dcc37" + integrity sha512-z0vtXSwucUJtANQWldhbtbt7BnL0vxiFjIdDLAatwhDYty2bad6s+rijD6Ri4YuYJubLzIJLUidCh09e1djEVQ== dependencies: hasown "^2.0.2" @@ -6085,6 +6408,11 @@ isbinaryfile@^4.0.8: resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-4.0.10.tgz#0c5b5e30c2557a2f06febd37b7322946aaee42b3" integrity sha512-iHrqe5shvBUcFbmZq9zOQHBoeOhZJu6RQGrDpBgenUm/Am+F3JM2MgQj+rK3Z601fzrL5gLZWtAPH2OBaSVcyw== +isbinaryfile@^5.0.0: + version "5.0.2" + resolved "https://registry.yarnpkg.com/isbinaryfile/-/isbinaryfile-5.0.2.tgz#fe6e4dfe2e34e947ffa240c113444876ba393ae0" + integrity sha512-GvcjojwonMjWbTkfMpnVHVqXW/wKMYDfEpY94/8zy8HFMOqb/VL6oeONq9v87q4ttVlaTLnGXnJD4B5B1OTGIg== + isexe@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/isexe/-/isexe-2.0.0.tgz#e8fbf374dc556ff8947a10dcb0572d633f2cfa10" @@ -6118,6 +6446,16 @@ jackspeak@^3.1.2: optionalDependencies: "@pkgjs/parseargs" "^0.11.0" +jake@^10.8.5: + version "10.9.2" + resolved "https://registry.yarnpkg.com/jake/-/jake-10.9.2.tgz#6ae487e6a69afec3a5e167628996b59f35ae2b7f" + integrity sha512-2P4SQ0HrLQ+fw6llpLnOaGAvN2Zu6778SJMrCUwns4fOoG9ayrTiZk3VV8sCPkVZF8ab0zksVpS8FDY5pRCNBA== + dependencies: + async "^3.2.3" + chalk "^4.0.2" + filelist "^1.0.4" + minimatch "^3.1.2" + jest-diff@^29.7.0: version "29.7.0" resolved "https://registry.yarnpkg.com/jest-diff/-/jest-diff-29.7.0.tgz#017934a66ebb7ecf6f205e84699be10afd70458a" @@ -6175,17 +6513,6 @@ jiti@^1.21.0: resolved "https://registry.yarnpkg.com/jiti/-/jiti-1.21.6.tgz#6c7f7398dd4b3142767f9a168af2f317a428d268" integrity sha512-2yTgeWTWzMWkHu6Jp9NKgePDaYHbntiwvYuuJLbbN9vl7DC9DvXKOB2BC3ZZ92D3cvV/aflH0osDfwpHepQ53w== -joi@^17.11.0: - version "17.13.3" - resolved "https://registry.yarnpkg.com/joi/-/joi-17.13.3.tgz#0f5cc1169c999b30d344366d384b12d92558bcec" - integrity sha512-otDA4ldcIx+ZXsKHWmp0YizCweVRZG96J10b0FevjfuncLO1oX59THoAmHkNubYJ+9gWsYsp5k8v4ib6oDv1fA== - dependencies: - "@hapi/hoek" "^9.3.0" - "@hapi/topo" "^5.1.0" - "@sideway/address" "^4.1.5" - "@sideway/formula" "^3.0.1" - "@sideway/pinpoint" "^2.0.0" - jose@^4.15.5: version "4.15.9" resolved "https://registry.yarnpkg.com/jose/-/jose-4.15.9.tgz#9b68eda29e9a0614c042fa29387196c7dd800100" @@ -6260,7 +6587,7 @@ json5@^1.0.2: dependencies: minimist "^1.2.0" -json5@^2.2.3: +json5@^2.2.0, json5@^2.2.3: version "2.2.3" resolved "https://registry.yarnpkg.com/json5/-/json5-2.2.3.tgz#78cd6f1a19bdc12b73db5ad0c61efd66c1e29283" integrity sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg== @@ -6330,6 +6657,11 @@ language-tags@^1.0.9: dependencies: language-subtag-registry "^0.3.20" +lazy-val@^1.0.4, lazy-val@^1.0.5: + version "1.0.5" + resolved "https://registry.yarnpkg.com/lazy-val/-/lazy-val-1.0.5.tgz#6cf3b9f5bc31cee7ee3e369c0832b7583dcd923d" + integrity sha512-0/BnGCCfyUMkBpeDgWihanIAF9JmZhHBgUhEqzvf+adhNGLoP6TaiI5oF8oyb3I45P+PcnrqihSf01M0l0G5+Q== + levn@^0.4.1: version "0.4.1" resolved "https://registry.yarnpkg.com/levn/-/levn-0.4.1.tgz#ae4562c007473b932a6200d403268dd2fffc6ade" @@ -6422,6 +6754,11 @@ lodash.debounce@^4.0.8: resolved "https://registry.yarnpkg.com/lodash.debounce/-/lodash.debounce-4.0.8.tgz#82d79bff30a67c4005ffd5e2515300ad9ca4d7af" integrity sha512-FT1yDzDYEoYWhnSGnpE/4Kj1fLZkDFyqRb7fNt6FdYOSxlUWAtp42Eh6Wb0rGIv/m9Bgo7x4GhQbm5Ys4SG5ow== +lodash.escaperegexp@^4.1.2: + version "4.1.2" + resolved "https://registry.yarnpkg.com/lodash.escaperegexp/-/lodash.escaperegexp-4.1.2.tgz#64762c48618082518ac3df4ccf5d5886dae20347" + integrity sha512-TM9YBvyC84ZxE3rgfefxUWiQKLilstD6k7PTGt6wfbtXF8ixIJLOL3VYyV/z+ZiPLsVxAsKAFVwWlWeb2Y8Yyw== + lodash.flow@^3.3.0: version "3.5.0" resolved "https://registry.yarnpkg.com/lodash.flow/-/lodash.flow-3.5.0.tgz#87bf40292b8cf83e4e8ce1a3ae4209e20071675a" @@ -6432,6 +6769,11 @@ lodash.get@^4.0.0: resolved "https://registry.yarnpkg.com/lodash.get/-/lodash.get-4.4.2.tgz#2d177f652fa31e939b4438d5341499dfa3825e99" integrity sha512-z+Uw/vLuy6gQe8cfaFWD7p0wVv8fJl3mbzXh33RS+0oW2wvUqiRXiQ69gLWSLpgB5/6sU+r6BlQR0MBILadqTQ== +lodash.isequal@^4.5.0: + version "4.5.0" + resolved "https://registry.yarnpkg.com/lodash.isequal/-/lodash.isequal-4.5.0.tgz#415c4478f2bcc30120c22ce10ed3226f7d3e18e0" + integrity sha512-pDo3lu8Jhfjqls6GkMgpahsF9kCyayhgykjyLMNFTKWrpVdAQtYyB4muAMWozBB4ig/dtWAmsMxLEI8wuz+DYQ== + lodash.merge@^4.6.2: version "4.6.2" resolved "https://registry.yarnpkg.com/lodash.merge/-/lodash.merge-4.6.2.tgz#558aa53b43b661e1925a0afdfa36a9a1085fe57a" @@ -6621,6 +6963,11 @@ mime@1.6.0, mime@^1.6.0: resolved "https://registry.yarnpkg.com/mime/-/mime-1.6.0.tgz#32cd9e5c64553bd58d19a568af452acff04981b1" integrity sha512-x0Vn8spI+wuJ1O6S7gnbaQg8Pxh4NNHb7KSINmEWKiPE4RKOplvijn+NkmYmmRgP68mc70j2EbeTFRsrswaQeg== +mime@^2.5.2: + version "2.6.0" + resolved "https://registry.yarnpkg.com/mime/-/mime-2.6.0.tgz#a2a682a95cd4d0cb1d6257e28f83da7e35800367" + integrity sha512-USPkMeET31rOMiarsBNIHZKLGgvKc/LrjofAnBlOttf5ajRvqiRA8QsenbcooctK6d6Ts6aqZXBA+XbkKthiQg== + mimic-fn@^2.0.0, mimic-fn@^2.1.0: version "2.1.0" resolved "https://registry.yarnpkg.com/mimic-fn/-/mimic-fn-2.1.0.tgz#7ed2c2ccccaf84d3ffcb7a69b57711fc2083401b" @@ -6653,7 +7000,7 @@ minimatch@^3.0.4, minimatch@^3.0.5, minimatch@^3.1.1, minimatch@^3.1.2: dependencies: brace-expansion "^1.1.7" -minimatch@^5.0.1: +minimatch@^5.0.1, minimatch@^5.1.1: version "5.1.6" resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-5.1.6.tgz#1cfcb8cf5522ea69952cd2af95ae09477f122a96" integrity sha512-lKwV/1brpG6mBUFHtb7NUmtABCb2WZZmm2wNiOA5hAb8VdCS4B3dtMWyvcoViccwAW/COERjXLt0zP1zXUN26g== @@ -6822,6 +7169,11 @@ node-abi@^3.45.0: dependencies: semver "^7.3.5" +node-addon-api@^1.6.3: + version "1.7.2" + resolved "https://registry.yarnpkg.com/node-addon-api/-/node-addon-api-1.7.2.tgz#3df30b95720b53c24e59948b49532b662444f54d" + integrity sha512-ibPK3iA+vaY1eEjESkQkM0BbCqFOaZMiXRTtdB0u7b4djtY6JnsjvPdUHVMg6xQt3B8fpTTWHI9A+ADjM9frzg== + node-api-version@^0.2.0: version "0.2.0" resolved "https://registry.yarnpkg.com/node-api-version/-/node-api-version-0.2.0.tgz#5177441da2b1046a4d4547ab9e0972eed7b1ac1d" @@ -6980,7 +7332,7 @@ object.entries@^1.1.8: define-properties "^1.2.1" es-object-atoms "^1.0.0" -object.fromentries@^2.0.7, object.fromentries@^2.0.8: +object.fromentries@^2.0.8: version "2.0.8" resolved "https://registry.yarnpkg.com/object.fromentries/-/object.fromentries-2.0.8.tgz#f7195d8a9b97bd95cbc1999ea939ecd1a2b00c65" integrity sha512-k6E21FzySsSK5a21KRADBd/NGneRegFO5pLHfdQLpRDETUNJueLXs3WCzyQ3tFRDYgbq3KHGXfTbi2bs8WQ6rQ== @@ -6990,7 +7342,7 @@ object.fromentries@^2.0.7, object.fromentries@^2.0.8: es-abstract "^1.23.2" es-object-atoms "^1.0.0" -object.groupby@^1.0.1: +object.groupby@^1.0.3: version "1.0.3" resolved "https://registry.yarnpkg.com/object.groupby/-/object.groupby-1.0.3.tgz#9b125c36238129f6f7b61954a1e7176148d5002e" integrity sha512-+Lhy3TQTuzXI5hevh8sBGqbmurHbbIjAi0Z4S63nthVLmLxfbj4T54a4CfZrXIrt9iP4mVAPYMo/v99taj3wjQ== @@ -6999,7 +7351,7 @@ object.groupby@^1.0.1: define-properties "^1.2.1" es-abstract "^1.23.2" -object.values@^1.1.6, object.values@^1.1.7, object.values@^1.2.0: +object.values@^1.1.6, object.values@^1.2.0: version "1.2.0" resolved "https://registry.yarnpkg.com/object.values/-/object.values-1.2.0.tgz#65405a9d92cee68ac2d303002e0b8470a4d9ab1b" integrity sha512-yBYjY9QX2hnRmZHAjG/f13MzmBzxzYgQhFrke06TTyKY5zSTEqkOeukBzIdVA3j3ulu8Qa3MbVFShV7T2RmGtQ== @@ -7369,7 +7721,7 @@ playwright@1.46.1: optionalDependencies: fsevents "2.3.2" -plist@^3.0.0, plist@^3.0.5, plist@^3.1.0: +plist@^3.0.0, plist@^3.0.4, plist@^3.0.5, plist@^3.1.0: version "3.1.0" resolved "https://registry.yarnpkg.com/plist/-/plist-3.1.0.tgz#797a516a93e62f5bde55e0b9cc9c967f860893c9" integrity sha512-uysumyrvkUX0rX/dEVqt8gC3sTBzd4zoWfLeS29nb53imdaXVvLINYXTI2GNqzaMuvacNx4uJQ8+b3zXR0pkgQ== @@ -7446,10 +7798,10 @@ postcss-value-parser@^4.0.0, postcss-value-parser@^4.2.0: resolved "https://registry.yarnpkg.com/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz#723c09920836ba6d3e5af019f92bc0971c02e514" integrity sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ== -postcss@^8.4.23, postcss@^8.4.31, postcss@^8.4.41: - version "8.4.41" - resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.41.tgz#d6104d3ba272d882fe18fc07d15dc2da62fa2681" - integrity sha512-TesUflQ0WKZqAvg52PWL6kHgLKP6xB6heTOdoYM0Wt2UHyxNa4K25EZZMgKns3BH1RLVbZCREPpLY0rhnNoHVQ== +postcss@^8.4.23, postcss@^8.4.41, postcss@^8.4.43: + version "8.4.43" + resolved "https://registry.yarnpkg.com/postcss/-/postcss-8.4.43.tgz#a5ddf22f4cc38e64c6ae030182b43e539d316419" + integrity sha512-gJAQVYbh5R3gYm33FijzCZj7CHyQ3hWMgJMprLUlIYqCwTeZhBQ19wp0e9mA25BUbEvY5+EXuuaAjqQsrBxQBQ== dependencies: nanoid "^3.3.7" picocolors "^1.0.1" @@ -7542,11 +7894,6 @@ proxy-addr@~2.0.7: forwarded "0.2.0" ipaddr.js "1.9.1" -proxy-from-env@^1.1.0: - version "1.1.0" - resolved "https://registry.yarnpkg.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz#e102f16ca355424865755d2c9e8ea4f24d58c3e2" - integrity sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg== - pump@^3.0.0: version "3.0.0" resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64" @@ -7646,10 +7993,10 @@ react-hot-toast@^2.4.1: dependencies: goober "^2.1.10" -react-hotkeys-hook@^4.5.0: - version "4.5.0" - resolved "https://registry.yarnpkg.com/react-hotkeys-hook/-/react-hotkeys-hook-4.5.0.tgz#807b389b15256daf6a813a1ec09e6698064fe97f" - integrity sha512-Samb85GSgAWFQNvVt3PS90LPPGSf9mkH/r4au81ZP1yOIFayLC3QAvqTgGtJ8YEDMXtPmaVBs6NgipHO6h4Mug== +react-hotkeys-hook@^4.5.1: + version "4.5.1" + resolved "https://registry.yarnpkg.com/react-hotkeys-hook/-/react-hotkeys-hook-4.5.1.tgz#990260ecc7e5a431414148a93b02a2f1a9707897" + integrity sha512-scAEJOh3Irm0g95NIn6+tQVf/OICCjsQsC9NBHfQws/Vxw4sfq1tDQut5fhTEvPraXhu/sHxRd9lOtxzyYuNAg== react-is@^16.13.1: version "16.13.1" @@ -7746,6 +8093,18 @@ read-cache@^1.0.0: dependencies: pify "^2.3.0" +read-config-file@6.3.2: + version "6.3.2" + resolved "https://registry.yarnpkg.com/read-config-file/-/read-config-file-6.3.2.tgz#556891aa6ffabced916ed57457cb192e61880411" + integrity sha512-M80lpCjnE6Wt6zb98DoW8WHR09nzMSpu8XHtPkiTHrJ5Az9CybfeQhTJ8D7saeBHpGhLPIVyA8lcL6ZmdKwY6Q== + dependencies: + config-file-ts "^0.2.4" + dotenv "^9.0.2" + dotenv-expand "^5.1.0" + js-yaml "^4.1.0" + json5 "^2.2.0" + lazy-val "^1.0.4" + read-pkg-up@^2.0.0: version "2.0.0" resolved "https://registry.yarnpkg.com/read-pkg-up/-/read-pkg-up-2.0.0.tgz#6b72a8048984e0c41e79510fd5e9fa99b3b549be" @@ -8047,13 +8406,6 @@ run-parallel@^1.1.9: dependencies: queue-microtask "^1.2.2" -rxjs@^7.8.1: - version "7.8.1" - resolved "https://registry.yarnpkg.com/rxjs/-/rxjs-7.8.1.tgz#6f6f3d99ea8044291efd92e7c7fcf562c4057543" - integrity sha512-AA3TVj+0A2iuIoQkWEK/tqFjBq2j+6PO6Y0zJcvzLAFhEFIO3HL0vls9hWLncZbAAbK0mar7oZ4V079I/qPMxg== - dependencies: - tslib "^2.1.0" - safe-array-concat@^1.1.2: version "1.1.2" resolved "https://registry.yarnpkg.com/safe-array-concat/-/safe-array-concat-1.1.2.tgz#81d77ee0c4e8b863635227c721278dd524c20edb" @@ -8088,6 +8440,18 @@ safe-regex-test@^1.0.3: resolved "https://registry.yarnpkg.com/safer-buffer/-/safer-buffer-2.1.2.tgz#44fa161b0187b9549dd84bb91802f9bd8385cd6a" integrity sha512-YZo3K82SD7Riyi0E1EQPojLz7kpepnSQI9IyPbHHg1XXXevb5dJI7tpyN2ADxGcQbHG7vcyRHk0cbwqcQriUtg== +sanitize-filename@^1.6.3: + version "1.6.3" + resolved "https://registry.yarnpkg.com/sanitize-filename/-/sanitize-filename-1.6.3.tgz#755ebd752045931977e30b2025d340d7c9090378" + integrity sha512-y/52Mcy7aw3gRm7IrcGDFx/bCk4AhRh2eI9luHOQM86nZsqwiRkkq2GekHXBBD+SmPidc8i2PqtYZl+pWJ8Oeg== + dependencies: + truncate-utf8-bytes "^1.0.0" + +sax@^1.2.4: + version "1.4.1" + resolved "https://registry.yarnpkg.com/sax/-/sax-1.4.1.tgz#44cc8988377f126304d3b3fc1010c733b929ef0f" + integrity sha512-+aWOz7yVScEGoKNd4PA10LZ8sk0A/z5+nXQG5giUO5rprX9jgYsTdov9qCchZiPIZezbZH+jRut8nPodFAX4Jg== + scheduler@^0.23.2: version "0.23.2" resolved "https://registry.yarnpkg.com/scheduler/-/scheduler-0.23.2.tgz#414ba64a3b282892e944cf2108ecc078d115cdc3" @@ -8115,7 +8479,7 @@ semver@^6.2.0, semver@^6.3.1: resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.1.tgz#556d2ef8689146e46dcea4bfdd095f3434dffcb4" integrity sha512-BR7VvDCVHO+q2xBEWskxS6DJE1qRnb7DxzUrogb71CWoSficBxYsiAGd+Kl0mmq/MprG9yArRkyrQxTO6XjMzA== -semver@^7.1.1, semver@^7.1.3, semver@^7.2.1, semver@^7.3.2, semver@^7.3.5, semver@^7.3.7, semver@^7.6.0: +semver@^7.1.1, semver@^7.1.3, semver@^7.2.1, semver@^7.3.2, semver@^7.3.5, semver@^7.3.7, semver@^7.3.8, semver@^7.5.3, semver@^7.6.0: version "7.6.3" resolved "https://registry.yarnpkg.com/semver/-/semver-7.6.3.tgz#980f7b5550bc175fb4dc09403085627f9eb33143" integrity sha512-oVekP1cKtI+CTDvHWYFUcMtsK/00wmAEfyqKfNdARm8u1wNVhSgaX7A8d4UuIlUI5e84iEwOhs7ZPYRmzU9U6A== @@ -8242,6 +8606,13 @@ signal-exit@^4.0.1, signal-exit@^4.1.0: resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-4.1.0.tgz#952188c1cbd546070e2dd20d0f41c0ae0530cb04" integrity sha512-bzyZ1e88w9O1iNJbKnOlvYTrWPDl46O1bG0D3XInv+9tkPrxrN8jUUTiFlDkkmKWgn1M6CfIA13SuGqOa9Korw== +simple-update-notifier@2.0.0: + version "2.0.0" + resolved "https://registry.yarnpkg.com/simple-update-notifier/-/simple-update-notifier-2.0.0.tgz#d70b92bdab7d6d90dfd73931195a30b6e3d7cebb" + integrity sha512-a2B9Y0KlNXl9u/vsW6sTIu9vGEpfKu2wRV6l1H3XEas/0gUIzGzBoP/IouTcUQbm9JWZLH3COxyn03TYlFax6w== + dependencies: + semver "^7.5.3" + sketch-helpers@^0.0.4: version "0.0.4" resolved "https://registry.yarnpkg.com/sketch-helpers/-/sketch-helpers-0.0.4.tgz#c6e4257451cd65483ab99ff7d3b10da04e98374d" @@ -8252,6 +8623,15 @@ slash@^3.0.0: resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== +slice-ansi@^3.0.0: + version "3.0.0" + resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-3.0.0.tgz#31ddc10930a1b7e0b67b08c96c2f49b77a789787" + integrity sha512-pSyv7bSTC7ig9Dcgbw9AuRNUb5k5V6oDudjZoMBSr13qpLBG7tB+zgCkARjq7xIUgdz5P1Qe8u+rSGdouOOIyQ== + dependencies: + ansi-styles "^4.0.0" + astral-regex "^2.0.0" + is-fullwidth-code-point "^3.0.0" + slice-ansi@^5.0.0: version "5.0.0" resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-5.0.0.tgz#b73063c57aa96f9cd881654b15294d95d285c42a" @@ -8260,7 +8640,7 @@ slice-ansi@^5.0.0: ansi-styles "^6.0.0" is-fullwidth-code-point "^4.0.0" -smart-buffer@^4.2.0: +smart-buffer@^4.0.2, smart-buffer@^4.2.0: version "4.2.0" resolved "https://registry.yarnpkg.com/smart-buffer/-/smart-buffer-4.2.0.tgz#6e1d71fa4f18c05f7d0ff216dd16a481d0e8d9ae" integrity sha512-94hK0Hh8rPqQl2xXc3HsaBoOXKV20MToPkcXvwbISWLEs+64sBq5kFgn2kJDHb1Pry9yrP0dxrCI9RRci7RXKg== @@ -8287,7 +8667,7 @@ source-map-js@^1.2.0: resolved "https://registry.yarnpkg.com/source-map-js/-/source-map-js-1.2.0.tgz#16b809c162517b5b8c3e7dcd315a2a5c2612b2af" integrity sha512-itJW8lvSA0TXEphiRoawsCksnlf8SyvmFzIhltqAHluXd88pkCd+cXJVHTDwdCr0IzwptSm035IHQktUu1QUMg== -source-map-support@^0.5.13: +source-map-support@^0.5.13, source-map-support@^0.5.19: version "0.5.21" resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.21.tgz#04fe7c7f9e1ed2d662233c28cb2b35b9f63f6e4f" integrity sha512-uBHU3L3czsIyYXKX88fdrGovxdSCoTGDRZ6SYXtSRxLZUzHg5P/66Ht6uoUlHu9EZod+inXhKo3qQgwXUT/y1w== @@ -8350,6 +8730,11 @@ stackback@0.0.2: resolved "https://registry.yarnpkg.com/stackback/-/stackback-0.0.2.tgz#1ac8a0d9483848d1695e418b6d031a3c3ce68e3b" integrity sha512-1XMJE5fQo1jGH6Y/7ebnwPOBEkIEnT4QF32d5R1+VXdXveM0IBMJt8zfaxX1P3QhVwrYe+576+jkANtSS2mBbw== +stat-mode@^1.0.0: + version "1.0.0" + resolved "https://registry.yarnpkg.com/stat-mode/-/stat-mode-1.0.0.tgz#68b55cb61ea639ff57136f36b216a291800d1465" + integrity sha512-jH9EhtKIjuXZ2cWxmXS8ZP80XyC3iasQxMDV8jzhNJpfDb7VbQLVW4Wvsxz9QZvzV+G4YoSfBUVKDOyxLzi/sg== + statuses@2.0.1: version "2.0.1" resolved "https://registry.yarnpkg.com/statuses/-/statuses-2.0.1.tgz#55cb000ccf1d48728bd23c685a063998cf1a1b63" @@ -8614,7 +8999,7 @@ tailwindcss@^3.4.1: resolve "^1.22.2" sucrase "^3.32.0" -tar@^6.0.5, tar@^6.1.11, tar@^6.1.2: +tar@^6.0.5, tar@^6.1.11, tar@^6.1.12, tar@^6.1.2: version "6.2.1" resolved "https://registry.yarnpkg.com/tar/-/tar-6.2.1.tgz#717549c541bc3c2af15751bea94b1dd068d4b03a" integrity sha512-DZ4yORTwrbTj/7MZYq2w+/ZFdI6OZ/f9SFHR+71gIVUZhOQPHzVCLpvRnPgyaMpfWxxk/4ONva3GQSyNIKRv6A== @@ -8626,6 +9011,14 @@ tar@^6.0.5, tar@^6.1.11, tar@^6.1.2: mkdirp "^1.0.3" yallist "^4.0.0" +temp-file@^3.4.0: + version "3.4.0" + resolved "https://registry.yarnpkg.com/temp-file/-/temp-file-3.4.0.tgz#766ea28911c683996c248ef1a20eea04d51652c7" + integrity sha512-C5tjlC/HCtVUOi3KWVokd4vHVViOmGjtLwIh4MuzPo/nMYTV/p1urt3RnMz2IWXDdKEGJH3k5+KPxtqRsUYGtg== + dependencies: + async-exit-hook "^2.0.1" + fs-extra "^10.0.0" + temp@^0.9.0: version "0.9.4" resolved "https://registry.yarnpkg.com/temp/-/temp-0.9.4.tgz#cd20a8580cb63635d0e4e9d4bd989d44286e7620" @@ -8680,6 +9073,11 @@ tiny-invariant@^1.3.3: resolved "https://registry.yarnpkg.com/tiny-invariant/-/tiny-invariant-1.3.3.tgz#46680b7a873a0d5d10005995eb90a70d74d60127" integrity sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg== +tiny-typed-emitter@^2.1.0: + version "2.1.0" + resolved "https://registry.yarnpkg.com/tiny-typed-emitter/-/tiny-typed-emitter-2.1.0.tgz#b3b027fdd389ff81a152c8e847ee2f5be9fad7b5" + integrity sha512-qVtvMxeXbVej0cQWKqVSSAHmKZEHAvxdF8HEUBFWts8h+xEo5m/lEiPakuyZ3BnCBjOD8i24kzNOiOLLgsSxhA== + tinybench@^2.5.1: version "2.8.0" resolved "https://registry.yarnpkg.com/tinybench/-/tinybench-2.8.0.tgz#30e19ae3a27508ee18273ffed9ac7018949acd7b" @@ -8736,6 +9134,13 @@ trim-repeated@^1.0.0: dependencies: escape-string-regexp "^1.0.2" +truncate-utf8-bytes@^1.0.0: + version "1.0.2" + resolved "https://registry.yarnpkg.com/truncate-utf8-bytes/-/truncate-utf8-bytes-1.0.2.tgz#405923909592d56f78a5818434b0b78489ca5f2b" + integrity sha512-95Pu1QXQvruGEhv62XCMO3Mm90GscOCClvrIUwCM0PYOXK3kaF3l3sIHxx71ThJfcbM2O5Au6SO3AWCSEfW4mQ== + dependencies: + utf8-byte-length "^1.0.1" + ts-interface-checker@^0.1.9: version "0.1.13" resolved "https://registry.yarnpkg.com/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz#784fd3d679722bc103b1b4b8030bcddb5db2a699" @@ -8780,7 +9185,7 @@ tslib@^1.8.1: resolved "https://registry.yarnpkg.com/tslib/-/tslib-1.14.1.tgz#cf2d38bdc34a134bcaf1091c41f6619e2f672d00" integrity sha512-Xni35NKzjgMrwevysHTCArtLDpPvye8zV/0E4EyYn43P7/7qvQwPh9BGkHewbMulVntbigmcT7rdX3BNo9wRJg== -tslib@^2.0.1, tslib@^2.1.0, tslib@^2.3.0: +tslib@^2.0.1, tslib@^2.3.0: version "2.6.3" resolved "https://registry.yarnpkg.com/tslib/-/tslib-2.6.3.tgz#0438f810ad7a9edcde7a241c3d80db693c8cbfe0" integrity sha512-xNvxJEOUiWPGhUuUdQgAJPKOOJfGnIyKySOc09XkKsgdUV/3E2zvwZYdejjmRgPCgcym1juLH3226yA7sEFJKQ== @@ -8876,7 +9281,7 @@ typed-array-length@^1.0.6: is-typed-array "^1.1.13" possible-typed-array-names "^1.0.0" -typescript@^5.0.0: +typescript@^5.0.0, typescript@^5.3.3: version "5.5.4" resolved "https://registry.yarnpkg.com/typescript/-/typescript-5.5.4.tgz#d9852d6c82bad2d2eda4fd74a5762a8f5909e9ba" integrity sha512-Mtq29sKDAEYP7aljRgtPOpTvOfbwRWlS6dPRzwjdE+C0R4brX/GUyhHSecbHMFLNBLcJIPt9nl9yG5TZ1weH+Q== @@ -9020,6 +9425,11 @@ username@^5.1.0: execa "^1.0.0" mem "^4.3.0" +utf8-byte-length@^1.0.1: + version "1.0.5" + resolved "https://registry.yarnpkg.com/utf8-byte-length/-/utf8-byte-length-1.0.5.tgz#f9f63910d15536ee2b2d5dd4665389715eac5c1e" + integrity sha512-Xn0w3MtiQ6zoz2vFyUVruaCL53O/DwUvkEeOvj+uulMm0BkUGYWmBYVyElqZaSLhY6ZD0ulfU3aBra2aVT4xfA== + util-deprecate@^1.0.1, util-deprecate@^1.0.2, util-deprecate@~1.0.1: version "1.0.2" resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" @@ -9060,6 +9470,15 @@ vary@~1.1.2: resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" integrity sha512-BNGbWLfd0eUPabhkXUVm0j8uuvREyTh5ovRa/dyow/BqAbZJyC+5fU+IzQOzmAKzYqYRAISoRhdQr3eIZ/PXqg== +verror@^1.10.0: + version "1.10.1" + resolved "https://registry.yarnpkg.com/verror/-/verror-1.10.1.tgz#4bf09eeccf4563b109ed4b3d458380c972b0cdeb" + integrity sha512-veufcmxri4e3XSrT0xwfUR7kguIkaxBeosDg00yDWhk49wdwkSUrvvsm7nc75e1PUyvIeZj6nS8VQRYz2/S4Xg== + dependencies: + assert-plus "^1.0.0" + core-util-is "1.0.2" + extsprintf "^1.2.0" + vite-node@1.6.0: version "1.6.0" resolved "https://registry.yarnpkg.com/vite-node/-/vite-node-1.6.0.tgz#2c7e61129bfecc759478fa592754fd9704aaba7f" @@ -9172,17 +9591,6 @@ w3c-keyname@^2.2.4: resolved "https://registry.yarnpkg.com/w3c-keyname/-/w3c-keyname-2.2.8.tgz#7b17c8c6883d4e8b86ac8aba79d39e880f8869c5" integrity sha512-dpojBhNsCNN7T82Tm7k26A6G9ML3NkhDsnw9n/eoxSRlVBB4CEtIQ/KTCLI2Fwf3ataSXRhYFkQi3SlnFwPvPQ== -wait-on@^7.2.0: - version "7.2.0" - resolved "https://registry.yarnpkg.com/wait-on/-/wait-on-7.2.0.tgz#d76b20ed3fc1e2bebc051fae5c1ff93be7892928" - integrity sha512-wCQcHkRazgjG5XoAq9jbTMLpNIjoSlZslrJ2+N9MxDsGEv1HnFoVjOCexL0ESva7Y9cu350j+DWADdk54s4AFQ== - dependencies: - axios "^1.6.1" - joi "^17.11.0" - lodash "^4.17.21" - minimist "^1.2.8" - rxjs "^7.8.1" - warning@^4.0.3: version "4.0.3" resolved "https://registry.yarnpkg.com/warning/-/warning-4.0.3.tgz#16e9e077eb8a86d6af7d64aa1e05fd85b4678ca3" @@ -9375,7 +9783,7 @@ ws@^8.17.0: resolved "https://registry.yarnpkg.com/ws/-/ws-8.18.0.tgz#0d7505a6eafe2b0e712d232b42279f53bc289bbc" integrity sha512-8VbfWfHLbbwu3+N6OKsOMpBdT4kXPDDB9cJk2bJ6mh9ucxdlnNvH1e+roYkKmN9Nxw2yjz7VzeO9oOz2zJ04Pw== -xmlbuilder@^15.1.1: +xmlbuilder@>=11.0.1, xmlbuilder@^15.1.1: version "15.1.1" resolved "https://registry.yarnpkg.com/xmlbuilder/-/xmlbuilder-15.1.1.tgz#9dcdce49eea66d8d10b42cae94a79c3c8d0c2ec5" integrity sha512-yMqGBqtXyeN1e3TGYvgNgDVZ3j84W4cwkOXQswghol6APgZWaff9lnbvN7MHYJOiXsvGPXtjTYJEiC9J2wv9Eg== @@ -9453,7 +9861,7 @@ yargs@^16.0.2: y18n "^5.0.5" yargs-parser "^20.2.2" -yargs@^17.0.1: +yargs@^17.0.1, yargs@^17.6.2: version "17.7.2" resolved "https://registry.yarnpkg.com/yargs/-/yargs-17.7.2.tgz#991df39aca675a192b816e1e0363f9d75d2aa269" integrity sha512-7dSzzRQ++CKnNI/krKnYRV7JKKPUXMEh61soaHKg9mrWEhzFWhFnxPxGl+69cD1Ou63C13NUPCnmIcrvqCuM6w==