diff --git a/packages/visualizer/package.json b/packages/visualizer/package.json index 7c57327..5153940 100644 --- a/packages/visualizer/package.json +++ b/packages/visualizer/package.json @@ -1,63 +1,52 @@ { - "name": "@midscene/visualizer", - "version": "0.6.2", - "repository": "https://github.com/web-infra-dev/midscene", - "homepage": "https://midscenejs.com/", - "types": "./dist/types/index.d.ts", - "main": "./dist/lib/index.js", - "module": "./dist/es/index.js", - "files": [ - "dist", - "html", - "README.md" - ], - "watch": { - "build": { - "patterns": [ - "src" - ], - "extensions": "tsx,less,scss,css,js,jsx,ts", - "quiet": false - } - }, - "scripts": { - "dev": "npm run build && npx npm-watch", - "build": "modern build && npx ts-node scripts/build-html.ts", - "build:watch": "modern build -w", - "serve": "http-server ./dist/ -p 3000", - "new": "modern new", - "upgrade": "modern upgrade" - }, - "devDependencies": { - "@ant-design/icons": "5.3.7", - "@midscene/core": "workspace:*", - "@modern-js/module-tools": "2.58.2", - "@modern-js/plugin-module-doc": "^2.33.1", - "@modern-js/runtime": "^2.56.2", - "@types/node": "^18.0.0", - "@types/react": "18.3.3", - "@types/react-dom": "18.3.0", - "antd": "5.19.3", - "dayjs": "1.11.11", - "http-server": "14.1.1", - "npm-watch": "0.13.0", - "pixi-filters": "6.0.5", - "pixi.js": "8.1.1", - "react": "^18.2.0", - "react-dom": "^18.2.0", - "react-resizable-panels": "2.0.22", - "rimraf": "~3.0.2", - "ts-node": "10.9.2", - "typescript": "~5.0.4", - "zustand": "4.5.2" - }, - "sideEffects": [ - "**/*.css", - "**/*.less", - "**/*.sass", - "**/*.scss" - ], - "publishConfig": { - "access": "public" - } -} \ No newline at end of file + "name": "@midscene/visualizer", + "version": "0.6.2", + "repository": "https://github.com/web-infra-dev/midscene", + "homepage": "https://midscenejs.com/", + "types": "./dist/types/index.d.ts", + "main": "./dist/lib/index.js", + "module": "./dist/es/index.js", + "files": ["dist", "html", "README.md"], + "watch": { + "build": { + "patterns": ["src"], + "extensions": "tsx,less,scss,css,js,jsx,ts", + "quiet": false + } + }, + "scripts": { + "dev": "npm run build && npx npm-watch", + "build": "modern build && npx ts-node scripts/build-html.ts", + "build:watch": "modern build -w", + "serve": "http-server ./dist/ -p 3000", + "new": "modern new", + "upgrade": "modern upgrade" + }, + "devDependencies": { + "@ant-design/icons": "5.3.7", + "@midscene/core": "workspace:*", + "@modern-js/module-tools": "2.58.2", + "@modern-js/plugin-module-doc": "^2.33.1", + "@modern-js/runtime": "^2.56.2", + "@types/node": "^18.0.0", + "@types/react": "18.3.3", + "@types/react-dom": "18.3.0", + "antd": "5.19.3", + "dayjs": "1.11.11", + "http-server": "14.1.1", + "npm-watch": "0.13.0", + "pixi-filters": "6.0.5", + "pixi.js": "8.1.1", + "react": "^18.2.0", + "react-dom": "^18.2.0", + "react-resizable-panels": "2.0.22", + "rimraf": "~3.0.2", + "ts-node": "10.9.2", + "typescript": "~5.0.4", + "zustand": "4.5.2" + }, + "sideEffects": ["**/*.css", "**/*.less", "**/*.sass", "**/*.scss"], + "publishConfig": { + "access": "public" + } +} diff --git a/packages/visualizer/src/component/detail-panel.tsx b/packages/visualizer/src/component/detail-panel.tsx index a20eb9a..be4c400 100644 --- a/packages/visualizer/src/component/detail-panel.tsx +++ b/packages/visualizer/src/component/detail-panel.tsx @@ -1,19 +1,18 @@ 'use client'; import './detail-panel.less'; import { useExecutionDump } from '@/component/store'; -// import Playground from '@/playground'; import { filterBase64Value, timeStr } from '@/utils'; import { CameraOutlined, - ExperimentFilled, FileTextOutlined, ScheduleOutlined, VideoCameraOutlined, } from '@ant-design/icons'; -import { Button, ConfigProvider, Segmented, message } from 'antd'; +import { ConfigProvider, Segmented } from 'antd'; import { useEffect, useState } from 'react'; import Blackboard from './blackboard'; import Player from './player'; +import SendToPlayground from './send-to-playground'; const ScreenshotItem = (props: { time: string; img: string }) => { return ( @@ -167,21 +166,6 @@ const DetailPanel = (): JSX.Element => { }; }); - const ifPlaygroundValid = Boolean(insightDump?.context); - let playgroundEl = null; - if (ifPlaygroundValid) { - // playgroundEl = ; - playgroundEl = ; - } - const launchPlayground = () => { - if (ifPlaygroundValid) { - // TODO - // setOpen(true); - } else { - message.error('No context available'); - } - }; - return (
@@ -203,17 +187,10 @@ const DetailPanel = (): JSX.Element => { }} /> - +
{content}
- {playgroundEl}
); }; diff --git a/packages/visualizer/src/component/playground-demo-ui-context.json b/packages/visualizer/src/component/playground-demo-ui-context.json index 81e4bfa..3012c1c 100644 --- a/packages/visualizer/src/component/playground-demo-ui-context.json +++ b/packages/visualizer/src/component/playground-demo-ui-context.json @@ -8,10 +8,7 @@ "width": 621, "height": 520 }, - "center": [ - 330, - 286 - ], + "center": [330, 286], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -30,10 +27,7 @@ "width": 186, "height": 26 }, - "center": [ - 114, - 39 - ], + "center": [114, 39], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -52,10 +46,7 @@ "width": 223, "height": 23 }, - "center": [ - 131, - 91 - ], + "center": [131, 91], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -74,10 +65,7 @@ "width": 203, "height": 20 }, - "center": [ - 122, - 141 - ], + "center": [122, 141], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -96,10 +84,7 @@ "width": 620, "height": 214 }, - "center": [ - 330, - 282 - ], + "center": [330, 282], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -118,10 +103,7 @@ "width": 619, "height": 36 }, - "center": [ - 330, - 193 - ], + "center": [330, 193], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -140,10 +122,7 @@ "width": 619, "height": 36 }, - "center": [ - 330, - 193 - ], + "center": [330, 193], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -162,10 +141,7 @@ "width": 51, "height": 35 }, - "center": [ - 46, - 193 - ], + "center": [46, 193], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -184,10 +160,7 @@ "width": 159, "height": 35 }, - "center": [ - 151, - 193 - ], + "center": [151, 193], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -206,10 +179,7 @@ "width": 136, "height": 35 }, - "center": [ - 298, - 193 - ], + "center": [298, 193], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -228,10 +198,7 @@ "width": 142, "height": 36 }, - "center": [ - 438, - 193 - ], + "center": [438, 193], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -250,10 +217,7 @@ "width": 130, "height": 35 }, - "center": [ - 574, - 193 - ], + "center": [574, 193], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -272,10 +236,7 @@ "width": 619, "height": 178 }, - "center": [ - 330, - 300 - ], + "center": [330, 300], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -294,10 +255,7 @@ "width": 619, "height": 36 }, - "center": [ - 330, - 229 - ], + "center": [330, 229], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -316,10 +274,7 @@ "width": 51, "height": 36 }, - "center": [ - 46, - 229 - ], + "center": [46, 229], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -338,10 +293,7 @@ "width": 159, "height": 36 }, - "center": [ - 151, - 229 - ], + "center": [151, 229], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -360,10 +312,7 @@ "width": 136, "height": 36 }, - "center": [ - 298, - 229 - ], + "center": [298, 229], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -382,10 +331,7 @@ "width": 143, "height": 36 }, - "center": [ - 438, - 229 - ], + "center": [438, 229], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -404,10 +350,7 @@ "width": 130, "height": 35 }, - "center": [ - 574, - 228 - ], + "center": [574, 228], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -426,10 +369,7 @@ "width": 619, "height": 36 }, - "center": [ - 330, - 264 - ], + "center": [330, 264], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -448,10 +388,7 @@ "width": 51, "height": 36 }, - "center": [ - 46, - 264 - ], + "center": [46, 264], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -470,10 +407,7 @@ "width": 159, "height": 35 }, - "center": [ - 151, - 264 - ], + "center": [151, 264], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -492,10 +426,7 @@ "width": 136, "height": 36 }, - "center": [ - 298, - 264 - ], + "center": [298, 264], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -514,10 +445,7 @@ "width": 143, "height": 36 }, - "center": [ - 438, - 264 - ], + "center": [438, 264], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -536,10 +464,7 @@ "width": 130, "height": 36 }, - "center": [ - 574, - 264 - ], + "center": [574, 264], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -558,10 +483,7 @@ "width": 619, "height": 36 }, - "center": [ - 330, - 300 - ], + "center": [330, 300], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -580,10 +502,7 @@ "width": 51, "height": 36 }, - "center": [ - 46, - 300 - ], + "center": [46, 300], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -602,10 +521,7 @@ "width": 159, "height": 36 }, - "center": [ - 151, - 300 - ], + "center": [151, 300], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -624,10 +540,7 @@ "width": 136, "height": 36 }, - "center": [ - 298, - 300 - ], + "center": [298, 300], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -646,10 +559,7 @@ "width": 143, "height": 36 }, - "center": [ - 438, - 300 - ], + "center": [438, 300], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -668,10 +578,7 @@ "width": 130, "height": 36 }, - "center": [ - 574, - 300 - ], + "center": [574, 300], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -690,10 +597,7 @@ "width": 619, "height": 36 }, - "center": [ - 330, - 335 - ], + "center": [330, 335], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -712,10 +616,7 @@ "width": 51, "height": 36 }, - "center": [ - 46, - 335 - ], + "center": [46, 335], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -734,10 +635,7 @@ "width": 159, "height": 35 }, - "center": [ - 151, - 335 - ], + "center": [151, 335], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -756,10 +654,7 @@ "width": 136, "height": 36 }, - "center": [ - 298, - 335 - ], + "center": [298, 335], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -778,10 +673,7 @@ "width": 143, "height": 36 }, - "center": [ - 438, - 335 - ], + "center": [438, 335], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -800,10 +692,7 @@ "width": 130, "height": 36 }, - "center": [ - 574, - 335 - ], + "center": [574, 335], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -822,10 +711,7 @@ "width": 619, "height": 36 }, - "center": [ - 330, - 371 - ], + "center": [330, 371], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -844,10 +730,7 @@ "width": 51, "height": 36 }, - "center": [ - 46, - 371 - ], + "center": [46, 371], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -866,10 +749,7 @@ "width": 159, "height": 36 }, - "center": [ - 151, - 371 - ], + "center": [151, 371], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -888,10 +768,7 @@ "width": 136, "height": 36 }, - "center": [ - 298, - 371 - ], + "center": [298, 371], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -910,10 +787,7 @@ "width": 143, "height": 36 }, - "center": [ - 438, - 371 - ], + "center": [438, 371], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -932,10 +806,7 @@ "width": 130, "height": 36 }, - "center": [ - 574, - 371 - ], + "center": [574, 371], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -954,10 +825,7 @@ "width": 473, "height": 109 }, - "center": [ - 256, - 464 - ], + "center": [256, 464], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -976,10 +844,7 @@ "width": 46, "height": 18 }, - "center": [ - 43, - 419 - ], + "center": [43, 419], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -998,10 +863,7 @@ "width": 47, "height": 16 }, - "center": [ - 43, - 456 - ], + "center": [43, 456], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1021,10 +883,7 @@ "width": 147, "height": 22 }, - "center": [ - 145, - 459 - ], + "center": [145, 459], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1045,10 +904,7 @@ "width": 67, "height": 22 }, - "center": [ - 256, - 459 - ], + "center": [256, 459], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1067,10 +923,7 @@ "width": 46, "height": 16 }, - "center": [ - 318, - 458 - ], + "center": [318, 458], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1089,10 +942,7 @@ "width": 147, "height": 22 }, - "center": [ - 418, - 459 - ], + "center": [418, 459], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1113,10 +963,7 @@ "width": 183, "height": 37 }, - "center": [ - 111, - 500 - ], + "center": [111, 500], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1136,10 +983,7 @@ "width": 459, "height": 23 }, - "center": [ - 249, - 535 - ], + "center": [249, 535], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1160,10 +1004,7 @@ "width": 112, "height": 16 }, - "center": [ - 76, - 534 - ], + "center": [76, 534], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1184,10 +1025,7 @@ "width": 347, "height": 21 }, - "center": [ - 305, - 535 - ], + "center": [305, 535], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1223,4 +1061,4 @@ }, "screenshotBase64": "", "url": "file:///Users/bytedance/workspace/midscene/packages/web-integration/tests/fixtures/extractor.html" -} \ No newline at end of file +} diff --git a/packages/visualizer/src/component/send-to-playground.tsx b/packages/visualizer/src/component/send-to-playground.tsx new file mode 100644 index 0000000..c40cd04 --- /dev/null +++ b/packages/visualizer/src/component/send-to-playground.tsx @@ -0,0 +1,84 @@ +import { SendOutlined } from '@ant-design/icons'; +import type { UIContext } from '@midscene/core/.'; +import { Button } from 'antd'; +import { useEffect, useState } from 'react'; + +export const serverBase = 'http://localhost:5800'; + +const checkServerStatus = async () => { + try { + const res = await fetch(`${serverBase}/status`); + return res.status === 200; + } catch (e) { + return false; + } +}; + +export const useServerValid = () => { + const [serverValid, setServerValid] = useState(false); + + useEffect(() => { + let interruptFlag = false; + Promise.resolve( + (async () => { + while (!interruptFlag) { + const status = await checkServerStatus(); + if (status) { + setServerValid(true); + } else { + setServerValid(false); + } + // sleep 1s + await new Promise((resolve) => setTimeout(resolve, 1000)); + } + })(), + ); + + return () => { + interruptFlag = true; + }; + }, []); + + return serverValid; +}; + +export default function SendToPlayground(props?: { context?: UIContext }) { + const serverValid = useServerValid(); + + let ifPlaygroundValid = true; + let invalidReason = ''; + if (!props?.context) { + ifPlaygroundValid = false; + invalidReason = 'No context'; + } else if (!serverValid) { + ifPlaygroundValid = false; + invalidReason = 'Cannot connect to playground server'; + } + + const launchPlayground = async () => { + // post a form to server, use a new window to open the playground + + const res = await fetch(`${serverBase}/playground-with-context`, { + method: 'POST', + body: JSON.stringify({ + context: JSON.stringify(props?.context), + }), + headers: { + 'Content-Type': 'application/json', + }, + credentials: 'omit', + }); + const data = await res.json(); + const location = data.location; + window.open(`${serverBase}${location}`, '_blank'); + }; + return ( + + ); +} diff --git a/packages/visualizer/src/index.tsx b/packages/visualizer/src/index.tsx index cc42c15..7c9faef 100644 --- a/packages/visualizer/src/index.tsx +++ b/packages/visualizer/src/index.tsx @@ -248,7 +248,7 @@ export function Visualizer(props: { }} > - Visualization - Midscene.js + Report - Midscene.js
{ - try { - const res = await fetch(`${serverBase}/playground/status`); - return res.status === 200; - } catch (e) { - return false; - } -}; - const cacheKeyForPrompt = 'playground-user-prompt'; const cacheKeyForType = 'playground-user-type'; const setCache = (prompt: string, type: string) => { @@ -57,8 +48,15 @@ const getCachedType = () => { return localStorage.getItem(cacheKeyForType); }; +const useContextId = () => { + const path = window.location.pathname; + const match = path.match(/^\/playground\/([a-zA-Z0-9-]+)$/); + return match ? match[1] : null; +}; + const { TextArea } = Input; function Playground() { + const contextId = useContextId(); const [uiContext, setUiContext] = useState(null); const [loading, setLoading] = useState(false); const [result, setResult] = useState<{ @@ -72,31 +70,18 @@ function Playground() { useState(null); const [replayCounter, setReplayCounter] = useState(0); - const [serverStatus, setServerStatus] = useState< - 'connected' | 'pending' | 'failed' - >('pending'); + const serverValid = useServerValid(); useEffect(() => { - let interruptFlag = false; - Promise.resolve( - (async () => { - while (!interruptFlag) { - const status = await checkServerStatus(); - if (status) { - setServerStatus('connected'); - } else { - setServerStatus('failed'); - } - // sleep 1s - await new Promise((resolve) => setTimeout(resolve, 1000)); - } - })(), - ); - - return () => { - interruptFlag = true; - }; - }, []); + if (contextId) { + fetch(`${serverBase}/context/${contextId}`) + .then((res) => res.json()) + .then((data) => { + const contextObj = JSON.parse(data.context); + setUiContext(contextObj); + }); + } + }, [contextId]); const handleRun = useCallback(async () => { const value = form.getFieldsValue(); @@ -135,8 +120,7 @@ function Playground() { placeholder = 'What do you want to assert?'; } - const runButtonDisabled = - !uiContext || loading || serverStatus !== 'connected'; + const runButtonDisabled = !uiContext || loading || !serverValid; // use cmd + enter to run useEffect(() => { @@ -180,17 +164,14 @@ function Playground() { resultDataToShow =
{result?.error}
; } - const serverTip = - serverStatus === 'failed' ? ( - <> - {iconForStatus(serverStatus)} Failed to connect to server. Please launch - the local server first. - - ) : ( - <> - {iconForStatus(serverStatus)} {serverStatus} - - ); + const serverTip = !serverValid ? ( + <> + {iconForStatus('failed')} Failed to connect to server. Please launch the + local server first. + + ) : ( + <>{iconForStatus('connected')} Connected to server + ); return (
diff --git a/packages/web-integration/package.json b/packages/web-integration/package.json index 591d884..eccbca8 100644 --- a/packages/web-integration/package.json +++ b/packages/web-integration/package.json @@ -1,129 +1,109 @@ { - "name": "@midscene/web", - "description": "An AI-powered automation SDK can control the page, perform assertions, and extract data in JSON format using natural language. See https://midscenejs.com/ for details.", - "version": "0.6.2", - "repository": "https://github.com/web-infra-dev/midscene", - "homepage": "https://midscenejs.com/", - "jsnext:source": "./src/index.ts", - "bin": { - "midscene-playground": "./bin/midscene-playground" + "name": "@midscene/web", + "description": "An AI-powered automation SDK can control the page, perform assertions, and extract data in JSON format using natural language. See https://midscenejs.com/ for details.", + "version": "0.6.2", + "repository": "https://github.com/web-infra-dev/midscene", + "homepage": "https://midscenejs.com/", + "jsnext:source": "./src/index.ts", + "bin": { + "midscene-playground": "./bin/midscene-playground" + }, + "main": "./dist/lib/index.js", + "types": "./dist/types/index.d.ts", + "exports": { + ".": "./dist/lib/index.js", + "./puppeteer": "./dist/lib/puppeteer.js", + "./playwright": "./dist/lib/playwright.js", + "./playwright-report": "./dist/lib/playwright-report.js", + "./playground": "./dist/lib/playground.js", + "./debug": "./dist/lib/debug.js", + "./constants": "./dist/lib/constants.js", + "./html-element": "./dist/lib/html-element/index.js" + }, + "typesVersions": { + "*": { + ".": ["./dist/types/index.d.ts"], + "puppeteer": ["./dist/types/puppeteer.d.ts"], + "playwright": ["./dist/types/playwright.d.ts"], + "playwright-report": ["./dist/types/playwright-report.d.ts"], + "playground": ["./dist/types/playground.d.ts"], + "debug": ["./dist/types/debug.d.ts"], + "constants": ["./dist/types/constants.d.ts"], + "html-element": ["./dist/types/html-element/index.d.ts"] + } + }, + "scripts": { + "dev": "modern dev", + "dev:server": "npm run build && ./bin/midscene-playground", + "build": "npm run build:pkg && npm run build:script", + "build:pkg": "modern build -c ./modern.config.ts", + "build:script": "modern build -c ./modern.inspect.config.ts", + "build:watch": "modern build -w -c ./modern.config.ts & modern build -w -c ./modern.inspect.config.ts", + "test": "vitest --run", + "test:u": "vitest --run -u", + "test:ai": "AI_TEST_TYPE=web npm run test", + "test:ai:cache": "MIDSCENE_CACHE=true AI_TEST_TYPE=web npm run test", + "test:ai:all": "npm run test:ai:web && npm run test:ai:native", + "test:ai:native": "MIDSCENE_CACHE=true AI_TEST_TYPE=native npm run test", + "new": "modern new", + "upgrade": "modern upgrade", + "prepublishOnly": "npm run build", + "e2e": "playwright test --config=playwright.config.ts", + "e2e:report": "MIDSCENE_REPORT=true playwright test --config=playwright.config.ts", + "e2e:cache": "MIDSCENE_CACHE=true playwright test --config=playwright.config.ts", + "e2e:ui": "playwright test --config=playwright.config.ts --ui", + "e2e:ui:cache": "MIDSCENE_CACHE=true playwright test --config=playwright.config.ts --ui" + }, + "files": ["static", "dist", "README.md", "bin"], + "dependencies": { + "express": "4.21.1", + "openai": "4.57.1", + "inquirer": "10.1.5", + "cors": "2.8.5", + "@midscene/core": "workspace:*", + "@midscene/shared": "workspace:*", + "@xmldom/xmldom": "0.8.10" + }, + "devDependencies": { + "@modern-js/module-tools": "2.58.2", + "@playwright/test": "1.44.1", + "@types/cors": "2.8.12", + "@types/fs-extra": "11.0.4", + "@types/node": "^18.0.0", + "@types/express": "4.17.14", + "@wdio/types": "9.0.4", + "dotenv": "16.4.5", + "fs-extra": "11.2.0", + "js-sha256": "0.11.0", + "playwright": "1.44.1", + "puppeteer": "23.0.2", + "typescript": "~5.0.4", + "vitest": "^1.6.0", + "webdriverio": "9.0.6" + }, + "peerDependencies": { + "@playwright/test": "^1.44.1", + "playwright": "^1.44.1", + "puppeteer": ">=20.0.0", + "webdriverio": ">=9.0.6" + }, + "peerDependenciesMeta": { + "@playwright/test": { + "optional": true }, - "main": "./dist/lib/index.js", - "types": "./dist/types/index.d.ts", - "exports": { - ".": "./dist/lib/index.js", - "./puppeteer": "./dist/lib/puppeteer.js", - "./playwright": "./dist/lib/playwright.js", - "./playwright-report": "./dist/lib/playwright-report.js", - "./playground": "./dist/lib/playground.js", - "./debug": "./dist/lib/debug.js", - "./constants": "./dist/lib/constants.js", - "./html-element": "./dist/lib/html-element/index.js" + "puppeteer": { + "optional": true }, - "typesVersions": { - "*": { - ".": [ - "./dist/types/index.d.ts" - ], - "puppeteer": [ - "./dist/types/puppeteer.d.ts" - ], - "playwright": [ - "./dist/types/playwright.d.ts" - ], - "playwright-report": [ - "./dist/types/playwright-report.d.ts" - ], - "playground": [ - "./dist/types/playground.d.ts" - ], - "debug": [ - "./dist/types/debug.d.ts" - ], - "constants": [ - "./dist/types/constants.d.ts" - ], - "html-element": [ - "./dist/types/html-element/index.d.ts" - ] - } - }, - "scripts": { - "dev": "modern dev", - "build": "npm run build:pkg && npm run build:script", - "build:pkg": "modern build -c ./modern.config.ts", - "build:script": "modern build -c ./modern.inspect.config.ts", - "build:watch": "modern build -w -c ./modern.config.ts & modern build -w -c ./modern.inspect.config.ts", - "test": "vitest --run", - "test:u": "vitest --run -u", - "test:ai": "AI_TEST_TYPE=web npm run test", - "test:ai:cache": "MIDSCENE_CACHE=true AI_TEST_TYPE=web npm run test", - "test:ai:all": "npm run test:ai:web && npm run test:ai:native", - "test:ai:native": "MIDSCENE_CACHE=true AI_TEST_TYPE=native npm run test", - "new": "modern new", - "upgrade": "modern upgrade", - "prepublishOnly": "npm run build", - "e2e": "playwright test --config=playwright.config.ts", - "e2e:report": "MIDSCENE_REPORT=true playwright test --config=playwright.config.ts", - "e2e:cache": "MIDSCENE_CACHE=true playwright test --config=playwright.config.ts", - "e2e:ui": "playwright test --config=playwright.config.ts --ui", - "e2e:ui:cache": "MIDSCENE_CACHE=true playwright test --config=playwright.config.ts --ui" - }, - "files": [ - "static", - "dist", - "README.md", - "bin" - ], - "dependencies": { - "express": "4.21.1", - "openai": "4.57.1", - "inquirer": "10.1.5", - "cors": "2.8.5", - "@midscene/core": "workspace:*", - "@midscene/shared": "workspace:*", - "@xmldom/xmldom": "0.8.10" - }, - "devDependencies": { - "@modern-js/module-tools": "2.58.2", - "@playwright/test": "1.44.1", - "@types/cors": "2.8.12", - "@types/fs-extra": "11.0.4", - "@types/node": "^18.0.0", - "@types/express": "4.17.14", - "@wdio/types": "9.0.4", - "dotenv": "16.4.5", - "fs-extra": "11.2.0", - "js-sha256": "0.11.0", - "playwright": "1.44.1", - "puppeteer": "23.0.2", - "typescript": "~5.0.4", - "vitest": "^1.6.0", - "webdriverio": "9.0.6" - }, - "peerDependencies": { - "@playwright/test": "^1.44.1", - "playwright": "^1.44.1", - "puppeteer": ">=20.0.0", - "webdriverio": ">=9.0.6" - }, - "peerDependenciesMeta": { - "@playwright/test": { - "optional": true - }, - "puppeteer": { - "optional": true - }, - "webdriverio": { - "optional": true - } - }, - "engines": { - "node": ">=16.0.0" - }, - "publishConfig": { - "access": "public", - "registry": "https://registry.npmjs.org" - }, - "license": "MIT" -} \ No newline at end of file + "webdriverio": { + "optional": true + } + }, + "engines": { + "node": ">=16.0.0" + }, + "publishConfig": { + "access": "public", + "registry": "https://registry.npmjs.org" + }, + "license": "MIT" +} diff --git a/packages/web-integration/src/playground/server.ts b/packages/web-integration/src/playground/server.ts index 0552c94..a48940f 100644 --- a/packages/web-integration/src/playground/server.ts +++ b/packages/web-integration/src/playground/server.ts @@ -43,25 +43,33 @@ export default class PlaygroundServer { async launch() { this.app.use(errorHandler); - // Serve static files from the staticPath - this.app.use(express.static(staticPath)); + + this.app.use( + cors({ + origin: '*', + credentials: true, + }), + ); + + this.app.get('/status', cors(), async (req, res) => { + res.send({ + status: 'ok', + }); + }); // Serve index.html for the root route this.app.get('/', (req, res) => { res.sendFile(join(staticPath, 'index.html')); }); - this.app.get('/playground/status', cors(), async (req, res) => { - res.send({ - status: 'ok', - }); + this.app.get('/playground/:uuid', async (req, res) => { + res.sendFile(join(staticPath, 'index.html')); }); this.app.get('/context/:uuid', async (req, res) => { const { uuid } = req.params; const contextFile = this.filePathForUuid(uuid); assert(existsSync(contextFile), 'Context not found'); - const context = readFileSync(contextFile, 'utf8'); res.json({ context, @@ -72,20 +80,21 @@ export default class PlaygroundServer { // actions from report file this.app.post( '/playground-with-context', - cors(), - express.urlencoded({ extended: false }), + express.json({ limit: '50mb' }), async (req, res) => { const context = req.body.context; assert(context, 'context is required'); const uuid = randomUUID(); this.saveContextFile(uuid, context); - return res.redirect(`/context/${uuid}`); + return res.json({ + location: `/playground/${uuid}`, + uuid, + }); }, ); this.app.post( '/execute', - cors(), express.json({ limit: '30mb' }), async (req, res) => { const { context, type, prompt } = req.body; diff --git a/packages/web-integration/tests/unit-test/playground-server.test.ts b/packages/web-integration/tests/unit-test/playground-server.test.ts index 7095a16..ef8ab2c 100644 --- a/packages/web-integration/tests/unit-test/playground-server.test.ts +++ b/packages/web-integration/tests/unit-test/playground-server.test.ts @@ -15,27 +15,25 @@ describe('Playground Server', () => { }); it('post context', async () => { - const contextData = JSON.stringify({ - foo: 'bar', - }); + const contextValue = 'bar'; const res = await fetch(`${serverBase}/playground-with-context`, { method: 'POST', headers: { - 'Content-Type': 'application/x-www-form-urlencoded', + 'Content-Type': 'application/json', }, - body: `context=${encodeURIComponent(contextData)}`, - redirect: 'manual', + body: JSON.stringify({ + context: contextValue, + }), }); - expect(res.status).toBe(302); - const location = res.headers.get('Location'); - expect(location).toBeDefined(); - expect(location).toContain('/context/'); + expect(res.status).toBe(200); + const data = await res.json(); + const contextId = data.uuid; // retrieve context - const contextRes = await fetch(`${serverBase}${location}`); + const contextRes = await fetch(`${serverBase}/context/${contextId}`); const context = await contextRes.json(); expect(context).toBeDefined(); - expect(context.context).toBe(contextData); + expect(context.context).toBe(contextValue); }); });