whiteboard
This commit is contained in:
parent
6dbbd74c01
commit
8e3ef39116
51
bun.lock
51
bun.lock
|
|
@ -142,6 +142,7 @@
|
|||
"@workshop/nano-remix": "workspace:*",
|
||||
"@workshop/shared": "workspace:*",
|
||||
"hono": "catalog:",
|
||||
"opencv4nodejs": "^5.6.0",
|
||||
"zod": "3.25.67",
|
||||
},
|
||||
"devDependencies": {
|
||||
|
|
@ -249,10 +250,16 @@
|
|||
|
||||
"amqplib": ["amqplib@0.5.2", "", { "dependencies": { "bitsyntax": "~0.0.4", "bluebird": "^3.4.6", "buffer-more-ints": "0.0.2", "readable-stream": "1.x >=1.1.9", "safe-buffer": "^5.0.1" } }, "sha512-l9mCs6LbydtHqRniRwYkKdqxVa6XMz3Vw1fh+2gJaaVgTM6Jk3o8RccAKWKtlhT1US5sWrFh+KKxsVUALURSIA=="],
|
||||
|
||||
"ansi-regex": ["ansi-regex@2.1.1", "", {}, "sha512-TIGnTpdo+E3+pCyAluZvtED5p5wCqLdezCyhPZzKPcxvFplEt4i+W7OONCKgeZFT3+y5NZZfOOS/Bdcanm1MYA=="],
|
||||
|
||||
"ansi-styles": ["ansi-styles@3.2.1", "", { "dependencies": { "color-convert": "^1.9.0" } }, "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA=="],
|
||||
|
||||
"app-root-path": ["app-root-path@2.1.0", "", {}, "sha512-z5BqVjscbjmJBybKlICogJR2jCr2q/Ixu7Pvui5D4y97i7FLsJlvEG9XOR/KJRlkxxZz7UaaS2TMwQh1dRJ2dA=="],
|
||||
|
||||
"aproba": ["aproba@1.2.0", "", {}, "sha512-Y9J6ZjXtoYh8RnXVCMOU/ttDmk1aBjunq9vO0ta5x85WDQiQfUF9sIPBITdbiiIVcBo03Hi3jMxigBtsddlXRw=="],
|
||||
|
||||
"are-we-there-yet": ["are-we-there-yet@1.1.7", "", { "dependencies": { "delegates": "^1.0.0", "readable-stream": "^2.0.6" } }, "sha512-nxwy40TuMiUGqMyRHgCSWZ9FM4VAoRP4xUYSTv5ImRog+h9yISPbVH7H8fASCIzYn9wlEv4zvFL7uKDMCFQm3g=="],
|
||||
|
||||
"array-buffer-byte-length": ["array-buffer-byte-length@1.0.2", "", { "dependencies": { "call-bound": "^1.0.3", "is-array-buffer": "^3.0.5" } }, "sha512-LHE+8BuR7RYGDKvnrmcuSq3tDcKv9OFEXQt/HpbZhY7V6h0zlUXutnAD82GiFx9rdieCMjkvtcsPqBwgUl1Iiw=="],
|
||||
|
||||
"array-flatten": ["array-flatten@1.1.1", "", {}, "sha512-PCVAQswWemu6UdxsDFFX/+gVeYqKAod3D3UVm91jHwynguOwAvYPhx8nNlM++NqRcK6CxxpUafjmhIdKiHibqg=="],
|
||||
|
|
@ -305,6 +312,8 @@
|
|||
|
||||
"chalk": ["chalk@2.4.1", "", { "dependencies": { "ansi-styles": "^3.2.1", "escape-string-regexp": "^1.0.5", "supports-color": "^5.3.0" } }, "sha512-ObN6h1v2fTJSmUXoS3nMQ92LbDK9be4TV+6G+omQlGJFdcUX5heKi1LZ1YnRMIgwTLEj3E24bT6tYni50rlCfQ=="],
|
||||
|
||||
"code-point-at": ["code-point-at@1.1.0", "", {}, "sha512-RpAVKQA5T63xEj6/giIbUEtZwJ4UFIc3ZtvEkiaUERylqe8xb5IvqcgOurZLahv93CLKfxcw5YI+DZcUBRyLXA=="],
|
||||
|
||||
"color-convert": ["color-convert@1.9.3", "", { "dependencies": { "color-name": "1.1.3" } }, "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg=="],
|
||||
|
||||
"color-name": ["color-name@1.1.3", "", {}, "sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw=="],
|
||||
|
|
@ -317,6 +326,8 @@
|
|||
|
||||
"compression": ["compression@1.7.3", "", { "dependencies": { "accepts": "~1.3.5", "bytes": "3.0.0", "compressible": "~2.0.14", "debug": "2.6.9", "on-headers": "~1.0.1", "safe-buffer": "5.1.2", "vary": "~1.1.2" } }, "sha512-HSjyBG5N1Nnz7tF2+O7A9XUhyjru71/fwgNb7oIsEVHR0WShfs2tIS/EySLgiTe98aOK18YDlMXpzjCXY/n9mg=="],
|
||||
|
||||
"console-control-strings": ["console-control-strings@1.1.0", "", {}, "sha512-ty/fTekppD2fIwRvnZAVdeOiGd1c7YXEixbgJTNzqcxJWKQnjJ/V1bNEEE6hygpM3WjwHFUVK6HTjWSzV4a8sQ=="],
|
||||
|
||||
"content-disposition": ["content-disposition@0.5.2", "", {}, "sha512-kRGRZw3bLlFISDBgwTSA1TMBFN6J6GWDeubmDE3AF+3+yXL8hTWv8r5rkLbqYXY4RjPk/EzHnClI3zQf1cFmHA=="],
|
||||
|
||||
"content-type": ["content-type@1.0.4", "", {}, "sha512-hIP3EEPs8tB9AT1L+NUqtwOAps4mk2Zob89MWXMHjHWg9milF/j4osnnQLXBCBFBk/tvIG/tUc9mOUJiPBhPXA=="],
|
||||
|
|
@ -353,6 +364,8 @@
|
|||
|
||||
"define-properties": ["define-properties@1.2.1", "", { "dependencies": { "define-data-property": "^1.0.1", "has-property-descriptors": "^1.0.0", "object-keys": "^1.1.1" } }, "sha512-8QmQKqEASLd5nx0U1B1okLElbUuuttJ/AnYmRXbbbGDWh6uS208EjD4Xqq/I9wK7u0v6O08XhTWnt5XtEbR6Dg=="],
|
||||
|
||||
"delegates": ["delegates@1.0.0", "", {}, "sha512-bd2L678uiWATM6m5Z1VzNCErI3jiGzt6HGY8OVICs40JQq/HALfbyNJmp0UDakEY4pMMaN0Ly5om/B1VI/+xfQ=="],
|
||||
|
||||
"depd": ["depd@1.1.2", "", {}, "sha512-7emPTl6Dpo6JRXOXjLRxck+FlLRX5847cLKEn00PLAgc3g2hTZZgr+e4c2v6QpSmLeFP3n5yUo7ft6avBK/5jQ=="],
|
||||
|
||||
"destroy": ["destroy@1.0.4", "", {}, "sha512-3NdhDuEXnfun/z7x9GOElY49LoqVHoGScmOKwmxhsS8N5Y+Z8KyPPDnaSzqWgYt/ji4mqwfTS34Htrk0zPIXVg=="],
|
||||
|
|
@ -427,6 +440,8 @@
|
|||
|
||||
"functions-have-names": ["functions-have-names@1.2.3", "", {}, "sha512-xckBUXyTIqT97tq2x2AMb+g163b5JFysYk0x4qxNFwbfQkmNZoiRHb6sPzI9/QV33WeuvVYBUIiD4NzNIyqaRQ=="],
|
||||
|
||||
"gauge": ["gauge@2.7.4", "", { "dependencies": { "aproba": "^1.0.3", "console-control-strings": "^1.0.0", "has-unicode": "^2.0.0", "object-assign": "^4.1.0", "signal-exit": "^3.0.0", "string-width": "^1.0.1", "strip-ansi": "^3.0.1", "wide-align": "^1.1.0" } }, "sha512-14x4kjc6lkD3ltw589k0NrPD6cCNTD6CWoVUNpB85+DrtONoZn+Rug6xZU5RvSC4+TZPxA5AnBibQYAvZn41Hg=="],
|
||||
|
||||
"gaxios": ["gaxios@6.7.1", "", { "dependencies": { "extend": "^3.0.2", "https-proxy-agent": "^7.0.1", "is-stream": "^2.0.0", "node-fetch": "^2.6.9", "uuid": "^9.0.1" } }, "sha512-LDODD4TMYx7XXdpwxAVRAIAuB0bzv0s+ywFonY46k126qzQHT9ygyoa9tncmOiQmmDrik65UYsEkv3lbfqQ3yQ=="],
|
||||
|
||||
"gcp-metadata": ["gcp-metadata@6.1.1", "", { "dependencies": { "gaxios": "^6.1.1", "google-logging-utils": "^0.0.2", "json-bigint": "^1.0.0" } }, "sha512-a4tiq7E0/5fTjxPAaH4jpjkSv/uCaU2p5KC6HVGrvl0cDjA8iBZv4vv1gyzlmK0ZUKqwpOyQMKzZQe3lTit77A=="],
|
||||
|
|
@ -461,6 +476,8 @@
|
|||
|
||||
"has-tostringtag": ["has-tostringtag@1.0.2", "", { "dependencies": { "has-symbols": "^1.0.3" } }, "sha512-NqADB8VjPFLM2V0VvHUewwwsw0ZWBaIdgo+ieHtK3hasLz4qeCRjYcqfB6AQrBggRKppKF8L52/VqdVsO47Dlw=="],
|
||||
|
||||
"has-unicode": ["has-unicode@2.0.1", "", {}, "sha512-8Rf9Y83NBReMnx0gFzA8JImQACstCYWUplepDa9xprwwtmgEZUF0h/i5xSA625zB/I37EtrswSST6OXxwaaIJQ=="],
|
||||
|
||||
"hase": ["hase@2.0.0", "", { "dependencies": { "@babel/runtime": "7.1.2", "amqplib": "0.5.2" } }, "sha512-L83pBR/oZvQQNjv4kw9aUpTqBxERPiY7B42jsmkt1VDeUaRVhYkEIKzkCqrppjtxHe2EZqzZJzuhMXsWsxYIsw=="],
|
||||
|
||||
"hasown": ["hasown@2.0.2", "", { "dependencies": { "function-bind": "^1.1.2" } }, "sha512-0hJU9SCPvmMzIBdZFqNPXWa6dqh7WdH0cII9y+CyS8rG3nL48Bclra9HmKhVVUHyPWNH5Y7xDwAB7bfgSjkUMQ=="],
|
||||
|
|
@ -495,6 +512,8 @@
|
|||
|
||||
"is-finalizationregistry": ["is-finalizationregistry@1.1.1", "", { "dependencies": { "call-bound": "^1.0.3" } }, "sha512-1pC6N8qWJbWoPtEjgcL2xyhQOP491EQjeUo3qTKcmV8YSDDJrOepfG8pcC7h/QgnQHYSv0mJ3Z/ZWxmatVrysg=="],
|
||||
|
||||
"is-fullwidth-code-point": ["is-fullwidth-code-point@1.0.0", "", { "dependencies": { "number-is-nan": "^1.0.0" } }, "sha512-1pqUqRjkhPJ9miNq9SwMfdvi6lBJcd6eFxvfaivQhaH3SgisfiuudvFntdKOmxuee/77l+FPjKrQjWvmPjWrRw=="],
|
||||
|
||||
"is-generator-function": ["is-generator-function@1.1.0", "", { "dependencies": { "call-bound": "^1.0.3", "get-proto": "^1.0.0", "has-tostringtag": "^1.0.2", "safe-regex-test": "^1.1.0" } }, "sha512-nPUB5km40q9e8UfN/Zc24eLlzdSf9OfKByBw9CIdw4H1giPMeA0OIJvbchsCu4npfI2QcMVBsGEBHKZ7wLTWmQ=="],
|
||||
|
||||
"is-map": ["is-map@2.0.3", "", {}, "sha512-1Qed0/Hr2m+YqxnM09CjA2d/i6YZNfF6R2oRAOj36eUdS6qIV/huPJNSEpKbupewFs+ZsJlxsjjPbc0/afW6Lw=="],
|
||||
|
|
@ -593,8 +612,12 @@
|
|||
|
||||
"ms": ["ms@2.1.3", "", {}, "sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA=="],
|
||||
|
||||
"nan": ["nan@2.23.0", "", {}, "sha512-1UxuyYGdoQHcGg87Lkqm3FzefucTa0NAiOcuRsDmysep3c1LVCRK2krrUDafMWtjSG04htvAmvg96+SDknOmgQ=="],
|
||||
|
||||
"nanoid": ["nanoid@5.1.5", "", { "bin": { "nanoid": "bin/nanoid.js" } }, "sha512-Ir/+ZpE9fDsNH0hQ3C68uyThDXzYcim2EqcZ8zn8Chtt1iylPT9xXJB0kPCnqzgcEGikO9RxSrh63MsmVCU7Fw=="],
|
||||
|
||||
"native-node-utils": ["native-node-utils@0.2.7", "", { "dependencies": { "nan": "^2.13.2" } }, "sha512-61v0G3uVxWlXHppSZGwZi+ZEIgGUKI8QvEkEJLb1GVePI7P8SBe+G747z+QMXSt4TxfgbVZP0DyobbRKYVIjdw=="],
|
||||
|
||||
"negotiator": ["negotiator@0.6.3", "", {}, "sha512-+EUsqGPLsM+j/zdChZjsnX51g4XrHFOIXwfnCVPGlQk/k5giakcKsuxCObBRu6DSm9opw/O6slWbJdghQM4bBg=="],
|
||||
|
||||
"nocache": ["nocache@2.0.0", "", {}, "sha512-YdKcy2x0dDwOh+8BEuHvA+mnOKAhmMQDgKBOCUGaLpewdmsRYguYZSom3yA+/OrE61O/q+NMQANnun65xpI1Hw=="],
|
||||
|
|
@ -605,6 +628,10 @@
|
|||
|
||||
"node-statsd": ["node-statsd@0.1.1", "", {}, "sha512-QDf6R8VXF56QVe1boek8an/Rb3rSNaxoFWb7Elpsv2m1+Noua1yy0F1FpKpK5VluF8oymWM4w764A4KsYL4pDg=="],
|
||||
|
||||
"npmlog": ["npmlog@4.1.2", "", { "dependencies": { "are-we-there-yet": "~1.1.2", "console-control-strings": "~1.1.0", "gauge": "~2.7.3", "set-blocking": "~2.0.0" } }, "sha512-2uUqazuKlTaSI/dC8AzicUck7+IrEaOnN/e0jd3Xtt1KcGpwx30v50mL7oPyr/h9bL3E4aZccVwpwP+5W9Vjkg=="],
|
||||
|
||||
"number-is-nan": ["number-is-nan@1.0.1", "", {}, "sha512-4jbtZXNAsfZbAHiiqjLPBiCl16dES1zI4Hpzzxw61Tk+loF+sBDBKx1ICKKKwIqQ7M0mFn1TmkN7euSncWgHiQ=="],
|
||||
|
||||
"object-assign": ["object-assign@4.1.1", "", {}, "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg=="],
|
||||
|
||||
"object-inspect": ["object-inspect@1.13.4", "", {}, "sha512-W67iLl4J2EXEGTbfeHCffrjDfitvLANg0UlX3wFUUSTx92KXRFegMHUVgSqE+wvhAbi4WqjGg9czysTV2Epbew=="],
|
||||
|
|
@ -623,6 +650,10 @@
|
|||
|
||||
"openai": ["openai@5.9.0", "", { "peerDependencies": { "ws": "^8.18.0", "zod": "^3.23.8" }, "optionalPeers": ["ws", "zod"], "bin": { "openai": "bin/cli" } }, "sha512-cmLC0pfqLLhBGxE4aZPyRPjydgYCncppV2ClQkKmW79hNjCvmzkfhz8rN5/YVDmjVQlFV+UsF1JIuNjNgeagyQ=="],
|
||||
|
||||
"opencv-build": ["opencv-build@0.1.9", "", { "dependencies": { "npmlog": "^4.1.2" } }, "sha512-tgT/bnJAcYROen9yaPynfK98IMl62mPSgMLmTx41911m5bczlq21xtE5r+UWLB/xEo/0hKk6tl5zHyxV/JS5Rg=="],
|
||||
|
||||
"opencv4nodejs": ["opencv4nodejs@5.6.0", "", { "dependencies": { "nan": "^2.14.0", "native-node-utils": "^0.2.7", "npmlog": "^4.1.2", "opencv-build": "^0.1.9" }, "optionalDependencies": { "@types/node": ">6" } }, "sha512-JvcT1hb2JUCdntcVABgD9Gprr+gkXBe+jhHKvrr0Ug51y087K4ybm0vHBQVzI2ei1aJxEc9tNknPL9rpyx5Xuw=="],
|
||||
|
||||
"own-keys": ["own-keys@1.0.1", "", { "dependencies": { "get-intrinsic": "^1.2.6", "object-keys": "^1.1.1", "safe-push-apply": "^1.0.0" } }, "sha512-qFOyK5PjiWZd+QQIh+1jhdb9LpxTF0qs7Pm8o5QHYZ0M3vKqSqzsZaEB6oWlxZ+q2sJBMI/Ktgd2N5ZwQoRHfg=="],
|
||||
|
||||
"parseurl": ["parseurl@1.3.3", "", {}, "sha512-CiyeOxFT/JZyN5m0z9PfXw4SCBJ6Sygz1Dpl0wqjlhDEGGBP1GnsUVEL0p63hoG1fcj3fHynXi9NYO4nWOL+qQ=="],
|
||||
|
|
@ -637,6 +668,8 @@
|
|||
|
||||
"possible-typed-array-names": ["possible-typed-array-names@1.1.0", "", {}, "sha512-/+5VFTchJDoVj3bhoqi6UeymcD00DAwb1nJwamzPvHEszJ4FpF6SNNbUbOS8yI56qHzdV8eK0qEfOSiodkTdxg=="],
|
||||
|
||||
"process-nextick-args": ["process-nextick-args@2.0.1", "", {}, "sha512-3ouUOpQhtgrbOa17J7+uxOTpITYWaGP7/AhoR3+A+/1e9skrzelGi/dXzEYyvbxubEF6Wn2ypscTKiKJFFn1ag=="],
|
||||
|
||||
"processenv": ["processenv@1.1.0", "", { "dependencies": { "babel-runtime": "6.26.0" } }, "sha512-SymqIsn8GjEUy8nG7HiyEjgbfk1xFosRIakUX1NHLpriq3vVpKniGrr9RdMWCaGYWByIovbRt2f/WvmP/IOApQ=="],
|
||||
|
||||
"proxy-addr": ["proxy-addr@2.0.7", "", { "dependencies": { "forwarded": "0.2.0", "ipaddr.js": "1.9.1" } }, "sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg=="],
|
||||
|
|
@ -679,6 +712,8 @@
|
|||
|
||||
"serve-static": ["serve-static@1.13.2", "", { "dependencies": { "encodeurl": "~1.0.2", "escape-html": "~1.0.3", "parseurl": "~1.3.2", "send": "0.16.2" } }, "sha512-p/tdJrO4U387R9oMjb1oj7qSMaMfmOyd4j9hOFoxZe2baQszgHcSWjuya/CiT5kgZZKRudHNOA0pYXOl8rQ5nw=="],
|
||||
|
||||
"set-blocking": ["set-blocking@2.0.0", "", {}, "sha512-KiKBS8AnWGEyLzofFfmvKwpdPzqiy16LvQfK3yv/fVH7Bj13/wl3JSR1J+rfgRE9q7xUJK4qvgS8raSOeLUehw=="],
|
||||
|
||||
"set-function-length": ["set-function-length@1.2.2", "", { "dependencies": { "define-data-property": "^1.1.4", "es-errors": "^1.3.0", "function-bind": "^1.1.2", "get-intrinsic": "^1.2.4", "gopd": "^1.0.1", "has-property-descriptors": "^1.0.2" } }, "sha512-pgRc4hJ4/sNjWCSS9AmnS40x3bNMDTknHgL5UaMBTMyJnU90EgWh1Rz+MC9eFu4BuN/UwZjKQuY/1v3rM7HMfg=="],
|
||||
|
||||
"set-function-name": ["set-function-name@2.0.2", "", { "dependencies": { "define-data-property": "^1.1.4", "es-errors": "^1.3.0", "functions-have-names": "^1.2.3", "has-property-descriptors": "^1.0.2" } }, "sha512-7PGFlmtwsEADb0WYyvCMa1t+yke6daIG4Wirafur5kcf+MhUnPms1UeR0CKQdTZD81yESwMHbtn+TR+dMviakQ=="],
|
||||
|
|
@ -701,6 +736,8 @@
|
|||
|
||||
"side-channel-weakmap": ["side-channel-weakmap@1.0.2", "", { "dependencies": { "call-bound": "^1.0.2", "es-errors": "^1.3.0", "get-intrinsic": "^1.2.5", "object-inspect": "^1.13.3", "side-channel-map": "^1.0.1" } }, "sha512-WPS/HvHQTYnHisLo9McqBHOJk2FkHO/tlpvldyrnem4aeQp4hai3gythswg6p01oSoTl58rcpiFAjF2br2Ak2A=="],
|
||||
|
||||
"signal-exit": ["signal-exit@3.0.7", "", {}, "sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ=="],
|
||||
|
||||
"split2": ["split2@3.0.0", "", { "dependencies": { "readable-stream": "^3.0.0" } }, "sha512-Cp7G+nUfKJyHCrAI8kze3Q00PFGEG1pMgrAlTFlDbn+GW24evSZHJuMl+iUJx1w/NTRDeBiTgvwnf6YOt94FMw=="],
|
||||
|
||||
"stack-trace": ["stack-trace@0.0.10", "", {}, "sha512-KGzahc7puUKkzyMt+IqAep+TVNbKP+k2Lmwhub39m1AsTSkaDutx56aDCo+HLDzf/D26BIHTJWNiTG1KAJiQCg=="],
|
||||
|
|
@ -711,6 +748,8 @@
|
|||
|
||||
"stop-iteration-iterator": ["stop-iteration-iterator@1.1.0", "", { "dependencies": { "es-errors": "^1.3.0", "internal-slot": "^1.1.0" } }, "sha512-eLoXW/DHyl62zxY4SCaIgnRhuMr6ri4juEYARS8E6sCEqzKpOiE521Ucofdx+KnDZl5xmvGYaaKCk5FEOxJCoQ=="],
|
||||
|
||||
"string-width": ["string-width@1.0.2", "", { "dependencies": { "code-point-at": "^1.0.0", "is-fullwidth-code-point": "^1.0.0", "strip-ansi": "^3.0.0" } }, "sha512-0XsVpQLnVCXHJfyEs8tC0zpTVIr5PKKsQtkT29IwupnPTjtPmQ3xT/4yCREF9hYkV/3M3kzcUTSAZT6a6h81tw=="],
|
||||
|
||||
"string.prototype.trim": ["string.prototype.trim@1.2.10", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.2", "define-data-property": "^1.1.4", "define-properties": "^1.2.1", "es-abstract": "^1.23.5", "es-object-atoms": "^1.0.0", "has-property-descriptors": "^1.0.2" } }, "sha512-Rs66F0P/1kedk5lyYyH9uBzuiI/kNRmwJAR9quK6VOtIpZ2G+hMZd+HQbbv25MgCA6gEffoMZYxlTod4WcdrKA=="],
|
||||
|
||||
"string.prototype.trimend": ["string.prototype.trimend@1.0.9", "", { "dependencies": { "call-bind": "^1.0.8", "call-bound": "^1.0.2", "define-properties": "^1.2.1", "es-object-atoms": "^1.0.0" } }, "sha512-G7Ok5C6E/j4SGfyLCloXTrngQIQU3PWtXGst3yM7Bea9FRURf1S42ZHlZZtsNque2FN2PoUhfZXYLNWwEr4dLQ=="],
|
||||
|
|
@ -721,6 +760,8 @@
|
|||
|
||||
"stringify-object": ["stringify-object@3.3.0", "", { "dependencies": { "get-own-enumerable-property-symbols": "^3.0.0", "is-obj": "^1.0.1", "is-regexp": "^1.0.0" } }, "sha512-rHqiFh1elqCQ9WPLIC8I0Q/g/wj5J1eMkyoiD6eoQApWHP0FtlK7rqnhmabL5VUY9JQCcqwwvlOaSuutekgyrw=="],
|
||||
|
||||
"strip-ansi": ["strip-ansi@3.0.1", "", { "dependencies": { "ansi-regex": "^2.0.0" } }, "sha512-VhumSSbBqDTP8p2ZLKj40UjBCV4+v8bUSEpUb4KjRgWk9pbqGF4REFj6KEagidb2f/M6AzC0EmFyDNGaw9OCzg=="],
|
||||
|
||||
"style-mod": ["style-mod@4.1.2", "", {}, "sha512-wnD1HyVqpJUI2+eKZ+eo1UwghftP6yuFheBqqe+bWCotBjC2K1YnteJILRMs3SM4V/0dLEW1SC27MWP5y+mwmw=="],
|
||||
|
||||
"supports-color": ["supports-color@5.5.0", "", { "dependencies": { "has-flag": "^3.0.0" } }, "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow=="],
|
||||
|
|
@ -795,6 +836,8 @@
|
|||
|
||||
"which-typed-array": ["which-typed-array@1.1.19", "", { "dependencies": { "available-typed-arrays": "^1.0.7", "call-bind": "^1.0.8", "call-bound": "^1.0.4", "for-each": "^0.3.5", "get-proto": "^1.0.1", "gopd": "^1.2.0", "has-tostringtag": "^1.0.2" } }, "sha512-rEvr90Bck4WZt9HHFC4DJMsjvu7x+r6bImz0/BrbWb7A2djJ8hnZMrWnHo9F8ssv0OMErasDhftrfROTyqSDrw=="],
|
||||
|
||||
"wide-align": ["wide-align@1.1.5", "", { "dependencies": { "string-width": "^1.0.2 || 2 || 3 || 4" } }, "sha512-eDMORYaPNZ4sQIuuYPDHdQvf4gyCF9rEEV/yPxGfwPkRodwEgiMUUXTx/dex+Me0wxx53S+NgUHaP7y3MGlDmg=="],
|
||||
|
||||
"wrappy": ["wrappy@1.0.2", "", {}, "sha512-l4Sp/DRseor9wL6EvV2+TuQn63dMkPjZ/sp9XkghTEbV9KlPS1xUsZ3u7/IQO4wxtcFB4bgpQPRcR3QCvezPcQ=="],
|
||||
|
||||
"ws": ["ws@6.2.0", "", { "dependencies": { "async-limiter": "~1.0.0" } }, "sha512-deZYUNlt2O4buFCa3t5bKLf8A7FPP/TVjwOeVNpw818Ma5nk4MLXls2eoEGS39o8119QIYxTrTDoPQ5B/gTD6w=="],
|
||||
|
|
@ -831,6 +874,8 @@
|
|||
|
||||
"amqplib/readable-stream": ["readable-stream@1.1.14", "", { "dependencies": { "core-util-is": "~1.0.0", "inherits": "~2.0.1", "isarray": "0.0.1", "string_decoder": "~0.10.x" } }, "sha512-+MeVjFf4L44XUkhM1eYbD8fyEsxcV81pqMSR5gblfcLCHfZvbrqy4/qYHE+/R5HoBUT11WV5O08Cr1n3YXkWVQ=="],
|
||||
|
||||
"are-we-there-yet/readable-stream": ["readable-stream@2.3.8", "", { "dependencies": { "core-util-is": "~1.0.0", "inherits": "~2.0.3", "isarray": "~1.0.0", "process-nextick-args": "~2.0.0", "safe-buffer": "~5.1.1", "string_decoder": "~1.1.1", "util-deprecate": "~1.0.1" } }, "sha512-8p0AUk4XODgIewSi0l8Epjs+EVnWiK7NoDIEGU0HhE7+ZyY8D1IMY7odu5lRrFXGg71L15KG8QrPmum45RTtdA=="],
|
||||
|
||||
"babel-runtime/regenerator-runtime": ["regenerator-runtime@0.11.1", "", {}, "sha512-MguG95oij0fC3QV3URf4V2SDYGJhJnJGqvIIgdECeODCT98wSWDAJ94SSuVpYQUoTcGUIL6L4yNB7j1DFFHSBg=="],
|
||||
|
||||
"body-parser/debug": ["debug@2.6.9", "", { "dependencies": { "ms": "2.0.0" } }, "sha512-bC7ElrdJaJnPbAP+1EotYvqZsb3ecl5wi6Bfi6BJTUcNowp6cvspg0jXznRTKDjm/E7AdgFBVeAPVMNcKGsHMA=="],
|
||||
|
|
@ -923,6 +968,12 @@
|
|||
|
||||
"amqplib/readable-stream/string_decoder": ["string_decoder@0.10.31", "", {}, "sha512-ev2QzSzWPYmy9GuqfIVildA4OdcGLeFZQrq5ys6RtiuF+RQQiZWr8TZNyAcuVXyQRYfEO+MsoB/1BuQVhOJuoQ=="],
|
||||
|
||||
"are-we-there-yet/readable-stream/inherits": ["inherits@2.0.4", "", {}, "sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ=="],
|
||||
|
||||
"are-we-there-yet/readable-stream/isarray": ["isarray@1.0.0", "", {}, "sha512-VLghIWNM6ELQzo7zwmcg0NmTVyWKYjvIeM83yjp0wRDTmUnrM678fQbcKBo6n2CJEF0szoG//ytg+TKla89ALQ=="],
|
||||
|
||||
"are-we-there-yet/readable-stream/string_decoder": ["string_decoder@1.1.1", "", { "dependencies": { "safe-buffer": "~5.1.0" } }, "sha512-n/ShnvDi6FHbbVfviro+WojiFzv+s8MPMHBczVePfUpDJLwoLT0ht1l4YwBCbi8pJAveEEdnkHyPyTP/mzRfwg=="],
|
||||
|
||||
"body-parser/debug/ms": ["ms@2.0.0", "", {}, "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="],
|
||||
|
||||
"compression/debug/ms": ["ms@2.0.0", "", {}, "sha512-Tpp60P6IUJDTuOq/5Z8cdskzJujfwqfOTkrwIwj7IRISpnkJnT6SyJ4PCPnGMoFjC9ddhal5KVIYtAt97ix05A=="],
|
||||
|
|
|
|||
|
|
@ -9,8 +9,36 @@ export type LoaderProps<T> = T extends Loader<infer R> ? Awaited<R> : never
|
|||
|
||||
export type Action<Data = unknown> = (req: Request, params: any) => Promise<Response | Data>
|
||||
|
||||
// I'd rather have used HTMLLinkElement and HTMLScriptElement, but onload didn't work with those because
|
||||
// it expected a function, not a string.
|
||||
type LinkAttributes = {
|
||||
rel?: string
|
||||
href?: string
|
||||
type?: string
|
||||
media?: string
|
||||
sizes?: string
|
||||
hreflang?: string
|
||||
crossOrigin?: HTMLLinkElement["crossOrigin"]
|
||||
integrity?: string
|
||||
referrerPolicy?: HTMLLinkElement["referrerPolicy"]
|
||||
onload?: string
|
||||
onerror?: string
|
||||
}
|
||||
|
||||
type ScriptAttributes = {
|
||||
src?: string
|
||||
type?: string
|
||||
async?: boolean
|
||||
defer?: boolean
|
||||
crossOrigin?: HTMLScriptElement["crossOrigin"]
|
||||
integrity?: string
|
||||
referrerPolicy?: HTMLScriptElement["referrerPolicy"]
|
||||
onload?: string
|
||||
onerror?: string
|
||||
}
|
||||
|
||||
export type Head = {
|
||||
title?: string
|
||||
links?: { rel: string; href: string; attributes?: string }[]
|
||||
scripts?: { src: string; type?: boolean }[]
|
||||
links?: LinkAttributes[]
|
||||
scripts?: ScriptAttributes[]
|
||||
}
|
||||
|
|
|
|||
|
|
@ -1,4 +1,5 @@
|
|||
import type { Action, Loader } from "./main"
|
||||
|
||||
export const renderServer = async (req: Request, route: Bun.MatchedRoute) => {
|
||||
const contentType = req.headers.get("Content-Type")
|
||||
|
||||
|
|
@ -42,12 +43,8 @@ const renderHtml = async (req: Request, route: Bun.MatchedRoute) => {
|
|||
|
||||
// Remove any < characters from the loader data to prevent XSS attacks
|
||||
const escapedLoaderData = JSON.stringify(loaderData).replace(/</g, "\\u003c")
|
||||
const headLinks = component.head?.links?.map(
|
||||
(link: any) => `<link rel="${link.rel}" href="${link.href}" ${link.attributes || ""}>`
|
||||
)
|
||||
const headScripts = component.head?.scripts?.map(
|
||||
(script: any) => `<script src="${script.src}" ${script.type ? `type="${script.type}"` : ""}></script>`
|
||||
)
|
||||
const headLinks = component.head?.links?.map((attrs: any) => <link {...attrs} />)
|
||||
const headScripts = component.head?.scripts?.map((attrs: any) => <script {...attrs} />)
|
||||
return new Response(
|
||||
`<!DOCTYPE html>
|
||||
<html lang="en">
|
||||
|
|
|
|||
|
|
@ -29,7 +29,6 @@ export const getGeminiResponse = async <T = StructuredResponse>(
|
|||
},
|
||||
})
|
||||
|
||||
console.log(`🌭 i am starting`)
|
||||
const responseText = response.text
|
||||
if (!responseText) {
|
||||
throw new Error("No response received from Gemini")
|
||||
|
|
@ -50,17 +49,12 @@ export const getGeminiResponse = async <T = StructuredResponse>(
|
|||
|
||||
// Zod schema for structured response
|
||||
const StructuredResponseSchema = z.object({
|
||||
description: z.string(),
|
||||
summary: z.string(),
|
||||
elements: z.array(
|
||||
z.object({
|
||||
type: z.string(),
|
||||
box_2d: z.object({
|
||||
ymin: z.number().describe("Normalized ymin coordinate (0-1000)"),
|
||||
xmin: z.number().describe("Normalized xmin coordinate (0-1000)"),
|
||||
ymax: z.number().describe("Normalized ymax coordinate (0-1000)"),
|
||||
xmax: z.number().describe("Normalized xmax coordinate (0-1000)"),
|
||||
}),
|
||||
ymin: z.number().describe("Normalized ymin coordinate (0-1000)"),
|
||||
xmin: z.number().describe("Normalized xmin coordinate (0-1000)"),
|
||||
ymax: z.number().describe("Normalized ymax coordinate (0-1000)"),
|
||||
xmax: z.number().describe("Normalized xmax coordinate (0-1000)"),
|
||||
label: z.string(),
|
||||
})
|
||||
),
|
||||
|
|
@ -109,49 +103,34 @@ export type StructuredResponse = z.infer<typeof StructuredResponseSchema>
|
|||
const whiteboardSchema = {
|
||||
type: Type.OBJECT,
|
||||
properties: {
|
||||
description: {
|
||||
type: Type.STRING,
|
||||
},
|
||||
summary: {
|
||||
type: Type.STRING,
|
||||
},
|
||||
elements: {
|
||||
type: Type.ARRAY,
|
||||
items: {
|
||||
type: Type.OBJECT,
|
||||
properties: {
|
||||
type: {
|
||||
type: Type.STRING,
|
||||
ymin: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized ymin coordinate (0-1000)",
|
||||
},
|
||||
box_2d: {
|
||||
type: Type.OBJECT,
|
||||
properties: {
|
||||
ymin: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized ymin coordinate (0-1000)",
|
||||
},
|
||||
xmin: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized xmin coordinate (0-1000)",
|
||||
},
|
||||
ymax: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized ymax coordinate (0-1000)",
|
||||
},
|
||||
xmax: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized xmax coordinate (0-1000)",
|
||||
},
|
||||
},
|
||||
required: ["ymin", "xmin", "ymax", "xmax"],
|
||||
xmin: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized xmin coordinate (0-1000)",
|
||||
},
|
||||
ymax: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized ymax coordinate (0-1000)",
|
||||
},
|
||||
xmax: {
|
||||
type: Type.NUMBER,
|
||||
description: "Normalized xmax coordinate (0-1000)",
|
||||
},
|
||||
label: {
|
||||
type: Type.STRING,
|
||||
},
|
||||
},
|
||||
required: ["type", "box_2d", "label"],
|
||||
required: ["ymin", "xmin", "ymax", "xmax", "label"],
|
||||
},
|
||||
},
|
||||
},
|
||||
required: ["description", "summary", "elements"],
|
||||
required: ["elements"],
|
||||
}
|
||||
|
|
|
|||
8
packages/whiteboard/src/ml.ts
Normal file
8
packages/whiteboard/src/ml.ts
Normal file
|
|
@ -0,0 +1,8 @@
|
|||
// This file has been replaced with AI-based shape detection
|
||||
// See ai.ts for the current implementation using Gemini/OpenAI
|
||||
// which is more suitable for hand-drawn shape detection on whiteboards
|
||||
|
||||
export const detectShapesFromBuffer = (arrayBuffer: ArrayBuffer) => {
|
||||
console.warn("detectShapesFromBuffer is deprecated. Use AI-based detection from ai.ts instead.")
|
||||
return []
|
||||
}
|
||||
|
|
@ -1,5 +1,5 @@
|
|||
import { ensure } from "@workshop/shared/utils"
|
||||
import { useAction, Form } from "@workshop/nano-remix"
|
||||
import { useAction, Form, type Head } from "@workshop/nano-remix"
|
||||
import { useEffect, useRef, useState } from "hono/jsx"
|
||||
import { getGeminiResponse } from "../ai"
|
||||
|
||||
|
|
@ -14,13 +14,9 @@ export const action = async (req: Request, params: {}) => {
|
|||
const imageUrl = new URL("whiteboard.jpeg", url.origin).toString()
|
||||
const imageResponse = await fetch(imageUrl)
|
||||
const imageBuffer = await imageResponse.arrayBuffer()
|
||||
const response = await getGeminiResponse(imageBuffer, prompts.default)
|
||||
// const response = await getGeminiResponse(imageBuffer, prompts.default)
|
||||
|
||||
return {
|
||||
summary: response?.summary,
|
||||
description: response?.description,
|
||||
elements: response?.elements || [],
|
||||
}
|
||||
return { elements: response?.elements || [] }
|
||||
}
|
||||
|
||||
export default function Index() {
|
||||
|
|
@ -54,7 +50,7 @@ export default function Index() {
|
|||
|
||||
// Draw AI detected elements with box_2d format
|
||||
data?.elements.forEach((element, index) => {
|
||||
const { ymin, xmin, ymax, xmax } = element.box_2d
|
||||
const { ymin, xmin, ymax, xmax } = element
|
||||
ensure(ymin && xmin && ymax && xmax, "Box 2D coordinates must be defined")
|
||||
|
||||
// Convert normalized coordinates (0-1000) to actual canvas coordinates
|
||||
|
|
@ -77,7 +73,6 @@ export default function Index() {
|
|||
<div class="p-5">
|
||||
<h1 class="text-3xl font-bold mb-5">PROJECT WHITEBOARD</h1>
|
||||
|
||||
<form>hi</form>
|
||||
<Form name="analyzeWhiteboard">
|
||||
<button
|
||||
type="submit"
|
||||
|
|
@ -89,20 +84,13 @@ export default function Index() {
|
|||
|
||||
<div class="mb-4">
|
||||
<h2 class="text-xl font-semibold mb-2">AI Analysis: {loading ? "Loading..." : ""}</h2>
|
||||
<p class="mb-2">
|
||||
<strong>Description:</strong> {data?.description}
|
||||
</p>
|
||||
<p class="mb-2">
|
||||
<strong>Summary:</strong> {data?.summary}
|
||||
</p>
|
||||
<p class="mb-2">
|
||||
<strong>Detected Elements:</strong> {data?.elements.length}
|
||||
</p>
|
||||
<ul class="list-disc list-inside mb-4">
|
||||
{data?.elements.map((element, index) => (
|
||||
<li key={index}>
|
||||
<span class="text-magenta-600 font-medium">{element.type}</span>: {element.label} at box_2d: [
|
||||
{element.box_2d.ymin}, {element.box_2d.xmin}, {element.box_2d.ymax}, {element.box_2d.xmax}]
|
||||
{element.label} at box_2d: [{element.ymin}, {element.xmin}, {element.ymax}, {element.xmax}]
|
||||
</li>
|
||||
))}
|
||||
</ul>
|
||||
|
|
@ -114,41 +102,3 @@ export default function Index() {
|
|||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
const sampleResponse = {
|
||||
description:
|
||||
"The image displays a whiteboard with various hand-drawn shapes and text in different colors. Shapes include several circles and one rectangle, along with an arrow. There are also written words and a wavy line, and a smiley face.",
|
||||
elements: [
|
||||
{
|
||||
box_2d: { ymin: 45, xmin: 70, ymax: 275, xmax: 300 },
|
||||
label: "blue circle",
|
||||
type: "hand drawn circle",
|
||||
},
|
||||
{
|
||||
box_2d: { ymin: 305, xmin: 160, ymax: 405, xmax: 295 },
|
||||
label: "green smiley face",
|
||||
type: "hand drawn circle",
|
||||
},
|
||||
{
|
||||
box_2d: { ymin: 290, xmin: 400, ymax: 380, xmax: 500 },
|
||||
label: "green circle with red cross",
|
||||
type: "hand drawn circle",
|
||||
},
|
||||
{
|
||||
box_2d: { ymin: 455, xmin: 450, ymax: 600, xmax: 560 },
|
||||
label: "grey circle",
|
||||
type: "hand drawn circle",
|
||||
},
|
||||
{
|
||||
box_2d: { ymin: 620, xmin: 100, ymax: 740, xmax: 350 },
|
||||
label: "blue rectangle",
|
||||
type: "hand drawn square",
|
||||
},
|
||||
{
|
||||
box_2d: { ymin: 460, xmin: 370, ymax: 530, xmax: 450 },
|
||||
label: "arrow pointing to circle",
|
||||
type: "hand drawn arrow",
|
||||
},
|
||||
],
|
||||
summary: "A whiteboard showing hand-drawn circles, a rectangle, an arrow, and text.",
|
||||
}
|
||||
|
|
|
|||
|
|
@ -2,10 +2,8 @@ import { nanoRemix } from "@workshop/nano-remix"
|
|||
|
||||
Bun.serve({
|
||||
routes: {
|
||||
"/*": {
|
||||
GET: (req) => {
|
||||
return nanoRemix(req)
|
||||
},
|
||||
"/*": (req) => {
|
||||
return nanoRemix(req)
|
||||
},
|
||||
},
|
||||
})
|
||||
|
|
|
|||
Loading…
Reference in New Issue
Block a user