Compare commits
	
		
			1 Commits
		
	
	
		
			kurt-messy
			...
			dependabot
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 6e2274a6c4 | 
| @ -1,3 +1,3 @@ | ||||
| [codespell] | ||||
| ignore-words-list: crate,everytime,inout,co-ordinate,ot,nwo,absolutey,atleast,ue,afterall | ||||
| skip: **/target,node_modules,build,**/Cargo.lock,./docs/kcl/*.md,./src-tauri/gen/schemas,.yarn.lock,**/yarn.lock | ||||
| skip: **/target,node_modules,build,**/Cargo.lock,./docs/kcl/*.md,./src-tauri/gen/schemas | ||||
|  | ||||
							
								
								
									
										404
									
								
								.github/workflows/build-test-publish-apps.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										404
									
								
								.github/workflows/build-test-publish-apps.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,404 +0,0 @@ | ||||
| name: build-test-publish-apps | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|   push: | ||||
|   release: | ||||
|     types: [published] | ||||
|   schedule: | ||||
|     - cron: '0 4 * * *' | ||||
|   # Daily at 04:00 AM UTC | ||||
|   # Will checkout the last commit from the default branch (main as of 2023-10-04) | ||||
|  | ||||
| env: | ||||
|   CUT_RELEASE_PR: ${{ github.event_name == 'pull_request' && (contains(github.event.pull_request.title, 'Cut release v')) }} | ||||
|   BUILD_RELEASE: ${{ github.event_name == 'release' || github.event_name == 'schedule' || github.event_name == 'pull_request' && (contains(github.event.pull_request.title, 'Cut release v')) }} | ||||
|  | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||
|   cancel-in-progress: true | ||||
|  | ||||
| jobs: | ||||
|   prepare-json-files: | ||||
|     runs-on: ubuntu-22.04  # seperate job on Ubuntu for easy string manipulations (compared to Windows) | ||||
|     outputs: | ||||
|       version: ${{ steps.export_version.outputs.version }} | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|  | ||||
|       - name: Set nightly version | ||||
|         if: github.event_name == 'schedule' | ||||
|         run: | | ||||
|           VERSION=$(date +'%-y.%-m.%-d') yarn bump-jsons | ||||
|  | ||||
|       # TODO: see if we need to inject updater nightly URL here https://dl.zoo.dev/releases/modeling-app/nightly/last_update.json | ||||
|       # 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: | ||||
|           path: | | ||||
|             package.json | ||||
|  | ||||
|       - 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 | ||||
|     env: | ||||
|       APPLE_ID: ${{ secrets.APPLE_ID }} | ||||
|       APPLE_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 }} | ||||
|     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' || ''}}" | ||||
|  | ||||
|       # 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 | ||||
|  | ||||
|       - name: Prepare certificate and variables (Windows only) | ||||
|         if: ${{ env.BUILD_RELEASE == 'true' }} | ||||
|         run: | | ||||
|           echo "${{secrets.SM_CLIENT_CERT_FILE_B64 }}" | base64 --decode > /d/Certificate_pkcs12.p12 | ||||
|           cat /d/Certificate_pkcs12.p12 | ||||
|           echo "::set-output name=version::${GITHUB_REF#refs/tags/v}" | ||||
|           echo "SM_HOST=${{ secrets.SM_HOST }}" >> "$GITHUB_ENV" | ||||
|           echo "SM_API_KEY=${{ secrets.SM_API_KEY }}" >> "$GITHUB_ENV" | ||||
|           echo "SM_CLIENT_CERT_FILE=D:\\Certificate_pkcs12.p12" >> "$GITHUB_ENV" | ||||
|           echo "SM_CLIENT_CERT_PASSWORD=${{ secrets.SM_CLIENT_CERT_PASSWORD }}" >> "$GITHUB_ENV" | ||||
|           echo "C:\Program Files (x86)\Windows Kits\10\App Certification Kit" >> $GITHUB_PATH | ||||
|           echo "C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8 Tools" >> $GITHUB_PATH | ||||
|           echo "C:\Program Files\DigiCert\DigiCert One Signing Manager Tools" >> $GITHUB_PATH | ||||
|         shell: bash | ||||
|  | ||||
|       - name: Setup certicate with SSM KSP (Windows only) | ||||
|         if: ${{ env.BUILD_RELEASE == 'true' }} | ||||
|         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 | ||||
|           smksp_registrar.exe list | ||||
|           smctl.exe keypair ls | ||||
|           C:\Windows\System32\certutil.exe -csp "DigiCert Signing Manager KSP" -key -user | ||||
|           smksp_cert_sync.exe | ||||
|         shell: cmd | ||||
|  | ||||
|       - name: Build the app for x64 | ||||
|         run: "yarn electron-forge make --arch x64" | ||||
|  | ||||
|       - 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 }}" | ||||
|  | ||||
|       - 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" | ||||
|  | ||||
|       # 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/*/*/*" | ||||
|  | ||||
|  | ||||
|   publish-apps-release: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     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] | ||||
|     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 }} | ||||
|       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' }} | ||||
|       URL_CODED_NAME: ${{ github.event_name == 'schedule' && 'Zoo%20Modeling%20App%20%28Nightly%29' || 'Zoo%20Modeling%20App' }} | ||||
|     steps: | ||||
|       - uses: actions/download-artifact@v3 | ||||
|  | ||||
|       - 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 | ||||
|  | ||||
|       - name: Generate the download static endpoint | ||||
|         run: | | ||||
|           RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} | ||||
|           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" \ | ||||
|             '{ | ||||
|               "version": $version, | ||||
|               "pub_date": $pub_date, | ||||
|               "notes": $notes, | ||||
|               "platforms": { | ||||
|                 "dmg-universal": { | ||||
|                   "url": $darwin_url | ||||
|                 }, | ||||
|                 "msi-x86_64": { | ||||
|                   "url": $windows_x86_64_url | ||||
|                 }, | ||||
|                 "msi-aarch64": { | ||||
|                   "url": $windows_aarch64_url | ||||
|                 } | ||||
|               } | ||||
|             }' > last_download.json | ||||
|             cat last_download.json | ||||
|  | ||||
|       - name: Authenticate to Google Cloud | ||||
|         uses: 'google-github-actions/auth@v2.1.3' | ||||
|         with: | ||||
|           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' | ||||
|  | ||||
|       - name: Set up Google Cloud SDK | ||||
|         uses: google-github-actions/setup-gcloud@v2.1.0 | ||||
|         with: | ||||
|           project_id: kittycadapi | ||||
|  | ||||
|       - name: Upload release files to public bucket | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.1.0 | ||||
|         with: | ||||
|           path: artifact | ||||
|           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.0 | ||||
|         with: | ||||
|           path: last_update.json | ||||
|           destination: ${{ env.BUCKET_DIR }} | ||||
|  | ||||
|       - name: Upload download endpoint to public bucket | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.1.0 | ||||
|         with: | ||||
|           path: last_download.json | ||||
|           destination: ${{ env.BUCKET_DIR }} | ||||
|  | ||||
|       - name: Upload release files to Github | ||||
|         if: ${{ github.event_name == 'release' }} | ||||
|         uses: softprops/action-gh-release@v2 | ||||
|         with: | ||||
|           files: 'artifact/*/Zoo*' | ||||
|  | ||||
|   announce_release: | ||||
|     needs: [publish-apps-release] | ||||
|     runs-on: ubuntu-22.04 | ||||
|     if: github.event_name == 'release' | ||||
|     steps: | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Set up Python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: '3.x' | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: | | ||||
|           python -m pip install --upgrade pip | ||||
|           pip install requests | ||||
|  | ||||
|       - name: Announce Release | ||||
|         env: | ||||
|           DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_WEBHOOK_URL }} | ||||
|           RELEASE_VERSION: ${{ github.event.release.tag_name }} | ||||
|           RELEASE_BODY: ${{ github.event.release.body}} | ||||
|         run: python public/announce_release.py | ||||
							
								
								
									
										76
									
								
								.github/workflows/build-test-web.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										76
									
								
								.github/workflows/build-test-web.yml
									
									
									
									
										vendored
									
									
								
							| @ -1,76 +0,0 @@ | ||||
| name: build-test-web | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|   push: | ||||
|  | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||
|   cancel-in-progress: true | ||||
|  | ||||
| jobs: | ||||
|   check-format: | ||||
|     runs-on: 'ubuntu-22.04' | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|       - run: yarn fmt-check | ||||
|  | ||||
|   check-types: | ||||
|     runs-on: ubuntu-22.04 | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|       - uses: Swatinem/rust-cache@v2 | ||||
|         with: | ||||
|           workspaces: './src/wasm-lib' | ||||
|  | ||||
|       - run: yarn build:wasm | ||||
|       - run: yarn xstate:typegen | ||||
|       - run: yarn tsc | ||||
|  | ||||
|  | ||||
|   check-typos: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Set up Python | ||||
|         uses: actions/setup-python@v5 | ||||
|       - name: Install codespell | ||||
|         run: | | ||||
|             python -m pip install codespell | ||||
|       - name: Run codespell | ||||
|         run: codespell --config .codespellrc # Edit this file to tweak the typo list and other configuration. | ||||
|  | ||||
|  | ||||
|   build-test-web: | ||||
|     runs-on: ubuntu-22.04 | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|  | ||||
|       - run: yarn install | ||||
|  | ||||
|       - uses: Swatinem/rust-cache@v2 | ||||
|         with: | ||||
|           workspaces: './src/wasm-lib' | ||||
|  | ||||
|       - run: yarn build:wasm | ||||
|  | ||||
|       - run: yarn simpleserver:ci | ||||
|  | ||||
|       - run: yarn test:nowatch | ||||
							
								
								
									
										586
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										586
									
								
								.github/workflows/ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @ -0,0 +1,586 @@ | ||||
