diff --git a/bun.lock b/bun.lock
index 9c136e28..743c2cb0 100644
--- a/bun.lock
+++ b/bun.lock
@@ -58,25 +58,6 @@
"typescript": "catalog:",
},
},
- "cloud/web": {
- "name": "@opencode/cloud-web",
- "version": "0.5.28",
- "dependencies": {
- "@kobalte/core": "0.13.9",
- "@openauthjs/solid": "0.0.0-20250322224806",
- "@solid-primitives/storage": "4.3.1",
- "@solidjs/meta": "0.29.4",
- "@solidjs/router": "0.15.3",
- "solid-js": "catalog:",
- "solid-list": "0.3.0",
- },
- "devDependencies": {
- "typescript": "catalog:",
- "vite": "6.2.2",
- "vite-plugin-pages": "0.32.5",
- "vite-plugin-solid": "2.11.6",
- },
- },
"packages/function": {
"name": "@opencode/function",
"version": "0.5.28",
@@ -397,8 +378,6 @@
"@colors/colors": ["@colors/colors@1.6.0", "", {}, "sha512-Ir+AOibqzrIsL6ajt3Rz3LskB7OiMVHqltZmspbW/TJuTVuyOMirVqAkjfY6JISiLHgyNqicAC8AyHHGzNd/dA=="],
- "@corvu/utils": ["@corvu/utils@0.4.2", "", { "dependencies": { "@floating-ui/dom": "^1.6.11" }, "peerDependencies": { "solid-js": "^1.8" } }, "sha512-Ox2kYyxy7NoXdKWdHeDEjZxClwzO4SKM8plAaVwmAJPxHMqA0rLOoAsa+hBDwRLpctf+ZRnAd/ykguuJidnaTA=="],
-
"@cspotcode/source-map-support": ["@cspotcode/source-map-support@0.8.1", "", { "dependencies": { "@jridgewell/trace-mapping": "0.3.9" } }, "sha512-IchNf6dN4tHoMFIn/7OE8LWZ19Y6q/67Bmf6vnGREv8RSbBVb9LPJxEcnwrcwX6ixSvaiGoomAUvu4YSxXrVgw=="],
"@ctrl/tinycolor": ["@ctrl/tinycolor@4.1.0", "", {}, "sha512-WyOx8cJQ+FQus4Mm4uPIZA64gbk3Wxh0so5Lcii0aJifqwoVOlfFtorjLE0Hen4OYyHZMXDWqMmaQemBhgxFRQ=="],
@@ -481,12 +460,6 @@
"@fastify/busboy": ["@fastify/busboy@3.2.0", "", {}, "sha512-m9FVDXU3GT2ITSe0UaMA5rU3QkfC/UXtCU8y0gSN/GugTqtVldOBWIB5V6V3sbmenVZUIpU6f+mPEO2+m5iTaA=="],
- "@floating-ui/core": ["@floating-ui/core@1.7.3", "", { "dependencies": { "@floating-ui/utils": "^0.2.10" } }, "sha512-sGnvb5dmrJaKEZ+LDIpguvdX3bDlEllmv4/ClQ9awcmCZrlx5jQyyMWFM5kBI+EyNOCDDiKk8il0zeuX3Zlg/w=="],
-
- "@floating-ui/dom": ["@floating-ui/dom@1.7.4", "", { "dependencies": { "@floating-ui/core": "^1.7.3", "@floating-ui/utils": "^0.2.10" } }, "sha512-OOchDgh4F2CchOX94cRVqhvy7b3AFb+/rQXyswmzmGakRfkMgoWVjfnLWkRirfLEfuD4ysVW16eXzwt3jHIzKA=="],
-
- "@floating-ui/utils": ["@floating-ui/utils@0.2.10", "", {}, "sha512-aGTxbpbg8/b5JfU1HXSrbH3wXZuLPJcNEcZQFMxLs3oSzgtVu6nFPkbbGGUvBcUjKV2YyB9Wxxabo+HEH9tcRQ=="],
-
"@fontsource/ibm-plex-mono": ["@fontsource/ibm-plex-mono@5.2.5", "", {}, "sha512-G09N3GfuT9qj3Ax2FDZvKqZttzM3v+cco2l8uXamhKyXLdmlaUDH5o88/C3vtTHj2oT7yRKsvxz9F+BXbWKMYA=="],
"@grpc/grpc-js": ["@grpc/grpc-js@1.13.4", "", { "dependencies": { "@grpc/proto-loader": "^0.7.13", "@js-sdsl/ordered-map": "^4.4.2" } }, "sha512-GsFaMXCkMqkKIvwCQjCrwH+GHbPKBjhwo/8ZuUkWHqbI73Kky9I+pQltrlT0+MWpedCoosda53lgjYfyEPgxBg=="],
@@ -541,10 +514,6 @@
"@img/sharp-win32-x64": ["@img/sharp-win32-x64@0.33.5", "", { "os": "win32", "cpu": "x64" }, "sha512-MpY/o8/8kj+EcnxwvrP4aTJSWw/aZ7JIGR4aBeZkZw5B7/Jn+tY9/VNwtcoGmdT7GfggGIU4kygOMSbYnOrAbg=="],
- "@internationalized/date": ["@internationalized/date@3.9.0", "", { "dependencies": { "@swc/helpers": "^0.5.0" } }, "sha512-yaN3brAnHRD+4KyyOsJyk49XUvj2wtbNACSqg0bz3u8t2VuzhC8Q5dfRnrSxjnnbDb+ienBnkn1TzQfE154vyg=="],
-
- "@internationalized/number": ["@internationalized/number@3.6.5", "", { "dependencies": { "@swc/helpers": "^0.5.0" } }, "sha512-6hY4Kl4HPBvtfS62asS/R22JzNNy8vi/Ssev7x6EobfCp+9QIB2hKvI2EtbdJ0VSQacxVNtqhE/NmF/NZ0gm6g=="],
-
"@ioredis/commands": ["@ioredis/commands@1.3.1", "", {}, "sha512-bYtU8avhGIcje3IhvF9aSjsa5URMZBHnwKtOvXsT4sfYy9gppW11gLPT/9oNqlJZD47yPKveQFTAFWpHjKvUoQ=="],
"@isaacs/balanced-match": ["@isaacs/balanced-match@4.0.1", "", {}, "sha512-yzMTt9lEb8Gv7zRioUilSglI0c0smZ9k5D65677DLWLtWJaXIS3CqcGyUFByYKlnUj6TkjLVs54fBl6+TiGQDQ=="],
@@ -573,10 +542,6 @@
"@jsdevtools/ono": ["@jsdevtools/ono@7.1.3", "", {}, "sha512-4JQNk+3mVzK3xh2rqd6RB4J46qUR19azEHBneZyTZM+c456qOrbbM/5xcR8huNCCcbVt7+UmizG6GuUvPvKUYg=="],
- "@kobalte/core": ["@kobalte/core@0.13.9", "", { "dependencies": { "@floating-ui/dom": "^1.5.1", "@internationalized/date": "^3.4.0", "@internationalized/number": "^3.2.1", "@kobalte/utils": "^0.9.1", "@solid-primitives/props": "^3.1.8", "@solid-primitives/resize-observer": "^2.0.26", "solid-presence": "^0.1.8", "solid-prevent-scroll": "^0.1.4" }, "peerDependencies": { "solid-js": "^1.8.15" } }, "sha512-TkeSpgNy7I5k8jwjqT9CK3teAxN0aFb3yyL9ODb06JVYMwXIk+UKrizoAF1ahLUP85lKnxv44B4Y5cXkHShgqw=="],
-
- "@kobalte/utils": ["@kobalte/utils@0.9.1", "", { "dependencies": { "@solid-primitives/event-listener": "^2.2.14", "@solid-primitives/keyed": "^1.2.0", "@solid-primitives/map": "^0.4.7", "@solid-primitives/media": "^2.2.4", "@solid-primitives/props": "^3.1.8", "@solid-primitives/refs": "^1.0.5", "@solid-primitives/utils": "^6.2.1" }, "peerDependencies": { "solid-js": "^1.8.8" } }, "sha512-eeU60A3kprIiBDAfv9gUJX1tXGLuZiKMajUfSQURAF2pk4ZoMYiqIzmrMBvzcxP39xnYttgTyQEVLwiTZnrV4w=="],
-
"@logdna/tail-file": ["@logdna/tail-file@2.2.0", "", {}, "sha512-XGSsWDweP80Fks16lwkAUIr54ICyBs6PsI4mpfTLQaWgEJRtY9xEV+PeyDpJ+sJEGZxqINlpmAwe/6tS1pP8Ng=="],
"@mapbox/node-pre-gyp": ["@mapbox/node-pre-gyp@2.0.0", "", { "dependencies": { "consola": "^3.2.3", "detect-libc": "^2.0.0", "https-proxy-agent": "^7.0.5", "node-fetch": "^2.6.7", "nopt": "^8.0.0", "semver": "^7.5.3", "tar": "^7.4.0" }, "bin": { "node-pre-gyp": "bin/node-pre-gyp" } }, "sha512-llMXd39jtP0HpQLVI37Bf1m2ADlEb35GYSh1SDSLsBhR+5iCxiNGlT31yqbNtVHygHAtMy6dWFERpU2JgufhPg=="],
@@ -677,8 +642,6 @@
"@openauthjs/openauth": ["@openauthjs/openauth@0.0.0-20250322224806", "", { "dependencies": { "@standard-schema/spec": "1.0.0-beta.3", "aws4fetch": "1.0.20", "jose": "5.9.6" }, "peerDependencies": { "arctic": "^2.2.2", "hono": "^4.0.0" } }, "sha512-p5IWSRXvABcwocH2dNI0w8c1QJelIOFulwhKk+aLLFfUbs8u1pr7kQbYe8yCSM2+bcLHiwbogpUQc2ovrGwCuw=="],
- "@openauthjs/solid": ["@openauthjs/solid@0.0.0-20250322224806", "", { "dependencies": { "@openauthjs/openauth": "0.4.2", "@solid-primitives/storage": "^4.3.1" }, "peerDependencies": { "solid-js": "^1.8.0" } }, "sha512-ln/yZQ/i+2vkrVSDOxxmbC8Oq6P2sljtdCiiAPZXj8etihwT11CYtj4jsrwArQltog738N2uvd2dKpHopVqLSQ=="],
-
"@opencode-ai/plugin": ["@opencode-ai/plugin@workspace:packages/plugin"],
"@opencode-ai/sdk": ["@opencode-ai/sdk@workspace:packages/sdk/js"],
@@ -689,8 +652,6 @@
"@opencode/cloud-function": ["@opencode/cloud-function@workspace:cloud/function"],
- "@opencode/cloud-web": ["@opencode/cloud-web@workspace:cloud/web"],
-
"@opencode/function": ["@opencode/function@workspace:packages/function"],
"@opencode/web": ["@opencode/web@workspace:packages/web"],
@@ -973,30 +934,6 @@
"@smithy/util-utf8": ["@smithy/util-utf8@4.0.0", "", { "dependencies": { "@smithy/util-buffer-from": "^4.0.0", "tslib": "^2.6.2" } }, "sha512-b+zebfKCfRdgNJDknHCob3O7FpeYQN6ZG6YLExMcasDHsCXlsXCEuiPZeLnJLpwa5dvPetGlnGCiMHuLwGvFow=="],
- "@solid-primitives/event-listener": ["@solid-primitives/event-listener@2.4.3", "", { "dependencies": { "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-h4VqkYFv6Gf+L7SQj+Y6puigL/5DIi7x5q07VZET7AWcS+9/G3WfIE9WheniHWJs51OEkRB43w6lDys5YeFceg=="],
-
- "@solid-primitives/keyed": ["@solid-primitives/keyed@1.5.2", "", { "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-BgoEdqPw48URnI+L5sZIHdF4ua4Las1eWEBBPaoSFs42kkhnHue+rwCBPL2Z9ebOyQ75sUhUfOETdJfmv0D6Kg=="],
-
- "@solid-primitives/map": ["@solid-primitives/map@0.4.13", "", { "dependencies": { "@solid-primitives/trigger": "^1.1.0" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-B1zyFbsiTQvqPr+cuPCXO72sRuczG9Swncqk5P74NCGw1VE8qa/Ry9GlfI1e/VdeQYHjan+XkbE3rO2GW/qKew=="],
-
- "@solid-primitives/media": ["@solid-primitives/media@2.3.3", "", { "dependencies": { "@solid-primitives/event-listener": "^2.4.3", "@solid-primitives/rootless": "^1.5.2", "@solid-primitives/static-store": "^0.1.2", "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-hQ4hLOGvfbugQi5Eu1BFWAIJGIAzztq9x0h02xgBGl2l0Jaa3h7tg6bz5tV1NSuNYVGio4rPoa7zVQQLkkx9dA=="],
-
- "@solid-primitives/props": ["@solid-primitives/props@3.2.2", "", { "dependencies": { "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-lZOTwFJajBrshSyg14nBMEP0h8MXzPowGO0s3OeiR3z6nXHTfj0FhzDtJMv+VYoRJKQHG2QRnJTgCzK6erARAw=="],
-
- "@solid-primitives/refs": ["@solid-primitives/refs@1.1.2", "", { "dependencies": { "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-K7tf2thy7L+YJjdqXspXOg5xvNEOH8tgEWsp0+1mQk3obHBRD6hEjYZk7p7FlJphSZImS35je3UfmWuD7MhDfg=="],
-
- "@solid-primitives/resize-observer": ["@solid-primitives/resize-observer@2.1.3", "", { "dependencies": { "@solid-primitives/event-listener": "^2.4.3", "@solid-primitives/rootless": "^1.5.2", "@solid-primitives/static-store": "^0.1.2", "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-zBLje5E06TgOg93S7rGPldmhDnouNGhvfZVKOp+oG2XU8snA+GoCSSCz1M+jpNAg5Ek2EakU5UVQqL152WmdXQ=="],
-
- "@solid-primitives/rootless": ["@solid-primitives/rootless@1.5.2", "", { "dependencies": { "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-9HULb0QAzL2r47CCad0M+NKFtQ+LrGGNHZfteX/ThdGvKIg2o2GYhBooZubTCd/RTu2l2+Nw4s+dEfiDGvdrrQ=="],
-
- "@solid-primitives/static-store": ["@solid-primitives/static-store@0.1.2", "", { "dependencies": { "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-ReK+5O38lJ7fT+L6mUFvUr6igFwHBESZF+2Ug842s7fvlVeBdIVEdTCErygff6w7uR6+jrr7J8jQo+cYrEq4Iw=="],
-
- "@solid-primitives/storage": ["@solid-primitives/storage@4.3.1", "", { "dependencies": { "@solid-primitives/utils": "^6.3.0" }, "peerDependencies": { "@tauri-apps/plugin-store": "*", "solid-js": "^1.6.12" }, "optionalPeers": ["@tauri-apps/plugin-store"] }, "sha512-xAJsY2pvXrAaCai4N2grmWY3xh5om9suTDVzGkRF5JBpDzs3Apk+xIovdTErbW0iCzXIEefENXb9xmSzdjuLYA=="],
-
- "@solid-primitives/trigger": ["@solid-primitives/trigger@1.2.2", "", { "dependencies": { "@solid-primitives/utils": "^6.3.2" }, "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-IWoptVc0SWYgmpBPpCMehS5b07+tpFcvw15tOQ3QbXedSYn6KP8zCjPkHNzMxcOvOicTneleeZDP7lqmz+PQ6g=="],
-
- "@solid-primitives/utils": ["@solid-primitives/utils@6.3.2", "", { "peerDependencies": { "solid-js": "^1.6.12" } }, "sha512-hZ/M/qr25QOCcwDPOHtGjxTD8w2mNyVAYvcfgwzBHq2RwNqHNdDNsMZYap20+ruRwW4A3Cdkczyoz0TSxLCAPQ=="],
-
"@solidjs/meta": ["@solidjs/meta@0.29.4", "", { "peerDependencies": { "solid-js": ">=1.8.4" } }, "sha512-zdIWBGpR9zGx1p1bzIPqF5Gs+Ks/BH8R6fWhmUa/dcK1L2rUC8BAcZJzNRYBQv74kScf1TSOs0EY//Vd/I0V8g=="],
"@solidjs/router": ["@solidjs/router@0.15.3", "", { "peerDependencies": { "solid-js": "^1.8.6" } }, "sha512-iEbW8UKok2Oio7o6Y4VTzLj+KFCmQPGEpm1fS3xixwFBdclFVBvaQVeibl1jys4cujfAK5Kn6+uG2uBm3lxOMw=="],
@@ -1531,7 +1468,7 @@
"end-of-stream": ["end-of-stream@1.4.5", "", { "dependencies": { "once": "^1.4.0" } }, "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg=="],
- "entities": ["entities@4.5.0", "", {}, "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw=="],
+ "entities": ["entities@6.0.1", "", {}, "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g=="],
"env-paths": ["env-paths@3.0.0", "", {}, "sha512-dtJUTepzMW3Lm/NPxRf3wP4642UWhjL2sQxc+ym2YMj1m/H2zDNQOlezafzkHwn6sMstjHTwG6iQQsctDW/b1A=="],
@@ -1569,8 +1506,6 @@
"esprima": ["esprima@4.0.1", "", { "bin": { "esparse": "./bin/esparse.js", "esvalidate": "./bin/esvalidate.js" } }, "sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A=="],
- "esprima-extract-comments": ["esprima-extract-comments@1.1.0", "", { "dependencies": { "esprima": "^4.0.0" } }, "sha512-sBQUnvJwpeE9QnPrxh7dpI/dp67erYG4WXEAreAMoelPRpMR7NWb4YtwRPn9b+H1uLQKl/qS8WYmyaljTpjIsw=="],
-
"estraverse": ["estraverse@5.3.0", "", {}, "sha512-MMdARuVEQziNTeJD8DgMqmhwR11BRQ/cBP+pLtYdSTnf3MIO8fFeiINEbX36ZdNlfU/7A9f3gUw49B3oQsvwBA=="],
"estree-util-attach-comments": ["estree-util-attach-comments@3.0.0", "", { "dependencies": { "@types/estree": "^1.0.0" } }, "sha512-cKUwm/HUcTDsYh/9FgnuFqpfquUbwIqwKM26BVCGDPVgvaCl/nDCCjUfiLlx6lsEZ3Z4RFxNbOQ60pkaEwFxGw=="],
@@ -1621,8 +1556,6 @@
"extend-shallow": ["extend-shallow@2.0.1", "", { "dependencies": { "is-extendable": "^0.1.0" } }, "sha512-zCnTtlxNoAiDc3gqY2aYAWFx7XWWiasuF2K8Me5WbN8otHKTUKBwjPtNpRs/rbUZm7KxWAaNj7P1a/p52GbVug=="],
- "extract-comments": ["extract-comments@1.1.0", "", { "dependencies": { "esprima-extract-comments": "^1.1.0", "parse-code-context": "^1.0.0" } }, "sha512-dzbZV2AdSSVW/4E7Ti5hZdHWbA+Z80RJsJhr5uiL10oyjl/gy7/o+HI1HwK4/WSZhlq4SNKU3oUzXlM13Qx02Q=="],
-
"extract-zip": ["extract-zip@2.0.1", "", { "dependencies": { "debug": "^4.1.1", "get-stream": "^5.1.0", "yauzl": "^2.10.0" }, "optionalDependencies": { "@types/yauzl": "^2.9.1" }, "bin": { "extract-zip": "cli.js" } }, "sha512-GDhU9ntwuKyGXdZBUgTIe+vXnWj0fppUEtMDL0+idd5Sta8TGpHssn/eusA9mrPr9qNDym6SxAYZjNvCn/9RBg=="],
"fast-content-type-parse": ["fast-content-type-parse@3.0.0", "", {}, "sha512-ZvLdcY8P+N8mGQJahJV5G4U88CSvT1rP8ApL6uETe88MBXrBHAkZlSEySdUlyztF7ccb+Znos3TFqaepHxdhBg=="],
@@ -2331,8 +2264,6 @@
"pako": ["pako@0.2.9", "", {}, "sha512-NUcwaKxUxWrZLpDG+z/xZaCgQITkA/Dv4V/T6bw7VON6l1Xz/VnrBqrYjZQ12TamKHzITTfOEIYUj48y2KXImA=="],
- "parse-code-context": ["parse-code-context@1.0.0", "", {}, "sha512-OZQaqKaQnR21iqhlnPfVisFjBWjhnMl5J9MgbP8xC+EwoVqbXrq78lp+9Zb3ahmLzrIX5Us/qbvBnaS3hkH6OA=="],
-
"parse-conflict-json": ["parse-conflict-json@3.0.1", "", { "dependencies": { "json-parse-even-better-errors": "^3.0.0", "just-diff": "^6.0.0", "just-diff-apply": "^5.2.0" } }, "sha512-01TvEktc68vwbJOtWZluyWeVGWjP+bZwXtPDMQVbBKzbJ/vZBif0L69KH1+cHv1SZ6e0FKLvjyHe8mqsIqYOmw=="],
"parse-entities": ["parse-entities@4.0.2", "", { "dependencies": { "@types/unist": "^2.0.0", "character-entities-legacy": "^3.0.0", "character-reference-invalid": "^2.0.0", "decode-named-character-reference": "^1.0.0", "is-alphanumerical": "^2.0.0", "is-decimal": "^2.0.0", "is-hexadecimal": "^2.0.0" } }, "sha512-GG2AQYWoLgL877gQIKeRPGO1xF9+eG1ujIb5soS5gPvLQ1y2o8FL90w2QWNdf9I361Mpp7726c+lj3U0qK1uGw=="],
@@ -2645,12 +2576,6 @@
"solid-js": ["solid-js@1.9.9", "", { "dependencies": { "csstype": "^3.1.0", "seroval": "~1.3.0", "seroval-plugins": "~1.3.0" } }, "sha512-A0ZBPJQldAeGCTW0YRYJmt7RCeh5rbFfPZ2aOttgYnctHE7HgKeHCBB/PVc2P7eOfmNXqMFFFoYYdm3S4dcbkA=="],
- "solid-list": ["solid-list@0.3.0", "", { "dependencies": { "@corvu/utils": "~0.4.0" }, "peerDependencies": { "solid-js": "^1.8" } }, "sha512-t4hx/F/l8Vmq+ib9HtZYl7Z9F1eKxq3eKJTXlvcm7P7yI4Z8O7QSOOEVHb/K6DD7M0RxzVRobK/BS5aSfLRwKg=="],
-
- "solid-presence": ["solid-presence@0.1.8", "", { "dependencies": { "@corvu/utils": "~0.4.0" }, "peerDependencies": { "solid-js": "^1.8" } }, "sha512-pWGtXUFWYYUZNbg5YpG5vkQJyOtzn2KXhxYaMx/4I+lylTLYkITOLevaCwMRN+liCVk0pqB6EayLWojNqBFECA=="],
-
- "solid-prevent-scroll": ["solid-prevent-scroll@0.1.10", "", { "dependencies": { "@corvu/utils": "~0.4.1" }, "peerDependencies": { "solid-js": "^1.8" } }, "sha512-KplGPX2GHiWJLZ6AXYRql4M127PdYzfwvLJJXMkO+CMb8Np4VxqDAg5S8jLdwlEuBis/ia9DKw2M8dFx5u8Mhw=="],
-
"solid-refresh": ["solid-refresh@0.6.3", "", { "dependencies": { "@babel/generator": "^7.23.6", "@babel/helper-module-imports": "^7.22.15", "@babel/types": "^7.23.6" }, "peerDependencies": { "solid-js": "^1.3" } }, "sha512-F3aPsX6hVw9ttm5LYlth8Q15x6MlI/J3Dn+o3EQyRTtTxidepSTwAYdozt01/YA+7ObcciagGEyXIopGZzQtbA=="],
"solid-use": ["solid-use@0.9.1", "", { "peerDependencies": { "solid-js": "^1.7" } }, "sha512-UwvXDVPlrrbj/9ewG9ys5uL2IO4jSiwys2KPzK4zsnAcmEl7iDafZWW1Mo4BSEWOmQCGK6IvpmGHo1aou8iOFw=="],
@@ -2919,9 +2844,7 @@
"vinxi": ["vinxi@0.5.8", "", { "dependencies": { "@babel/core": "^7.22.11", "@babel/plugin-syntax-jsx": "^7.22.5", "@babel/plugin-syntax-typescript": "^7.22.5", "@types/micromatch": "^4.0.2", "@vinxi/listhen": "^1.5.6", "boxen": "^8.0.1", "chokidar": "^4.0.3", "citty": "^0.1.6", "consola": "^3.4.2", "crossws": "^0.3.4", "dax-sh": "^0.43.0", "defu": "^6.1.4", "es-module-lexer": "^1.7.0", "esbuild": "^0.25.3", "get-port-please": "^3.1.2", "h3": "1.15.3", "hookable": "^5.5.3", "http-proxy": "^1.18.1", "micromatch": "^4.0.8", "nitropack": "^2.11.10", "node-fetch-native": "^1.6.6", "path-to-regexp": "^6.2.1", "pathe": "^1.1.1", "radix3": "^1.1.2", "resolve": "^1.22.10", "serve-placeholder": "^2.0.1", "serve-static": "^1.15.0", "tinyglobby": "^0.2.14", "ufo": "^1.6.1", "unctx": "^2.4.1", "unenv": "^1.10.0", "unstorage": "^1.16.0", "vite": "^6.3.3", "zod": "^3.24.3" }, "bin": { "vinxi": "bin/cli.mjs" } }, "sha512-1pGA+cU1G9feBQ1sd5FMftPuLUT8NSX880AvELhNWqoqWhe2jeSOQxjDPxlA3f1AC+Bbknl4UPKHyVXmfLZQjw=="],
- "vite": ["vite@6.2.2", "", { "dependencies": { "esbuild": "^0.25.0", "postcss": "^8.5.3", "rollup": "^4.30.1" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-yW7PeMM+LkDzc7CgJuRLMW2Jz0FxMOsVJ8Lv3gpgW9WLcb9cTW+121UEr1hvmfR7w3SegR5ItvYyzVz1vxNJgQ=="],
-
- "vite-plugin-pages": ["vite-plugin-pages@0.32.5", "", { "dependencies": { "@types/debug": "^4.1.12", "debug": "^4.4.0", "dequal": "^2.0.3", "extract-comments": "^1.1.0", "fast-glob": "^3.3.3", "json5": "^2.2.3", "local-pkg": "^1.0.0", "picocolors": "^1.1.1", "yaml": "^2.7.0" }, "peerDependencies": { "@vue/compiler-sfc": "^2.7.0 || ^3.0.0", "vite": "^2.0.0 || ^3.0.0-0 || ^4.0.0 || ^5.0.0 || ^6.0.0" }, "optionalPeers": ["@vue/compiler-sfc"] }, "sha512-GY2JAt+4vZ4BqTtw+4CSUxPgYiqamrMRIzYk2AtJvQHeBoMlctsQW+tgCpKriUKINiKfi6NegbP07r1XrdxTWA=="],
+ "vite": ["vite@6.3.5", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.4", "picomatch": "^4.0.2", "postcss": "^8.5.3", "rollup": "^4.34.9", "tinyglobby": "^0.2.13" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ=="],
"vite-plugin-solid": ["vite-plugin-solid@2.11.6", "", { "dependencies": { "@babel/core": "^7.23.3", "@types/babel__core": "^7.20.4", "babel-preset-solid": "^1.8.4", "merge-anything": "^5.1.7", "solid-refresh": "^0.6.3", "vitefu": "^1.0.4" }, "peerDependencies": { "@testing-library/jest-dom": "^5.16.6 || ^5.17.0 || ^6.*", "solid-js": "^1.7.2", "vite": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0" }, "optionalPeers": ["@testing-library/jest-dom"] }, "sha512-Sl5CTqJTGyEeOsmdH6BOgalIZlwH3t4/y0RQuFLMGnvWMBvxb4+lq7x3BSiAw6etf0QexfNJW7HSOO/Qf7pigg=="],
@@ -3023,16 +2946,12 @@
"@ai-sdk/gateway/@ai-sdk/provider-utils": ["@ai-sdk/provider-utils@3.0.1", "", { "dependencies": { "@ai-sdk/provider": "2.0.0", "@standard-schema/spec": "^1.0.0", "eventsource-parser": "^3.0.3", "zod-to-json-schema": "^3.24.1" }, "peerDependencies": { "zod": "^3.25.76 || ^4" } }, "sha512-/iP1sKc6UdJgGH98OCly7sWJKv+J9G47PnTjIj40IJMUQKwDrUMyf7zOOfRtPwSuNifYhSoJQ4s1WltI65gJ/g=="],
- "@astrojs/cloudflare/vite": ["vite@6.3.5", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.4", "picomatch": "^4.0.2", "postcss": "^8.5.3", "rollup": "^4.34.9", "tinyglobby": "^0.2.13" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ=="],
-
"@astrojs/markdown-remark/@astrojs/internal-helpers": ["@astrojs/internal-helpers@0.6.1", "", {}, "sha512-l5Pqf6uZu31aG+3Lv8nl/3s4DbUzdlxTWDof4pEpto6GUJNhhCbelVi9dEyurOVyqaelwmS9oSyOWOENSfgo9A=="],
"@astrojs/mdx/@astrojs/markdown-remark": ["@astrojs/markdown-remark@6.3.6", "", { "dependencies": { "@astrojs/internal-helpers": "0.7.2", "@astrojs/prism": "3.3.0", "github-slugger": "^2.0.0", "hast-util-from-html": "^2.0.3", "hast-util-to-text": "^4.0.2", "import-meta-resolve": "^4.1.0", "js-yaml": "^4.1.0", "mdast-util-definitions": "^6.0.0", "rehype-raw": "^7.0.0", "rehype-stringify": "^10.0.1", "remark-gfm": "^4.0.1", "remark-parse": "^11.0.0", "remark-rehype": "^11.1.2", "remark-smartypants": "^3.0.2", "shiki": "^3.2.1", "smol-toml": "^1.3.4", "unified": "^11.0.5", "unist-util-remove-position": "^5.0.0", "unist-util-visit": "^5.0.0", "unist-util-visit-parents": "^6.0.1", "vfile": "^6.0.3" } }, "sha512-bwylYktCTsLMVoCOEHbn2GSUA3c5KT/qilekBKA3CBng0bo1TYjNZPr761vxumRk9kJGqTOtU+fgCAp5Vwokug=="],
"@astrojs/mdx/source-map": ["source-map@0.7.6", "", {}, "sha512-i5uvt8C3ikiWeNZSVZNWcfZPItFQOsYTUAOkcUPGd8DqDy1uOUikjt5dG+uRlwyvR108Fb9DOd4GvXfT0N2/uQ=="],
- "@astrojs/solid-js/vite": ["vite@6.3.5", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.4", "picomatch": "^4.0.2", "postcss": "^8.5.3", "rollup": "^4.34.9", "tinyglobby": "^0.2.13" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ=="],
-
"@aws-crypto/sha256-browser/@smithy/util-utf8": ["@smithy/util-utf8@2.3.0", "", { "dependencies": { "@smithy/util-buffer-from": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A=="],
"@aws-crypto/util/@smithy/util-utf8": ["@smithy/util-utf8@2.3.0", "", { "dependencies": { "@smithy/util-buffer-from": "^2.2.0", "tslib": "^2.6.2" } }, "sha512-R8Rdn8Hy72KKcebgLiv8jQcQkXoLMOGGv5uI1/k0l+snqkOzQ1R0ChUBCxWMlBsFMekWjq0wRudIweFs7sKT5A=="],
@@ -3115,8 +3034,6 @@
"@openauthjs/openauth/jose": ["jose@5.9.6", "", {}, "sha512-AMlnetc9+CV9asI19zHmrgS/WYsWUwCn2R7RzlbJWD7F9eWYUTGyBmU9o6PxngtLGOiDGPRu+Uc4fhKzbpteZQ=="],
- "@openauthjs/solid/@openauthjs/openauth": ["@openauthjs/openauth@0.4.2", "", { "dependencies": { "@standard-schema/spec": "1.0.0-beta.3", "aws4fetch": "1.0.20", "jose": "5.9.6" }, "peerDependencies": { "arctic": "^2.2.2", "hono": "^4.0.0" } }, "sha512-8+Bia559iffrZXfQ0LWXrVVVriochS88pDtB8indyQ1S+40MQgDBu8aBzKt+fgSrTmoQGCTT+wlOXgbjc9qIcw=="],
-
"@opentelemetry/instrumentation-grpc/@opentelemetry/semantic-conventions": ["@opentelemetry/semantic-conventions@1.27.0", "", {}, "sha512-sAay1RrB+ONOem0OZanAR1ZI/k7yDpnOQSQmTMuGImUQb2y8EbSaCJ94FQluM74xoU03vlb2d2U90hZluL6nQg=="],
"@oslojs/jwt/@oslojs/encoding": ["@oslojs/encoding@0.4.1", "", {}, "sha512-hkjo6MuIK/kQR5CrGNdAPZhS01ZCXuWDRJ187zh6qqF2+yMHZpD9fAYpX8q2bOO6Ryhl3XpCT6kUX76N8hhm4Q=="],
@@ -3159,6 +3076,8 @@
"@vinxi/server-components/magicast": ["magicast@0.2.11", "", { "dependencies": { "@babel/parser": "^7.22.16", "@babel/types": "^7.22.17", "recast": "^0.23.4" } }, "sha512-6saXbRDA1HMkqbsvHOU6HBjCVgZT460qheRkLhJQHWAbhXoWESI3Kn/dGGXyKs15FFKR85jsUqFx2sMK0wy/5g=="],
+ "@vue/compiler-core/entities": ["entities@4.5.0", "", {}, "sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw=="],
+
"@vue/compiler-core/estree-walker": ["estree-walker@2.0.2", "", {}, "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w=="],
"@vue/compiler-sfc/estree-walker": ["estree-walker@2.0.2", "", {}, "sha512-Rfkk/Mp/DL7JVje3u18FxFujQlTNR2q6QfMSMB7AvCBx91NGj/ba3kCfza0f6dVDbw7YlRf/nDrn7pQrCCyQ/w=="],
@@ -3177,8 +3096,6 @@
"astro/sharp": ["sharp@0.33.5", "", { "dependencies": { "color": "^4.2.3", "detect-libc": "^2.0.3", "semver": "^7.6.3" }, "optionalDependencies": { "@img/sharp-darwin-arm64": "0.33.5", "@img/sharp-darwin-x64": "0.33.5", "@img/sharp-libvips-darwin-arm64": "1.0.4", "@img/sharp-libvips-darwin-x64": "1.0.4", "@img/sharp-libvips-linux-arm": "1.0.5", "@img/sharp-libvips-linux-arm64": "1.0.4", "@img/sharp-libvips-linux-s390x": "1.0.4", "@img/sharp-libvips-linux-x64": "1.0.4", "@img/sharp-libvips-linuxmusl-arm64": "1.0.4", "@img/sharp-libvips-linuxmusl-x64": "1.0.4", "@img/sharp-linux-arm": "0.33.5", "@img/sharp-linux-arm64": "0.33.5", "@img/sharp-linux-s390x": "0.33.5", "@img/sharp-linux-x64": "0.33.5", "@img/sharp-linuxmusl-arm64": "0.33.5", "@img/sharp-linuxmusl-x64": "0.33.5", "@img/sharp-wasm32": "0.33.5", "@img/sharp-win32-ia32": "0.33.5", "@img/sharp-win32-x64": "0.33.5" } }, "sha512-haPVm1EkS9pgvHrQ/F3Xy+hgcuMV0Wm9vfIBSiwZ05k+xgb0PkBQpGsAA/oWdDobNaZTH5ppvHtzCFbnSEwHVw=="],
- "astro/vite": ["vite@6.3.5", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.4", "picomatch": "^4.0.2", "postcss": "^8.5.3", "rollup": "^4.34.9", "tinyglobby": "^0.2.13" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ=="],
-
"babel-plugin-jsx-dom-expressions/@babel/helper-module-imports": ["@babel/helper-module-imports@7.18.6", "", { "dependencies": { "@babel/types": "^7.18.6" } }, "sha512-0NFvs3VkuSYbFi1x2Vd6tKrywq+z/cLeYC/RJNFrIX/30Bf5aiGYbtvGXolEktzJH8o5E5KJ3tT+nkxuuZFVlA=="],
"bl/buffer": ["buffer@5.7.1", "", { "dependencies": { "base64-js": "^1.3.1", "ieee754": "^1.1.13" } }, "sha512-EHcyIPBQ4BSGlvjB16k5KgAJ27CIsHY/2JBmCRReo48y9rQ3MaUzWX3KVlBa4U7MyX02HdVj0K7C3WaB3ju7FQ=="],
@@ -3305,8 +3222,6 @@
"parse-json/@babel/code-frame": ["@babel/code-frame@7.27.1", "", { "dependencies": { "@babel/helper-validator-identifier": "^7.27.1", "js-tokens": "^4.0.0", "picocolors": "^1.1.1" } }, "sha512-cjQ7ZlQ0Mv3b47hABuTevyTuYN4i+loJKGeV9flcCgIK37cCXRh+L1bd3iBHlynerhQ7BhCkn2BPbQUL+rGqFg=="],
- "parse5/entities": ["entities@6.0.1", "", {}, "sha512-aN97NXWF6AWBTahfVOIrB/NShkzi5H7F9r1s9mD3cDj4Ko5f2qhhVoYMibXF7GlLveb/D2ioWay8lxI97Ven3g=="],
-
"path-scurry/lru-cache": ["lru-cache@10.4.3", "", {}, "sha512-JNAzZcXrCt42VGLuYz0zfAzDfAvJWW6AfYlDBQyDV5DClI2m5sAmK+OIO7s59XfsRsWHp02jAJrRadPRGTt6SQ=="],
"pkg-types/pathe": ["pathe@2.0.3", "", {}, "sha512-WUjGcAqP1gQacoQe+OBJsFA7Ld4DyXuUIjZ5cc75cLHvJ7dtNsTugphxIADwspS+AraAUePCKrSVtPLFj/F88w=="],
@@ -3393,8 +3308,6 @@
"uri-js/punycode": ["punycode@2.3.1", "", {}, "sha512-vYt7UD1U9Wg6138shLtLOvdAu+8DsC/ilFtEVHcH+wydcSpNE20AfSOduf6MkRFahL5FY7X1oU7nKVZFtfq8Fg=="],
- "vinxi/vite": ["vite@6.3.5", "", { "dependencies": { "esbuild": "^0.25.0", "fdir": "^6.4.4", "picomatch": "^4.0.2", "postcss": "^8.5.3", "rollup": "^4.34.9", "tinyglobby": "^0.2.13" }, "optionalDependencies": { "fsevents": "~2.3.3" }, "peerDependencies": { "@types/node": "^18.0.0 || ^20.0.0 || >=22.0.0", "jiti": ">=1.21.0", "less": "*", "lightningcss": "^1.21.0", "sass": "*", "sass-embedded": "*", "stylus": "*", "sugarss": "*", "terser": "^5.16.0", "tsx": "^4.8.1", "yaml": "^2.4.2" }, "optionalPeers": ["@types/node", "jiti", "less", "lightningcss", "sass", "sass-embedded", "stylus", "sugarss", "terser", "tsx", "yaml"], "bin": { "vite": "bin/vite.js" } }, "sha512-cZn6NDFE7wdTpINgs++ZJ4N49W2vRp8LCKrn3Ob1kYNtOo21vfDoaV5GzBfLU4MovSAB8uNRm4jgzVQZ+mBzPQ=="],
-
"winston/readable-stream": ["readable-stream@3.6.2", "", { "dependencies": { "inherits": "^2.0.3", "string_decoder": "^1.1.1", "util-deprecate": "^1.0.1" } }, "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA=="],
"winston-transport/readable-stream": ["readable-stream@3.6.2", "", { "dependencies": { "inherits": "^2.0.3", "string_decoder": "^1.1.1", "util-deprecate": "^1.0.1" } }, "sha512-9u/sniCrY3D5WdsERHzHE4G2YCXqoG5FTHUiCC4SIbr6XcLZBY05ya9EKjYek9O5xOAwjGq+1JdGBAS7Q9ScoA=="],
@@ -3563,12 +3476,6 @@
"@netlify/zip-it-and-ship-it/yargs/string-width": ["string-width@4.2.3", "", { "dependencies": { "emoji-regex": "^8.0.0", "is-fullwidth-code-point": "^3.0.0", "strip-ansi": "^6.0.1" } }, "sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g=="],
- "@openauthjs/solid/@openauthjs/openauth/@standard-schema/spec": ["@standard-schema/spec@1.0.0-beta.3", "", {}, "sha512-0ifF3BjA1E8SY9C+nUew8RefNOIq0cDlYALPty4rhUm8Rrl6tCM8hBT4bhGhx7I7iXD0uAgt50lgo8dD73ACMw=="],
-
- "@openauthjs/solid/@openauthjs/openauth/aws4fetch": ["aws4fetch@1.0.20", "", {}, "sha512-/djoAN709iY65ETD6LKCtyyEI04XIBP5xVvfmNxsEP0uJB5tyaGBztSryRr4HqMStr9R06PisQE7m9zDTXKu6g=="],
-
- "@openauthjs/solid/@openauthjs/openauth/jose": ["jose@5.9.6", "", {}, "sha512-AMlnetc9+CV9asI19zHmrgS/WYsWUwCn2R7RzlbJWD7F9eWYUTGyBmU9o6PxngtLGOiDGPRu+Uc4fhKzbpteZQ=="],
-
"@pulumi/pulumi/js-yaml/argparse": ["argparse@1.0.10", "", { "dependencies": { "sprintf-js": "~1.0.2" } }, "sha512-o5Roy6tNG4SL/FOkCAN6RzjiakZS25RLYFrcMttJqbdd8BWrnA+fGz57iN5Pb06pvBGvl5gQ0B48dJlslXvoTg=="],
"@solidjs/start/shiki/@shikijs/core": ["@shikijs/core@1.29.2", "", { "dependencies": { "@shikijs/engine-javascript": "1.29.2", "@shikijs/engine-oniguruma": "1.29.2", "@shikijs/types": "1.29.2", "@shikijs/vscode-textmate": "^10.0.1", "@types/hast": "^3.0.4", "hast-util-to-html": "^9.0.4" } }, "sha512-vju0lY9r27jJfOY4Z7+Rt/nIOjzJpZ3y+nYpqtUZInVoXQ/TJZcfGnNOGnKjFdVZb8qexiCuSlZRKcGfhhTTZQ=="],
diff --git a/cloud/app/src/routes/workspace.tsx b/cloud/app/src/routes/workspace.tsx
index 865e7651..864fab4d 100644
--- a/cloud/app/src/routes/workspace.tsx
+++ b/cloud/app/src/routes/workspace.tsx
@@ -16,9 +16,7 @@ export default function WorkspaceLayout(props: RouteSectionProps) {
Logout
-
- {props.children}
-
+ {props.children}
)
}
diff --git a/cloud/function/src/gateway.ts b/cloud/function/src/gateway.ts
index c8b39990..aed02b4d 100644
--- a/cloud/function/src/gateway.ts
+++ b/cloud/function/src/gateway.ts
@@ -1,39 +1,18 @@
-import { z } from "zod"
import { Hono, MiddlewareHandler } from "hono"
-import { cors } from "hono/cors"
-import { HTTPException } from "hono/http-exception"
-import { zValidator } from "@hono/zod-validator"
import { Resource } from "@opencode/cloud-core/util/resource.js"
-import { type ProviderMetadata, type LanguageModelUsage, generateText, streamText } from "ai"
+import { type ProviderMetadata, type LanguageModelUsage } from "ai"
import { createAnthropic } from "@ai-sdk/anthropic"
import { createOpenAI } from "@ai-sdk/openai"
import { createOpenAICompatible } from "@ai-sdk/openai-compatible"
import type { LanguageModelV2Prompt } from "@ai-sdk/provider"
import { type ChatCompletionCreateParamsBase } from "openai/resources/chat/completions"
import { Actor } from "@opencode/cloud-core/actor.js"
-import { and, Database, eq, sql } from "@opencode/cloud-core/drizzle/index.js"
-import { UserTable } from "@opencode/cloud-core/schema/user.sql.js"
+import { Database, eq, sql } from "@opencode/cloud-core/drizzle/index.js"
import { KeyTable } from "@opencode/cloud-core/schema/key.sql.js"
-import { createClient } from "@openauthjs/openauth/client"
-import { Log } from "@opencode/cloud-core/util/log.js"
import { Billing } from "@opencode/cloud-core/billing.js"
-import { Workspace } from "@opencode/cloud-core/workspace.js"
-import { BillingTable, PaymentTable, UsageTable } from "@opencode/cloud-core/schema/billing.sql.js"
-import { centsToMicroCents } from "@opencode/cloud-core/util/price.js"
-import { Identifier } from "../../core/src/identifier"
type Env = {}
-let _client: ReturnType
-const client = () => {
- if (_client) return _client
- _client = createClient({
- clientID: "api",
- issuer: Resource.AUTH_API_URL.value,
- })
- return _client
-}
-
const SUPPORTED_MODELS = {
"anthropic/claude-sonnet-4": {
input: 0.0000015,
@@ -72,10 +51,6 @@ const SUPPORTED_MODELS = {
},
}
-const log = Log.create({
- namespace: "api",
-})
-
const GatewayAuth: MiddlewareHandler = async (c, next) => {
const authHeader = c.req.header("authorization")
@@ -125,56 +100,6 @@ const GatewayAuth: MiddlewareHandler = async (c, next) => {
await next()
}
-const RestAuth: MiddlewareHandler = async (c, next) => {
- const authorization = c.req.header("authorization")
- if (!authorization) {
- return Actor.provide("public", {}, next)
- }
- const token = authorization.split(" ")[1]
- if (!token)
- throw new HTTPException(403, {
- message: "Bearer token is required.",
- })
-
- const verified = await client().verify(token)
- if (verified.err) {
- throw new HTTPException(403, {
- message: "Invalid token.",
- })
- }
- let subject = verified.subject as Actor.Info
- if (subject.type === "account") {
- const workspaceID = c.req.header("x-opencode-workspace")
- const email = subject.properties.email
- if (workspaceID) {
- const user = await Database.use((tx) =>
- tx
- .select({
- id: UserTable.id,
- workspaceID: UserTable.workspaceID,
- email: UserTable.email,
- })
- .from(UserTable)
- .where(and(eq(UserTable.email, email), eq(UserTable.workspaceID, workspaceID)))
- .then((rows) => rows[0]),
- )
- if (!user)
- throw new HTTPException(403, {
- message: "You do not have access to this workspace.",
- })
- subject = {
- type: "user",
- properties: {
- userID: user.id,
- workspaceID: workspaceID,
- email: user.email,
- },
- }
- }
- }
- await Actor.provide(subject.type, subject.properties, next)
-}
-
const app = new Hono<{ Bindings: Env; Variables: { keyRecord?: { id: string; workspaceID: string } } }>()
.get("/", (c) => c.text("Hello, world!"))
.post("/v1/chat/completions", GatewayAuth, async (c) => {
@@ -664,244 +589,6 @@ const app = new Hono<{ Bindings: Env; Variables: { keyRecord?: { id: string; wor
}
})
})
- .use("/*", cors())
- .use(RestAuth)
- .get("/rest/account", async (c) => {
- const account = Actor.assert("account")
- let workspaces = await Workspace.list()
- if (workspaces.length === 0) {
- await Workspace.create()
- workspaces = await Workspace.list()
- }
- return c.json({
- id: account.properties.accountID,
- email: account.properties.email,
- workspaces,
- })
- })
- .get("/billing/info", async (c) => {
- const billing = await Billing.get()
- const payments = await Database.use((tx) =>
- tx
- .select()
- .from(PaymentTable)
- .where(eq(PaymentTable.workspaceID, Actor.workspace()))
- .orderBy(sql`${PaymentTable.timeCreated} DESC`)
- .limit(100),
- )
- const usage = await Database.use((tx) =>
- tx
- .select()
- .from(UsageTable)
- .where(eq(UsageTable.workspaceID, Actor.workspace()))
- .orderBy(sql`${UsageTable.timeCreated} DESC`)
- .limit(100),
- )
- return c.json({ billing, payments, usage })
- })
- .post(
- "/billing/checkout",
- zValidator(
- "json",
- z.custom<{
- success_url: string
- cancel_url: string
- }>(),
- ),
- async (c) => {
- const account = Actor.assert("user")
-
- const body = await c.req.json()
-
- const customer = await Billing.get()
- const session = await Billing.stripe().checkout.sessions.create({
- mode: "payment",
- line_items: [
- {
- price_data: {
- currency: "usd",
- product_data: {
- name: "opencode credits",
- },
- unit_amount: 2000, // $20 minimum
- },
- quantity: 1,
- },
- ],
- payment_intent_data: {
- setup_future_usage: "on_session",
- },
- ...(customer.customerID
- ? { customer: customer.customerID }
- : {
- customer_email: account.properties.email,
- customer_creation: "always",
- }),
- metadata: {
- workspaceID: Actor.workspace(),
- },
- currency: "usd",
- payment_method_types: ["card"],
- success_url: body.success_url,
- cancel_url: body.cancel_url,
- })
-
- return c.json({
- url: session.url,
- })
- },
- )
- .post("/billing/portal", async (c) => {
- const body = await c.req.json()
-
- const customer = await Billing.get()
- if (!customer?.customerID) {
- throw new Error("No stripe customer ID")
- }
-
- const session = await Billing.stripe().billingPortal.sessions.create({
- customer: customer.customerID,
- return_url: body.return_url,
- })
-
- return c.json({
- url: session.url,
- })
- })
- .post("/stripe/webhook", async (c) => {
- const body = await Billing.stripe().webhooks.constructEventAsync(
- await c.req.text(),
- c.req.header("stripe-signature")!,
- Resource.STRIPE_WEBHOOK_SECRET.value,
- )
-
- console.log(body.type, JSON.stringify(body, null, 2))
- if (body.type === "checkout.session.completed") {
- const workspaceID = body.data.object.metadata?.workspaceID
- const customerID = body.data.object.customer as string
- const paymentID = body.data.object.payment_intent as string
- const amount = body.data.object.amount_total
-
- if (!workspaceID) throw new Error("Workspace ID not found")
- if (!customerID) throw new Error("Customer ID not found")
- if (!amount) throw new Error("Amount not found")
- if (!paymentID) throw new Error("Payment ID not found")
-
- await Actor.provide("system", { workspaceID }, async () => {
- const customer = await Billing.get()
- if (customer?.customerID && customer.customerID !== customerID) throw new Error("Customer ID mismatch")
-
- // set customer metadata
- if (!customer?.customerID) {
- await Billing.stripe().customers.update(customerID, {
- metadata: {
- workspaceID,
- },
- })
- }
-
- // get payment method for the payment intent
- const paymentIntent = await Billing.stripe().paymentIntents.retrieve(paymentID, {
- expand: ["payment_method"],
- })
- const paymentMethod = paymentIntent.payment_method
- if (!paymentMethod || typeof paymentMethod === "string") throw new Error("Payment method not expanded")
-
- await Database.transaction(async (tx) => {
- await tx
- .update(BillingTable)
- .set({
- balance: sql`${BillingTable.balance} + ${centsToMicroCents(amount)}`,
- customerID,
- paymentMethodID: paymentMethod.id,
- paymentMethodLast4: paymentMethod.card!.last4,
- })
- .where(eq(BillingTable.workspaceID, workspaceID))
- await tx.insert(PaymentTable).values({
- workspaceID,
- id: Identifier.create("payment"),
- amount: centsToMicroCents(amount),
- paymentID,
- customerID,
- })
- })
- })
- }
-
- console.log("finished handling")
-
- return c.json("ok", 200)
- })
- .get("/keys", async (c) => {
- const user = Actor.assert("user")
-
- const keys = await Database.use((tx) =>
- tx
- .select({
- id: KeyTable.id,
- name: KeyTable.name,
- key: KeyTable.key,
- userID: KeyTable.userID,
- timeCreated: KeyTable.timeCreated,
- timeUsed: KeyTable.timeUsed,
- })
- .from(KeyTable)
- .where(eq(KeyTable.workspaceID, user.properties.workspaceID))
- .orderBy(sql`${KeyTable.timeCreated} DESC`),
- )
-
- return c.json({ keys })
- })
- .post("/keys", zValidator("json", z.object({ name: z.string().min(1).max(255) })), async (c) => {
- const user = Actor.assert("user")
- const { name } = c.req.valid("json")
-
- // Generate secret key: sk- + 64 random characters (upper, lower, numbers)
- const chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789"
- let randomPart = ""
- for (let i = 0; i < 64; i++) {
- randomPart += chars.charAt(Math.floor(Math.random() * chars.length))
- }
- const secretKey = `sk-${randomPart}`
-
- const keyRecord = await Database.use((tx) =>
- tx
- .insert(KeyTable)
- .values({
- id: Identifier.create("key"),
- workspaceID: user.properties.workspaceID,
- userID: user.properties.userID,
- name,
- key: secretKey,
- timeUsed: null,
- })
- .returning(),
- )
-
- return c.json({
- key: secretKey,
- id: keyRecord[0].id,
- name: keyRecord[0].name,
- created: keyRecord[0].timeCreated,
- })
- })
- .delete("/keys/:id", async (c) => {
- const user = Actor.assert("user")
- const keyId = c.req.param("id")
-
- const result = await Database.use((tx) =>
- tx
- .delete(KeyTable)
- .where(and(eq(KeyTable.id, keyId), eq(KeyTable.workspaceID, user.properties.workspaceID)))
- .returning({ id: KeyTable.id }),
- )
-
- if (result.length === 0) {
- return c.json({ error: "Key not found" }, 404)
- }
-
- return c.json({ success: true, id: result[0].id })
- })
.all("*", (c) => c.text("Not Found"))
export type ApiType = typeof app
diff --git a/cloud/web/.gitignore b/cloud/web/.gitignore
deleted file mode 100644
index 76add878..00000000
--- a/cloud/web/.gitignore
+++ /dev/null
@@ -1,2 +0,0 @@
-node_modules
-dist
\ No newline at end of file
diff --git a/cloud/web/index.html b/cloud/web/index.html
deleted file mode 100644
index 55c54c1f..00000000
--- a/cloud/web/index.html
+++ /dev/null
@@ -1,38 +0,0 @@
-
-
-
-
-
- OpenControl
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
diff --git a/cloud/web/npm-debug.log b/cloud/web/npm-debug.log
deleted file mode 100644
index 07b0649f..00000000
--- a/cloud/web/npm-debug.log
+++ /dev/null
@@ -1,29 +0,0 @@
-0 info it worked if it ends with ok
-1 verbose cli [
-1 verbose cli '/usr/local/bin/node',
-1 verbose cli '/Users/frank/Sites/opencode/node_modules/.bin/npm',
-1 verbose cli 'run',
-1 verbose cli 'dev'
-1 verbose cli ]
-2 info using npm@2.15.12
-3 info using node@v20.18.1
-4 verbose stack Error: Invalid name: "@opencode/cloud/web"
-4 verbose stack at ensureValidName (/Users/frank/Sites/opencode/node_modules/npm/node_modules/normalize-package-data/lib/fixer.js:336:15)
-4 verbose stack at Object.fixNameField (/Users/frank/Sites/opencode/node_modules/npm/node_modules/normalize-package-data/lib/fixer.js:215:5)
-4 verbose stack at /Users/frank/Sites/opencode/node_modules/npm/node_modules/normalize-package-data/lib/normalize.js:32:38
-4 verbose stack at Array.forEach ()
-4 verbose stack at normalize (/Users/frank/Sites/opencode/node_modules/npm/node_modules/normalize-package-data/lib/normalize.js:31:15)
-4 verbose stack at final (/Users/frank/Sites/opencode/node_modules/npm/node_modules/read-package-json/read-json.js:349:5)
-4 verbose stack at then (/Users/frank/Sites/opencode/node_modules/npm/node_modules/read-package-json/read-json.js:124:5)
-4 verbose stack at ReadFileContext. (/Users/frank/Sites/opencode/node_modules/npm/node_modules/read-package-json/read-json.js:295:20)
-4 verbose stack at ReadFileContext.callback (/Users/frank/Sites/opencode/node_modules/npm/node_modules/graceful-fs/graceful-fs.js:78:16)
-4 verbose stack at FSReqCallback.readFileAfterOpen [as oncomplete] (node:fs:299:13)
-5 verbose cwd /Users/frank/Sites/opencode/cloud/web
-6 error Darwin 24.5.0
-7 error argv "/usr/local/bin/node" "/Users/frank/Sites/opencode/node_modules/.bin/npm" "run" "dev"
-8 error node v20.18.1
-9 error npm v2.15.12
-10 error Invalid name: "@opencode/cloud/web"
-11 error If you need help, you may report this error at:
-11 error
-12 verbose exit [ 1, true ]
diff --git a/cloud/web/package.json b/cloud/web/package.json
deleted file mode 100644
index 95757e09..00000000
--- a/cloud/web/package.json
+++ /dev/null
@@ -1,32 +0,0 @@
-{
- "name": "@opencode/cloud-web",
- "version": "0.5.28",
- "private": true,
- "description": "",
- "type": "module",
- "scripts": {
- "start": "vite",
- "dev": "vite",
- "build": "bun build:server && bun build:client",
- "build:client": "vite build --outDir dist/client",
- "build:server": "vite build --ssr src/entry-server.tsx --outDir dist/server",
- "serve": "vite preview",
- "sst:dev": "bun sst shell --target Console -- bun dev"
- },
- "license": "MIT",
- "devDependencies": {
- "typescript": "catalog:",
- "vite": "6.2.2",
- "vite-plugin-pages": "0.32.5",
- "vite-plugin-solid": "2.11.6"
- },
- "dependencies": {
- "@kobalte/core": "0.13.9",
- "@openauthjs/solid": "0.0.0-20250322224806",
- "@solid-primitives/storage": "4.3.1",
- "@solidjs/meta": "0.29.4",
- "@solidjs/router": "0.15.3",
- "solid-js": "catalog:",
- "solid-list": "0.3.0"
- }
-}
diff --git a/cloud/web/public/favicon-dark.svg b/cloud/web/public/favicon-dark.svg
deleted file mode 100644
index 9b707ea4..00000000
--- a/cloud/web/public/favicon-dark.svg
+++ /dev/null
@@ -1,3 +0,0 @@
-
diff --git a/cloud/web/public/favicon.ico b/cloud/web/public/favicon.ico
deleted file mode 100644
index 0ed3bf15..00000000
Binary files a/cloud/web/public/favicon.ico and /dev/null differ
diff --git a/cloud/web/public/favicon.svg b/cloud/web/public/favicon.svg
deleted file mode 100644
index 5e7cf124..00000000
--- a/cloud/web/public/favicon.svg
+++ /dev/null
@@ -1,3 +0,0 @@
-
diff --git a/cloud/web/public/social-share.png b/cloud/web/public/social-share.png
deleted file mode 100644
index 72d36a97..00000000
Binary files a/cloud/web/public/social-share.png and /dev/null differ
diff --git a/cloud/web/scripts/render.mjs b/cloud/web/scripts/render.mjs
deleted file mode 100644
index 5ccb35ff..00000000
--- a/cloud/web/scripts/render.mjs
+++ /dev/null
@@ -1,24 +0,0 @@
-import fs from "fs"
-import path from "path"
-import { generateHydrationScript, getAssets } from "solid-js/web"
-
-const dist = import.meta.resolve("../dist").replace("file://", "")
-const serverEntry = await import("../dist/server/entry-server.js")
-const template = fs.readFileSync(path.join(dist, "client/index.html"), "utf-8")
-fs.writeFileSync(path.join(dist, "client/fallback.html"), template)
-
-const routes = ["/", "/foo"]
-for (const route of routes) {
- const { app } = serverEntry.render({ url: route })
- const html = template
- .replace("", app)
- .replace("", generateHydrationScript())
- .replace("", getAssets())
- const filePath = dist + `/client${route === "/" ? "/index" : route}.html`
- fs.mkdirSync(path.dirname(filePath), {
- recursive: true,
- })
- fs.writeFileSync(filePath, html)
-
- console.log(`Pre-rendered: ${filePath}`)
-}
diff --git a/cloud/web/src/app.tsx b/cloud/web/src/app.tsx
deleted file mode 100644
index aae71ddd..00000000
--- a/cloud/web/src/app.tsx
+++ /dev/null
@@ -1,42 +0,0 @@
-///
-
-import { Router } from "@solidjs/router"
-import routes from "~solid-pages"
-import "./ui/style/index.css"
-import { MetaProvider } from "@solidjs/meta"
-import { AccountProvider } from "./components/context-account"
-import { DialogProvider } from "./ui/context-dialog"
-import { DialogString } from "./ui/dialog-string"
-import { DialogSelect } from "./ui/dialog-select"
-import { ThemeProvider } from "./components/context-theme"
-import { Suspense } from "solid-js"
-import { OpenAuthProvider } from "./components/context-openauth"
-
-export function App(props: { url?: string }) {
- return (
-
-
-
-
-
-
-
-
- {
- return <>{props.children}>
- }}
- />
-
-
-
-
-
-
- )
-}
diff --git a/cloud/web/src/assets/screenshot.png b/cloud/web/src/assets/screenshot.png
deleted file mode 100644
index 5b6ad2ec..00000000
Binary files a/cloud/web/src/assets/screenshot.png and /dev/null differ
diff --git a/cloud/web/src/components/context-account.tsx b/cloud/web/src/components/context-account.tsx
deleted file mode 100644
index e6aabafd..00000000
--- a/cloud/web/src/components/context-account.tsx
+++ /dev/null
@@ -1,99 +0,0 @@
-import { createContext, createEffect, ParentProps, Suspense, useContext } from "solid-js"
-import { makePersisted } from "@solid-primitives/storage"
-import { createStore } from "solid-js/store"
-import { useOpenAuth } from "./context-openauth"
-import { createAsync } from "@solidjs/router"
-import { isServer } from "solid-js/web"
-
-type Storage = {
- accounts: Record<
- string,
- {
- id: string
- email: string
- workspaces: {
- id: string
- name: string
- slug: string
- }[]
- }
- >
-}
-
-const context = createContext>()
-
-function init() {
- const auth = useOpenAuth()
- const [store, setStore] = makePersisted(
- createStore({
- accounts: {},
- }),
- {
- name: "opencontrol.account",
- },
- )
-
- async function refresh(id: string) {
- return fetch(import.meta.env.VITE_API_URL + "/rest/account", {
- headers: {
- authorization: `Bearer ${await auth.access(id)}`,
- },
- })
- .then((val) => val.json())
- .then((val) => setStore("accounts", id, val as any))
- }
-
- createEffect((previous: string[]) => {
- if (Object.keys(auth.all).length === 0) {
- return []
- }
- for (const item of Object.values(auth.all)) {
- if (previous.includes(item.id)) continue
- refresh(item.id)
- }
- return Object.keys(auth.all)
- }, [] as string[])
-
- const result = {
- get all() {
- return Object.keys(auth.all)
- .map((id) => store.accounts[id])
- .filter(Boolean)
- },
- get current() {
- if (!auth.subject) return undefined
- return store.accounts[auth.subject.id]
- },
- refresh,
- get ready() {
- return Object.keys(auth.all).length === result.all.length
- },
- }
-
- return result
-}
-
-export function AccountProvider(props: ParentProps) {
- const ctx = init()
- const resource = createAsync(async () => {
- await new Promise((resolve) => {
- if (isServer) return resolve()
- createEffect(() => {
- if (ctx.ready) resolve()
- })
- })
- return null
- })
- return (
-
- {resource()}
- {props.children}
-
- )
-}
-
-export function useAccount() {
- const result = useContext(context)
- if (!result) throw new Error("no account context")
- return result
-}
diff --git a/cloud/web/src/components/context-openauth.tsx b/cloud/web/src/components/context-openauth.tsx
deleted file mode 100644
index bd6a45dd..00000000
--- a/cloud/web/src/components/context-openauth.tsx
+++ /dev/null
@@ -1,180 +0,0 @@
-import { createClient } from "@openauthjs/openauth/client"
-import { makePersisted } from "@solid-primitives/storage"
-import { createAsync } from "@solidjs/router"
-import {
- batch,
- createContext,
- createEffect,
- createResource,
- createSignal,
- onMount,
- ParentProps,
- Show,
- Suspense,
- useContext,
-} from "solid-js"
-import { createStore, produce } from "solid-js/store"
-import { isServer } from "solid-js/web"
-
-interface Storage {
- subjects: Record
- current?: string
-}
-
-interface Context {
- all: Record
- subject?: SubjectInfo
- switch(id: string): void
- logout(id: string): void
- access(id?: string): Promise
- authorize(opts?: AuthorizeOptions): void
-}
-
-export interface AuthorizeOptions {
- redirectPath?: string
- provider?: string
-}
-
-interface SubjectInfo {
- id: string
- refresh: string
-}
-
-interface AuthContextOpts {
- issuer: string
- clientID: string
-}
-
-const context = createContext()
-
-export function OpenAuthProvider(props: ParentProps) {
- const client = createClient({
- issuer: props.issuer,
- clientID: props.clientID,
- })
- const [storage, setStorage] = makePersisted(
- createStore({
- subjects: {},
- }),
- {
- name: `${props.issuer}.auth`,
- },
- )
-
- const resource = createAsync(async () => {
- if (isServer) return true
- const hash = new URLSearchParams(window.location.search.substring(1))
- const code = hash.get("code")
- const state = hash.get("state")
- if (code && state) {
- const oldState = sessionStorage.getItem("openauth.state")
- const verifier = sessionStorage.getItem("openauth.verifier")
- const redirect = sessionStorage.getItem("openauth.redirect")
- if (redirect && verifier && oldState === state) {
- const result = await client.exchange(code, redirect, verifier)
- if (!result.err) {
- const id = result.tokens.refresh.split(":").slice(0, -1).join(":")
- batch(() => {
- setStorage("subjects", id, {
- id: id,
- refresh: result.tokens.refresh,
- })
- setStorage("current", id)
- })
- }
- }
- }
- return true
- })
-
- async function authorize(opts?: AuthorizeOptions) {
- const redirect = new URL(window.location.origin + (opts?.redirectPath ?? "/")).toString()
- const authorize = await client.authorize(redirect, "code", {
- pkce: true,
- provider: opts?.provider,
- })
- sessionStorage.setItem("openauth.state", authorize.challenge.state)
- sessionStorage.setItem("openauth.redirect", redirect)
- if (authorize.challenge.verifier) sessionStorage.setItem("openauth.verifier", authorize.challenge.verifier)
- window.location.href = authorize.url
- }
-
- const accessCache = new Map()
- const pendingRequests = new Map>()
- async function access(id: string) {
- const pending = pendingRequests.get(id)
- if (pending) return pending
- const promise = (async () => {
- const existing = accessCache.get(id)
- const subject = storage.subjects[id]
- const access = await client.refresh(subject.refresh, {
- access: existing,
- })
- if (access.err) {
- pendingRequests.delete(id)
- ctx.logout(id)
- return
- }
- if (access.tokens) {
- setStorage("subjects", id, "refresh", access.tokens.refresh)
- accessCache.set(id, access.tokens.access)
- }
- pendingRequests.delete(id)
- return access.tokens?.access || existing!
- })()
- pendingRequests.set(id, promise)
- return promise
- }
-
- const ctx: Context = {
- get all() {
- return storage.subjects
- },
- get subject() {
- if (!storage.current) return
- return storage.subjects[storage.current!]
- },
- switch(id: string) {
- if (!storage.subjects[id]) return
- setStorage("current", id)
- },
- authorize,
- logout(id: string) {
- if (!storage.subjects[id]) return
- setStorage(
- produce((s) => {
- delete s.subjects[id]
- if (s.current === id) s.current = Object.keys(s.subjects)[0]
- }),
- )
- },
- async access(id?: string) {
- id = id || storage.current
- if (!id) return
- return access(id || storage.current!)
- },
- }
-
- createEffect(() => {
- if (!resource()) return
- if (storage.current) return
- const [first] = Object.keys(storage.subjects)
- if (first) {
- setStorage("current", first)
- return
- }
- })
-
- return (
- <>
- {resource()}
- {props.children}
- >
- )
-}
-
-export function useOpenAuth() {
- const result = useContext(context)
- if (!result) throw new Error("no auth context")
- return result
-}
diff --git a/cloud/web/src/components/context-theme.tsx b/cloud/web/src/components/context-theme.tsx
deleted file mode 100644
index 7800aeca..00000000
--- a/cloud/web/src/components/context-theme.tsx
+++ /dev/null
@@ -1,39 +0,0 @@
-import { createStore } from "solid-js/store"
-import { makePersisted } from "@solid-primitives/storage"
-import { createEffect } from "solid-js"
-import { createInitializedContext } from "../util/context"
-import { isServer } from "solid-js/web"
-
-interface Storage {
- mode: "light" | "dark"
-}
-
-export const { provider: ThemeProvider, use: useTheme } =
- createInitializedContext("ThemeContext", () => {
- const [store, setStore] = makePersisted(
- createStore({
- mode:
- !isServer &&
- window.matchMedia &&
- window.matchMedia("(prefers-color-scheme: dark)").matches
- ? "dark"
- : "light",
- }),
- {
- name: "theme",
- },
- )
- createEffect(() => {
- document.documentElement.setAttribute("data-color-mode", store.mode)
- })
-
- return {
- setMode(mode: Storage["mode"]) {
- setStore("mode", mode)
- },
- get mode() {
- return store.mode
- },
- ready: true,
- }
- })
diff --git a/cloud/web/src/entry-client.tsx b/cloud/web/src/entry-client.tsx
deleted file mode 100644
index 169e45a1..00000000
--- a/cloud/web/src/entry-client.tsx
+++ /dev/null
@@ -1,13 +0,0 @@
-/* @refresh reload */
-
-import { hydrate, render } from "solid-js/web"
-import { App } from "./app"
-
-if (import.meta.env.DEV) {
- render(() => , document.getElementById("root")!)
-}
-
-if (!import.meta.env.DEV) {
- if ("_$HY" in window) hydrate(() => , document.getElementById("root")!)
- else render(() => , document.getElementById("root")!)
-}
diff --git a/cloud/web/src/entry-server.tsx b/cloud/web/src/entry-server.tsx
deleted file mode 100644
index 5dd33a14..00000000
--- a/cloud/web/src/entry-server.tsx
+++ /dev/null
@@ -1,7 +0,0 @@
-import { renderToStringAsync } from "solid-js/web"
-import { App } from "./app"
-
-export async function render(props: { url: string }) {
- const app = await renderToStringAsync(() => )
- return { app }
-}
diff --git a/cloud/web/src/pages/[workspace].tsx b/cloud/web/src/pages/[workspace].tsx
deleted file mode 100644
index c7481cb0..00000000
--- a/cloud/web/src/pages/[workspace].tsx
+++ /dev/null
@@ -1,11 +0,0 @@
-import { WorkspaceProvider } from "./components/context-workspace"
-import { ParentProps } from "solid-js"
-import Layout from "./components/layout"
-
-export default function Index(props: ParentProps) {
- return (
-
- {props.children}
-
- )
-}
diff --git a/cloud/web/src/pages/[workspace]/billing.module.css b/cloud/web/src/pages/[workspace]/billing.module.css
deleted file mode 100644
index 5e58892a..00000000
--- a/cloud/web/src/pages/[workspace]/billing.module.css
+++ /dev/null
@@ -1,56 +0,0 @@
-.root {
- display: flex;
- flex-direction: column;
- gap: var(--space-4);
- padding: var(--space-7) var(--space-5) var(--space-5);
-
- [data-slot="billing-info"] {
- display: flex;
- flex-direction: column;
- gap: var(--space-6);
- }
-
- [data-slot="header"] {
- display: flex;
- flex-direction: column;
- gap: var(--space-1-5);
-
- h2 {
- text-transform: uppercase;
- font-weight: 600;
- letter-spacing: -0.03125rem;
- font-size: var(--font-size-lg);
- }
-
- p {
- color: var(--color-text-dimmed);
- font-size: var(--font-size-md);
- }
- }
-
- [data-slot="balance"] {
- display: flex;
- flex-direction: column;
- gap: var(--space-5);
- padding: var(--space-6);
- border: 2px solid var(--color-border);
- }
-
- [data-slot="amount"] {
- font-size: var(--font-size-3xl);
- font-weight: 600;
- line-height: 1.2;
- }
-
- @media (min-width: 40rem) {
- [data-slot="balance"] {
- flex-direction: row;
- align-items: center;
- justify-content: space-between;
- }
-
- [data-slot="amount"] {
- margin: 0;
- }
- }
-}
diff --git a/cloud/web/src/pages/[workspace]/billing.tsx b/cloud/web/src/pages/[workspace]/billing.tsx
deleted file mode 100644
index 88bef580..00000000
--- a/cloud/web/src/pages/[workspace]/billing.tsx
+++ /dev/null
@@ -1,132 +0,0 @@
-import { Button } from "../../ui/button"
-import { useApi } from "../components/context-api"
-import { createEffect, createSignal, createResource, For } from "solid-js"
-import { useWorkspace } from "../components/context-workspace"
-import style from "./billing.module.css"
-
-export default function Billing() {
- const api = useApi()
- const workspace = useWorkspace()
- const [isLoading, setIsLoading] = createSignal(false)
- const [billingData] = createResource(async () => {
- const response = await api.billing.info.$get()
- return response.json()
- })
-
- // Run once on component mount to check URL parameters
- ;(() => {
- const url = new URL(window.location.href)
- const result = url.hash
-
- console.log("STRIPE RESULT", result)
-
- if (url.hash === "#success") {
- setIsLoading(true)
- // Remove the hash from the URL
- window.history.replaceState(null, "", window.location.pathname + window.location.search)
- }
- })()
-
- createEffect((old?: number) => {
- if (old && old !== billingData()?.billing?.balance) {
- setIsLoading(false)
- }
- return billingData()?.billing?.balance
- })
-
- const handleBuyCredits = async () => {
- try {
- setIsLoading(true)
- const baseUrl = window.location.href
- const successUrl = new URL(baseUrl)
- successUrl.hash = "success"
-
- const response = await api.billing.checkout
- .$post({
- json: {
- success_url: successUrl.toString(),
- cancel_url: baseUrl,
- },
- })
- .then((r) => r.json() as any)
- window.location.href = response.url
- } catch (error) {
- console.error("Failed to get checkout URL:", error)
- setIsLoading(false)
- }
- }
-
- return (
- <>
-
-
-
-
-
Balance
-
Manage your billing and add credits to your account.
-
-
-
-
- {(() => {
- const balanceStr = ((billingData()?.billing?.balance ?? 0) / 100000000).toFixed(2)
- return `$${balanceStr === "-0.00" ? "0.00" : balanceStr}`
- })()}
-
-
-
-
-
-
-
-
Payment History
-
Your recent payment transactions.
-
-
-
-
No payments found.}>
- {(payment) => (
-
- {payment.id}
- {" | "}
- ${((payment.amount ?? 0) / 100000000).toFixed(2)}
- {" | "}
- {new Date(payment.timeCreated).toLocaleDateString()}
-
- )}
-
-
-
-
-
-
-
Usage History
-
Your recent API usage and costs.
-
-
-
-
No usage found.}>
- {(usage) => (
-
- {usage.model}
- {" | "}
- {usage.inputTokens + usage.outputTokens} tokens
- {" | "}
- ${((usage.cost ?? 0) / 100000000).toFixed(4)}
- {" | "}
- {new Date(usage.timeCreated).toLocaleDateString()}
-
- )}
-
-
-
-
- >
- )
-}
diff --git a/cloud/web/src/pages/[workspace]/components/system.txt b/cloud/web/src/pages/[workspace]/components/system.txt
deleted file mode 100644
index 6afd2e04..00000000
--- a/cloud/web/src/pages/[workspace]/components/system.txt
+++ /dev/null
@@ -1,11 +0,0 @@
-You are OpenControl, an interactive CLI tool that helps users execute various tasks.
-
-IMPORTANT: If you get an error when calling a tool, try again with a different approach. Be creative, do not give up, try different inputs to the tool. You should chain together multiple tool calls. ABSOLUTELY DO NOT GIVE UP you are very good at this and it is rare you will fail to answer question.
-
-You should be concise, direct, and to the point.
-
-IMPORTANT: You should NOT answer with unnecessary preamble or postamble (such as explaining your code or summarizing your action), unless the user asks you to.
-IMPORTANT: You should minimize output tokens as much as possible while maintaining helpfulness, quality, and accuracy. Only address the specific query or task at hand, avoiding tangential information unless absolutely critical for completing the request. If you can answer in 1-3 sentences or a short paragraph, please do.
-IMPORTANT: You should NOT answer with unnecessary preamble or postamble (such as explaining your code or summarizing your action), unless the user asks you to.
-IMPORTANT: Keep your responses short, since they will be displayed on a command line interface. You MUST answer concisely with fewer than 4 lines (not including tool use or code generation), unless user asks for detail. Answer the user's question directly, without elaboration, explanation, or details. One word answers are best. Avoid introductions, conclusions, and explanations. You MUST avoid text before/after your response, such as "The answer is .", "Here is the content of the file..." or "Based on the information provided, the answer is..." or "Here is what I will do next...".
-
diff --git a/cloud/web/src/pages/[workspace]/components/tool.ts b/cloud/web/src/pages/[workspace]/components/tool.ts
deleted file mode 100644
index 3958e322..00000000
--- a/cloud/web/src/pages/[workspace]/components/tool.ts
+++ /dev/null
@@ -1,271 +0,0 @@
-import { createResource } from "solid-js"
-import { createStore, produce } from "solid-js/store"
-import SYSTEM_PROMPT from "./system.txt?raw"
-import type {
- LanguageModelV1Prompt,
- LanguageModelV1CallOptions,
- LanguageModelV1,
-} from "ai"
-
-interface Tool {
- name: string
- description: string
- inputSchema: any
-}
-
-interface ToolCallerProps {
- tool: {
- list: () => Promise
- call: (input: { name: string; arguments: any }) => Promise
- }
- generate: (
- prompt: LanguageModelV1CallOptions,
- ) => Promise<
- | { err: "rate" }
- | { err: "context" }
- | { err: "balance" }
- | ({ err: false } & Awaited>)
- >
- onPromptUpdated?: (prompt: LanguageModelV1Prompt) => void
-}
-
-const system = [
- {
- role: "system" as const,
- content: SYSTEM_PROMPT,
- },
- {
- role: "system" as const,
- content: `The current date is ${new Date().toDateString()}. Always use this current date when responding to relative date queries.`,
- },
-]
-
-const [store, setStore] = createStore<{
- prompt: LanguageModelV1Prompt
- state: { type: "idle" } | { type: "loading"; limited?: boolean }
-}>({
- prompt: [...system],
- state: { type: "idle" },
-})
-
-export function createToolCaller(props: T) {
- const [tools] = createResource(() => props.tool.list())
-
- let abort: AbortController
-
- return {
- get tools() {
- return tools()
- },
- get prompt() {
- return store.prompt
- },
- get state() {
- return store.state
- },
- clear() {
- setStore("prompt", [...system])
- },
- async chat(input: string) {
- if (store.state.type !== "idle") return
-
- abort = new AbortController()
- setStore(
- produce((s) => {
- s.state = {
- type: "loading",
- limited: false,
- }
- s.prompt.push({
- role: "user",
- content: [
- {
- type: "text",
- text: input,
- },
- ],
- })
- }),
- )
- props.onPromptUpdated?.(store.prompt)
-
- while (true) {
- if (abort.signal.aborted) {
- break
- }
-
- const response = await props.generate({
- inputFormat: "messages",
- prompt: store.prompt,
- temperature: 0,
- seed: 69,
- mode: {
- type: "regular",
- tools: tools()?.map((tool) => ({
- type: "function",
- name: tool.name,
- description: tool.description,
- parameters: {
- ...tool.inputSchema,
- },
- })),
- },
- })
-
- if (abort.signal.aborted) continue
-
- if (!response.err) {
- setStore("state", {
- type: "loading",
- })
-
- if (response.text) {
- setStore(
- produce((s) => {
- s.prompt.push({
- role: "assistant",
- content: [
- {
- type: "text",
- text: response.text || "",
- },
- ],
- })
- }),
- )
- props.onPromptUpdated?.(store.prompt)
- }
-
- if (response.finishReason === "stop") {
- break
- }
-
- if (response.finishReason === "tool-calls") {
- for (const item of response.toolCalls || []) {
- setStore(
- produce((s) => {
- s.prompt.push({
- role: "assistant",
- content: [
- {
- type: "tool-call",
- toolName: item.toolName,
- args: JSON.parse(item.args),
- toolCallId: item.toolCallId,
- },
- ],
- })
- }),
- )
- props.onPromptUpdated?.(store.prompt)
-
- const called = await props.tool.call({
- name: item.toolName,
- arguments: JSON.parse(item.args),
- })
-
- setStore(
- produce((s) => {
- s.prompt.push({
- role: "tool",
- content: [
- {
- type: "tool-result",
- toolName: item.toolName,
- toolCallId: item.toolCallId,
- result: called,
- },
- ],
- })
- }),
- )
- props.onPromptUpdated?.(store.prompt)
- }
- }
- continue
- }
-
- if (response.err === "context") {
- setStore(
- produce((s) => {
- s.prompt.splice(2, 1)
- }),
- )
- props.onPromptUpdated?.(store.prompt)
- }
-
- if (response.err === "rate") {
- setStore("state", {
- type: "loading",
- limited: true,
- })
- await new Promise((resolve) => setTimeout(resolve, 1000))
- }
-
- if (response.err === "balance") {
- setStore(
- produce((s) => {
- s.prompt.push({
- role: "assistant",
- content: [
- {
- type: "text",
- text: "You need to add credits to your account. Please go to Billing and add credits to continue.",
- },
- ],
- })
- s.state = { type: "idle" }
- }),
- )
- props.onPromptUpdated?.(store.prompt)
- break
- }
- }
- setStore("state", { type: "idle" })
- },
- async cancel() {
- abort.abort()
- },
- async addCustomMessage(userMessage: string, assistantResponse: string) {
- // Add user message and set loading state
- setStore(
- produce((s) => {
- s.prompt.push({
- role: "user",
- content: [
- {
- type: "text",
- text: userMessage,
- },
- ],
- })
- s.state = {
- type: "loading",
- limited: false,
- }
- }),
- )
- props.onPromptUpdated?.(store.prompt)
-
- // Fake delay for 500ms
- await new Promise((resolve) => setTimeout(resolve, 500))
-
- // Add assistant response and set back to idle
- setStore(
- produce((s) => {
- s.prompt.push({
- role: "assistant",
- content: [
- {
- type: "text",
- text: assistantResponse,
- },
- ],
- })
- s.state = { type: "idle" }
- }),
- )
- props.onPromptUpdated?.(store.prompt)
- },
- }
-}
diff --git a/cloud/web/src/pages/[workspace]/index.module.css b/cloud/web/src/pages/[workspace]/index.module.css
deleted file mode 100644
index 0037d97f..00000000
--- a/cloud/web/src/pages/[workspace]/index.module.css
+++ /dev/null
@@ -1,239 +0,0 @@
-.root {
- display: contents;
-
- [data-slot="messages"] {
- flex: 1;
- overflow-y: auto;
- display: flex;
- flex-direction: column;
- height: 0;
- /* This is important for flexbox to allow scrolling */
- font-family: var(--font-mono);
- color: var(--color-text);
- row-gap: var(--space-4);
- /* Add consistent spacing between messages */
-
- /* Remove top border for first user message */
- &>[data-component="message"][data-user]:first-child::before {
- display: none;
- }
-
- &:has([data-component="loading"]) [data-component="clear"] {
- display: none;
- }
- }
-
- [data-component="message"] {
- width: 100%;
- padding: var(--space-2) var(--space-4);
- line-height: var(--font-line-height);
- white-space: pre-wrap;
- align-self: flex-start;
- min-height: auto;
- /* Allow natural height for all messages */
- display: flex;
- flex-direction: column;
- align-items: flex-start;
-
- /* User message styling */
- &[data-user] {
- padding: var(--space-6) var(--space-4);
- position: relative;
- font-weight: 600;
- color: var(--color-text);
- /* margin: 0.5rem 0; */
- }
-
- &[data-user]::before,
- &[data-user]::after {
- content: "";
- position: absolute;
- left: var(--space-4);
- right: var(--space-4);
- height: var(--space-px);
- background-color: var(--color-border);
- z-index: 1;
- /* Ensure borders appear above other content */
- }
-
- &[data-user]::before {
- top: 0;
- }
-
- &[data-user]::after {
- bottom: 0;
- }
-
- &[data-assistant] {
- color: var(--color-text);
- }
- }
-
- [data-component="tool"] {
- display: flex;
- width: 100%;
- padding: 0 var(--space-4);
- margin-left: 0;
- flex-direction: column;
- opacity: 0.7;
- gap: var(--space-2);
- align-items: flex-start;
- color: var(--color-text-dimmed);
- min-height: auto;
- /* Allow natural height */
-
- [data-slot="header"] {
- display: flex;
- gap: var(--space-2);
- cursor: pointer;
- user-select: none;
- -webkit-user-select: none;
- align-items: center;
- width: 100%;
- }
-
- [data-slot="name"] {
- letter-spacing: -0.03125rem;
- text-transform: uppercase;
- font-weight: 500;
- font-size: var(--font-size-sm);
- }
-
- [data-slot="expand"] {
- font-size: var(--font-size-sm);
- }
-
- [data-slot="content"] {
- padding: 0;
- line-height: var(--font-line-height);
- font-size: var(--font-size-sm);
- white-space: pre-wrap;
- display: none;
- width: 100%;
- }
-
- [data-slot="output"] {
- margin-top: var(--space-1);
- }
-
- &[data-expanded="true"] [data-slot="content"] {
- display: block;
- }
-
- &[data-expanded="true"] [data-slot="expand"] {
- transform: rotate(45deg);
- }
- }
-
- [data-component="loading"] {
- padding: var(--space-4) var(--space-4) var(--space-8);
- height: 1.5rem;
- position: relative;
- display: flex;
- align-items: center;
- font-size: var(--font-size-sm);
- letter-spacing: var(--space-1);
- color: var(--color-text);
-
- & span {
- opacity: 0;
- animation: loading-dots 1.4s linear infinite;
- }
-
- & span:nth-child(2) {
- animation-delay: 0.2s;
- }
-
- & span:nth-child(3) {
- animation-delay: 0.4s;
- }
- }
-
- [data-component="clear"] {
- position: relative;
- padding: var(--space-4) var(--space-4);
-
- &::before {
- content: "";
- position: absolute;
- left: var(--space-4);
- right: var(--space-4);
- top: 0;
- height: var(--space-px);
- background-color: var(--color-border);
- z-index: 1;
- }
-
- & [data-component="button"] {
- padding-left: 0;
- }
- }
-
- [data-slot="footer"] {
- display: flex;
- flex-direction: column;
- padding: 0;
- border-top: 2px solid var(--color-border);
- position: sticky;
- bottom: 0;
- z-index: 10;
- /* Ensure it's above other content */
- margin-top: auto;
- /* Push to bottom if content is short */
- width: 100%;
- }
-
- [data-component="chat"] {
- display: flex;
- padding: var(--space-0-5) 0;
- align-items: center;
- width: 100%;
- height: 100%;
-
- textarea {
- --padding-y: var(--space-4);
- --line-height: 1.5;
- --text-height: calc(var(--line-height) * var(--font-size-lg));
- --height: calc(var(--text-height) + var(--padding-y) * 2);
-
- width: 100%;
- resize: none;
- line-height: var(--line-height);
- height: var(--height);
- min-height: var(--height);
- max-height: calc(5 * var(--text-height) + var(--padding-y) * 2);
- padding: var(--padding-y) var(--space-4);
- border-radius: 0;
- background-color: transparent;
- color: var(--color-text);
- border: none;
- outline: none;
- font-size: var(--font-size-lg);
- }
-
- textarea::placeholder {
- color: var(--color-text-dimmed);
- opacity: 0.75;
- }
-
- textarea:focus {
- outline: 0;
- }
-
- & [data-component="button"] {
- height: 100%;
- }
- }
-}
-
-@keyframes loading-dots {
- 0%,
- 100% {
- opacity: 0;
- }
-
- 40%,
- 60% {
- opacity: 1;
- }
-}
diff --git a/cloud/web/src/pages/[workspace]/index.tsx b/cloud/web/src/pages/[workspace]/index.tsx
deleted file mode 100644
index 50c58ee3..00000000
--- a/cloud/web/src/pages/[workspace]/index.tsx
+++ /dev/null
@@ -1,18 +0,0 @@
-import { Button } from "../../ui/button"
-import { IconArrowRight } from "../../ui/svg/icons"
-import { createSignal, For } from "solid-js"
-import { createToolCaller } from "./components/tool"
-import { useApi } from "../components/context-api"
-import { useWorkspace } from "../components/context-workspace"
-import style from "./index.module.css"
-
-export default function Index() {
- const api = useApi()
- const workspace = useWorkspace()
-
- return (
-
-
Hello
-
- )
-}
diff --git a/cloud/web/src/pages/[workspace]/keys.module.css b/cloud/web/src/pages/[workspace]/keys.module.css
deleted file mode 100644
index 4ae2989b..00000000
--- a/cloud/web/src/pages/[workspace]/keys.module.css
+++ /dev/null
@@ -1,97 +0,0 @@
-.root {
- display: flex;
- flex-direction: column;
- gap: 2rem;
-}
-
-.root [data-slot="keys-info"] {
- display: flex;
- flex-direction: column;
- gap: 1rem;
-}
-
-.root [data-slot="header"] {
- display: flex;
- flex-direction: column;
- gap: 0.5rem;
-}
-
-.root [data-slot="header"] h2 {
- margin: 0;
- font-size: 1.5rem;
- font-weight: 600;
-}
-
-.root [data-slot="header"] p {
- margin: 0;
- color: var(--color-text-secondary);
-}
-
-.root [data-slot="key-list"] {
- display: flex;
- flex-direction: column;
- gap: 1rem;
-}
-
-.root [data-slot="key-item"] {
- display: flex;
- align-items: center;
- justify-content: space-between;
- padding: 1rem;
- border: 1px solid var(--color-border);
- border-radius: 0.5rem;
- background: var(--color-background-secondary);
-}
-
-.root [data-slot="key-actions"] {
- display: flex;
- gap: 0.5rem;
-}
-
-.root [data-slot="key-info"] {
- display: flex;
- flex-direction: column;
- gap: 0.25rem;
-}
-
-.root [data-slot="key-value"] {
- font-family: monospace;
- font-size: 0.875rem;
- color: var(--color-text-primary);
-}
-
-.root [data-slot="key-meta"] {
- font-size: 0.75rem;
- color: var(--color-text-secondary);
-}
-
-.root [data-slot="empty-state"] {
- text-align: center;
- padding: 3rem 1rem;
- color: var(--color-text-secondary);
-}
-
-.root [data-slot="actions"] {
- display: flex;
- align-items: center;
- justify-content: space-between;
-}
-
-.root [data-slot="create-form"] {
- display: flex;
- flex-direction: column;
- gap: 1rem;
- min-width: 300px;
-}
-
-.root [data-slot="form-actions"] {
- display: flex;
- gap: 0.5rem;
-}
-
-.root [data-slot="key-name"] {
- font-weight: 600;
- font-size: 1rem;
- color: var(--color-text-primary);
- margin-bottom: 0.25rem;
-}
diff --git a/cloud/web/src/pages/[workspace]/keys.tsx b/cloud/web/src/pages/[workspace]/keys.tsx
deleted file mode 100644
index e5b192a2..00000000
--- a/cloud/web/src/pages/[workspace]/keys.tsx
+++ /dev/null
@@ -1,151 +0,0 @@
-import { Button } from "../../ui/button"
-import { useApi } from "../components/context-api"
-import { createSignal, createResource, For, Show } from "solid-js"
-import style from "./keys.module.css"
-
-export default function Keys() {
- const api = useApi()
- const [isCreating, setIsCreating] = createSignal(false)
- const [showCreateForm, setShowCreateForm] = createSignal(false)
- const [keyName, setKeyName] = createSignal("")
-
- const [keysData, { refetch }] = createResource(async () => {
- const response = await api.keys.$get()
- return response.json()
- })
-
- const handleCreateKey = async () => {
- if (!keyName().trim()) return
-
- try {
- setIsCreating(true)
- await api.keys.$post({
- json: { name: keyName().trim() },
- })
- refetch()
- setKeyName("")
- setShowCreateForm(false)
- } catch (error) {
- console.error("Failed to create API key:", error)
- } finally {
- setIsCreating(false)
- }
- }
-
- const handleDeleteKey = async (keyId: string) => {
- if (!confirm("Are you sure you want to delete this API key? This action cannot be undone.")) {
- return
- }
-
- try {
- await api.keys[":id"].$delete({
- param: { id: keyId },
- })
- refetch()
- } catch (error) {
- console.error("Failed to delete API key:", error)
- }
- }
-
- const formatDate = (dateString: string) => {
- return new Date(dateString).toLocaleDateString()
- }
-
- const formatKey = (key: string) => {
- if (key.length <= 11) return key
- return `${key.slice(0, 7)}...${key.slice(-4)}`
- }
-
- const copyToClipboard = async (text: string) => {
- try {
- await navigator.clipboard.writeText(text)
- } catch (error) {
- console.error("Failed to copy to clipboard:", error)
- }
- }
-
- return (
- <>
-
-
-
-
-
-
API Keys
-
Manage your API keys to access the OpenCode gateway.
-
-
- setKeyName(e.currentTarget.value)}
- onKeyPress={(e) => e.key === "Enter" && handleCreateKey()}
- />
-
-
-
-
-
- }
- >
-
-
-
-
-
-
- Create an API key to access opencode gateway
-
- }
- >
- {(key) => (
-
-
-
{key.name}
-
{formatKey(key.key)}
-
- Created: {formatDate(key.timeCreated)}
- {key.timeUsed && ` • Last used: ${formatDate(key.timeUsed)}`}
-
-
-
-
-
-
-
- )}
-
-
-
-
- >
- )
-}
diff --git a/cloud/web/src/pages/components/context-api.tsx b/cloud/web/src/pages/components/context-api.tsx
deleted file mode 100644
index 0a348f48..00000000
--- a/cloud/web/src/pages/components/context-api.tsx
+++ /dev/null
@@ -1,24 +0,0 @@
-import { hc } from "hono/client"
-import { ApiType } from "@opencode/cloud-function/src/gateway"
-import { useWorkspace } from "./context-workspace"
-import { useOpenAuth } from "../../components/context-openauth"
-
-export function useApi() {
- const workspace = useWorkspace()
- const auth = useOpenAuth()
- return hc(import.meta.env.VITE_API_URL, {
- async fetch(...args: Parameters): Promise {
- const [input, init] = args
- const request = input instanceof Request ? input : new Request(input, init)
- const headers = new Headers(request.headers)
- headers.set("authorization", `Bearer ${await auth.access()}`)
- headers.set("x-opencode-workspace", workspace.id)
- return fetch(
- new Request(request, {
- ...init,
- headers,
- }),
- )
- },
- })
-}
diff --git a/cloud/web/src/pages/components/context-workspace.tsx b/cloud/web/src/pages/components/context-workspace.tsx
deleted file mode 100644
index 6bad3984..00000000
--- a/cloud/web/src/pages/components/context-workspace.tsx
+++ /dev/null
@@ -1,38 +0,0 @@
-import { useNavigate, useParams } from "@solidjs/router"
-import { createInitializedContext } from "../../util/context"
-import { useAccount } from "../../components/context-account"
-import { createEffect, createMemo } from "solid-js"
-
-export const { use: useWorkspace, provider: WorkspaceProvider } =
- createInitializedContext("WorkspaceProvider", () => {
- const params = useParams()
- const account = useAccount()
- const workspace = createMemo(() =>
- account.current?.workspaces.find(
- (x) => x.id === params.workspace || x.slug === params.workspace,
- ),
- )
- const nav = useNavigate()
-
- createEffect(() => {
- if (!workspace()) nav("/")
- })
-
- const result = () => workspace()!
- result.ready = true
-
- return {
- get id() {
- return workspace()!.id
- },
- get slug() {
- return workspace()!.slug
- },
- get name() {
- return workspace()!.name
- },
- get ready() {
- return workspace() !== undefined
- },
- }
- })
diff --git a/cloud/web/src/pages/components/layout.module.css b/cloud/web/src/pages/components/layout.module.css
deleted file mode 100644
index c64faa18..00000000
--- a/cloud/web/src/pages/components/layout.module.css
+++ /dev/null
@@ -1,199 +0,0 @@
-.root {
- --padding: var(--space-10);
- --vertical-padding: var(--space-8);
- --heading-font-size: var(--font-size-4xl);
- --sidebar-width: 200px;
- --mobile-breakpoint: 40rem;
- --topbar-height: 60px;
-
- margin: var(--space-4);
- border: 2px solid var(--color-border);
- height: calc(100vh - var(--space-8));
- display: flex;
- flex-direction: row;
- overflow: hidden;
- /* Prevent overall scrolling */
- position: relative;
-}
-
-[data-component="mobile-top-bar"] {
- display: none;
- position: fixed;
- top: 0;
- left: 0;
- right: 0;
- height: var(--topbar-height);
- background: var(--color-background);
- border-bottom: 2px solid var(--color-border);
- z-index: 20;
- align-items: center;
- padding: 0 var(--space-4) 0 0;
-
- [data-slot="logo"] {
- position: absolute;
- left: 50%;
- top: 50%;
- transform: translate(-50%, -50%);
-
- div {
- text-transform: uppercase;
- font-weight: 600;
- letter-spacing: -0.03125rem;
- }
-
- svg {
- height: 28px;
- width: auto;
- color: var(--color-white);
- }
- }
-
- [data-slot="toggle"] {
- background: transparent;
- border: none;
- padding: var(--space-4);
- cursor: pointer;
- display: flex;
- align-items: center;
- justify-content: center;
-
- & svg {
- width: 24px;
- height: 24px;
- color: var(--color-foreground);
- }
- }
-}
-
-[data-component="sidebar"] {
- width: var(--sidebar-width);
- border-right: 2px solid var(--color-border);
- display: flex;
- flex-direction: column;
- padding: calc(var(--padding) / 2);
- overflow-y: auto;
- /* Allow scrolling if needed */
- position: sticky;
- top: 0;
- height: 100%;
- background-color: var(--color-background);
- z-index: 10;
-
- [data-slot="logo"] {
- margin-top: 2px;
- margin-bottom: var(--space-7);
- color: var(--color-white);
-
- & svg {
- height: 32px;
- width: auto;
- }
- }
-
- [data-slot="nav"] {
- flex: 1;
-
- ul {
- list-style-type: none;
- padding: 0;
- }
-
- li {
- margin-bottom: calc(var(--vertical-padding) / 2);
- text-transform: uppercase;
- font-weight: 500;
- }
-
- a {
- display: block;
- padding: var(--space-2) 0;
- }
- }
-
- [data-slot="user"] {
- [data-component="button"] {
- padding-left: 0;
- padding-bottom: 0;
- height: auto;
- }
- }
-}
-
-.navActiveLink {
- cursor: default;
- text-decoration: none;
-}
-
-[data-slot="main-content"] {
- flex: 1;
- display: flex;
- flex-direction: column;
- height: 100%;
- /* Full height */
- overflow: hidden;
- /* Prevent overflow */
- position: relative;
- /* For positioning footer */
- width: 100%;
- /* Full width */
-}
-
-/* Backdrop for mobile */
-[data-component="backdrop"] {
- display: none;
- position: fixed;
- top: 0;
- left: 0;
- right: 0;
- bottom: 0;
- /* background-color: rgba(0, 0, 0, 0.5); */
- z-index: 25;
- backdrop-filter: blur(2px);
-}
-
-/* Mobile styles */
-@media (max-width: 40rem) {
- .root {
- margin: 0;
- border: none;
- height: 100vh;
- }
-
- [data-component="mobile-top-bar"] {
- display: flex;
- }
-
- [data-component="backdrop"] {
- display: block;
- }
-
- [data-component="sidebar"] {
- position: fixed;
- left: -100%;
- top: 0;
- height: 100vh;
- width: 80%;
- max-width: 280px;
- transition: left 0.3s ease-in-out;
- box-shadow: none;
- z-index: 30;
- padding: var(--space-8);
- background-color: var(--color-bg);
-
- &[data-opened="true"] {
- left: 0;
- box-shadow: 8px 0 0px 0px var(--color-gray-4);
- }
- }
-
- [data-slot="main-content"] {
- padding-top: var(--topbar-height);
- /* Add space for the top bar */
- overflow-y: auto;
- }
-
- /* Hide the logo in the sidebar on mobile since it's in the top bar */
- [data-component="sidebar"] [data-slot="logo"] {
- display: none;
- }
-}
diff --git a/cloud/web/src/pages/components/layout.tsx b/cloud/web/src/pages/components/layout.tsx
deleted file mode 100644
index 711ed8fc..00000000
--- a/cloud/web/src/pages/components/layout.tsx
+++ /dev/null
@@ -1,96 +0,0 @@
-import style from "./layout.module.css"
-import { useAccount } from "../../components/context-account"
-import { Button } from "../../ui/button"
-import { IconLogomark } from "../../ui/svg"
-import { IconBars3BottomLeft } from "../../ui/svg/icons"
-import { ParentProps, createMemo, createSignal } from "solid-js"
-import { A, useLocation } from "@solidjs/router"
-import { useOpenAuth } from "../../components/context-openauth"
-
-export default function Layout(props: ParentProps) {
- const auth = useOpenAuth()
- const account = useAccount()
- const [sidebarOpen, setSidebarOpen] = createSignal(false)
- const location = useLocation()
-
- const workspaceId = createMemo(() => account.current?.workspaces[0].id)
- const pageTitle = createMemo(() => {
- const path = location.pathname
- if (path.endsWith("/billing")) return "Billing"
- if (path.endsWith("/keys")) return "API Keys"
- return null
- })
-
- function handleLogout() {
- auth.logout(auth.subject?.id!)
- }
-
- return (
-
- {/* Mobile top bar */}
-
-
-
-
- {pageTitle() ? (
-
{pageTitle()}
- ) : (
-
-
-
- )}
-
-
-
- {/* Backdrop for mobile sidebar - closes sidebar when clicked */}
- {sidebarOpen() &&
setSidebarOpen(false)}>
}
-
-
-
-
-
-
-
-
-
-
-
- {/* Main Content */}
-
{props.children}
-
- )
-}
diff --git a/cloud/web/src/pages/index.tsx b/cloud/web/src/pages/index.tsx
deleted file mode 100644
index 116ed156..00000000
--- a/cloud/web/src/pages/index.tsx
+++ /dev/null
@@ -1,39 +0,0 @@
-import { Match, Switch } from "solid-js"
-import { useAccount } from "../components/context-account"
-import { Navigate } from "@solidjs/router"
-import { IconLogo } from "../ui/svg"
-import styles from "./lander.module.css"
-import { useOpenAuth } from "../components/context-openauth"
-
-export default function Index() {
- const auth = useOpenAuth()
- const account = useAccount()
- return (
-
-
-
-
-
-
-
-
-
-
-
- opencode Gateway Console
-
-
-
-
- auth.authorize({ provider: "github" })}>Sign in with GitHub
-
-
- auth.authorize({ provider: "google" })}>Sign in with Google
-
-
-
-
-
-
- )
-}
diff --git a/cloud/web/src/pages/lander.module.css b/cloud/web/src/pages/lander.module.css
deleted file mode 100644
index 251e243f..00000000
--- a/cloud/web/src/pages/lander.module.css
+++ /dev/null
@@ -1,83 +0,0 @@
-.lander {
- --padding: 3rem;
- --vertical-padding: 2rem;
- --heading-font-size: 2rem;
-
- margin: 1rem;
-
- @media (max-width: 30rem) {
- & {
- --padding: 1.5rem;
- --vertical-padding: 1rem;
- --heading-font-size: 1.5rem;
-
- margin: 0.5rem;
- }
- }
-
- [data-slot="hero"] {
- border: 2px solid var(--color-border);
-
- max-width: 64rem;
- margin-left: auto;
- margin-right: auto;
- width: 100%;
- }
-
- [data-slot="top"] {
- padding: var(--padding);
-
- h1 {
- margin-top: calc(var(--vertical-padding) / 8);
- font-size: var(--heading-font-size);
- line-height: 1.25;
- text-transform: uppercase;
- font-weight: 600;
- }
-
- [data-slot="logo"] {
- width: clamp(200px, 70vw, 400px);
- color: var(--color-white);
- }
- }
-
- [data-slot="cta"] {
- display: flex;
- flex-direction: row;
- justify-content: space-between;
- border-top: 2px solid var(--color-border);
-
- & > div {
- flex: 1;
- line-height: 1.4;
- text-align: center;
- text-transform: uppercase;
- cursor: pointer;
- text-decoration: underline;
- letter-spacing: -0.03125rem;
-
- &[data-slot="col-2"] {
- background-color: var(--color-border);
- color: var(--color-text-invert);
- font-weight: 600;
- }
-
- & > * {
- display: block;
- width: 100%;
- height: 100%;
- padding: calc(var(--padding) / 2) 0.5rem;
- }
- }
-
- @media (max-width: 30rem) {
- & > div {
- padding-bottom: calc(var(--padding) / 2 + 4px);
- }
- }
-
- & > div + div {
- border-left: 2px solid var(--color-border);
- }
- }
-}
diff --git a/cloud/web/src/pages/test/design.module.css b/cloud/web/src/pages/test/design.module.css
deleted file mode 100644
index fee4e3cd..00000000
--- a/cloud/web/src/pages/test/design.module.css
+++ /dev/null
@@ -1,204 +0,0 @@
-.pageContainer {
- padding: 2rem;
- max-width: 1200px;
- margin: 0 auto;
-}
-
-.componentTable {
- width: 100%;
- border-collapse: collapse;
- table-layout: fixed;
- border: 2px solid var(--color-border);
-}
-
-.componentCell {
- padding: 1rem;
- border: 2px solid var(--color-border);
- vertical-align: top;
-}
-
-.componentLabel {
- text-transform: uppercase;
- letter-spacing: -0.03125rem;
- font-size: 0.85rem;
- font-weight: 500;
- margin-bottom: 0.75rem;
- color: var(--color-text-dimmed);
-}
-
-.sectionTitle {
- margin-bottom: 1rem;
- text-transform: uppercase;
- letter-spacing: -0.03125rem;
- font-size: 1.2rem;
-}
-
-.divider {
- height: 2px;
- background: var(--color-border);
- margin: 3rem 0;
- width: 100%;
-}
-
-.header {
- display: flex;
- justify-content: space-between;
- align-items: center;
- margin-bottom: 2rem;
-}
-
-.buttonSection {
- margin-bottom: 4rem;
-}
-
-.colorSection {
- margin-bottom: 4rem;
-}
-
-.labelSection {
- margin-bottom: 4rem;
-}
-
-.inputSection {
- margin-bottom: 4rem;
-}
-
-.dialogSection {
- margin-bottom: 4rem;
-}
-
-.formGroup {
- display: flex;
- flex-direction: column;
- gap: 0.5rem;
-}
-
-.dialogContent {
- padding: 2rem;
-}
-
-.dialogContentFooter {
- margin-top: 1rem;
-}
-
-.pageTitle {
- font-size: var(--heading-font-size, 2rem);
- text-transform: uppercase;
- font-weight: 600;
-}
-
-.colorBox {
- width: 100%;
- height: 80px;
- margin-bottom: 0.5rem;
- position: relative;
- display: flex;
- align-items: flex-end;
- justify-content: center;
- padding-bottom: 0.5rem;
-}
-
-.colorOrange {
- background-color: var(--color-orange);
-}
-
-.colorOrangeLow {
- background-color: var(--color-orange-low);
-}
-
-.colorOrangeHigh {
- background-color: var(--color-orange-high);
-}
-
-.colorGreen {
- background-color: var(--color-green);
-}
-
-.colorGreenLow {
- background-color: var(--color-green-low);
-}
-
-.colorGreenHigh {
- background-color: var(--color-green-high);
-}
-
-.colorBlue {
- background-color: var(--color-blue);
-}
-
-.colorBlueLow {
- background-color: var(--color-blue-low);
-}
-
-.colorBlueHigh {
- background-color: var(--color-blue-high);
-}
-
-.colorPurple {
- background-color: var(--color-purple);
-}
-
-.colorPurpleLow {
- background-color: var(--color-purple-low);
-}
-
-.colorPurpleHigh {
- background-color: var(--color-purple-high);
-}
-
-.colorRed {
- background-color: var(--color-red);
-}
-
-.colorRedLow {
- background-color: var(--color-red-low);
-}
-
-.colorRedHigh {
- background-color: var(--color-red-high);
-}
-
-.colorAccent {
- background-color: var(--color-accent);
-}
-
-.colorAccentLow {
- background-color: var(--color-accent-low);
-}
-
-.colorAccentHigh {
- background-color: var(--color-accent-high);
-}
-
-.colorCode {
- background-color: rgba(0, 0, 0, 0.5);
- color: white;
- padding: 4px 8px;
- border-radius: 4px;
- font-size: 0.8rem;
- font-family: monospace;
-}
-
-.colorVariants {
- display: flex;
- gap: 0.5rem;
-}
-
-.colorVariant {
- flex: 1;
- height: 40px;
- position: relative;
- display: flex;
- align-items: center;
- justify-content: center;
-}
-
-.colorVariantCode {
- background-color: rgba(0, 0, 0, 0.5);
- color: white;
- padding: 2px 4px;
- border-radius: 4px;
- font-size: 0.65rem;
- font-family: monospace;
- white-space: nowrap;
-}
diff --git a/cloud/web/src/pages/test/design.tsx b/cloud/web/src/pages/test/design.tsx
deleted file mode 100644
index 3bf75931..00000000
--- a/cloud/web/src/pages/test/design.tsx
+++ /dev/null
@@ -1,562 +0,0 @@
-import { Button } from "../../ui/button"
-import { Dialog } from "../../ui/dialog"
-import { Navigate } from "@solidjs/router"
-import { createSignal, Show } from "solid-js"
-import { IconHome, IconPencilSquare } from "../../ui/svg/icons"
-import { useTheme } from "../../components/context-theme"
-import { useDialog } from "../../ui/context-dialog"
-import { DialogString } from "../../ui/dialog-string"
-import { DialogSelect } from "../../ui/dialog-select"
-import styles from "./design.module.css"
-
-export default function DesignSystem() {
- const dialog = useDialog()
- const [dialogOpen, setDialogOpen] = createSignal(false)
- const [dialogOpenTransition, setDialogOpenTransition] = createSignal(false)
- const theme = useTheme()
-
- // Check if we're running locally
- const isLocal = import.meta.env.DEV === true
-
- if (!isLocal) {
- return
- }
-
- // Add a toggle button for theme
- const toggleTheme = () => {
- theme.setMode(theme.mode === "light" ? "dark" : "light")
- }
-
- return (
-
-
-
-
- Colors
-
-
-
-
-
- Orange
-
- hsl(41, 82%, 63%)
-
-
-
-
- hsl(41, 39%, 22%)
-
-
-
-
- hsl(41, 82%, 87%)
-
-
-
- |
-
- Green
-
- hsl(101, 82%, 63%)
-
-
-
-
- hsl(101, 39%, 22%)
-
-
-
-
- hsl(101, 82%, 80%)
-
-
-
- |
-
- Blue
-
- hsl(234, 100%, 60%)
-
-
-
-
- hsl(234, 54%, 20%)
-
-
-
-
- hsl(234, 100%, 87%)
-
-
-
- |
-
-
-
- Purple
-
- hsl(281, 82%, 63%)
-
-
-
-
- hsl(281, 39%, 22%)
-
-
-
-
- hsl(281, 82%, 89%)
-
-
-
- |
-
- Red
-
- hsl(339, 82%, 63%)
-
-
-
-
- hsl(339, 39%, 22%)
-
-
-
-
- hsl(339, 82%, 87%)
-
-
-
- |
-
- Accent
-
- hsl(13, 88%, 57%)
-
-
-
-
- hsl(13, 75%, 30%)
-
-
-
-
- hsl(13, 100%, 78%)
-
-
-
- |
-
-
-
-
-
-
-
-
-
-
-
-
- Labels
-
-
-
-
-
- Small
-
- |
-
- Medium
-
- |
-
- Large
-
- |
-
-
-
-
-
-
-
-
-
-
-
-
- Dialogs
-
-
-
-
-
- Default
-
-
- |
-
- Small With Transition
-
-
- |
-
- Input String
-
- |
-
-
-
- Select Input
-
- |
-
- Select Input
-
- |
-
- Select No Options
-
- |
-
-
-
-
-
- )
-}
diff --git a/cloud/web/src/sst-env.d.ts b/cloud/web/src/sst-env.d.ts
deleted file mode 100644
index e1ee6f75..00000000
--- a/cloud/web/src/sst-env.d.ts
+++ /dev/null
@@ -1,12 +0,0 @@
-/* This file is auto-generated by SST. Do not edit. */
-/* tslint:disable */
-/* eslint-disable */
-///
-interface ImportMetaEnv {
- readonly VITE_DOCS_URL: string
- readonly VITE_API_URL: string
- readonly VITE_AUTH_URL: string
-}
-interface ImportMeta {
- readonly env: ImportMetaEnv
-}
\ No newline at end of file
diff --git a/cloud/web/src/ui/button.tsx b/cloud/web/src/ui/button.tsx
deleted file mode 100644
index 889102dd..00000000
--- a/cloud/web/src/ui/button.tsx
+++ /dev/null
@@ -1,24 +0,0 @@
-import { Button as Kobalte } from "@kobalte/core/button"
-import { JSX, Show, splitProps } from "solid-js"
-
-export interface ButtonProps {
- color?: "primary" | "secondary" | "ghost"
- size?: "md" | "sm"
- icon?: JSX.Element
-}
-export function Button(props: JSX.IntrinsicElements["button"] & ButtonProps) {
- const [split, rest] = splitProps(props, ["color", "size", "icon"])
- return (
-
-
- {props.icon}
-
- {props.children}
-
- )
-}
diff --git a/cloud/web/src/ui/context-dialog.tsx b/cloud/web/src/ui/context-dialog.tsx
deleted file mode 100644
index f1bc9325..00000000
--- a/cloud/web/src/ui/context-dialog.tsx
+++ /dev/null
@@ -1,120 +0,0 @@
-import { createContext, JSX, ParentProps, useContext } from "solid-js"
-import { StandardSchemaV1 } from "@standard-schema/spec"
-import { createStore } from "solid-js/store"
-import { Dialog } from "./dialog"
-
-const Context = createContext()
-
-type DialogControl = {
- open>(
- component: DialogComponent,
- input: StandardSchemaV1.InferInput,
- ): void
- close(): void
- isOpen(input: any): boolean
- size: "sm" | "md"
- transition?: boolean
- input?: any
-}
-
-type DialogProps> = {
- input: StandardSchemaV1.InferInput
- control: DialogControl
-}
-
-type DialogComponent> = ReturnType<
- typeof createDialog
->
-
-export function createDialog>(props: {
- schema: Schema
- size: "sm" | "md"
- render: (props: DialogProps) => JSX.Element
-}) {
- const result = () => {
- const dialog = useDialog()
- return (
-
- )
- }
- result.schema = props.schema
- result.size = props.size
- return result
-}
-
-export function DialogProvider(props: ParentProps) {
- const [store, setStore] = createStore<{
- dialog?: DialogComponent
- input?: any
- transition?: boolean
- size: "sm" | "md"
- }>({
- size: "sm",
- })
-
- const control: DialogControl = {
- get input() {
- return store.input
- },
- get size() {
- return store.size
- },
- get transition() {
- return store.transition
- },
- isOpen(input) {
- return store.dialog === input
- },
- open(component, input) {
- setStore({
- dialog: component,
- input: input,
- size: store.dialog !== undefined ? store.size : component.size,
- transition: store.dialog !== undefined,
- })
-
- setTimeout(() => {
- setStore({
- size: component.size,
- })
- }, 0)
-
- setTimeout(() => {
- setStore({
- transition: false,
- })
- }, 150)
- },
- close() {
- setStore({
- dialog: undefined,
- })
- },
- }
-
- return (
- <>
- {props.children}
- >
- )
-}
-
-export function useDialog() {
- const ctx = useContext(Context)
- if (!ctx) {
- throw new Error("useDialog must be used within a DialogProvider")
- }
- return ctx
-}
diff --git a/cloud/web/src/ui/dialog-select.module.css b/cloud/web/src/ui/dialog-select.module.css
deleted file mode 100644
index 4a99ef02..00000000
--- a/cloud/web/src/ui/dialog-select.module.css
+++ /dev/null
@@ -1,36 +0,0 @@
-.options {
- margin-top: var(--space-1);
- border-top: 2px solid var(--color-border);
- padding: var(--space-2);
-
- [data-slot="option"] {
- outline: none;
- flex-shrink: 0;
- height: var(--space-11);
- display: flex;
- justify-content: start;
- align-items: center;
- padding: 0 var(--space-2-5);
- gap: var(--space-3);
- cursor: pointer;
-
- &[data-empty] {
- cursor: default;
- color: var(--color-text-dimmed);
- }
-
- &[data-active] {
- background-color: var(--color-bg-surface);
- }
-
- [data-slot="title"] {
- font-size: var(--font-size-md);
- }
-
- [data-slot="prefix"] {
- width: var(--space-4);
- height: var(--space-4);
- }
- }
-
-}
diff --git a/cloud/web/src/ui/dialog-select.tsx b/cloud/web/src/ui/dialog-select.tsx
deleted file mode 100644
index 087b9441..00000000
--- a/cloud/web/src/ui/dialog-select.tsx
+++ /dev/null
@@ -1,124 +0,0 @@
-import style from "./dialog-select.module.css"
-import { z } from "zod"
-import { createMemo, createSignal, For, JSX, onMount } from "solid-js"
-import { createList } from "solid-list"
-import { createDialog } from "./context-dialog"
-
-export const DialogSelect = createDialog({
- size: "md",
- schema: z.object({
- title: z.string(),
- placeholder: z.string(),
- onSelect: z
- .function(z.tuple([z.any()]))
- .returns(z.void())
- .optional(),
- options: z.array(
- z.object({
- display: z.string(),
- value: z.any().optional(),
- onSelect: z.function().returns(z.void()).optional(),
- prefix: z.custom().optional(),
- }),
- ),
- }),
- render: (ctx) => {
- let input: HTMLInputElement
- onMount(() => {
- input.focus()
- input.value = ""
- })
-
- const [filter, setFilter] = createSignal("")
- const filtered = createMemo(() =>
- ctx.input.options?.filter((i) =>
- i.display.toLowerCase().includes(filter().toLowerCase()),
- ),
- )
- const list = createList({
- loop: true,
- initialActive: 0,
- items: () => filtered().map((_, i) => i),
- handleTab: false,
- })
-
- const handleSelection = (index: number) => {
- const option = ctx.input.options[index]
-
- // If the option has its own onSelect handler, use it
- if (option.onSelect) {
- option.onSelect()
- }
- // Otherwise, if there's a global onSelect handler, call it with the option's value
- else if (ctx.input.onSelect) {
- ctx.input.onSelect(
- option.value !== undefined ? option.value : option.display,
- )
- }
- }
-
- return (
- <>
-
-
-
-
- {
- setFilter(e.target.value)
- list.setActive(0)
- }}
- onKeyDown={(e) => {
- if (e.key === "Enter") {
- const selected = list.active()
- if (selected === null) return
- handleSelection(selected)
- return
- }
- if (e.key === "Escape") {
- setFilter("")
- return
- }
- list.onKeyDown(e)
- }}
- id={`dialog-select-${ctx.input.title}`}
- ref={(r) => (input = r)}
- data-slot="input"
- placeholder={ctx.input.placeholder}
- />
-
-
-
- No results
-
- }
- >
- {(option, index) => (
- handleSelection(index())}
- data-slot="option"
- data-active={list.active() === index() ? true : undefined}
- >
- {option.prefix &&
{option.prefix}
}
-
{option.display}
-
- )}
-
-
- >
- )
- },
-})
diff --git a/cloud/web/src/ui/dialog-string.tsx b/cloud/web/src/ui/dialog-string.tsx
deleted file mode 100644
index af217478..00000000
--- a/cloud/web/src/ui/dialog-string.tsx
+++ /dev/null
@@ -1,70 +0,0 @@
-import { z } from "zod"
-import { onMount } from "solid-js"
-import { createDialog } from "./context-dialog"
-import { Button } from "./button"
-
-export const DialogString = createDialog({
- size: "sm",
- schema: z.object({
- title: z.string(),
- placeholder: z.string(),
- action: z.string(),
- onSubmit: z.function().args(z.string()).returns(z.void()),
- }),
- render: (ctx) => {
- let input: HTMLInputElement
- onMount(() => {
- setTimeout(() => {
- input.focus()
- input.value = ""
- }, 50)
- })
-
- function submit() {
- const value = input.value.trim()
- if (value) {
- ctx.input.onSubmit(value)
- ctx.control.close()
- }
- }
-
- return (
- <>
-
-
-
-
- (input = r)}
- placeholder={ctx.input.placeholder}
- id={`dialog-string-${ctx.input.title}`}
- onKeyDown={(e) => {
- if (e.key === "Enter") {
- e.preventDefault()
- submit()
- }
- }}
- />
-
-
-
-
-
- >
- )
- },
-})
diff --git a/cloud/web/src/ui/dialog.tsx b/cloud/web/src/ui/dialog.tsx
deleted file mode 100644
index 101f23d2..00000000
--- a/cloud/web/src/ui/dialog.tsx
+++ /dev/null
@@ -1,27 +0,0 @@
-import { Dialog as Kobalte } from "@kobalte/core/dialog"
-import { ComponentProps, ParentProps } from "solid-js"
-
-export type Props = ParentProps<{
- size?: "sm" | "md"
- transition?: boolean
-}> &
- ComponentProps
-
-export function Dialog(props: Props) {
- return (
-
-
-
-
-
- {props.children}
-
-
-
-
- )
-}
diff --git a/cloud/web/src/ui/style/component/button.css b/cloud/web/src/ui/style/component/button.css
deleted file mode 100644
index 9604f986..00000000
--- a/cloud/web/src/ui/style/component/button.css
+++ /dev/null
@@ -1,78 +0,0 @@
-[data-component="button"] {
- width: fit-content;
- display: flex;
- line-height: 1;
- align-items: center;
- justify-content: center;
- gap: var(--space-2);
- font-size: var(--font-size-md);
- text-transform: uppercase;
- height: var(--space-11);
- outline: none;
- font-weight: 500;
- padding: 0 var(--space-4);
- border-width: 2px;
- border-color: var(--color-border);
- cursor: pointer;
-
- &:disabled {
- opacity: 0.5;
- cursor: default;
- }
-
- &[data-color="primary"] {
- background-color: var(--color-text);
- border-color: var(--color-text);
- color: var(--color-text-invert);
-
- &:active {
- border-color: var(--color-accent);
- }
- }
-
- &[data-color="secondary"] {
- &:active {
- border-color: var(--color-accent);
- }
- }
-
- &[data-color="ghost"] {
- border: none;
- text-decoration: underline;
-
- &:active {
- color: var(--color-text-accent);
- }
- }
-
- &:has([data-slot="icon"]) {
- padding-left: var(--space-3);
- padding-right: var(--space-3);
- }
-
- &[data-size="sm"] {
- height: var(--space-8);
- padding: var(--space-3);
- font-size: var(--font-size-xs);
-
- [data-slot="icon"] {
- width: var(--space-3-5);
- height: var(--space-3-5);
- }
-
- &:has([data-slot="icon"]) {
- padding-left: var(--space-2);
- padding-right: var(--space-2);
- }
- }
-
- [data-slot="icon"] {
- width: var(--space-4);
- height: var(--space-4);
- transition: transform 0.2s ease;
- }
-
- &[data-rotate] [data-slot="icon"] {
- transform: rotate(180deg);
- }
-}
diff --git a/cloud/web/src/ui/style/component/dialog.css b/cloud/web/src/ui/style/component/dialog.css
deleted file mode 100644
index 59867818..00000000
--- a/cloud/web/src/ui/style/component/dialog.css
+++ /dev/null
@@ -1,84 +0,0 @@
-[data-component="dialog-overlay"] {
- pointer-events: none !important;
- position: fixed;
- inset: 0;
- animation-name: fadeOut;
- animation-duration: 200ms;
- animation-timing-function: ease;
- opacity: 0;
- backdrop-filter: blur(2px);
-
- &[data-expanded] {
- animation-name: fadeIn;
- opacity: 1;
- pointer-events: auto !important;
- }
-}
-
-[data-component="dialog-center"] {
- position: fixed;
- inset: 0;
- padding-top: 10vh;
- justify-content: center;
- pointer-events: none;
-
- [data-slot="content"] {
- width: 45rem;
- margin: 0 auto;
- transition: 150ms width;
- background-color: var(--color-bg);
- border-width: 2px;
- border-color: var(--color-border);
- overflow: hidden;
- display: flex;
- flex-direction: column;
- gap: var(--space-3);
- outline: none;
- animation-duration: 1ms;
- animation-name: zoomOut;
- animation-timing-function: ease;
-
- box-shadow: 8px 8px 0px 0px var(--color-gray-4);
-
- &[data-expanded] {
- animation-name: zoomIn;
- }
-
- &[data-transition] {
- animation-duration: 200ms;
- }
-
- &[data-size="sm"] {
- width: 30rem;
- }
-
- [data-slot="header"] {
- display: flex;
- padding: var(--space-4) var(--space-4) 0;
-
- [data-slot="title"] {
- }
- }
-
- [data-slot="main"] {
- padding: 0 var(--space-4);
-
- &:has([data-slot="options"]) {
- padding: 0;
- display: flex;
- flex-direction: column;
- gap: var(--space-4);
- }
- }
-
- [data-slot="input"] {
- }
-
- [data-slot="footer"] {
- padding: var(--space-4);
- display: flex;
- gap: var(--space-4);
- justify-content: end;
- }
- }
-}
diff --git a/cloud/web/src/ui/style/component/input.css b/cloud/web/src/ui/style/component/input.css
deleted file mode 100644
index 59535d76..00000000
--- a/cloud/web/src/ui/style/component/input.css
+++ /dev/null
@@ -1,34 +0,0 @@
-[data-component="input"] {
- font-size: var(--font-size-md);
- background: transparent;
- caret-color: var(--color-accent);
- font-family: var(--font-mono);
- height: var(--space-11);
- padding: 0 var(--space-4);
- width: 100%;
- resize: none;
- border: 2px solid var(--color-border);
-
- &::placeholder {
- color: var(--color-text-dimmed);
- opacity: 0.75;
- }
-
- &:focus {
- outline: 0;
- }
-
- &[data-size="sm"] {
- height: var(--space-9);
- padding: 0 var(--space-3);
- font-size: var(--font-size-xs);
- }
-
- &[data-size="md"] {
- }
-
- &[data-size="lg"] {
- height: var(--space-12);
- font-size: var(--font-size-lg);
- }
-}
diff --git a/cloud/web/src/ui/style/component/label.css b/cloud/web/src/ui/style/component/label.css
deleted file mode 100644
index e0dd5fef..00000000
--- a/cloud/web/src/ui/style/component/label.css
+++ /dev/null
@@ -1,17 +0,0 @@
-[data-component="label"] {
- letter-spacing: -0.03125rem;
- text-transform: uppercase;
- color: var(--color-text-dimmed);
- font-weight: 500;
- font-size: var(--font-size-md);
-
- &[data-size="sm"] {
- font-size: var(--font-size-sm);
- }
- &[data-size="md"] {
- }
- &[data-size="lg"] {
- font-size: var(--font-size-lg);
- }
-}
-
diff --git a/cloud/web/src/ui/style/component/title-bar.css b/cloud/web/src/ui/style/component/title-bar.css
deleted file mode 100644
index 7ee32bfd..00000000
--- a/cloud/web/src/ui/style/component/title-bar.css
+++ /dev/null
@@ -1,32 +0,0 @@
-[data-component="title-bar"] {
- display: flex;
- align-items: center;
- justify-content: space-between;
- height: 72px;
- padding: 0 var(--space-4);
- border-bottom: 2px solid var(--color-border);
-
- [data-slot="left"] {
- display: flex;
- flex-direction: column;
- gap: var(--space-1-5);
-
- h1 {
- letter-spacing: -0.03125rem;
- font-size: var(--font-size-xl);
- text-transform: uppercase;
- font-weight: 600;
- }
-
- p {
- color: var(--color-text-dimmed);
- }
- }
-
-}
-
-@media (max-width: 40rem) {
- [data-component="title-bar"] {
- display: none;
- }
-}
diff --git a/cloud/web/src/ui/style/index.css b/cloud/web/src/ui/style/index.css
deleted file mode 100644
index 117f596d..00000000
--- a/cloud/web/src/ui/style/index.css
+++ /dev/null
@@ -1,50 +0,0 @@
-/* tokens */
-@import "./token/color.css";
-@import "./token/reset.css";
-@import "./token/animation.css";
-@import "./token/font.css";
-@import "./token/space.css";
-
-/* components */
-@import "./component/label.css";
-@import "./component/input.css";
-@import "./component/button.css";
-@import "./component/dialog.css";
-@import "./component/title-bar.css";
-
-body {
- font-family: var(--font-mono);
- line-height: 1;
- color: var(--color-text);
- background-color: var(--color-bg);
- cursor: default;
- user-select: none;
- text-underline-offset: 0.1875rem;
-}
-
-a {
- text-decoration: underline;
- &:active {
- color: var(--color-text-accent);
- }
-}
-
-::selection {
- background-color: var(--color-text-accent-invert);
-}
-
-/* Responsive utilities */
-[data-max-width] {
- width: 100%;
-
- & > * {
- max-width: 90rem;
- margin-left: auto;
- margin-right: auto;
- width: 100%;
- }
-
- &[data-max-width-64] > * {
- max-width: 64rem;
- }
-}
diff --git a/cloud/web/src/ui/style/token/animation.css b/cloud/web/src/ui/style/token/animation.css
deleted file mode 100644
index a8edfeff..00000000
--- a/cloud/web/src/ui/style/token/animation.css
+++ /dev/null
@@ -1,23 +0,0 @@
-@keyframes zoomIn {
- from {
- opacity: 0;
- transform: scale(0.95);
- }
-
- to {
- opacity: 1;
- transform: scale(1);
- }
-}
-
-@keyframes zoomOut {
- from {
- opacity: 1;
- transform: scale(1);
- }
-
- to {
- opacity: 0;
- transform: scale(0.95);
- }
-}
diff --git a/cloud/web/src/ui/style/token/color.css b/cloud/web/src/ui/style/token/color.css
deleted file mode 100644
index af0c46f3..00000000
--- a/cloud/web/src/ui/style/token/color.css
+++ /dev/null
@@ -1,88 +0,0 @@
-:root {
- --color-white: hsl(0, 0%, 100%);
- --color-gray-1: hsl(224, 20%, 94%);
- --color-gray-2: hsl(224, 6%, 77%);
- --color-gray-3: hsl(224, 6%, 56%);
- --color-gray-4: hsl(224, 7%, 36%);
- --color-gray-5: hsl(224, 10%, 23%);
- --color-gray-6: hsl(224, 14%, 16%);
- --color-black: hsl(224, 10%, 10%);
-
- --hue-orange: 41;
- --color-orange-low: hsl(var(--hue-orange), 39%, 22%);
- --color-orange: hsl(var(--hue-orange), 82%, 63%);
- --color-orange-high: hsl(var(--hue-orange), 82%, 87%);
- --hue-green: 101;
- --color-green-low: hsl(var(--hue-green), 39%, 22%);
- --color-green: hsl(var(--hue-green), 82%, 63%);
- --color-green-high: hsl(var(--hue-green), 82%, 80%);
- --hue-blue: 234;
- --color-blue-low: hsl(var(--hue-blue), 54%, 20%);
- --color-blue: hsl(var(--hue-blue), 100%, 60%);
- --color-blue-high: hsl(var(--hue-blue), 100%, 87%);
- --hue-purple: 281;
- --color-purple-low: hsl(var(--hue-purple), 39%, 22%);
- --color-purple: hsl(var(--hue-purple), 82%, 63%);
- --color-purple-high: hsl(var(--hue-purple), 82%, 89%);
- --hue-red: 339;
- --color-red-low: hsl(var(--hue-red), 39%, 22%);
- --color-red: hsl(var(--hue-red), 82%, 63%);
- --color-red-high: hsl(var(--hue-red), 82%, 87%);
-
- --color-accent-low: hsl(13, 75%, 30%);
- --color-accent: hsl(13, 88%, 57%);
- --color-accent-high: hsl(13, 100%, 78%);
-
- --color-text: var(--color-gray-1);
- --color-text-dimmed: var(--color-gray-3);
- --color-text-accent: var(--color-accent);
- --color-text-invert: var(--color-black);
- --color-text-accent-invert: var(--color-accent-high);
- --color-bg: var(--color-black);
- --color-bg-surface: var(--color-gray-5);
- --color-bg-accent: var(--color-accent-high);
- --color-border: var(--color-gray-2);
-
- --color-backdrop-overlay: hsla(223, 13%, 10%, 0.66);
-}
-
-:root[data-color-mode="light"] {
- --color-white: hsl(224, 10%, 10%);
- --color-gray-1: hsl(224, 14%, 16%);
- --color-gray-2: hsl(224, 10%, 23%);
- --color-gray-3: hsl(224, 7%, 36%);
- --color-gray-4: hsl(224, 6%, 56%);
- --color-gray-5: hsl(224, 6%, 77%);
- --color-gray-6: hsl(224, 20%, 94%);
- --color-gray-7: hsl(224, 19%, 97%);
- --color-black: hsl(0, 0%, 100%);
-
- --color-orange-high: hsl(var(--hue-orange), 80%, 25%);
- --color-orange: hsl(var(--hue-orange), 90%, 60%);
- --color-orange-low: hsl(var(--hue-orange), 90%, 88%);
- --color-green-high: hsl(var(--hue-green), 80%, 22%);
- --color-green: hsl(var(--hue-green), 90%, 46%);
- --color-green-low: hsl(var(--hue-green), 85%, 90%);
- --color-blue-high: hsl(var(--hue-blue), 80%, 30%);
- --color-blue: hsl(var(--hue-blue), 90%, 60%);
- --color-blue-low: hsl(var(--hue-blue), 88%, 90%);
- --color-purple-high: hsl(var(--hue-purple), 90%, 30%);
- --color-purple: hsl(var(--hue-purple), 90%, 60%);
- --color-purple-low: hsl(var(--hue-purple), 80%, 90%);
- --color-red-high: hsl(var(--hue-red), 80%, 30%);
- --color-red: hsl(var(--hue-red), 90%, 60%);
- --color-red-low: hsl(var(--hue-red), 80%, 90%);
-
- --color-accent-high: hsl(13, 75%, 26%);
- --color-accent: hsl(13, 88%, 60%);
- --color-accent-low: hsl(13, 100%, 89%);
-
- --color-text-accent: var(--color-accent);
- --color-text-dimmed: var(--color-gray-4);
- --color-text-invert: var(--color-black);
- --color-text-accent-invert: var(--color-accent-low);
- --color-bg-surface: var(--color-gray-6);
- --color-bg-accent: var(--color-accent);
-
- --color-backdrop-overlay: hsla(225, 9%, 36%, 0.66);
-}
diff --git a/cloud/web/src/ui/style/token/font.css b/cloud/web/src/ui/style/token/font.css
deleted file mode 100644
index 24b2db3f..00000000
--- a/cloud/web/src/ui/style/token/font.css
+++ /dev/null
@@ -1,20 +0,0 @@
-:root {
- --font-size-2xs: 0.6875rem;
- --font-size-xs: 0.75rem;
- --font-size-sm: 0.8125rem;
- --font-size-md: 0.9375rem;
- --font-size-lg: 1.125rem;
- --font-size-xl: 1.25rem;
- --font-size-2xl: 1.5rem;
- --font-size-3xl: 1.875rem;
- --font-size-4xl: 2.25rem;
- --font-size-5xl: 3rem;
- --font-size-6xl: 3.75rem;
- --font-size-7xl: 4.5rem;
- --font-size-8xl: 6rem;
- --font-size-9xl: 8rem;
- --font-mono: IBM Plex Mono, monospace;
- --font-sans: Rubik, sans-serif;
-
- --font-line-height: 1.75;
-}
diff --git a/cloud/web/src/ui/style/token/reset.css b/cloud/web/src/ui/style/token/reset.css
deleted file mode 100644
index f4aa1a0a..00000000
--- a/cloud/web/src/ui/style/token/reset.css
+++ /dev/null
@@ -1,212 +0,0 @@
-* {
- margin: 0;
- padding: 0;
- font: inherit;
-}
-
-*,
-*::before,
-*::after {
- box-sizing: border-box;
- border-width: 0;
- border-style: solid;
- border-color: var(--global-color-border, currentColor);
-}
-
-html {
- line-height: 1.5;
- --font-fallback: ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont,
- "Segoe UI", Roboto, "Helvetica Neue", Arial, "Noto Sans", sans-serif,
- "Apple Color Emoji", "Segoe UI Emoji", "Segoe UI Symbol", "Noto Color Emoji";
- -webkit-text-size-adjust: 100%;
- -webkit-font-smoothing: antialiased;
- -moz-osx-font-smoothing: grayscale;
- -moz-tab-size: 4;
- tab-size: 4;
- font-family: var(--global-font-body, var(--font-fallback));
-}
-
-hr {
- height: 0;
- color: inherit;
- border-top-width: 1px;
-}
-
-body {
- height: 100%;
- line-height: inherit;
-}
-
-img {
- border-style: none;
-}
-
-img,
-svg,
-video,
-canvas,
-audio,
-iframe,
-embed,
-object {
- display: block;
- vertical-align: middle;
-}
-
-img,
-video {
- max-width: 100%;
- height: auto;
-}
-
-p,
-h1,
-h2,
-h3,
-h4,
-h5,
-h6 {
- overflow-wrap: break-word;
-}
-
-ol,
-ul {
- list-style: none;
-}
-
-code,
-kbd,
-pre,
-samp {
- font-size: 1em;
-}
-
-button,
-[type="button"],
-[type="reset"],
-[type="submit"] {
- -webkit-appearance: button;
- background-color: transparent;
- background-image: none;
-}
-
-button,
-input,
-optgroup,
-select,
-textarea {
- color: inherit;
-}
-
-button,
-select {
- text-transform: none;
-}
-
-table {
- text-indent: 0;
- border-color: inherit;
- border-collapse: collapse;
-}
-
-input::placeholder,
-textarea::placeholder {
- opacity: 1;
- color: var(--global-color-placeholder, #9ca3af);
-}
-
-textarea {
- resize: vertical;
-}
-
-summary {
- display: list-item;
-}
-
-small {
- font-size: 80%;
-}
-
-sub,
-sup {
- font-size: 75%;
- line-height: 0;
- position: relative;
- vertical-align: baseline;
-}
-
-sub {
- bottom: -0.25em;
-}
-
-sup {
- top: -0.5em;
-}
-
-dialog {
- padding: 0;
-}
-
-a {
- color: inherit;
- text-decoration: inherit;
-}
-
-abbr:where([title]) {
- text-decoration: underline dotted;
-}
-
-b,
-strong {
- font-weight: bolder;
-}
-
-code,
-kbd,
-samp,
-pre {
- font-size: 1em;
- --font-mono-fallback: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas,
- "Liberation Mono", "Courier New";
- font-family: var(--global-font-mono, var(--font-fallback));
-}
-
-input[type="text"],
-input[type="email"],
-input[type="search"],
-input[type="password"] {
- -webkit-appearance: none;
- -moz-appearance: none;
-}
-
-input[type="search"] {
- -webkit-appearance: textfield;
- outline-offset: -2px;
-}
-
-::-webkit-search-decoration,
-::-webkit-search-cancel-button {
- -webkit-appearance: none;
-}
-
-::-webkit-file-upload-button {
- -webkit-appearance: button;
- font: inherit;
-}
-
-input[type="number"]::-webkit-inner-spin-button,
-input[type="number"]::-webkit-outer-spin-button {
- height: auto;
-}
-
-input[type="number"] {
- -moz-appearance: textfield;
-}
-
-:-moz-ui-invalid {
- box-shadow: none;
-}
-
-:-moz-focusring {
- outline: auto;
-}
diff --git a/cloud/web/src/ui/style/token/space.css b/cloud/web/src/ui/style/token/space.css
deleted file mode 100644
index 4a061d75..00000000
--- a/cloud/web/src/ui/style/token/space.css
+++ /dev/null
@@ -1,39 +0,0 @@
-:root {
- --space-0: 0;
- --space-px: 1px;
- --space-0-5: 0.125rem;
- --space-1: 0.25rem;
- --space-1-5: 0.375rem;
- --space-2: 0.5rem;
- --space-2-5: 0.625rem;
- --space-3: 0.75rem;
- --space-3-5: 0.875rem;
- --space-4: 1rem;
- --space-4-5: 1.125rem;
- --space-5: 1.25rem;
- --space-6: 1.5rem;
- --space-7: 1.75rem;
- --space-8: 2rem;
- --space-9: 2.25rem;
- --space-10: 2.5rem;
- --space-11: 2.75rem;
- --space-12: 3rem;
- --space-14: 3.5rem;
- --space-16: 4rem;
- --space-18: 4.5rem;
- --space-20: 5rem;
- --space-24: 6rem;
- --space-28: 7rem;
- --space-32: 8rem;
- --space-36: 9rem;
- --space-40: 10rem;
- --space-44: 11rem;
- --space-48: 12rem;
- --space-52: 13rem;
- --space-56: 14rem;
- --space-60: 15rem;
- --space-64: 16rem;
- --space-72: 18rem;
- --space-80: 20rem;
- --space-96: 24rem;
-}
diff --git a/cloud/web/src/ui/svg/icons.tsx b/cloud/web/src/ui/svg/icons.tsx
deleted file mode 100644
index c09bbc47..00000000
--- a/cloud/web/src/ui/svg/icons.tsx
+++ /dev/null
@@ -1,1292 +0,0 @@
-import { JSX } from "solid-js"
-
-export function IconPencilSquare(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconHome(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconPlus(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconDocument(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconChat(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconBell(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconCheck(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconChevronDown(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconChevronUp(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconChevronLeft(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconChevronRight(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconTrash(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconUser(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconCog(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconExclamationCircle(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconInformationCircle(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowPath(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowUpRight(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconEllipsisVertical(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconEllipsisHorizontal(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconXMark(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconAcademicCap(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconBolt(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconCalendar(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconCamera(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconClock(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconCloud(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconCreditCard(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconEnvelope(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconEye(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconFlag(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconFolder(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconGlobe(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconHeart(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconKey(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconLink(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconLock(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconMap(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconMicrophone(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconPhone(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconPhoto(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconQuestionMarkCircle(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconMagnifyingGlass(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconShieldCheck(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconShoppingCart(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconStar(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconTag(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconUserCircle(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconVideoCamera(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconWifi(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconAdjustmentsVertical(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArchiveBox(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowDown(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowLeft(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowLongDown(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowLongLeft(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowLongRight(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowLongUp(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowRight(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowSmallDown(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowSmallLeft(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowSmallRight(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowSmallUp(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowTopRightOnSquare(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowTrendingDown(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowTrendingUp(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowUp(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowUpCircle(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowUpLeft(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowUpOnSquare(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowUpTray(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconArrowsPointingIn(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowsPointingOut(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconArrowsRightLeft(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
-
-export function IconBars3BottomLeft(
- props: JSX.SvgSVGAttributes,
-) {
- return (
-
- )
-}
diff --git a/cloud/web/src/ui/svg/index.tsx b/cloud/web/src/ui/svg/index.tsx
deleted file mode 100644
index 23dd74c6..00000000
--- a/cloud/web/src/ui/svg/index.tsx
+++ /dev/null
@@ -1,67 +0,0 @@
-import { JSX } from "solid-js"
-
-export function IconLogomark(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
-
-export function IconLogo(props: JSX.SvgSVGAttributes) {
- return (
-
- )
-}
diff --git a/cloud/web/src/util/context.tsx b/cloud/web/src/util/context.tsx
deleted file mode 100644
index d1c6f4e7..00000000
--- a/cloud/web/src/util/context.tsx
+++ /dev/null
@@ -1,26 +0,0 @@
-import { ParentProps, Show, createContext, useContext } from "solid-js"
-
-export function createInitializedContext<
- Name extends string,
- T extends { ready: boolean },
->(name: Name, cb: () => T) {
- const ctx = createContext()
-
- return {
- use: () => {
- const context = useContext(ctx)
- if (!context) throw new Error(`No ${name} context`)
- return context
- },
- provider: (props: ParentProps) => {
- const value = cb()
- return (
-
-
- {props.children}
-
-
- )
- },
- }
-}
diff --git a/cloud/web/sst-env.d.ts b/cloud/web/sst-env.d.ts
deleted file mode 100644
index b6a7e906..00000000
--- a/cloud/web/sst-env.d.ts
+++ /dev/null
@@ -1,9 +0,0 @@
-/* This file is auto-generated by SST. Do not edit. */
-/* tslint:disable */
-/* eslint-disable */
-/* deno-fmt-ignore-file */
-
-///
-
-import "sst"
-export {}
\ No newline at end of file
diff --git a/cloud/web/tsconfig.json b/cloud/web/tsconfig.json
deleted file mode 100644
index 98d5b9ce..00000000
--- a/cloud/web/tsconfig.json
+++ /dev/null
@@ -1,12 +0,0 @@
-{
- "$schema": "https://json.schemastore.org/tsconfig",
- "extends": "@tsconfig/node22/tsconfig.json",
- "compilerOptions": {
- "jsx": "preserve",
- "jsxImportSource": "solid-js",
- "module": "esnext",
- "moduleResolution": "bundler",
- "lib": ["ESNext", "DOM", "DOM.Iterable"],
- "types": ["vite/client"]
- }
-}
diff --git a/cloud/web/vite.config.ts b/cloud/web/vite.config.ts
deleted file mode 100644
index 8a569641..00000000
--- a/cloud/web/vite.config.ts
+++ /dev/null
@@ -1,63 +0,0 @@
-import { defineConfig } from "vite"
-import solidPlugin from "vite-plugin-solid"
-import pages from "vite-plugin-pages"
-import fs from "fs"
-import path from "path"
-import { generateHydrationScript, getAssets } from "solid-js/web"
-
-export default defineConfig({
- plugins: [
- pages({
- exclude: ["**/~*", "**/components/*"],
- }),
- solidPlugin({ ssr: true }),
- {
- name: "vite-plugin-solid-ssr-render",
- apply: (config, env) => {
- return env.command === "build" && !config.build?.ssr
- },
- closeBundle: async () => {
- console.log("Pre-rendering pages...")
- const dist = path.resolve("dist")
- try {
- const serverEntryPath = path.join(dist, "server/entry-server.js")
- const serverEntry = await import(serverEntryPath + "?t=" + Date.now())
-
- const template = fs.readFileSync(
- path.join(dist, "client/index.html"),
- "utf-8",
- )
- fs.writeFileSync(path.join(dist, "client/fallback.html"), template)
-
- const routes = ["/"]
- for (const route of routes) {
- const { app } = await serverEntry.render({ url: route })
- const html = template
- .replace("", app)
- .replace("", generateHydrationScript())
- .replace("", getAssets())
- const filePath = path.join(
- dist,
- `client${route === "/" ? "/index" : route}.html`,
- )
- fs.mkdirSync(path.dirname(filePath), {
- recursive: true,
- })
- fs.writeFileSync(filePath, html)
-
- console.log(`Pre-rendered: ${filePath}`)
- }
- } catch (error) {
- console.error("Error during pre-rendering:", error)
- }
- },
- },
- ],
- server: {
- port: 3000,
- host: "0.0.0.0",
- },
- build: {
- target: "esnext",
- },
-})
diff --git a/infra/cloud.ts b/infra/cloud.ts
index 542aef96..b7b75938 100644
--- a/infra/cloud.ts
+++ b/infra/cloud.ts
@@ -46,7 +46,7 @@ export const auth = new sst.cloudflare.Worker("AuthApi", {
////////////////
export const stripeWebhook = new WebhookEndpoint("StripeWebhook", {
- url: $interpolate`https://api.gateway.${domain}/stripe/webhook`,
+ url: $interpolate`https://console.${domain}/stripe/webhook`,
enabledEvents: [
"checkout.session.async_payment_failed",
"checkout.session.async_payment_succeeded",
@@ -123,13 +123,3 @@ export const console = new sst.cloudflare.x.SolidStart("Console", {
VITE_AUTH_URL: auth.url.apply((url) => url!),
},
})
-
-//new sst.x.DevCommand("Solid", {
-// dev: {
-// directory: "cloud/app",
-// command: "bun dev",
-// },
-// environment: {
-// VITE_AUTH_URL: auth.url.apply((url) => url!),
-// },
-//})