| name: CI | ||||
|  | ||||
| on: | ||||
|   pull_request: | ||||
|   push: | ||||
|     branches: | ||||
|       - main | ||||
|   release: | ||||
|     types: [published] | ||||
|   schedule: | ||||
|     - cron: '0 4 * * *' | ||||
|   # Daily at 04:00 AM UTC | ||||
|   # Will checkout the last commit from the default branch (main as of 2023-10-04) | ||||
|  | ||||
| env: | ||||
|   CUT_RELEASE_PR: ${{ github.event_name == 'pull_request' && (contains(github.event.pull_request.title, 'Cut release v')) }} | ||||
|   BUILD_RELEASE: ${{ github.event_name == 'release' || github.event_name == 'schedule' || github.event_name == 'pull_request' && (contains(github.event.pull_request.title, 'Cut release v')) }} | ||||
|  | ||||
| concurrency: | ||||
|   group: ${{ github.workflow }}-${{ github.head_ref || github.run_id }} | ||||
|   cancel-in-progress: true | ||||
|  | ||||
| permissions: | ||||
|   contents: write | ||||
|   pull-requests: write | ||||
|   actions: read | ||||
|  | ||||
| jobs: | ||||
|   check-format: | ||||
|     runs-on: 'ubuntu-latest' | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|       - run: yarn fmt-check | ||||
|  | ||||
|   check-types: | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|       - run: yarn install | ||||
|       - uses: Swatinem/rust-cache@v2 | ||||
|         with: | ||||
|           workspaces: './src/wasm-lib' | ||||
|  | ||||
|       - run: yarn build:wasm | ||||
|       - run: yarn xstate:typegen | ||||
|       - run: yarn tsc | ||||
|  | ||||
|  | ||||
|   check-typos: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - name: Checkout | ||||
|         uses: actions/checkout@v4 | ||||
|       - name: Set up Python | ||||
|         uses: actions/setup-python@v5 | ||||
|       - name: Install codespell | ||||
|         run: | | ||||
|             python -m pip install codespell | ||||
|       - name: Run codespell | ||||
|         run: codespell --config .codespellrc # Edit this file to tweak the typo list and other configuration. | ||||
|  | ||||
|  | ||||
|   build-test-web: | ||||
|     runs-on: ubuntu-latest | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|  | ||||
|       - run: yarn install | ||||
|  | ||||
|       - uses: Swatinem/rust-cache@v2 | ||||
|         with: | ||||
|           workspaces: './src/wasm-lib' | ||||
|  | ||||
|       - run: yarn build:wasm | ||||
|  | ||||
|       - run: yarn simpleserver:ci | ||||
|         if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
|  | ||||
|       - name: Install Chromium Browser | ||||
|         if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
|         run: yarn playwright install chromium --with-deps | ||||
|  | ||||
|       - name: run unit tests | ||||
|         if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
|         run: yarn test:nowatch | ||||
|         env: | ||||
|           VITE_KC_DEV_TOKEN: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|  | ||||
|       - name: check for changes | ||||
|         if: ${{ github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
|         id: git-check | ||||
|         run: | | ||||
|             git add src/lang/std/artifactMapGraphs | ||||
|             if git status src/lang/std/artifactMapGraphs | grep -q "Changes to be committed" | ||||
|             then echo "modified=true" >> $GITHUB_OUTPUT | ||||
|             else echo "modified=false" >> $GITHUB_OUTPUT | ||||
|             fi | ||||
|       - name: Commit changes, if any | ||||
|         if: ${{ github.event_name != 'release' && github.event_name != 'schedule' && steps.git-check.outputs.modified == 'true' }} | ||||
|         run: | | ||||
|           git config --local user.email "github-actions[bot]@users.noreply.github.com" | ||||
|           git config --local user.name "github-actions[bot]" | ||||
|           git remote set-url origin https://${{ github.actor }}:${{ secrets.GITHUB_TOKEN }}@github.com/${{ github.repository }}.git | ||||
|           git fetch origin | ||||
|           echo ${{ github.head_ref }} | ||||
|           git checkout ${{ github.head_ref }} | ||||
|           # TODO when webkit works on ubuntu remove the os part of the commit message | ||||
|           git commit -am "Look at this (photo)Graph *in the voice of Nickelback*" || true | ||||
|           git push | ||||
|           git push origin ${{ github.head_ref }} | ||||
|          | ||||
|  | ||||
|  | ||||
|  | ||||
|  | ||||
|   prepare-json-files: | ||||
|     runs-on: ubuntu-latest  # seperate job on Ubuntu for easy string manipulations (compared to Windows) | ||||
|     outputs: | ||||
|       version: ${{ steps.export_version.outputs.version }} | ||||
|     steps: | ||||
|       - uses: actions/checkout@v4 | ||||
|  | ||||
|       - uses: actions/setup-node@v4 | ||||
|         with: | ||||
|           node-version-file: '.nvmrc' | ||||
|           cache: 'yarn' | ||||
|  | ||||
|       - name: Set nightly version | ||||
|         if: github.event_name == 'schedule' | ||||
|         run: | | ||||
|           VERSION=$(date +'%-y.%-m.%-d') yarn bump-jsons | ||||
|           echo "$(jq --arg url 'https://dl.zoo.dev/releases/modeling-app/nightly/last_update.json' \ | ||||
|             '.plugins.updater.endpoints[]=$url' src-tauri/tauri.release.conf.json --indent 2)" > src-tauri/tauri.release.conf.json | ||||
|           echo "$(jq --arg id 'dev.zoo.modeling-app-nightly' \ | ||||
|             '.identifier=$id' src-tauri/tauri.release.conf.json --indent 2)" > src-tauri/tauri.release.conf.json | ||||
|           echo "$(jq --arg name 'Zoo Modeling App (Nightly)' \ | ||||
|             '.productName=$name' src-tauri/tauri.release.conf.json --indent 2)" > src-tauri/tauri.release.conf.json | ||||
|  | ||||
|       - name: Set updater test version | ||||
|         if: ${{ env.CUT_RELEASE_PR == 'true' }} | ||||
|         run: | | ||||
|           echo "$(jq --arg url 'https://dl.zoo.dev/releases/modeling-app/test/last_update.json' \ | ||||
|             '.plugins.updater.endpoints[]=$url' src-tauri/tauri.release.conf.json --indent 2)" > src-tauri/tauri.release.conf.json | ||||
|  | ||||
|       - uses: actions/upload-artifact@v3 | ||||
|         if: ${{ github.event_name == 'schedule' || env.CUT_RELEASE_PR == 'true' }} | ||||
|         with: | ||||
|           path: | | ||||
|             package.json | ||||
|             src-tauri/tauri.conf.json | ||||
|             src-tauri/tauri.release.conf.json | ||||
|  | ||||
|       - id: export_version | ||||
|         run: echo "version=`cat package.json | jq -r '.version'`" >> "$GITHUB_OUTPUT" | ||||
|  | ||||
|  | ||||
|   build-test-apps: | ||||
|     needs: [prepare-json-files] | ||||
|     runs-on: ${{ matrix.os }} | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
|         os: [macos-14, ubuntu-latest, windows-latest] | ||||
|     env: | ||||
|       # Specific Apple Universal target for macos | ||||
|       TAURI_ARGS_MACOS: ${{ matrix.os == 'macos-14' && '--target universal-apple-darwin' || '' }} | ||||
|       # Only build executable on linux (no appimage or deb) | ||||
|       TAURI_ARGS_UBUNTU: ${{ matrix.os == 'ubuntu-latest' && '--bundles' || '' }} | ||||
|     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 | ||||
|           cp artifact/src-tauri/tauri.conf.json src-tauri/tauri.conf.json | ||||
|           cp artifact/src-tauri/tauri.release.conf.json src-tauri/tauri.release.conf.json | ||||
|  | ||||
|       - name: Update WebView2 on Windows | ||||
|         if: matrix.os == 'windows-latest' | ||||
|         # Workaround needed to build the tauri windows app with matching edge version. | ||||
|         # From https://github.com/actions/runner-images/issues/9538 | ||||
|         run: | | ||||
|           Invoke-WebRequest -Uri 'https://go.microsoft.com/fwlink/p/?LinkId=2124703' -OutFile 'setup.exe' | ||||
|           Start-Process -FilePath setup.exe -Verb RunAs -Wait | ||||
|  | ||||
|       - name: Install ubuntu system dependencies | ||||
|         if: matrix.os == 'ubuntu-latest' | ||||
|         run: | | ||||
|           sudo apt-get update | ||||
|           sudo apt-get install -y \ | ||||
|             libgtk-3-dev \ | ||||
|             libayatana-appindicator3-dev \ | ||||
|             webkit2gtk-driver \ | ||||
|             libsoup-3.0-dev \ | ||||
|             libjavascriptcoregtk-4.1-dev \ | ||||
|             libwebkit2gtk-4.1-dev \ | ||||
|             at-spi2-core \ | ||||
|             xvfb | ||||
|  | ||||
|       - 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 | ||||
|  | ||||
|       - name: Setup Rust cache | ||||
|         uses: swatinem/rust-cache@v2 | ||||
|         with: | ||||
|           workspaces: './src-tauri -> target' | ||||
|  | ||||
|       - 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 | ||||
|  | ||||
|       - name: Run vite build (build:both) | ||||
|         run: yarn vite build --mode ${{ env.BUILD_RELEASE == 'true' && 'production' || 'development' }} | ||||
|  | ||||
|       - name: Fix format | ||||
|         run: yarn fmt | ||||
|  | ||||
|       - name: Install x86 target for Universal builds (MacOS only) | ||||
|         if: matrix.os == 'macos-14' | ||||
|         run: | | ||||
|           rustup target add x86_64-apple-darwin | ||||
|  | ||||
|       - name: Prepare certificate and variables (Windows only) | ||||
|         if: ${{ matrix.os == 'windows-latest' && env.BUILD_RELEASE == 'true' }} | ||||
|         run: | | ||||
|           echo "${{secrets.SM_CLIENT_CERT_FILE_B64 }}" | base64 --decode > /d/Certificate_pkcs12.p12 | ||||
|           cat /d/Certificate_pkcs12.p12 | ||||
|           echo "::set-output name=version::${GITHUB_REF#refs/tags/v}" | ||||
|           echo "SM_HOST=${{ secrets.SM_HOST }}" >> "$GITHUB_ENV" | ||||
|           echo "SM_API_KEY=${{ secrets.SM_API_KEY }}" >> "$GITHUB_ENV" | ||||
|           echo "SM_CLIENT_CERT_FILE=D:\\Certificate_pkcs12.p12" >> "$GITHUB_ENV" | ||||
|           echo "SM_CLIENT_CERT_PASSWORD=${{ secrets.SM_CLIENT_CERT_PASSWORD }}" >> "$GITHUB_ENV" | ||||
|           echo "C:\Program Files (x86)\Windows Kits\10\App Certification Kit" >> $GITHUB_PATH | ||||
|           echo "C:\Program Files (x86)\Microsoft SDKs\Windows\v10.0A\bin\NETFX 4.8 Tools" >> $GITHUB_PATH | ||||
|           echo "C:\Program Files\DigiCert\DigiCert One Signing Manager Tools" >> $GITHUB_PATH | ||||
|         shell: bash | ||||
|  | ||||
|       - name: Setup certicate with SSM KSP (Windows only) | ||||
|         if: ${{ matrix.os == 'windows-latest' && env.BUILD_RELEASE == 'true' }} | ||||
|         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 | ||||
|           smksp_registrar.exe list | ||||
|           smctl.exe keypair ls | ||||
|           C:\Windows\System32\certutil.exe -csp "DigiCert Signing Manager KSP" -key -user | ||||
|           smksp_cert_sync.exe | ||||
|         shell: cmd | ||||
|  | ||||
|       - name: Build the app (debug) | ||||
|         if: ${{ env.BUILD_RELEASE == 'false' }} | ||||
|         run: "yarn tauri build --debug ${{ env.TAURI_ARGS_MACOS }} ${{ env.TAURI_ARGS_UBUNTU }}" | ||||
|  | ||||
|       - name: Build for Mac TestFlight (nightly) | ||||
|         if: ${{ github.event_name == 'schedule' && matrix.os == 'macos-14' }} | ||||
|         shell: bash | ||||
|         run: | | ||||
|           unset APPLE_SIGNING_IDENTITY | ||||
|           unset APPLE_CERTIFICATE | ||||
|           sign_app="3rd Party Mac Developer Application: KittyCAD Inc (${APPLE_TEAM_ID})" | ||||
|           sign_install="3rd Party Mac Developer Installer: KittyCAD Inc (${APPLE_TEAM_ID})" | ||||
|           profile="src-tauri/entitlements/Mac_App_Distribution.provisionprofile" | ||||
|  | ||||
|           mkdir -p src-tauri/entitlements | ||||
|           echo -n "${APPLE_STORE_PROVISIONING_PROFILE}" | base64 --decode -o "${profile}" | ||||
|  | ||||
|           echo -n "${APPLE_STORE_DISTRIBUTION_CERT}" | base64 --decode -o "dist.cer" | ||||
|           echo -n "${APPLE_STORE_INSTALLER_CERT}" | base64 --decode -o "installer.cer" | ||||
|  | ||||
|           KEYCHAIN_PATH=$RUNNER_TEMP/app-signing.keychain-db | ||||
|           KEYCHAIN_PASSWORD="password" | ||||
|  | ||||
|           # create temporary keychain | ||||
|           security create-keychain -p "$KEYCHAIN_PASSWORD" $KEYCHAIN_PATH | ||||
|           security set-keychain-settings -lut 21600 $KEYCHAIN_PATH | ||||
|           security unlock-keychain -p "$KEYCHAIN_PASSWORD" $KEYCHAIN_PATH | ||||
|  | ||||
|           # import certificate to keychain | ||||
|           security import "dist.cer" -P "$APPLE_STORE_P12_PASSWORD" -k $KEYCHAIN_PATH -f pkcs12 -t cert -A | ||||
|           security import "installer.cer" -P "$APPLE_STORE_P12_PASSWORD" -k $KEYCHAIN_PATH -f pkcs12 -t cert -A | ||||
|  | ||||
|           security set-key-partition-list -S apple-tool:,apple: -k "$KEYCHAIN_PASSWORD" $KEYCHAIN_PATH | ||||
|           security list-keychain -d user -s $KEYCHAIN_PATH | ||||
|  | ||||
|           target="universal-apple-darwin" | ||||
|  | ||||
|           # Turn off the default target | ||||
|           # We don't want to install the updater for the apple store build | ||||
|           sed -i.bu "s/default =/# default =/" src-tauri/Cargo.toml | ||||
|           rm src-tauri/Cargo.toml.bu | ||||
|           git diff src-tauri/Cargo.toml | ||||
|  | ||||
|           yarn tauri build --target "${target}" --verbose --config src-tauri/tauri.app-store.conf.json | ||||
|  | ||||
|           app_path="src-tauri/target/${target}/release/bundle/macos/Zoo Modeling App.app" | ||||
|           build_name="src-tauri/target/${target}/release/bundle/macos/Zoo Modeling App.pkg" | ||||
|           cp_dir="src-tauri/target/${target}/release/bundle/macos/Zoo Modeling App.app/Contents/embedded.provisionprofile" | ||||
|           entitlements="src-tauri/entitlements/app-store.entitlements" | ||||
|  | ||||
|           cp "${profile}" "${cp_dir}" | ||||
|  | ||||
|           codesign --deep --force -s "${sign_app}" --entitlements "${entitlements}" "${app_path}" | ||||
|  | ||||
|           productbuild --component "${app_path}" /Applications/ --sign "${sign_install}" "${build_name}" | ||||
|  | ||||
|           # Undo the changes to the Cargo.toml | ||||
|           git checkout src-tauri/Cargo.toml | ||||
|  | ||||
|         env: | ||||
|           APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} | ||||
|           APPLE_STORE_PROVISIONING_PROFILE: ${{ secrets.APPLE_STORE_PROVISIONING_PROFILE }} | ||||
|           APPLE_STORE_DISTRIBUTION_CERT: ${{ secrets.APPLE_STORE_DISTRIBUTION_CERT }} | ||||
|           APPLE_STORE_INSTALLER_CERT: ${{ secrets.APPLE_STORE_INSTALLER_CERT }} | ||||
|           APPLE_STORE_P12_PASSWORD: ${{ secrets.APPLE_STORE_P12_PASSWORD }} | ||||
|  | ||||
|  | ||||
|       - name: 'Upload to Mac TestFlight (nightly)' | ||||
|         uses: apple-actions/upload-testflight-build@v1 | ||||
|         if: ${{ github.event_name == 'schedule' && matrix.os == 'macos-14' }} | ||||
|         with: | ||||
|           app-path: 'src-tauri/target/universal-apple-darwin/release/bundle/macos/Zoo Modeling App.pkg' | ||||
|           issuer-id: ${{ secrets.APPLE_STORE_ISSUER_ID }} | ||||
|           api-key-id: ${{ secrets.APPLE_STORE_API_KEY_ID }} | ||||
|           api-private-key: ${{ secrets.APPLE_STORE_API_PRIVATE_KEY }} | ||||
|           app-type: osx | ||||
|  | ||||
|  | ||||
|       - name: Clean up after Mac TestFlight (nightly) | ||||
|         if: ${{ github.event_name == 'schedule' && matrix.os == 'macos-14' }} | ||||
|         shell: bash | ||||
|         run: | | ||||
|           git status | ||||
|           # remove our target builds because we want to make sure the later build | ||||
|           # includes the updater, and that anything we changed with the target | ||||
|           # does not persist | ||||
|           rm -rf src-tauri/target | ||||
|           # Lets get rid of the info.plist for the normal mac builds since its | ||||
|           # being sketchy. | ||||
|           rm src-tauri/Info.plist | ||||
|  | ||||
|       # We do this after the apple store because the apple store build is | ||||
|       # specific and we want to overwrite it with the this new build after and | ||||
|       # not upload the apple store build to the public bucket | ||||
|       - name: Build the app (release) and sign | ||||
|         if: ${{ env.BUILD_RELEASE == 'true' }} | ||||
|         env: | ||||
|           TAURI_SIGNING_PRIVATE_KEY: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY }} | ||||
|           TAURI_SIGNING_PRIVATE_KEY_PASSWORD: ${{ secrets.TAURI_SIGNING_PRIVATE_KEY_PASSWORD }} | ||||
|           APPLE_CERTIFICATE: ${{ secrets.APPLE_CERTIFICATE }} | ||||
|           APPLE_CERTIFICATE_PASSWORD: ${{ secrets.APPLE_CERTIFICATE_PASSWORD }} | ||||
|           APPLE_SIGNING_IDENTITY: ${{ secrets.APPLE_SIGNING_IDENTITY }} | ||||
|           APPLE_ID: ${{ secrets.APPLE_ID }} | ||||
|           APPLE_PASSWORD: ${{ secrets.APPLE_PASSWORD }} | ||||
|           APPLE_TEAM_ID: ${{ secrets.APPLE_TEAM_ID }} | ||||
|           TAURI_CONF_ARGS: "--config ${{ matrix.os == 'windows-latest' && 'src-tauri\\tauri.release.conf.json' || 'src-tauri/tauri.release.conf.json' }}" | ||||
|         run: "yarn tauri build ${{ env.TAURI_CONF_ARGS }} ${{ env.TAURI_ARGS_MACOS }} ${{ env.TAURI_ARGS_UBUNTU }}" | ||||
|  | ||||
|       - uses: actions/upload-artifact@v3 | ||||
|         if: matrix.os != 'ubuntu-latest' | ||||
|         env: | ||||
|           PREFIX: ${{ matrix.os == 'macos-14' && 'src-tauri/target/universal-apple-darwin' || 'src-tauri/target' }} | ||||
|           MODE: ${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }} | ||||
|         with: | ||||
|           path: "${{ env.PREFIX }}/${{ env.MODE }}/bundle/*/*" | ||||
|  | ||||
|       - name: Run e2e tests (linux only) | ||||
|         if: ${{ matrix.os == 'ubuntu-latest' && github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
|         run: | | ||||
|           cargo install tauri-driver --force | ||||
|           source .env.${{ env.BUILD_RELEASE == 'true' && 'production' || 'development' }} | ||||
|           export VITE_KC_API_BASE_URL | ||||
|           xvfb-run yarn test:e2e:tauri | ||||
|         env: | ||||
|           E2E_APPLICATION: "./src-tauri/target/${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }}/zoo-modeling-app" | ||||
|           KITTYCAD_API_TOKEN: ${{ env.BUILD_RELEASE == 'true' && secrets.KITTYCAD_API_TOKEN || secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|  | ||||
|       - name: Run e2e tests (windows only) | ||||
|         if: ${{ matrix.os == 'windows-latest' && github.event_name != 'release' && github.event_name != 'schedule' }} | ||||
|         run: | | ||||
|           cargo install tauri-driver --force | ||||
|           yarn wdio run wdio.conf.ts | ||||
|         env: | ||||
|           E2E_APPLICATION: ".\\src-tauri\\target\\${{ env.BUILD_RELEASE == 'true' && 'release' || 'debug' }}\\Zoo Modeling App.exe" | ||||
|           KITTYCAD_API_TOKEN: ${{ env.BUILD_RELEASE == 'true' && secrets.KITTYCAD_API_TOKEN || secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|           VITE_KC_API_BASE_URL: ${{ env.BUILD_RELEASE == 'true' && 'https://api.zoo.dev' || 'https://api.dev.zoo.dev' }} | ||||
|           E2E_TAURI_ENABLED: true | ||||
|           TS_NODE_COMPILER_OPTIONS: '{"module": "commonjs"}' | ||||
|  | ||||
|       - uses: actions/download-artifact@v3 | ||||
|         if: ${{ env.CUT_RELEASE_PR == 'true' }} | ||||
|  | ||||
|       - name: Copy updated .json file for updater test | ||||
|         if: ${{ env.CUT_RELEASE_PR == 'true' }} | ||||
|         run: | | ||||
|           ls -l artifact | ||||
|           cp artifact/src-tauri/tauri.release.conf.json src-tauri/tauri.release.conf.json | ||||
|           cat src-tauri/tauri.release.conf.json | ||||
|  | ||||
|       - name: Build the app (release, updater test) | ||||
|         if: ${{ env.CUT_RELEASE_PR == 'true' && matrix.os != 'ubuntu-latest' }} | ||||
|         env: | ||||
|           TAURI_CONF_ARGS: "-c ${{ matrix.os == 'windows-latest' && 'src-tauri\\tauri.release.conf.json' || 'src-tauri/tauri.release.conf.json' }}" | ||||
|           TAURI_BUNDLE_ARGS: "-b ${{ matrix.os == 'windows-latest' && 'msi' || 'dmg' }}" | ||||
|         run: "yarn tauri build ${{ env.TAURI_CONF_ARGS }} ${{ env.TAURI_BUNDLE_ARGS }} ${{ env.TAURI_ARGS_MACOS }}" | ||||
|  | ||||
|       - uses: actions/upload-artifact@v3 | ||||
|         if: ${{ env.CUT_RELEASE_PR == 'true' && matrix.os != 'ubuntu-latest' }} | ||||
|         with: | ||||
|           path: "${{ matrix.os == 'macos-14' && 'src-tauri/target/universal-apple-darwin/release/bundle/dmg/*.dmg' || 'src-tauri/target/release/bundle/msi/*.msi' }}" | ||||
|           name: updater-test | ||||
|  | ||||
|  | ||||
|   publish-apps-release: | ||||
|     permissions: | ||||
|       contents: write | ||||
|     runs-on: ubuntu-latest | ||||
|     if: ${{ github.event_name == 'release' || github.event_name == 'schedule' }} | ||||
|     needs: [check-format, check-types, check-typos, build-test-web, prepare-json-files, build-test-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 }} | ||||
|       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' }} | ||||
|       URL_CODED_NAME: ${{ github.event_name == 'schedule' && 'Zoo%20Modeling%20App%20%28Nightly%29' || 'Zoo%20Modeling%20App' }} | ||||
|     steps: | ||||
|       - uses: actions/download-artifact@v3 | ||||
|  | ||||
|       - name: Generate the update static endpoint | ||||
|         run: | | ||||
|           ls -l artifact/*/*oo* | ||||
|           DARWIN_SIG=`cat artifact/macos/*.app.tar.gz.sig` | ||||
|           WINDOWS_SIG=`cat artifact/msi/*.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_sig "$WINDOWS_SIG" \ | ||||
|             --arg windows_url "$RELEASE_DIR/msi/${{ env.URL_CODED_NAME }}_${VERSION_NO_V}_x64_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_sig, | ||||
|                   "url": $windows_url | ||||
|                 } | ||||
|               } | ||||
|             }' > last_update.json | ||||
|             cat last_update.json | ||||
|  | ||||
|       - name: Generate the download static endpoint | ||||
|         run: | | ||||
|           RELEASE_DIR=https://${WEBSITE_DIR}/${VERSION} | ||||
|           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_url "$RELEASE_DIR/msi/${{ env.URL_CODED_NAME }}_${VERSION_NO_V}_x64_en-US.msi" \ | ||||
|             '{ | ||||
|               "version": $version, | ||||
|               "pub_date": $pub_date, | ||||
|               "notes": $notes, | ||||
|               "platforms": { | ||||
|                 "dmg-universal": { | ||||
|                   "url": $darwin_url | ||||
|                 }, | ||||
|                 "msi-x86_64": { | ||||
|                   "url": $windows_url | ||||
|                 } | ||||
|               } | ||||
|             }' > last_download.json | ||||
|             cat last_download.json | ||||
|  | ||||
|       - name: Authenticate to Google Cloud | ||||
|         uses: 'google-github-actions/auth@v2.1.3' | ||||
|         with: | ||||
|           credentials_json: '${{ secrets.GOOGLE_CLOUD_DL_SA }}' | ||||
|  | ||||
|       - name: Set up Google Cloud SDK | ||||
|         uses: google-github-actions/setup-gcloud@v2.1.0 | ||||
|         with: | ||||
|           project_id: kittycadapi | ||||
|  | ||||
|       - name: Upload release files to public bucket | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.1.1 | ||||
|         with: | ||||
|           path: artifact | ||||
|           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.1 | ||||
|         with: | ||||
|           path: last_update.json | ||||
|           destination: ${{ env.BUCKET_DIR }} | ||||
|  | ||||
|       - name: Upload download endpoint to public bucket | ||||
|         uses: google-github-actions/upload-cloud-storage@v2.1.1 | ||||
|         with: | ||||
|           path: last_download.json | ||||
|           destination: ${{ env.BUCKET_DIR }} | ||||
|  | ||||
|       - name: Upload release files to Github | ||||
|         if: ${{ github.event_name == 'release' }} | ||||
|         uses: softprops/action-gh-release@v2 | ||||
|         with: | ||||
|           files: 'artifact/*/Zoo*' | ||||
|  | ||||
|   announce_release: | ||||
|     needs: [publish-apps-release] | ||||
|     runs-on: ubuntu-latest | ||||
|     if: github.event_name == 'release' | ||||
|     steps: | ||||
|       - name: Check out code | ||||
|         uses: actions/checkout@v4 | ||||
|  | ||||
|       - name: Set up Python | ||||
|         uses: actions/setup-python@v5 | ||||
|         with: | ||||
|           python-version: '3.x' | ||||
|  | ||||
|       - name: Install dependencies | ||||
|         run: | | ||||
|           python -m pip install --upgrade pip | ||||
|           pip install requests | ||||
|  | ||||
|       - name: Announce Release | ||||
|         env: | ||||
|           DISCORD_WEBHOOK_URL: ${{ secrets.DISCORD_WEBHOOK_URL }} | ||||
|           RELEASE_VERSION: ${{ github.event.release.tag_name }} | ||||
|           RELEASE_BODY: ${{ github.event.release.body}} | ||||
|         run: python public/announce_release.py | ||||
							
								
								
									
										166
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										166
									
								
								.github/workflows/playwright.yml
									
									
									
									
										vendored
									
									
								
							| @ -35,7 +35,7 @@ jobs: | ||||
|  | ||||
|   playwright-ubuntu: | ||||
|     timeout-minutes: 30 | ||||
|     runs-on: ubuntu-latest | ||||
|     runs-on: ubuntu-latest-8-cores | ||||
|     strategy: | ||||
|       fail-fast: false | ||||
|       matrix: | ||||
| @ -112,7 +112,7 @@ jobs: | ||||
|       run: yarn build:local | ||||
|     - name: Run ubuntu/chrome snapshots | ||||
|       run: | | ||||
|         yarn playwright test --project="Google Chrome" --config=playwright.ci.config.ts --retries="3" --update-snapshots --grep=@snapshot  --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} | ||||
|         yarn playwright test --project="Google Chrome" --retries="3" --update-snapshots --grep=@snapshot  --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} | ||||
|       env: | ||||
|         CI: true | ||||
|         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
| @ -171,7 +171,7 @@ jobs: | ||||
|         if [[ ! -f "test-results/.last-run.json" ]]; then | ||||
|             # if no last run artifact, than run plawright normally | ||||
|             echo "run playwright normally" | ||||
|             yarn playwright test --project="Google Chrome" --config=playwright.ci.config.ts --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} --grep-invert="@snapshot|@electron" || true | ||||
|             yarn playwright test --project="Google Chrome" --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} --grep-invert=@snapshot || true | ||||
|             # # send to axiom | ||||
|             node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
|         fi | ||||
| @ -186,7 +186,7 @@ jobs: | ||||
|                 if [[ $failed_tests -gt 0 ]]; then | ||||
|                     echo "retried=true" >>$GITHUB_OUTPUT | ||||
|                     echo "run playwright with last failed tests and retry $retry" | ||||
|                     yarn playwright test --project="Google Chrome" --config=playwright.ci.config.ts --last-failed --grep-invert="@snapshot|@electron" || true | ||||
|                     yarn playwright test --project="Google Chrome" --last-failed --grep-invert=@snapshot || true | ||||
|                     # send to axiom | ||||
|                     node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
|                     retry=$((retry + 1)) | ||||
| @ -233,7 +233,6 @@ jobs: | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
|  | ||||
|  | ||||
|   playwright-macos: | ||||
|     timeout-minutes: 30 | ||||
|     runs-on: macos-14 | ||||
| @ -326,7 +325,7 @@ jobs: | ||||
|         if [[ ! -f "test-results/.last-run.json" ]]; then | ||||
|             # if no last run artifact, than run plawright normally | ||||
|             echo "run playwright normally" | ||||
|             yarn playwright test --project="webkit" --config=playwright.ci.config.ts --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} --grep-invert="@snapshot|@electron" || true | ||||
|             yarn playwright test --project="webkit" --shard=${{ matrix.shardIndex }}/${{ matrix.shardTotal }} --grep-invert=@snapshot || true | ||||
|             # # send to axiom | ||||
|             node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
|         fi | ||||
| @ -341,7 +340,7 @@ jobs: | ||||
|                 if [[ $failed_tests -gt 0 ]]; then | ||||
|                     echo "retried=true" >>$GITHUB_OUTPUT | ||||
|                     echo "run playwright with last failed tests and retry $retry" | ||||
|                     yarn playwright test --project="webkit" --config=playwright.ci.config.ts --last-failed --grep-invert="@snapshot|@electron" || true | ||||
|                     yarn playwright test --project="webkit" --last-failed --grep-invert=@snapshot || true | ||||
|                     # send to axiom | ||||
|                     node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
|                     retry=$((retry + 1)) | ||||
| @ -382,156 +381,3 @@ jobs: | ||||
|         path: playwright-report/ | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
|  | ||||
|   playwright-electron: | ||||
|     timeout-minutes: 30 | ||||
|     runs-on: ubuntu-latest | ||||
|     needs: check-rust-changes | ||||
|     steps: | ||||
|     - name: Tune GitHub-hosted runner network | ||||
|       uses: smorimoto/tune-github-hosted-runner-network@v1 | ||||
|     - uses: actions/checkout@v4 | ||||
|     - uses: actions/setup-node@v4 | ||||
|       with: | ||||
|         node-version-file: '.nvmrc' | ||||
|         cache: 'yarn' | ||||
|     - uses: KittyCAD/action-install-cli@main | ||||
|     - name: Install dependencies | ||||
|       run: yarn | ||||
|     - name: Cache Playwright Browsers | ||||
|       uses: actions/cache@v4 | ||||
|       with: | ||||
|         path: | | ||||
|           ~/.cache/ms-playwright/ | ||||
|         key: ${{ runner.os }}-playwright-${{ hashFiles('yarn.lock') }} | ||||
|     - name: Install Playwright Browsers | ||||
|       run: yarn playwright install chromium --with-deps | ||||
|     - name: Download Wasm Cache | ||||
|       id: download-wasm | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
|       uses: dawidd6/action-download-artifact@v6 | ||||
|       continue-on-error: true | ||||
|       with: | ||||
|         github_token: ${{secrets.GITHUB_TOKEN}} | ||||
|         name: wasm-bundle | ||||
|         workflow: build-and-store-wasm.yml | ||||
|         branch: main | ||||
|         path: src/wasm-lib/pkg | ||||
|     - name: copy wasm blob | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'false' | ||||
|       run: cp src/wasm-lib/pkg/wasm_lib_bg.wasm public | ||||
|       continue-on-error: true | ||||
|     - name: Setup Rust | ||||
|       uses: dtolnay/rust-toolchain@stable | ||||
|     - name: Cache Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
|     - name: OR Cache Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       uses: Swatinem/rust-cache@v2 | ||||
|       with: | ||||
|         workspaces: './src/wasm-lib' | ||||
|     - name: Install vector | ||||
|       run: | | ||||
|         curl --proto '=https' --tlsv1.2 -sSfL https://sh.vector.dev > /tmp/vector.sh | ||||
|         chmod +x /tmp/vector.sh | ||||
|         /tmp/vector.sh -y -no-modify-path | ||||
|         mkdir -p /tmp/vector | ||||
|         cp .github/workflows/vector.toml /tmp/vector.toml | ||||
|         sed -i "s#GITHUB_WORKFLOW#${GITHUB_WORKFLOW}#g" /tmp/vector.toml | ||||
|         sed -i "s#GITHUB_REPOSITORY#${GITHUB_REPOSITORY}#g" /tmp/vector.toml | ||||
|         sed -i "s#GITHUB_SHA#${GITHUB_SHA}#g" /tmp/vector.toml | ||||
|         sed -i "s#GITHUB_REF_NAME#${GITHUB_REF_NAME}#g" /tmp/vector.toml | ||||
|         sed -i "s#GH_ACTIONS_AXIOM_TOKEN#${{secrets.GH_ACTIONS_AXIOM_TOKEN}}#g" /tmp/vector.toml | ||||
|         cat /tmp/vector.toml | ||||
|         ${HOME}/.vector/bin/vector --config /tmp/vector.toml & | ||||
|     - name: Build Wasm (because rust diff) | ||||
|       if: needs.check-rust-changes.outputs.rust-changed == 'true' | ||||
|       run: yarn build:wasm | ||||
|     - name: OR Build Wasm (because wasm cache failed) | ||||
|       if: steps.download-wasm.outcome == 'failure' | ||||
|       run: yarn build:wasm | ||||
|     - name: build web | ||||
|       run: yarn build:local | ||||
|     - uses: actions/download-artifact@v4 | ||||
|       if: always() | ||||
|       continue-on-error: true | ||||
|       with: | ||||
|         name: test-results-ubuntu-${{ github.sha }} | ||||
|         path: test-results/ | ||||
|     - name: run electron | ||||
|       run: | | ||||
|         yarn electron:start > electron.log 2>&1 & | ||||
|         while ! grep -q "built in" electron.log; do | ||||
|           sleep 1 | ||||
|         done | ||||
|     - name: Run ubuntu/chrome flow (with retries) | ||||
|       id: retry | ||||
|       if: always() | ||||
|       run: | | ||||
|         if [[ ! -f "test-results/.last-run.json" ]]; then | ||||
|             # if no last run artifact, than run plawright normally | ||||
|             echo "run playwright normally" | ||||
|             yarn playwright test --project="Google Chrome" --grep=@electron || true | ||||
|             # # send to axiom | ||||
|             node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
|         fi | ||||
|  | ||||
|         retry=1 | ||||
|         max_retrys=4 | ||||
|  | ||||
|         # retry failed tests, doing our own retries because using inbuilt playwright retries causes connection issues | ||||
|         while [[ $retry -le $max_retrys ]]; do | ||||
|             if [[ -f "test-results/.last-run.json" ]]; then | ||||
|                 failed_tests=$(jq '.failedTests | length' test-results/.last-run.json) | ||||
|                 if [[ $failed_tests -gt 0 ]]; then | ||||
|                     echo "retried=true" >>$GITHUB_OUTPUT | ||||
|                     echo "run playwright with last failed tests and retry $retry" | ||||
|                     yarn playwright test --project="Google Chrome" --last-failed --grep=@electron || true | ||||
|                     # send to axiom | ||||
|                     node playwrightProcess.mjs | tee /tmp/github-actions.log > /dev/null 2>&1 | ||||
|                     retry=$((retry + 1)) | ||||
|                 else | ||||
|                     echo "retried=false" >>$GITHUB_OUTPUT | ||||
|                     exit 0 | ||||
|                 fi | ||||
|             else | ||||
|                 echo "retried=false" >>$GITHUB_OUTPUT | ||||
|                 exit 0 | ||||
|             fi | ||||
|         done | ||||
|  | ||||
|         echo "retried=false" >>$GITHUB_OUTPUT | ||||
|  | ||||
|         if [[ -f "test-results/.last-run.json" ]]; then | ||||
|             failed_tests=$(jq '.failedTests | length' test-results/.last-run.json) | ||||
|             if [[ $failed_tests -gt 0 ]]; then | ||||
|                 # if it still fails after 3 retrys, then fail the job | ||||
|                 exit 1 | ||||
|             fi | ||||
|         fi | ||||
|         exit 0 | ||||
|       env: | ||||
|         CI: true | ||||
|         token: ${{ secrets.KITTYCAD_API_TOKEN_DEV }} | ||||
|     - name: send to axiom | ||||
|       if: always() | ||||
|       shell: bash | ||||
|       run: | | ||||
|         node playwrightProcess.mjs | tee /tmp/github-actions.log | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: always() | ||||
|       with: | ||||
|         name: test-results-electron-${{ github.sha }} | ||||
|         path: test-results/ | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
|     - uses: actions/upload-artifact@v4 | ||||
|       if: always() | ||||
|       with: | ||||
|         name: playwright-report-electron-${{ github.sha }} | ||||
|         path: playwright-report/ | ||||
|         retention-days: 30 | ||||
|         overwrite: true | ||||
							
								
								
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										7
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @ -62,10 +62,3 @@ Mac_App_Distribution.provisionprofile | ||||
| *.tsbuildinfo | ||||
|  | ||||
| venv | ||||
| .vite/ | ||||
|  | ||||
| # electron | ||||
| out/ | ||||
|  | ||||
| src-tauri/target | ||||
| electron-test-projects-dir | ||||
							
								
								
									
										19
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										19
									
								
								README.md
									
									
									
									
									
								
							| @ -89,19 +89,26 @@ enable third-party cookies. You can enable third-party cookies by clicking on | ||||
| the eye with a slash through it in the URL bar, and clicking on "Enable | ||||
| Third-Party Cookies". | ||||
|  | ||||
| ## Desktop | ||||
| ## Tauri | ||||
|  | ||||
| To spin up the desktop app, `yarn install` and `yarn build:wasm-dev` need to have been done before hand then | ||||
| To spin up up tauri dev, `yarn install` and `yarn build:wasm-dev` need to have been done before hand then | ||||
|  | ||||
| ``` | ||||
| yarn electron:start | ||||
| yarn tauri dev | ||||
| ``` | ||||
|  | ||||
| This will start the application and hot-reload on changed. | ||||
| Will spin up the web app before opening up the tauri dev desktop app. Note that it's probably a good idea to close the browser tab that gets opened since at the time of writing they can conflict. | ||||
|  | ||||
| Devtools can be opened with the usual Cmd/Ctrl-Shift-I. | ||||
| The dev instance automatically opens up the browser devtools which can be disabled by [commenting it out](https://github.com/KittyCAD/modeling-app/blob/main/src-tauri/src/main.rs#L92.) | ||||
|  | ||||
| To build, run `yarn electron:package`. | ||||
| To build, run `yarn tauri build`, or `yarn tauri build --debug` to keep access to the devtools. | ||||
|  | ||||
| Note that these became separate apps on Macos, so make sure you open the right one after a build 😉 | ||||
|  | ||||
|  | ||||
| <img width="1232" alt="image" src="https://user-images.githubusercontent.com/29681384/211947063-46164bb4-7bdd-45cb-9a76-2f40c71a24aa.png"> | ||||
|  | ||||
| <img width="1232" alt="image (1)" src="https://user-images.githubusercontent.com/29681384/211947073-e76b4933-bef5-4636-bc4d-e930ac8e290f.png"> | ||||
|  | ||||
| ## Checking out commits / Bisecting | ||||
|  | ||||
|  | ||||
| @ -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 | ||||
| @ -25,5 +25,5 @@ once fixed in engine will just start working here with no language changes. | ||||
|  | ||||
|     Sketching on the chamfered face does not currently work. | ||||
|  | ||||
| - **Shell**: Shell sometimes does not work when arcs or fillets are involved. | ||||
|     We are tracking the engine side bug on this. | ||||
| - **Shell**: Shell is only working for `end` faces, not for `side` or `start`  | ||||
|     faces. We are tracking the engine side bug on this. | ||||
|  | ||||
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
							
								
								
									
										1352
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
							
						
						
									
										1352
									
								
								docs/kcl/std.json
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							
										
											
												File diff suppressed because one or more lines are too long
											
										
									
								
							| @ -98,16 +98,14 @@ const extrude001 = extrude(-10, sketch001)` | ||||
|  | ||||
|     const commandBarButton = page.getByRole('button', { name: 'Commands' }) | ||||
|     const cmdSearchBar = page.getByPlaceholder('Search commands') | ||||
|     const commandName = 'debug panel' | ||||
|     const commandOption = page.getByRole('option', { | ||||
|       name: commandName, | ||||
|     const themeOption = page.getByRole('option', { | ||||
|       name: 'theme', | ||||
|       exact: false, | ||||
|     }) | ||||
|     const commandLevelArgButton = page.getByRole('button', { name: 'level' }) | ||||
|     const commandThemeArgButton = page.getByRole('button', { name: 'value' }) | ||||
|     const paneSelector = page.getByRole('button', { name: 'debug panel' }) | ||||
|     // This selector changes after we set the setting | ||||
|     let commandOptionInput = page.getByPlaceholder('On') | ||||
|     let commandOptionInput = page.getByPlaceholder('Select an option') | ||||
|  | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'Start Sketch' }) | ||||
| @ -129,16 +127,17 @@ const extrude001 = extrude(-10, sketch001)` | ||||
|     await expect(cmdSearchBar).toBeFocused() | ||||
|  | ||||
|     // Try typing in the command bar | ||||
|     await cmdSearchBar.fill(commandName) | ||||
|     await expect(commandOption).toBeVisible() | ||||
|     await commandOption.click() | ||||
|     const toggleInput = page.getByPlaceholder('On') | ||||
|     await expect(toggleInput).toBeVisible() | ||||
|     await expect(toggleInput).toBeFocused() | ||||
|     // Select On | ||||
|     await cmdSearchBar.fill('theme') | ||||
|     await expect(themeOption).toBeVisible() | ||||
|     await themeOption.click() | ||||
|     const themeInput = page.getByPlaceholder('Select an option') | ||||
|     await expect(themeInput).toBeVisible() | ||||
|     await expect(themeInput).toBeFocused() | ||||
|     // Select dark theme | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await expect(page.getByRole('option', { name: 'Off' })).toHaveAttribute( | ||||
|     await page.keyboard.press('ArrowDown') | ||||
|     await expect(page.getByRole('option', { name: 'system' })).toHaveAttribute( | ||||
|       'data-headlessui-state', | ||||
|       'active' | ||||
|     ) | ||||
| @ -146,21 +145,21 @@ const extrude001 = extrude(-10, sketch001)` | ||||
|  | ||||
|     // Check the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set show debug panel to "false" for this project`) | ||||
|       page.getByText(`Set theme to "system" for this project`) | ||||
|     ).toBeVisible() | ||||
|     // Check that the visibility changed | ||||
|     await expect(paneSelector).not.toBeVisible() | ||||
|     // Check that the theme changed | ||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||
|  | ||||
|     commandOptionInput = page.getByPlaceholder('off') | ||||
|     commandOptionInput = page.getByPlaceholder('system') | ||||
|  | ||||
|     // Test case for https://github.com/KittyCAD/modeling-app/issues/2882 | ||||
|     await commandBarButton.click() | ||||
|     await cmdSearchBar.focus() | ||||
|     await cmdSearchBar.fill(commandName) | ||||
|     await commandOption.click() | ||||
|     await cmdSearchBar.fill('theme') | ||||
|     await themeOption.click() | ||||
|     await expect(commandThemeArgButton).toBeDisabled() | ||||
|     await commandOptionInput.focus() | ||||
|     await commandOptionInput.fill('on') | ||||
|     await commandOptionInput.fill('lig') | ||||
|     await commandLevelArgButton.click() | ||||
|     await expect(commandLevelArgButton).toBeDisabled() | ||||
|  | ||||
| @ -198,7 +197,7 @@ const extrude001 = extrude(-10, sketch001)` | ||||
|     }) | ||||
|     await expect(themeOption).toBeVisible() | ||||
|     await themeOption.click() | ||||
|     const themeInput = page.getByPlaceholder('dark') | ||||
|     const themeInput = page.getByPlaceholder('Select an option') | ||||
|     await expect(themeInput).toBeVisible() | ||||
|     await expect(themeInput).toBeFocused() | ||||
|     // Select dark theme | ||||
| @ -213,7 +212,7 @@ const extrude001 = extrude(-10, sketch001)` | ||||
|  | ||||
|     // Check the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set theme to "system" as a user default`) | ||||
|       page.getByText(`Set theme to "system" for this project`) | ||||
|     ).toBeVisible() | ||||
|     // Check that the theme changed | ||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||
|  | ||||
| @ -1,48 +0,0 @@ | ||||
| import test, { _electron } from '@playwright/test' | ||||
| import { TEST_SETTINGS_KEY } from './storageStates' | ||||
| import { _electron as electron } from '@playwright/test' | ||||
| import * as TOML from '@iarna/toml' | ||||
| import fs from 'node:fs' | ||||
| import { secrets } from './secrets' | ||||
|  | ||||
| test('Electron setup', { tag: '@electron' }, async () => { | ||||
|   // create or otherwise clear the folder ./electron-test-projects-dir | ||||
|   const fileName = './electron-test-projects-dir' | ||||
|   try { | ||||
|     fs.rmSync(fileName, { recursive: true }) | ||||
|   } catch (e) { | ||||
|     console.error(e) | ||||
|   } | ||||
|  | ||||
|   fs.mkdirSync(fileName) | ||||
|  | ||||
|   // get full path for ./electron-test-projects-dir | ||||
|   const fullPath = fs.realpathSync(fileName) | ||||
|  | ||||
|   const electronApp = await electron.launch({ | ||||
|     args: ['.'], | ||||
|   }) | ||||
|  | ||||
|   const page = await electronApp.firstWindow() | ||||
|  | ||||
|   // Set local storage directly using evaluate | ||||
|   await page.evaluate( | ||||
|     (token) => localStorage.setItem('TOKEN_PERSIST_KEY', token), | ||||
|     secrets.token | ||||
|   ) | ||||
|  | ||||
|   // Override settings with electron temporary project directory | ||||
|   await page.addInitScript( | ||||
|     async ({ settingsKey, settings }) => { | ||||
|       localStorage.setItem(settingsKey, settings) | ||||
|     }, | ||||
|     { | ||||
|       settingsKey: TEST_SETTINGS_KEY, | ||||
|       settings: TOML.stringify({ | ||||
|         settings: { | ||||
|           app: { projectDirectory: fullPath }, | ||||
|         }, | ||||
|       }), | ||||
|     } | ||||
|   ) | ||||
| }) | ||||
| @ -1,591 +0,0 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
| import { getUtils, setupElectron, tearDown } from './test-utils' | ||||
| import fsp from 'fs/promises' | ||||
|  | ||||
| test.afterEach(async ({ page }, testInfo) => { | ||||
|   await tearDown(page, testInfo) | ||||
| }) | ||||
|  | ||||
| test( | ||||
|   'Rename and delete projects, also spam arrow keys when renaming', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         await fsp.mkdir(`${dir}/router-template-slate`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/router-template-slate.kcl', | ||||
|           `${dir}/router-template-slate/main.kcl` | ||||
|         ) | ||||
|  | ||||
|         await fsp.mkdir(`${dir}/bracket`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl', | ||||
|           `${dir}/bracket/main.kcl` | ||||
|         ) | ||||
|  | ||||
|         await fsp.mkdir(`${dir}/lego`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/lego.kcl', | ||||
|           `${dir}/lego/main.kcl` | ||||
|         ) | ||||
|       }, | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     await page.waitForTimeout(1_000) | ||||
|  | ||||
|     await test.step('rename a project clicking buttons checking left and right arrow does not impact the text', async () => { | ||||
|       const routerTemplate = page.getByText('router-template-slate') | ||||
|  | ||||
|       await routerTemplate.hover() | ||||
|       await routerTemplate.focus() | ||||
|  | ||||
|       await expect(page.getByLabel('sketch').last()).toBeVisible() | ||||
|       await page.getByLabel('sketch').last().click() | ||||
|  | ||||
|       const selectedText = await page.evaluate(() => { | ||||
|         const selection = window.getSelection() | ||||
|         return selection ? selection.toString() : '' | ||||
|       }) | ||||
|  | ||||
|       expect(selectedText).toBe('router-template-slate') | ||||
|  | ||||
|       // type "updated project name" | ||||
|       await page.keyboard.press('Backspace') | ||||
|       await page.keyboard.type('updated project name') | ||||
|  | ||||
|       for (let i = 0; i < 10; i++) { | ||||
|         await page.keyboard.press('ArrowRight') | ||||
|       } | ||||
|       for (let i = 0; i < 30; i++) { | ||||
|         await page.keyboard.press('ArrowLeft') | ||||
|       } | ||||
|  | ||||
|       await page.getByLabel('checkmark').last().click() | ||||
|  | ||||
|       await expect(page.getByText('Successfully renamed')).toBeVisible() | ||||
|       await expect(page.getByText('Successfully renamed')).not.toBeVisible() | ||||
|       await expect(page.getByText('updated project name')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('update a project by hitting enter', async () => { | ||||
|       const project = page.getByText('updated project name') | ||||
|  | ||||
|       await project.hover() | ||||
|       await project.focus() | ||||
|  | ||||
|       await expect(page.getByLabel('sketch').last()).toBeVisible() | ||||
|       await page.getByLabel('sketch').last().click() | ||||
|  | ||||
|       const selectedText = await page.evaluate(() => { | ||||
|         const selection = window.getSelection() | ||||
|         return selection ? selection.toString() : '' | ||||
|       }) | ||||
|  | ||||
|       expect(selectedText).toBe('updated project name') | ||||
|  | ||||
|       // type "updated project name" | ||||
|       await page.keyboard.press('Backspace') | ||||
|       await page.keyboard.type('updated name again') | ||||
|  | ||||
|       await page.keyboard.press('Enter') | ||||
|  | ||||
|       await expect(page.getByText('Successfully renamed')).toBeVisible() | ||||
|       await expect(page.getByText('Successfully renamed')).not.toBeVisible() | ||||
|  | ||||
|       await expect(page.getByText('updated name again')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Cancel and edit by clicking the x button', async () => { | ||||
|       const project = page.getByText('updated name again') | ||||
|  | ||||
|       await project.hover() | ||||
|       await project.focus() | ||||
|  | ||||
|       await expect(page.getByLabel('sketch').last()).toBeVisible() | ||||
|       await page.getByLabel('sketch').last().click() | ||||
|  | ||||
|       const selectedText = await page.evaluate(() => { | ||||
|         const selection = window.getSelection() | ||||
|         return selection ? selection.toString() : '' | ||||
|       }) | ||||
|  | ||||
|       expect(selectedText).toBe('updated name again') | ||||
|  | ||||
|       await page.keyboard.press('Backspace') | ||||
|       await page.keyboard.type('dismiss this text') | ||||
|  | ||||
|       await page.getByLabel('close').last().click() | ||||
|  | ||||
|       await expect(page.getByText('updated name again')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Cancel and edit by pressing esc', async () => { | ||||
|       const project = page.getByText('updated name again') | ||||
|  | ||||
|       await project.hover() | ||||
|       await project.focus() | ||||
|  | ||||
|       await expect(page.getByLabel('sketch').last()).toBeVisible() | ||||
|       await page.getByLabel('sketch').last().click() | ||||
|  | ||||
|       const selectedText = await page.evaluate(() => { | ||||
|         const selection = window.getSelection() | ||||
|         return selection ? selection.toString() : '' | ||||
|       }) | ||||
|  | ||||
|       expect(selectedText).toBe('updated name again') | ||||
|  | ||||
|       await page.keyboard.press('Backspace') | ||||
|       await page.keyboard.type('dismiss this text') | ||||
|  | ||||
|       await page.keyboard.press('Escape') | ||||
|  | ||||
|       await expect(page.getByText('updated name again')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('delete a project by clicking the trash button', async () => { | ||||
|       const project = page.getByText('updated name again') | ||||
|  | ||||
|       await project.hover() | ||||
|       await project.focus() | ||||
|  | ||||
|       await expect(page.getByLabel('trash').last()).toBeVisible() | ||||
|       await page.getByLabel('trash').last().click() | ||||
|  | ||||
|       await expect(page.getByText('This will permanently delete')).toBeVisible() | ||||
|  | ||||
|       await page.getByTestId('delete-confirmation').click() | ||||
|  | ||||
|       await expect(page.getByText('Successfully deleted')).toBeVisible() | ||||
|       await expect(page.getByText('Successfully deleted')).not.toBeVisible() | ||||
|  | ||||
|       await expect(page.getByText('updated name again')).not.toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('rename a project to an empty string should make the field complain', async () => { | ||||
|       const routerTemplate = page.getByText('bracket') | ||||
|  | ||||
|       await routerTemplate.hover() | ||||
|       await routerTemplate.focus() | ||||
|  | ||||
|       await expect(page.getByLabel('sketch').last()).toBeVisible() | ||||
|       await page.getByLabel('sketch').last().click() | ||||
|  | ||||
|       const selectedText = await page.evaluate(() => { | ||||
|         const selection = window.getSelection() | ||||
|         return selection ? selection.toString() : '' | ||||
|       }) | ||||
|  | ||||
|       expect(selectedText).toBe('bracket') | ||||
|  | ||||
|       // type "updated project name" | ||||
|       await page.keyboard.press('Backspace') | ||||
|  | ||||
|       await page.keyboard.press('Enter') | ||||
|       await page.waitForTimeout(100) | ||||
|       await page.keyboard.press('Enter') | ||||
|       await page.waitForTimeout(100) | ||||
|       await page.keyboard.press('Escape') | ||||
|  | ||||
|       // expect the name not to have changed | ||||
|       await expect(page.getByText('bracket')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| test( | ||||
|   'pressing "delete" on home screen should do nothing', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         await fsp.mkdir(`${dir}/router-template-slate`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/router-template-slate.kcl', | ||||
|           `${dir}/router-template-slate/main.kcl` | ||||
|         ) | ||||
|       }, | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     await expect(page.getByText('router-template-slate')).toBeVisible() | ||||
|     await expect(page.getByText('Your Projects')).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Delete') | ||||
|     await page.waitForTimeout(200) | ||||
|     await page.keyboard.press('Delete') | ||||
|  | ||||
|     // expect to still be on the home page | ||||
|     await expect(page.getByText('router-template-slate')).toBeVisible() | ||||
|     await expect(page.getByText('Your Projects')).toBeVisible() | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
| test.fixme( | ||||
|   'File in the file pane should open with a single click', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         await fsp.mkdir(`${dir}/router-template-slate`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/router-template-slate.kcl', | ||||
|           `${dir}/router-template-slate/main.kcl` | ||||
|         ) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl', | ||||
|           `${dir}/router-template-slate/otherThingToClickOn.kcl` | ||||
|         ) | ||||
|       }, | ||||
|     }) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     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(u.codeLocator).toContainText('routerDiameter') | ||||
|     await expect(u.codeLocator).toContainText('templateGap') | ||||
|     await expect(u.codeLocator).toContainText('minClampingDistance') | ||||
|  | ||||
|     await page.getByRole('button', { name: 'Project Files' }).click() | ||||
|  | ||||
|     const file = page.getByRole('button', { name: 'otherThingToClickOn.kcl' }) | ||||
|     await expect(file).toBeVisible() | ||||
|  | ||||
|     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' | ||||
|     ) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
| test( | ||||
|   'Can sort projects on home page', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         await fsp.mkdir(`${dir}/router-template-slate`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/router-template-slate.kcl', | ||||
|           `${dir}/router-template-slate/main.kcl` | ||||
|         ) | ||||
|  | ||||
|         // wait more than a second so the timestamp is different | ||||
|         await new Promise((r) => setTimeout(r, 1_200)) | ||||
|         await fsp.mkdir(`${dir}/bracket`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl', | ||||
|           `${dir}/bracket/main.kcl` | ||||
|         ) | ||||
|  | ||||
|         // wait more than a second so the timestamp is different | ||||
|         await new Promise((r) => setTimeout(r, 1_200)) | ||||
|         await fsp.mkdir(`${dir}/lego`, { recursive: true }) | ||||
|         await fsp.copyFile( | ||||
|           'src/wasm-lib/tests/executor/inputs/lego.kcl', | ||||
|           `${dir}/lego/main.kcl` | ||||
|         ) | ||||
|       }, | ||||
|     }) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     const getAllProjects = () => page.getByTestId('project-link').all() | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     await test.step('should be shorted by modified initially', async () => { | ||||
|       const lastModifiedButton = page.getByRole('button', { | ||||
|         name: 'Last Modified', | ||||
|       }) | ||||
|       await expect(lastModifiedButton).toBeVisible() | ||||
|       await expect(lastModifiedButton.getByLabel('arrow down')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     const projectNamesOrderedByModified = [ | ||||
|       'lego', | ||||
|       'bracket', | ||||
|       'router-template-slate', | ||||
|     ] | ||||
|     await test.step('Check the order of the projects is correct', async () => { | ||||
|       for (const [index, projectLink] of (await getAllProjects()).entries()) { | ||||
|         await expect(projectLink).toContainText( | ||||
|           projectNamesOrderedByModified[index] | ||||
|         ) | ||||
|       } | ||||
|     }) | ||||
|  | ||||
|     await test.step('Reverse modified order', async () => { | ||||
|       const lastModifiedButton = page.getByRole('button', { | ||||
|         name: 'Last Modified', | ||||
|       }) | ||||
|       await lastModifiedButton.click() | ||||
|       await expect(lastModifiedButton).toBeVisible() | ||||
|       await expect(lastModifiedButton.getByLabel('arrow up')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Check the order of the projects is has reversed', async () => { | ||||
|       for (const [index, projectLink] of (await getAllProjects()).entries()) { | ||||
|         await expect(projectLink).toContainText( | ||||
|           [...projectNamesOrderedByModified].reverse()[index] | ||||
|         ) | ||||
|       } | ||||
|     }) | ||||
|  | ||||
|     await test.step('Change order to by name', async () => { | ||||
|       const nameButton = page.getByRole('button', { | ||||
|         name: 'Name', | ||||
|       }) | ||||
|       await nameButton.click() | ||||
|       await expect(nameButton).toBeVisible() | ||||
|       await expect(nameButton.getByLabel('arrow down')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     const projectNamesOrderedByName = [ | ||||
|       'bracket', | ||||
|       'lego', | ||||
|       'router-template-slate', | ||||
|     ] | ||||
|     await test.step('Check the order of the projects is by name', async () => { | ||||
|       for (const [index, projectLink] of (await getAllProjects()).entries()) { | ||||
|         await expect(projectLink).toContainText( | ||||
|           projectNamesOrderedByName[index] | ||||
|         ) | ||||
|       } | ||||
|     }) | ||||
|  | ||||
|     await test.step('Reverse name order', async () => { | ||||
|       const nameButton = page.getByRole('button', { | ||||
|         name: 'Name', | ||||
|       }) | ||||
|       await nameButton.click() | ||||
|       await expect(nameButton).toBeVisible() | ||||
|       await expect(nameButton.getByLabel('arrow up')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Check the order of the projects is by name reversed', async () => { | ||||
|       for (const [index, projectLink] of (await getAllProjects()).entries()) { | ||||
|         await expect(projectLink).toContainText( | ||||
|           [...projectNamesOrderedByName].reverse()[index] | ||||
|         ) | ||||
|       } | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| test( | ||||
|   'When the project folder is empty, user can create new project and open it.', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ testInfo }) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     // expect to see text "No Projects found" | ||||
|     await expect(page.getByText('No Projects found')).toBeVisible() | ||||
|  | ||||
|     await page.getByRole('button', { name: 'New project' }).click() | ||||
|  | ||||
|     await expect(page.getByText('Successfully created')).toBeVisible() | ||||
|     await expect(page.getByText('Successfully created')).not.toBeVisible() | ||||
|  | ||||
|     await expect(page.getByText('project-000')).toBeVisible() | ||||
|  | ||||
|     await page.getByText('project-000').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 page.locator('.cm-content') | ||||
|       .fill(`const sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([-87.4, 282.92], %) | ||||
|   |> line([324.07, 27.199], %, $seg01) | ||||
|   |> line([118.328, -291.754], %) | ||||
|   |> line([-180.04, -202.08], %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %) | ||||
|   |> close(%) | ||||
| const extrude001 = extrude(200, sketch001)`) | ||||
|  | ||||
|     const pointOnModel = { x: 660, y: 250 } | ||||
|  | ||||
|     // 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]), { | ||||
|         timeout: 10_000, | ||||
|       }) | ||||
|       .toBeLessThan(10) | ||||
|  | ||||
|     await expect(async () => { | ||||
|       await page.mouse.move(0, 0, { steps: 5 }) | ||||
|       await page.mouse.move(pointOnModel.x, pointOnModel.y, { steps: 5 }) | ||||
|       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) | ||||
|     }).toPass({ timeout: 40_000, intervals: [1_000] }) | ||||
|  | ||||
|     await page.getByTestId('app-logo').click() | ||||
|  | ||||
|     await expect( | ||||
|       page.getByRole('button', { name: 'New project' }) | ||||
|     ).toBeVisible() | ||||
|  | ||||
|     const createProject = async (projectNum: number) => { | ||||
|       await page.getByRole('button', { name: 'New project' }).click() | ||||
|       await expect(page.getByText('Successfully created')).toBeVisible() | ||||
|       await expect(page.getByText('Successfully created')).not.toBeVisible() | ||||
|  | ||||
|       const projectNumStr = projectNum.toString().padStart(3, '0') | ||||
|       await expect(page.getByText(`project-${projectNumStr}`)).toBeVisible() | ||||
|     } | ||||
|     for (let i = 1; i <= 10; i++) { | ||||
|       await createProject(i) | ||||
|     } | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
|  | ||||
| test( | ||||
|   'Check you can go home with two different methods, and that switching between projects does not harm the stream', | ||||
|   { tag: '@electron' }, | ||||
|   async ({ browserName }, testInfo) => { | ||||
|     const { electronApp, page } = await setupElectron({ | ||||
|       testInfo, | ||||
|       folderSetupFn: async (dir) => { | ||||
|         await Promise.all([ | ||||
|           fsp.mkdir(`${dir}/router-template-slate`, { recursive: true }), | ||||
|           fsp.mkdir(`${dir}/bracket`, { recursive: true }), | ||||
|         ]) | ||||
|         await Promise.all([ | ||||
|           fsp.copyFile( | ||||
|             'src/wasm-lib/tests/executor/inputs/router-template-slate.kcl', | ||||
|             `${dir}/router-template-slate/main.kcl` | ||||
|           ), | ||||
|           fsp.copyFile( | ||||
|             'src/wasm-lib/tests/executor/inputs/focusrite_scarlett_mounting_braket.kcl', | ||||
|             `${dir}/bracket/main.kcl` | ||||
|           ), | ||||
|         ]) | ||||
|       }, | ||||
|     }) | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|  | ||||
|     page.on('console', console.log) | ||||
|  | ||||
|     const pointOnModel = { x: 630, y: 280 } | ||||
|  | ||||
|     await test.step('Opening the bracket project should load the stream', async () => { | ||||
|       // expect to see the text bracket | ||||
|       await expect(page.getByText('bracket')).toBeVisible() | ||||
|  | ||||
|       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, | ||||
|       }) | ||||
|  | ||||
|       // 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]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(10) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Clicking the logo takes us back to the projects page / home', async () => { | ||||
|       await page.getByTestId('app-logo').click() | ||||
|  | ||||
|       await expect(page.getByText('bracket')).toBeVisible() | ||||
|       await expect(page.getByText('router-template-slate')).toBeVisible() | ||||
|       await expect(page.getByText('New Project')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Opening the router-template project should load the stream', async () => { | ||||
|       // expect to see the text bracket | ||||
|       await expect(page.getByText('router-template-slate')).toBeVisible() | ||||
|  | ||||
|       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, | ||||
|       }) | ||||
|  | ||||
|       // 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]), { | ||||
|           timeout: 10_000, | ||||
|         }) | ||||
|         .toBeLessThan(10) | ||||
|     }) | ||||
|  | ||||
|     await test.step('Opening the router-template project should load the stream', async () => { | ||||
|       await page.getByTestId('project-sidebar-toggle').click() | ||||
|       await expect( | ||||
|         page.getByRole('button', { name: 'Go to Home' }) | ||||
|       ).toBeVisible() | ||||
|       await page.getByRole('button', { name: 'Go to Home' }).click() | ||||
|  | ||||
|       await expect(page.getByText('bracket')).toBeVisible() | ||||
|       await expect(page.getByText('router-template-slate')).toBeVisible() | ||||
|       await expect(page.getByText('New Project')).toBeVisible() | ||||
|     }) | ||||
|  | ||||
|     await electronApp.close() | ||||
|   } | ||||
| ) | ||||
| @ -1,8 +1,6 @@ | ||||
| import { test, expect } from '@playwright/test' | ||||
|  | ||||
| import { getUtils, setup, tearDown } from './test-utils' | ||||
| import { TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR } from './storageStates' | ||||
| import { bracket } from 'lib/exampleKcl' | ||||
|  | ||||
| test.beforeEach(async ({ context, page }) => { | ||||
|   await setup(context, page) | ||||
| @ -225,210 +223,4 @@ const sketch001 = startSketchAt([-0, -0]) | ||||
|  | ||||
|     await expect(page.locator('.cm-lint-marker-error')).toBeVisible() | ||||
|   }) | ||||
|   test('when engine fails export we handle the failure and alert the user', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async (code) => { | ||||
|       localStorage.setItem('persistCode', code) | ||||
|     }, TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     // expect zero errors in guter | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     // export the model | ||||
|     const exportButton = page.getByTestId('export-pane-button') | ||||
|     await expect(exportButton).toBeVisible() | ||||
|  | ||||
|     // Click the export button | ||||
|     exportButton.click() | ||||
|  | ||||
|     // Click the stl. | ||||
|     const stlOption = page.getByText('glTF') | ||||
|     await expect(stlOption).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Click the checkbox | ||||
|     const submitButton = page.getByText('Confirm Export') | ||||
|     await expect(submitButton).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Find the toast. | ||||
|     // Look out for the toast message | ||||
|     const exportingToastMessage = page.getByText(`Exporting...`) | ||||
|     await expect(exportingToastMessage).toBeVisible() | ||||
|  | ||||
|     const errorToastMessage = page.getByText(`Error while exporting`) | ||||
|     await expect(errorToastMessage).toBeVisible() | ||||
|  | ||||
|     const engineErrorToastMessage = page.getByText(`Nothing to export`) | ||||
|     await expect(engineErrorToastMessage).toBeVisible() | ||||
|  | ||||
|     // Make sure the exporting toast is gone | ||||
|     await expect(exportingToastMessage).not.toBeVisible() | ||||
|  | ||||
|     // Click the code editor | ||||
|     page.locator('.cm-content').click() | ||||
|  | ||||
|     await page.waitForTimeout(2000) | ||||
|  | ||||
|     // Expect the toast to be gone | ||||
|     await expect(errorToastMessage).not.toBeVisible() | ||||
|     await expect(engineErrorToastMessage).not.toBeVisible() | ||||
|  | ||||
|     // Now add in code that works. | ||||
|     page.locator('.cm-content').fill(bracket) | ||||
|     page.locator('.cm-content').click() | ||||
|     await page.keyboard.press('End') | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
|     await u.clearCommandLogs() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     // Now try exporting | ||||
|  | ||||
|     // Click the export button | ||||
|     exportButton.click() | ||||
|  | ||||
|     // Click the stl. | ||||
|     await expect(stlOption).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Click the checkbox | ||||
|     await expect(submitButton).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Find the toast. | ||||
|     // Look out for the toast message | ||||
|     await expect(exportingToastMessage).toBeVisible() | ||||
|  | ||||
|     // Expect it to succeed. | ||||
|     await expect(exportingToastMessage).not.toBeVisible() | ||||
|     await expect(errorToastMessage).not.toBeVisible() | ||||
|     await expect(engineErrorToastMessage).not.toBeVisible() | ||||
|  | ||||
|     const successToastMessage = page.getByText(`Exported successfully`) | ||||
|     await expect(successToastMessage).toBeVisible() | ||||
|   }) | ||||
|   test('ensure you can not export while an export is already going', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.addInitScript(async (code) => { | ||||
|       localStorage.setItem('persistCode', code) | ||||
|     }, bracket) | ||||
|  | ||||
|     await page.setViewportSize({ width: 1000, height: 500 }) | ||||
|  | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|  | ||||
|     // wait for execution done | ||||
|     await u.openDebugPanel() | ||||
|     await u.expectCmdLog('[data-message-type="execution-done"]') | ||||
|     await u.closeDebugPanel() | ||||
|  | ||||
|     // expect zero errors in guter | ||||
|     await expect(page.locator('.cm-lint-marker-error')).not.toBeVisible() | ||||
|  | ||||
|     // export the model | ||||
|     const exportButton = page.getByTestId('export-pane-button') | ||||
|     await expect(exportButton).toBeVisible() | ||||
|  | ||||
|     // Click the export button | ||||
|     exportButton.click() | ||||
|  | ||||
|     // Click the stl. | ||||
|     const stlOption = page.getByText('glTF') | ||||
|     await expect(stlOption).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Click the checkbox | ||||
|     const submitButton = page.getByText('Confirm Export') | ||||
|     await expect(submitButton).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Find the toast. | ||||
|     // Look out for the toast message | ||||
|     const exportingToastMessage = page.getByText(`Exporting...`) | ||||
|     await expect(exportingToastMessage).toBeVisible() | ||||
|  | ||||
|     const errorToastMessage = page.getByText(`Error while exporting`) | ||||
|  | ||||
|     const engineErrorToastMessage = page.getByText(`Nothing to export`) | ||||
|  | ||||
|     const alreadyExportingToastMessage = page.getByText(`Already exporting`) | ||||
|  | ||||
|     // Try exporting again. | ||||
|     // Click the export button | ||||
|     exportButton.click() | ||||
|  | ||||
|     // Click the stl. | ||||
|     await expect(stlOption).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Click the checkbox | ||||
|     await expect(submitButton).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Find the toast. | ||||
|     // Look out for the toast message | ||||
|     await expect(exportingToastMessage).toBeVisible() | ||||
|     await expect(alreadyExportingToastMessage).toBeVisible() | ||||
|  | ||||
|     // Expect it to succeed. | ||||
|     await expect(exportingToastMessage).not.toBeVisible() | ||||
|     await expect(errorToastMessage).not.toBeVisible() | ||||
|     await expect(engineErrorToastMessage).not.toBeVisible() | ||||
|  | ||||
|     const successToastMessage = page.getByText(`Exported successfully`) | ||||
|     await expect(successToastMessage).toBeVisible() | ||||
|  | ||||
|     await expect(alreadyExportingToastMessage).not.toBeVisible() | ||||
|  | ||||
|     // Try exporting again. | ||||
|     // Click the export button | ||||
|     exportButton.click() | ||||
|  | ||||
|     // Click the stl. | ||||
|     await expect(stlOption).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Click the checkbox | ||||
|     await expect(submitButton).toBeVisible() | ||||
|  | ||||
|     await page.keyboard.press('Enter') | ||||
|  | ||||
|     // Find the toast. | ||||
|     // Look out for the toast message | ||||
|     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 expect(successToastMessage).toBeVisible() | ||||
|   }) | ||||
| }) | ||||
|  | ||||
| @ -151,12 +151,11 @@ test.describe('Sketch tests', () => { | ||||
|  | ||||
|     await page.mouse.click(700, 200) | ||||
|  | ||||
|     await expect.poll(u.normalisedEditorCode) | ||||
|       .toBe(`const sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([12.34, -12.34], %) | ||||
|   |> line([-12.34, 12.34], %) | ||||
|  | ||||
| `) | ||||
|     await expect(page.locator('.cm-content')).toHaveText( | ||||
|       `const sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([4.61, -14.01], %) | ||||
|   |> line([0.31, 16.47], %)` | ||||
|     ) | ||||
|   }) | ||||
|   test('Can exit selection of face', async ({ page }) => { | ||||
|     // Load the app with the code panes | ||||
|  | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 49 KiB After Width: | Height: | Size: 49 KiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 59 KiB After Width: | Height: | Size: 59 KiB | 
| @ -368,5 +368,3 @@ startSketchOn(keychain, 'end') | ||||
|        height - (keychainHoleSize + 1.5) | ||||
|      ], keychainHoleSize, %) | ||||
|   |> extrude(-thickness, %)` | ||||
|  | ||||
| export const TEST_CODE_TRIGGER_ENGINE_EXPORT_ERROR = `const thing = 1` | ||||
|  | ||||
| @ -204,24 +204,19 @@ test.describe('Test network and connection issues', () => { | ||||
|  | ||||
|     // Ensure we can continue sketching | ||||
|     await page.mouse.click(startXPx + PUR * 20, 500 - PUR * 20) | ||||
|     await expect.poll(u.normalisedEditorCode) | ||||
|       .toBe(`const sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([12.34, -12.34], %) | ||||
|   |> line([12.34, 0], %) | ||||
|   |> line([-12.34, 12.34], %) | ||||
|  | ||||
| `) | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`const sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %) | ||||
|     |> line([${commonPoints.num1}, 0], %) | ||||
|     |> line([-8.84, 8.75], %)`) | ||||
|     await page.waitForTimeout(100) | ||||
|     await page.mouse.click(startXPx, 500 - PUR * 20) | ||||
|  | ||||
|     await expect.poll(u.normalisedEditorCode) | ||||
|       .toBe(`const sketch001 = startSketchOn('XZ') | ||||
|   |> startProfileAt([12.34, -12.34], %) | ||||
|   |> line([12.34, 0], %) | ||||
|   |> line([-12.34, 12.34], %) | ||||
|   |> line([-12.34, 0], %) | ||||
|  | ||||
| `) | ||||
|     await expect(page.locator('.cm-content')) | ||||
|       .toHaveText(`const sketch001 = startSketchOn('XZ') | ||||
|     |> startProfileAt(${commonPoints.startAt}, %) | ||||
|     |> line([${commonPoints.num1}, 0], %) | ||||
|     |> line([-8.84, 8.75], %) | ||||
|     |> line([-5.6, 0], %)`) | ||||
|  | ||||
|     // Unequip line tool | ||||
|     await page.keyboard.press('Escape') | ||||
|  | ||||
| @ -4,24 +4,19 @@ import { | ||||
|   Download, | ||||
|   TestInfo, | ||||
|   BrowserContext, | ||||
|   _electron as electron, | ||||
| } from '@playwright/test' | ||||
| import { EngineCommand } from 'lang/std/artifactGraph' | ||||
| import os from 'os' | ||||
| import fsp from 'fs/promises' | ||||
| import fsSync from 'fs' | ||||
| import { join } from 'path' | ||||
| import pixelMatch from 'pixelmatch' | ||||
| import { PNG } from 'pngjs' | ||||
| import { Protocol } from 'playwright-core/types/protocol' | ||||
| import type { Models } from '@kittycad/lib' | ||||
| import { APP_NAME, TEST_SETTINGS_FILE_KEY } from 'lib/constants' | ||||
| import { APP_NAME } from 'lib/constants' | ||||
| import waitOn from 'wait-on' | ||||
| import { secrets } from './secrets' | ||||
| import { TEST_SETTINGS_KEY, TEST_SETTINGS } from './storageStates' | ||||
| import * as TOML from '@iarna/toml' | ||||
| import { SaveSettingsPayload } from 'lib/settings/settingsTypes' | ||||
| import { SETTINGS_FILE_NAME } from 'lib/constants' | ||||
|  | ||||
| type TestColor = [number, number, number] | ||||
| export const TEST_COLORS = { | ||||
| @ -273,18 +268,6 @@ async function waitForAuthAndLsp(page: Page) { | ||||
|   return waitForLspPromise | ||||
| } | ||||
|  | ||||
| export function normaliseKclNumbers(code: string, ignoreZero = true): string { | ||||
|   const numberRegexp = /(?<!\w)-?\b\d+(\.\d+)?\b(?!\w)/g | ||||
|   const replaceNumber = (number: string) => { | ||||
|     if (ignoreZero && (number === '0' || number === '-0')) return number | ||||
|     const sign = number.startsWith('-') ? '-' : '' | ||||
|     return `${sign}12.34` | ||||
|   } | ||||
|   const replaceNumbers = (text: string) => | ||||
|     text.replace(numberRegexp, replaceNumber) | ||||
|   return replaceNumbers(code) | ||||
| } | ||||
|  | ||||
| export async function getUtils(page: Page) { | ||||
|   // Chrome devtools protocol session only works in Chromium | ||||
|   const browserType = page.context().browser()?.browserType().name() | ||||
| @ -347,11 +330,6 @@ export async function getUtils(page: Page) { | ||||
|         .boundingBox() | ||||
|         .then((box) => ({ ...box, x: box?.x || 0, y: box?.y || 0 })), | ||||
|     codeLocator: page.locator('.cm-content'), | ||||
|     normalisedEditorCode: async () => { | ||||
|       const code = await page.locator('.cm-content').innerText() | ||||
|       return normaliseKclNumbers(code) | ||||
|     }, | ||||
|     normalisedCode: (code: string) => normaliseKclNumbers(code), | ||||
|     canvasLocator: page.getByTestId('client-side-scene'), | ||||
|     doAndWaitForCmd: async ( | ||||
|       fn: () => Promise<void>, | ||||
| @ -628,96 +606,26 @@ export async function tearDown(page: Page, testInfo: TestInfo) { | ||||
|   await page.waitForTimeout(3000) | ||||
| } | ||||
|  | ||||
| // settingsOverrides may need to be augmented to take more generic items, | ||||
| // but we'll be strict for now | ||||
| export async function setup( | ||||
|   context: BrowserContext, | ||||
|   page: Page, | ||||
|   overrideDirectory?: string | ||||
| ) { | ||||
| export async function setup(context: BrowserContext, page: Page) { | ||||
|   // wait for Vite preview server to be up | ||||
|   await waitOn({ | ||||
|     resources: ['tcp:3000'], | ||||
|     timeout: 5000, | ||||
|   }) | ||||
|  | ||||
|   await context.addInitScript( | ||||
|     async ({ | ||||
|       token, | ||||
|       settingsKey, | ||||
|       settings, | ||||
|       // appSettingsFileKey, | ||||
|       // appSettingsFileContent, | ||||
|     }) => { | ||||
|     async ({ token, settingsKey, settings }) => { | ||||
|       localStorage.setItem('TOKEN_PERSIST_KEY', token) | ||||
|       localStorage.setItem('persistCode', ``) | ||||
|       localStorage.setItem(settingsKey, settings) | ||||
|       // localStorage.setItem(appSettingsFileKey, appSettingsFileContent) | ||||
|       localStorage.setItem('playwright', 'true') | ||||
|     }, | ||||
|     { | ||||
|       token: secrets.token, | ||||
|       // appSettingsFileKey: TEST_SETTINGS_FILE_KEY, | ||||
|       // appSettingsFileContent: | ||||
|       //   overrideDirectory || TEST_SETTINGS.app.projectDirectory, | ||||
|       settingsKey: TEST_SETTINGS_KEY, | ||||
|       settings: TOML.stringify({ | ||||
|         ...TEST_SETTINGS, | ||||
|         app: { | ||||
|           ...TEST_SETTINGS.projects, | ||||
|           projectDirectory: | ||||
|             overrideDirectory || TEST_SETTINGS.app.projectDirectory, | ||||
|         }, | ||||
|       } as Partial<SaveSettingsPayload>), | ||||
|       settings: TOML.stringify({ settings: TEST_SETTINGS }), | ||||
|     } | ||||
|   ) | ||||
|   // kill animations, speeds up tests and reduced flakiness | ||||
|   await page.emulateMedia({ reducedMotion: 'reduce' }) | ||||
| } | ||||
|  | ||||
| export async function setupElectron({ | ||||
|   testInfo, | ||||
|   folderSetupFn, | ||||
|   overrideDirectory, | ||||
| }: { | ||||
|   testInfo: TestInfo | ||||
|   folderSetupFn?: (projectDirName: string) => Promise<void> | ||||
|   overrideDirectory?: string | ||||
| }) { | ||||
|   // create or otherwise clear the folder | ||||
|   const projectDirName = testInfo.outputPath('electron-test-projects-dir') | ||||
|   try { | ||||
|     if (fsSync.existsSync(projectDirName)) { | ||||
|       await fsp.rm(projectDirName, { recursive: true }) | ||||
|     } | ||||
|   } catch (e) { | ||||
|     console.error(e) | ||||
|   } | ||||
|  | ||||
|   await fsp.mkdir(projectDirName) | ||||
|  | ||||
|   const electronApp = await electron.launch({ | ||||
|     args: ['.', '--no-sandbox'], | ||||
|     env: { | ||||
|       ...process.env, | ||||
|       TEST_SETTINGS_FILE_KEY: | ||||
|         overrideDirectory || TEST_SETTINGS.app.projectDirectory, | ||||
|     }, | ||||
|   }) | ||||
|   const context = electronApp.context() | ||||
|   const page = await electronApp.firstWindow() | ||||
|   context.on('console', console.log) | ||||
|   page.on('console', console.log) | ||||
|  | ||||
|   const tempSettingsFilePath = join(projectDirName, SETTINGS_FILE_NAME) | ||||
|   const settingsOverrides = TOML.stringify({ | ||||
|     ...TEST_SETTINGS, | ||||
|     settings: { | ||||
|       app: { | ||||
|         ...TEST_SETTINGS.app, | ||||
|         projectDirectory: projectDirName, | ||||
|       }, | ||||
|     }, | ||||
|   }) | ||||
|   await fsp.writeFile(tempSettingsFilePath, settingsOverrides) | ||||
|  | ||||
|   await folderSetupFn?.(projectDirName) | ||||
|  | ||||
|   await setup(context, page, projectDirName) | ||||
|  | ||||
|   return { electronApp, page } | ||||
| } | ||||
|  | ||||
| @ -80,11 +80,11 @@ const part001 = startSketchOn('XZ') | ||||
|   |> line([74.36, 130.4], %, $seg01) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> angledLine([segAng(seg01), yo], %) | ||||
|   |> line([41.19, 58.97 + 5], %) | ||||
|   |> line([41.19, 28.97 + 5], %) | ||||
| const part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 120], %) | ||||
|   |> xLine(-385.34, %, $seg_what) | ||||
|   |> yLine(-170.06, %) | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|   |> yLine(-264.06, %) | ||||
|   |> xLine(segLen(seg_what), %) | ||||
|   |> lineTo([profileStartX(%), profileStartY(%)], %)` | ||||
|       ) | ||||
| @ -138,7 +138,7 @@ const part001 = startSketchOn('XZ') | ||||
|   |> line([74.36, 130.4], %, $seg01) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> angledLine([segAng(seg01), 78.33], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
|   |> line([41.19, 28.97], %) | ||||
| const part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
| @ -237,7 +237,7 @@ const part001 = startSketchOn('XZ') | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
|   |> line([41.19, 28.97], %) | ||||
| const part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
| @ -343,7 +343,7 @@ const part001 = startSketchOn('XZ') | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
|   |> line([41.19, 28.97], %) | ||||
| const part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
| @ -450,7 +450,7 @@ const part001 = startSketchOn('XZ') | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
|   |> line([41.19, 28.97], %) | ||||
| const part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
| @ -560,7 +560,7 @@ const part001 = startSketchOn('XZ') | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
|   |> line([41.19, 28.97], %) | ||||
| const part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
| @ -613,14 +613,14 @@ const part002 = startSketchOn('XZ') | ||||
|         codeAfter: [ | ||||
|           `|> yLine(130.4, %)`, | ||||
|           `|> yLine(77.79, %)`, | ||||
|           `|> yLine(48.97, %)`, | ||||
|           `|> yLine(28.97, %)`, | ||||
|         ], | ||||
|       }, | ||||
|       { | ||||
|         codeAfter: [ | ||||
|           `|> xLine(74.36, %)`, | ||||
|           `|> xLine(9.16, %)`, | ||||
|           `|> xLine(51.19, %)`, | ||||
|           `|> xLine(41.19, %)`, | ||||
|         ], | ||||
|         constraintName: 'Horizontal', | ||||
|       }, | ||||
| @ -636,7 +636,7 @@ const part001 = startSketchOn('XZ') | ||||
|   |> line([74.36, 130.4], %) | ||||
|   |> line([78.92, -120.11], %) | ||||
|   |> line([9.16, 77.79], %) | ||||
|   |> line([51.19, 48.97], %) | ||||
|   |> line([41.19, 28.97], %) | ||||
| const part002 = startSketchOn('XZ') | ||||
|   |> startProfileAt([299.05, 231.45], %) | ||||
|   |> xLine(-425.34, %, $seg_what) | ||||
|  | ||||
| @ -64,55 +64,95 @@ test.describe('Testing settings', () => { | ||||
|       .getByRole('button', { name: 'Start Sketch' }) | ||||
|       .waitFor({ state: 'visible' }) | ||||
|  | ||||
|     const paneButtonLocator = page.getByTestId('debug-pane-button') | ||||
|     const headingLocator = page.getByRole('heading', { | ||||
|       name: 'Settings', | ||||
|       exact: true, | ||||
|     // Open the settings modal with the browser keyboard shortcut | ||||
|     await page.keyboard.press('Meta+Shift+,') | ||||
|  | ||||
|     await expect( | ||||
|       page.getByRole('heading', { name: 'Settings', exact: true }) | ||||
|     ).toBeVisible() | ||||
|     await page | ||||
|       .locator('select[name="app-theme"]') | ||||
|       .selectOption({ value: 'light' }) | ||||
|  | ||||
|     // Verify the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set theme to "light" for this project`) | ||||
|     ).toBeVisible() | ||||
|     // Check that the theme changed | ||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||
|  | ||||
|     // Check that the user setting was not changed | ||||
|     await page.getByRole('radio', { name: 'User' }).click() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('dark') | ||||
|  | ||||
|     // Roll back to default "system" theme | ||||
|     await page | ||||
|       .getByText( | ||||
|         'themeRoll back themeRoll back to match defaultThe overall appearance of the appl' | ||||
|       ) | ||||
|       .hover() | ||||
|     await page | ||||
|       .getByRole('button', { | ||||
|         name: 'Roll back theme', | ||||
|       }) | ||||
|     const inputLocator = page.locator('input[name="modeling-showDebugPanel"]') | ||||
|       .click() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('system') | ||||
|  | ||||
|     // Check that the project setting did not change | ||||
|     await page.getByRole('radio', { name: 'Project' }).click() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('light') | ||||
|   }) | ||||
|  | ||||
|   test('Project settings can be opened with keybinding from the editor', async ({ | ||||
|     page, | ||||
|   }) => { | ||||
|     const u = await getUtils(page) | ||||
|     await page.setViewportSize({ width: 1200, height: 500 }) | ||||
|     await u.waitForAuthSkipAppStart() | ||||
|     await page | ||||
|       .getByRole('button', { name: 'Start Sketch' }) | ||||
|       .waitFor({ state: 'visible' }) | ||||
|  | ||||
|     // Put the cursor in the editor | ||||
|     await page.locator('.cm-content').click() | ||||
|  | ||||
|     // Open the settings modal with the browser keyboard shortcut | ||||
|     await page.keyboard.press('Meta+Shift+,') | ||||
|  | ||||
|     await expect(headingLocator).toBeVisible() | ||||
|     await page.locator('#showDebugPanel').getByText('OffOn').click() | ||||
|  | ||||
|     // Close it and open again with keyboard shortcut, while KCL editor is focused | ||||
|     // Put the cursor in the editor | ||||
|     await test.step('Open settings with keyboard shortcut', async () => { | ||||
|       await page.getByTestId('settings-close-button').click() | ||||
|       await page.locator('.cm-content').click() | ||||
|       await page.keyboard.press('Meta+Shift+,') | ||||
|       await expect(headingLocator).toBeVisible() | ||||
|     }) | ||||
|     await expect( | ||||
|       page.getByRole('heading', { name: 'Settings', exact: true }) | ||||
|     ).toBeVisible() | ||||
|     await page | ||||
|       .locator('select[name="app-theme"]') | ||||
|       .selectOption({ value: 'light' }) | ||||
|  | ||||
|     // Verify the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set show debug panel to "false" for this project`) | ||||
|       page.getByText(`Set theme to "light" for this project`) | ||||
|     ).toBeVisible() | ||||
|     // Check that the theme changed | ||||
|     await expect(paneButtonLocator).not.toBeVisible() | ||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||
|  | ||||
|     // Check that the user setting was not changed | ||||
|     await page.getByRole('radio', { name: 'User' }).click() | ||||
|     await expect(inputLocator).toBeChecked() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('dark') | ||||
|  | ||||
|     // Roll back to default of "off" | ||||
|     await await page | ||||
|       .getByText('show debug panelRoll back show debug panelRoll back to match') | ||||
|     // Roll back to default "system" theme | ||||
|     await page | ||||
|       .getByText( | ||||
|         'themeRoll back themeRoll back to match defaultThe overall appearance of the appl' | ||||
|       ) | ||||
|       .hover() | ||||
|     await page | ||||
|       .getByRole('button', { | ||||
|         name: 'Roll back show debug panel', | ||||
|         name: 'Roll back theme', | ||||
|       }) | ||||
|       .click() | ||||
|     await expect(inputLocator).not.toBeChecked() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('system') | ||||
|  | ||||
|     // Check that the project setting did not change | ||||
|     await page.getByRole('radio', { name: 'Project' }).click() | ||||
|     await expect( | ||||
|       page.locator('input[name="modeling-showDebugPanel"]') | ||||
|     ).not.toBeChecked() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('light') | ||||
|   }) | ||||
|  | ||||
|   test('Project and user settings can be reset', async ({ page }) => { | ||||
| @ -123,67 +163,69 @@ test.describe('Testing settings', () => { | ||||
|       .getByRole('button', { name: 'Start Sketch' }) | ||||
|       .waitFor({ state: 'visible' }) | ||||
|  | ||||
|     const projectSettingsTab = page.getByRole('radio', { name: 'Project' }) | ||||
|     const userSettingsTab = page.getByRole('radio', { name: 'User' }) | ||||
|     const resetButton = page.getByRole('button', { | ||||
|       name: 'Restore default settings', | ||||
|     }) | ||||
|     const themeColorSetting = page.locator('#themeColor').getByRole('slider') | ||||
|     const settingValues = { | ||||
|       default: '259', | ||||
|       user: '120', | ||||
|       project: '50', | ||||
|     } | ||||
|     // Put the cursor in the editor | ||||
|     await page.locator('.cm-content').click() | ||||
|  | ||||
|     // Open the settings modal with the browser keyboard shortcut | ||||
|     await page.keyboard.press('Meta+Shift+,') | ||||
|  | ||||
|     // Open the settings modal with lower-right button | ||||
|     await page.getByRole('link', { name: 'Settings' }).last().click() | ||||
|     await expect( | ||||
|       page.getByRole('heading', { name: 'Settings', exact: true }) | ||||
|     ).toBeVisible() | ||||
|  | ||||
|     await test.step('Set up theme color', async () => { | ||||
|       // Verify we're looking at the project-level settings, | ||||
|       // and it's set to default value | ||||
|       await expect(projectSettingsTab).toBeChecked() | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|  | ||||
|       // Set project-level value to 50 | ||||
|       await themeColorSetting.fill(settingValues.project) | ||||
|  | ||||
|       // Set user-level value to 120 | ||||
|       await userSettingsTab.click() | ||||
|       await themeColorSetting.fill(settingValues.user) | ||||
|       await projectSettingsTab.click() | ||||
|     }) | ||||
|  | ||||
|     await test.step('Reset project settings', async () => { | ||||
|     // Click the reset settings button. | ||||
|       await resetButton.click() | ||||
|     await page.getByRole('button', { name: 'Restore default settings' }).click() | ||||
|  | ||||
|       // Verify it is now set to the inherited user value | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|     await page | ||||
|       .locator('select[name="app-theme"]') | ||||
|       .selectOption({ value: 'light' }) | ||||
|  | ||||
|       // Check that the user setting also rolled back | ||||
|       await userSettingsTab.click() | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|       await projectSettingsTab.click() | ||||
|     // Verify the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set theme to "light" for this project`) | ||||
|     ).toBeVisible() | ||||
|     // Check that the theme changed | ||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('light') | ||||
|  | ||||
|       // Set project-level value to 50 again to test the user-level reset | ||||
|       await themeColorSetting.fill(settingValues.project) | ||||
|       await userSettingsTab.click() | ||||
|     }) | ||||
|     // Check that the user setting was not changed | ||||
|     await page.getByRole('radio', { name: 'User' }).click() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('system') | ||||
|  | ||||
|     await test.step('Reset user settings', async () => { | ||||
|       // Change the setting and click the reset settings button. | ||||
|       await themeColorSetting.fill(settingValues.user) | ||||
|       await resetButton.click() | ||||
|     // Click the reset settings button. | ||||
|     await page.getByRole('button', { name: 'Restore default settings' }).click() | ||||
|  | ||||
|     // Verify it is now set to the default value | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('system') | ||||
|  | ||||
|       // Check that the project setting also changed | ||||
|       await projectSettingsTab.click() | ||||
|       await expect(themeColorSetting).toHaveValue(settingValues.default) | ||||
|     }) | ||||
|     // Set the user theme to light. | ||||
|     await page | ||||
|       .locator('select[name="app-theme"]') | ||||
|       .selectOption({ value: 'light' }) | ||||
|  | ||||
|     // Verify the toast appeared | ||||
|     await expect( | ||||
|       page.getByText(`Set theme to "light" as a user default`) | ||||
|     ).toBeVisible() | ||||
|     // Check that the theme changed | ||||
|     await expect(page.locator('body')).not.toHaveClass(`body-bg dark`) | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('light') | ||||
|  | ||||
|     await page.getByRole('radio', { name: 'Project' }).click() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('light') | ||||
|  | ||||
|     // Click the reset settings button. | ||||
|     await page.getByRole('button', { name: 'Restore default settings' }).click() | ||||
|     // Verify it is now set to the default value | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('system') | ||||
|  | ||||
|     await page.getByRole('radio', { name: 'User' }).click() | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('system') | ||||
|  | ||||
|     // Click the reset settings button. | ||||
|     await page.getByRole('button', { name: 'Restore default settings' }).click() | ||||
|  | ||||
|     // Verify it is now set to the default value | ||||
|     await expect(page.locator('select[name="app-theme"]')).toHaveValue('system') | ||||
|   }) | ||||
| }) | ||||
|  | ||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	![dependabot[bot]](/assets/img/avatar_default.png)