From e1ae31ac170dd00826dfacc6ccd2c5a32f36b406 Mon Sep 17 00:00:00 2001 From: Zhou xiao Date: Sun, 4 Aug 2024 08:28:19 +0800 Subject: [PATCH] workflow(ci): optimize biome lint (#31) * workflow(ci): format code * workflow(ci): format code * workflow(ci): format code * workflow(ci): format code * workflow(ci): format code --- .github/workflows/lint.yml | 11 +- .vscode/settings.json | 28 +- apps/site/docs/en/_meta.json | 2 +- apps/site/docs/en/docs/_meta.json | 2 +- .../docs/en/docs/getting-started/_meta.json | 6 +- apps/site/docs/en/docs/more/_meta.json | 5 +- apps/site/docs/en/docs/usage/_meta.json | 4 +- apps/site/docs/zh/_meta.json | 2 +- apps/site/docs/zh/docs/_meta.json | 2 +- .../docs/zh/docs/getting-started/_meta.json | 6 +- apps/site/docs/zh/docs/more/_meta.json | 5 +- apps/site/docs/zh/docs/usage/_meta.json | 4 +- apps/site/i18n.json | 2 +- apps/site/route.json | 2 +- apps/site/rspress.config.ts | 10 +- biome.json | 6 +- nx.json | 26 +- packages/midscene/demo_data/demo.actions.json | 22 +- packages/midscene/demo_data/demo.insight.json | 5226 ++++++++--------- packages/midscene/demo_data/index.js | 3 +- packages/midscene/modern.config.ts | 4 +- packages/midscene/package.json | 21 +- packages/midscene/src/action/executor.ts | 40 +- packages/midscene/src/ai-model/inspect.ts | 34 +- packages/midscene/src/ai-model/openai.ts | 17 +- .../src/ai-model/prompt/element_inspector.ts | 5 +- packages/midscene/src/ai-model/prompt/util.ts | 94 +- packages/midscene/src/automation/planning.ts | 9 +- packages/midscene/src/image/info.ts | 5 +- packages/midscene/src/image/transform.ts | 36 +- packages/midscene/src/image/visualization.ts | 14 +- packages/midscene/src/index.ts | 2 +- packages/midscene/src/insight/index.ts | 1 - packages/midscene/src/insight/utils.ts | 45 +- .../query/fixture/script_get_all_texts.tmp.js | 48 +- packages/midscene/src/query/index.ts | 9 +- packages/midscene/src/types.ts | 85 +- packages/midscene/src/utils.ts | 27 +- .../inspector/online_order_inspector.test.ts | 43 +- .../githubstatus/element-snapshot.json | 327 +- .../online_order/element-snapshot.json | 152 +- .../test-data/todo/element-snapshot.json | 102 +- .../visualstudio/element-snapshot.json | 222 +- .../ai-model/inspector/todo_inspector.test.ts | 43 +- .../midscene/tests/ai-model/inspector/util.ts | 212 +- .../midscene/tests/ai-model/openai.test.ts | 36 +- .../tests/automation/planning.test.ts | 2 +- .../midscene/tests/executor/index.test.ts | 18 +- packages/midscene/tests/fixtures/dump.json | 2 +- packages/midscene/tests/image/index.test.ts | 20 +- packages/midscene/tests/insight/index.test.ts | 15 +- .../midscene/tests/insight/prompts.test.ts | 4 +- packages/midscene/tests/insight/utils.test.ts | 8 +- packages/midscene/tests/query.test.ts | 2 +- packages/midscene/tests/utils.test.ts | 17 +- packages/midscene/tests/utils.ts | 61 +- packages/midscene/vitest.config.ts | 11 +- packages/playwright-demo/e2e/ai-xicha.spec.ts | 2 +- packages/playwright-demo/e2e/fixture.ts | 2 +- packages/playwright-demo/modern.config.ts | 9 +- packages/playwright-demo/tsconfig.json | 2 +- .../config/public/test-data-list.json | 41 +- packages/visualizer-report/modern.config.ts | 2 +- packages/visualizer-report/package.json | 10 +- packages/visualizer-report/src/App.tsx | 2 +- packages/visualizer-report/src/pages/Home.tsx | 28 +- .../visualizer-report/src/pages/Report.tsx | 6 +- packages/visualizer-report/tsconfig.json | 2 +- packages/visualizer/docs/index.tsx | 2 +- packages/visualizer/modern.config.ts | 2 +- packages/visualizer/package.json | 12 +- .../visualizer/src/component/blackboard.tsx | 72 +- packages/visualizer/src/component/color.tsx | 11 +- .../visualizer/src/component/detail-panel.tsx | 46 +- .../visualizer/src/component/detail-side.tsx | 131 +- .../src/component/global-hover-preview.tsx | 20 +- packages/visualizer/src/component/sidebar.tsx | 48 +- packages/visualizer/src/component/store.tsx | 15 +- .../visualizer/src/component/timeline.tsx | 89 +- packages/visualizer/src/index.tsx | 39 +- packages/visualizer/src/utils.ts | 38 +- packages/visualizer/tsconfig.json | 3 +- packages/web-integration/modern.config.ts | 2 +- .../web-integration/modern.inspect.config.ts | 10 +- packages/web-integration/package.json | 21 +- packages/web-integration/report-script.mjs | 11 +- packages/web-integration/src/common/agent.ts | 18 +- packages/web-integration/src/common/cdp.ts | 322 - packages/web-integration/src/common/page.d.ts | 2 +- .../web-integration/src/common/task-cache.ts | 21 +- packages/web-integration/src/common/tasks.ts | 232 +- packages/web-integration/src/common/utils.ts | 26 +- .../src/extractor/constants.ts | 8 +- .../src/extractor/extractor.ts | 35 +- .../web-integration/src/extractor/util.ts | 35 +- packages/web-integration/src/img/img.ts | 31 +- packages/web-integration/src/img/util.ts | 21 +- .../web-integration/src/playwright/cache.ts | 24 +- .../web-integration/src/playwright/index.ts | 76 +- .../src/playwright/reporter/index.ts | 9 +- .../playwright/reporter/select-cache-file.ts | 15 +- .../src/playwright/reporter/util.ts | 27 +- packages/web-integration/src/web-element.ts | 17 +- .../tests/e2e/ai-auto-todo.spec.ts | 17 +- packages/web-integration/tests/e2e/tool.ts | 71 +- .../tests/puppeteer/showcase.spec.ts | 4 +- .../web-integration/tests/puppeteer/utils.ts | 9 +- .../web-integration/tests/task-cache.test.ts | 70 +- packages/web-integration/tsconfig.json | 2 +- packages/web-integration/vitest.config.ts | 7 +- scripts/release.js | 4 +- 111 files changed, 4149 insertions(+), 4799 deletions(-) delete mode 100644 packages/web-integration/src/common/cdp.ts diff --git a/.github/workflows/lint.yml b/.github/workflows/lint.yml index ceafc43a..c41e4710 100644 --- a/.github/workflows/lint.yml +++ b/.github/workflows/lint.yml @@ -44,13 +44,6 @@ jobs: - name: Check Dependency Version run: pnpm run check-dependency-version - - name: Check for changed files - run: | - git fetch origin - git branch -a - git diff --name-only main - - - name: Debug Biome lint - run: | - npx biome lint --changed --since main --diagnostic-level=warn --no-errors-on-unmatched --verbose + - name: Biome lint + run: npx biome check . --diagnostic-level=warn --no-errors-on-unmatched diff --git a/.vscode/settings.json b/.vscode/settings.json index 96e19e5d..bea381df 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -1,16 +1,16 @@ { - "editor.codeActionsOnSave": { - "source.organizeImports.biome": "explicit" - }, - "editor.defaultFormatter": "biomejs.biome", - "editor.formatOnSave": true, - "eslint.validate": [ - "javascript", - "javascriptreact", - "typescript", - "typescriptreact", - "vue", - "html" - ], - "editor.formatOnSaveMode": "modifications" + "editor.codeActionsOnSave": { + "source.organizeImports.biome": "explicit" + }, + "editor.defaultFormatter": "biomejs.biome", + "editor.formatOnSave": true, + "eslint.validate": [ + "javascript", + "javascriptreact", + "typescript", + "typescriptreact", + "vue", + "html" + ], + "editor.formatOnSaveMode": "modifications" } diff --git a/apps/site/docs/en/_meta.json b/apps/site/docs/en/_meta.json index 6b8548c7..7c5dc749 100644 --- a/apps/site/docs/en/_meta.json +++ b/apps/site/docs/en/_meta.json @@ -9,4 +9,4 @@ "link": "/visualization/", "activeMatch": "/visualization/" } -] \ No newline at end of file +] diff --git a/apps/site/docs/en/docs/_meta.json b/apps/site/docs/en/docs/_meta.json index 30c2e079..45ee93e9 100644 --- a/apps/site/docs/en/docs/_meta.json +++ b/apps/site/docs/en/docs/_meta.json @@ -20,4 +20,4 @@ "collapsible": false, "collapsed": false } -] \ No newline at end of file +] diff --git a/apps/site/docs/en/docs/getting-started/_meta.json b/apps/site/docs/en/docs/getting-started/_meta.json index cee77653..b3532f6a 100644 --- a/apps/site/docs/en/docs/getting-started/_meta.json +++ b/apps/site/docs/en/docs/getting-started/_meta.json @@ -1,5 +1 @@ -[ - "introduction", - "quick-start", - "demo.md" -] \ No newline at end of file +["introduction", "quick-start", "demo.md"] diff --git a/apps/site/docs/en/docs/more/_meta.json b/apps/site/docs/en/docs/more/_meta.json index 526724ee..1888802d 100644 --- a/apps/site/docs/en/docs/more/_meta.json +++ b/apps/site/docs/en/docs/more/_meta.json @@ -1,4 +1 @@ -[ - "prompting-tips", - "faq" -] \ No newline at end of file +["prompting-tips", "faq"] diff --git a/apps/site/docs/en/docs/usage/_meta.json b/apps/site/docs/en/docs/usage/_meta.json index e1b87c9b..b1834241 100644 --- a/apps/site/docs/en/docs/usage/_meta.json +++ b/apps/site/docs/en/docs/usage/_meta.json @@ -1,3 +1 @@ -[ - "API.md" -] \ No newline at end of file +["API.md"] diff --git a/apps/site/docs/zh/_meta.json b/apps/site/docs/zh/_meta.json index 11ff9658..505b299a 100644 --- a/apps/site/docs/zh/_meta.json +++ b/apps/site/docs/zh/_meta.json @@ -9,4 +9,4 @@ "link": "/visualization/", "activeMatch": "/visualization/" } -] \ No newline at end of file +] diff --git a/apps/site/docs/zh/docs/_meta.json b/apps/site/docs/zh/docs/_meta.json index b8757c02..1f108537 100644 --- a/apps/site/docs/zh/docs/_meta.json +++ b/apps/site/docs/zh/docs/_meta.json @@ -20,4 +20,4 @@ "collapsible": false, "collapsed": false } -] \ No newline at end of file +] diff --git a/apps/site/docs/zh/docs/getting-started/_meta.json b/apps/site/docs/zh/docs/getting-started/_meta.json index 55125d73..7983df20 100644 --- a/apps/site/docs/zh/docs/getting-started/_meta.json +++ b/apps/site/docs/zh/docs/getting-started/_meta.json @@ -1,5 +1 @@ -[ - "introduction", - "quick-start", - "demo" -] \ No newline at end of file +["introduction", "quick-start", "demo"] diff --git a/apps/site/docs/zh/docs/more/_meta.json b/apps/site/docs/zh/docs/more/_meta.json index 526724ee..1888802d 100644 --- a/apps/site/docs/zh/docs/more/_meta.json +++ b/apps/site/docs/zh/docs/more/_meta.json @@ -1,4 +1 @@ -[ - "prompting-tips", - "faq" -] \ No newline at end of file +["prompting-tips", "faq"] diff --git a/apps/site/docs/zh/docs/usage/_meta.json b/apps/site/docs/zh/docs/usage/_meta.json index e1b87c9b..b1834241 100644 --- a/apps/site/docs/zh/docs/usage/_meta.json +++ b/apps/site/docs/zh/docs/usage/_meta.json @@ -1,3 +1 @@ -[ - "API.md" -] \ No newline at end of file +["API.md"] diff --git a/apps/site/i18n.json b/apps/site/i18n.json index 56f57b20..b4772deb 100644 --- a/apps/site/i18n.json +++ b/apps/site/i18n.json @@ -3,4 +3,4 @@ "en": "Getting Started", "zh": "开始" } -} \ No newline at end of file +} diff --git a/apps/site/route.json b/apps/site/route.json index 2f80c816..dc858b34 100644 --- a/apps/site/route.json +++ b/apps/site/route.json @@ -15,4 +15,4 @@ "regexDomain": false } ] -} \ No newline at end of file +} diff --git a/apps/site/rspress.config.ts b/apps/site/rspress.config.ts index 7dc58e20..835ce7ea 100644 --- a/apps/site/rspress.config.ts +++ b/apps/site/rspress.config.ts @@ -1,4 +1,4 @@ -import * as path from 'path'; +import * as path from 'node:path'; import { defineConfig } from 'rspress/config'; export default defineConfig({ @@ -12,7 +12,13 @@ export default defineConfig({ }, themeConfig: { darkMode: false, - socialLinks: [{ icon: 'github', mode: 'link', content: 'https://github.com/web-infra-dev/midscene' }], + socialLinks: [ + { + icon: 'github', + mode: 'link', + content: 'https://github.com/web-infra-dev/midscene', + }, + ], locales: [ { lang: 'en', diff --git a/biome.json b/biome.json index 30d8f3cf..ee07b73f 100644 --- a/biome.json +++ b/biome.json @@ -8,7 +8,10 @@ ".nx", "**/dist", "**/doc_build", - "*-dump.json" + "*-dump.json", + "script_get_all_texts.tmp.js", + "**/playwright-report/**", + "packages/visualizer/**" ] }, "javascript": { @@ -21,6 +24,7 @@ "rules": { "recommended": true, "style": {}, + "complexity": { "noForEach": "off" }, "suspicious": { "noExplicitAny": "off" } } }, diff --git a/nx.json b/nx.json index 8eb1a8b6..9c805f64 100644 --- a/nx.json +++ b/nx.json @@ -2,37 +2,25 @@ "$schema": "./node_modules/nx/schemas/nx-schema.json", "targetDefaults": { "dev": { - "dependsOn": [ - "^build" - ] + "dependsOn": ["^build"] }, "build": { - "dependsOn": [ - "^build" - ], + "dependsOn": ["^build"], "cache": true }, "build:watch": { - "dependsOn": [ - "^build" - ] + "dependsOn": ["^build"] }, "test": { - "dependsOn": [ - "^build" - ], + "dependsOn": ["^build"], "cache": false }, "e2e": { - "dependsOn": [ - "^build" - ] + "dependsOn": ["^build"] }, "e2e:ui": { - "dependsOn": [ - "^build" - ] + "dependsOn": ["^build"] } }, "defaultBase": "main" -} \ No newline at end of file +} diff --git a/packages/midscene/demo_data/demo.actions.json b/packages/midscene/demo_data/demo.actions.json index 70228d14..aff15d99 100644 --- a/packages/midscene/demo_data/demo.actions.json +++ b/packages/midscene/demo_data/demo.actions.json @@ -24,10 +24,7 @@ "top": 200, "left": 200 }, - "center": [ - 250, - 250 - ] + "center": [250, 250] } }, "log": { @@ -52,10 +49,7 @@ "top": 200, "left": 200 }, - "center": [ - 250, - 250 - ] + "center": [250, 250] } ] }, @@ -72,10 +66,7 @@ "top": 200, "left": 200 }, - "center": [ - 250, - 250 - ] + "center": [250, 250] } ], "data": null, @@ -136,10 +127,7 @@ "top": 200, "left": 200 }, - "center": [ - 250, - 250 - ] + "center": [250, 250] } ] }, @@ -157,4 +145,4 @@ } } ] -} \ No newline at end of file +} diff --git a/packages/midscene/demo_data/demo.insight.json b/packages/midscene/demo_data/demo.insight.json index 240233bd..0706f25d 100644 --- a/packages/midscene/demo_data/demo.insight.json +++ b/packages/midscene/demo_data/demo.insight.json @@ -1,1720 +1,751 @@ [ -{ - "logId": "e5875ae8-6aab-4df0-9d12-1cc36621252a", - "sdkVersion": "0.1.1", - "logTime": 1721022602461, - "type": "find", - "context": { - "screenshotBase64": "", - "size": { - "width": 1920, - "height": 1080 - }, - "timestamp": 1721022598776, - "content": [ - { - "content": "We use optional cookies to improve your experience on our websites, such as through social media connections, and to display personalized advertising based on your online activity. If you reject optional cookies, only cookies necessary to provide you the services will be used. You may change your selection by clicking “Manage Cookies” at the bottom of the page.", - "rect": { - "left": 144, - "top": 8, - "width": 1102, - "height": 49 - }, - "center": [ - 695, - 32 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='0']" - }, - { - "content": "Privacy Statement", - "rect": { - "left": 144, - "top": 25, - "width": 1060, - "height": 32 - }, - "center": [ - 674, - 41 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='1']" - }, - { - "content": "Third-Party Cookies", - "rect": { - "left": 210, - "top": 41, - "width": 121, - "height": 15 - }, - "center": [ - 270, - 48 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='2']" - }, - { - "content": "Accept", - "rect": { - "left": 1397, - "top": 26, - "width": 48, - "height": 15 - }, - "center": [ - 1421, - 33 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='3']" - }, - { - "content": "Reject", - "rect": { - "left": 1561, - "top": 26, - "width": 44, - "height": 16 - }, - "center": [ - 1583, - 34 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='4']" - }, - { - "content": "Manage cookies", - "rect": { - "left": 1689, - "top": 26, - "width": 110, - "height": 16 - }, - "center": [ - 1744, - 34 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='5']" - }, - { - "content": "Visual Studio Code", - "rect": { - "left": 376, - "top": 82, - "width": 170, - "height": 24 - }, - "center": [ - 461, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='6']" - }, - { - "content": "Docs", - "rect": { - "left": 573, - "top": 87, - "width": 38, - "height": 15 - }, - "center": [ - 592, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='7']" - }, - { - "content": "Updates", - "rect": { - "left": 634, - "top": 87, - "width": 64, - "height": 17 - }, - "center": [ - 666, - 95 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='8']" - }, - { - "content": "Blog", - "rect": { - "left": 721, - "top": 86, - "width": 34, - "height": 17 - }, - "center": [ - 738, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='9']" - }, - { - "content": "API", - "rect": { - "left": 779, - "top": 87, - "width": 24, - "height": 14 - }, - "center": [ - 791, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='10']" - }, - { - "content": "Extensions", - "rect": { - "left": 827, - "top": 87, - "width": 83, - "height": 15 - }, - "center": [ - 868, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='11']" - }, - { - "content": "FAQ", - "rect": { - "left": 932, - "top": 87, - "width": 31, - "height": 15 - }, - "center": [ - 947, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='12']" - }, - { - "content": "Learn", - "rect": { - "left": 987, - "top": 88, - "width": 42, - "height": 15 - }, - "center": [ - 1008, - 95 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='13']" - }, - { - "content": "Search Docs", - "rect": { - "left": 1265, - "top": 78, - "width": 200, - "height": 32 - }, - "center": [ - 1365, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='14']" - }, - { - "content": "Download", - "rect": { - "left": 1489, - "top": 85, - "width": 75, - "height": 19 - }, - "center": [ - 1526, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='15']" - }, - { - "content": "Version 1.91", - "rect": { - "left": 684, - "top": 86, - "width": 70, - "height": 8 - }, - "center": [ - 719, - 90 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='16']" - }, - { - "content": "is now available! Read about the new features and fixes from June.", - "rect": { - "left": 339, - "top": 74, - "width": 1225, - "height": 22 - }, - "center": [ - 951, - 85 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='17']" - }, - { - "content": "Free. Built on open source. Runs everywhere.", - "rect": { - "left": 340, - "top": 224, - "width": 331, - "height": 38 - }, - "center": [ - 505, - 243 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='18']" - }, - { - "content": "Code Editing. Redefined.", - "rect": { - "left": 342, - "top": 293, - "width": 326, - "height": 114 - }, - "center": [ - 505, - 350 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='19']" - }, - { - "content": "Download for macOS", - "rect": { - "left": 340, - "top": 452, - "width": 225, - "height": 50 - }, - "center": [ - 452, - 477 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='20']" - }, - { - "content": "Web", - "rect": { - "left": 340, - "top": 520, - "width": 29, - "height": 17 - }, - "center": [ - 354, - 528 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='21']" - }, - { - "content": ",", - "rect": { - "left": 340, - "top": 518, - "width": 272, - "height": 22 - }, - "center": [ - 476, - 529 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='22']" - }, - { - "content": "Insiders edition", - "rect": { - "left": 377, - "top": 520, - "width": 101, - "height": 17 - }, - "center": [ - 427, - 528 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='23']" - }, - { - "content": ", or", - "rect": { - "left": 340, - "top": 518, - "width": 272, - "height": 22 - }, - "center": [ - 476, - 529 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='24']" - }, - { - "content": "other platforms", - "rect": { - "left": 504, - "top": 520, - "width": 102, - "height": 17 - }, - "center": [ - 555, - 528 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='25']" - }, - { - "content": "By using VS Code, you agree to its", - "rect": { - "left": 340, - "top": 556, - "width": 344, - "height": 17 - }, - "center": [ - 512, - 564 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='26']" - }, - { - "content": "license", - "rect": { - "left": 524, - "top": 557, - "width": 37, - "height": 13 - }, - "center": [ - 542, - 563 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='27']" - }, - { - "content": "and", - "rect": { - "left": 524, - "top": 556, - "width": 160, - "height": 17 - }, - "center": [ - 604, - 564 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='28']" - }, - { - "content": "privacy statement", - "rect": { - "left": 587, - "top": 557, - "width": 94, - "height": 13 - }, - "center": [ - 634, - 563 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='29']" - }, - { - "content": ".", - "rect": { - "left": 524, - "top": 556, - "width": 160, - "height": 17 - }, - "center": [ - 604, - 564 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='30']" - }, - { - "content": "Code in any language", - "rect": { - "left": 340, - "top": 856, - "width": 289, - "height": 29 - }, - "center": [ - 484, - 870 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='31']" - }, - { - "content": "VS Code supports almost every major programming language. Several ship in the box, like JavaScript, TypeScript, CSS, and HTML, but extensions for others can be found in the VS Code Marketplace.", - "rect": { - "left": 340, - "top": 905, - "width": 355, - "height": 119 - }, - "center": [ - 517, - 964 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='32']" - }, - { - "content": "JavaScript", - "rect": { - "left": 832, - "top": 857, - "width": 95, - "height": 18 - }, - "center": [ - 879, - 866 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='33']" - }, - { - "content": "TypeScript", - "rect": { - "left": 1111, - "top": 855, - "width": 101, - "height": 20 - }, - "center": [ - 1161, - 865 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='34']" - }, - { - "content": "Python", - "rect": { - "left": 1387, - "top": 855, - "width": 66, - "height": 20 - }, - "center": [ - 1420, - 865 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='35']" - }, - { - "content": "C#", - "rect": { - "left": 832, - "top": 903, - "width": 21, - "height": 17 - }, - "center": [ - 842, - 911 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='36']" - }, - { - "content": "C++", - "rect": { - "left": 1111, - "top": 903, - "width": 29, - "height": 18 - }, - "center": [ - 1125, - 912 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='37']" - }, - { - "content": "HTML", - "rect": { - "left": 1390, - "top": 903, - "width": 38, - "height": 18 - }, - "center": [ - 1409, - 912 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='38']" - }, - { - "content": "Java", - "rect": { - "left": 832, - "top": 955, - "width": 38, - "height": 13 - }, - "center": [ - 851, - 961 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='39']" - }, - { - "content": "JSON", - "rect": { - "left": 1111, - "top": 951, - "width": 41, - "height": 20 - }, - "center": [ - 1131, - 961 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='40']" - }, - { - "content": "PHP", - "rect": { - "left": 1388, - "top": 951, - "width": 36, - "height": 18 - }, - "center": [ - 1406, - 960 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='41']" - }, - { - "content": "Markdown", - "rect": { - "left": 829, - "top": 999, - "width": 84, - "height": 20 - }, - "center": [ - 871, - 1009 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='42']" - }, - { - "content": "Powershell", - "rect": { - "left": 1108, - "top": 999, - "width": 104, - "height": 20 - }, - "center": [ - 1160, - 1009 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='43']" - }, - { - "content": "YAML", - "rect": { - "left": 1387, - "top": 999, - "width": 47, - "height": 20 - }, - "center": [ - 1410, - 1009 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='44']" - } - ] - }, - "userQuery": { - "element": "download buttons on the page" - }, - "matchedSection": [], - "matchedElement": [ - { - "content": "Download", - "rect": { - "left": 1489, - "top": 85, - "width": 75, - "height": 19 - }, - "center": [ - 1526, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='15']" - }, - { - "content": "Download for macOS", - "rect": { - "left": 340, - "top": 452, - "width": 225, - "height": 50 - }, - "center": [ - 452, - 477 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='20']" - } - ], - "data": null, - "taskInfo": { - "url": "https://code.visualstudio.com/", - "name": "download buttons of vscode", - "durationMs": 3678, - "rawResponse": "{\"elements\":[{\"id\":\"15\"},{\"id\":\"20\"}]}", - "systemPrompt": "\nYou are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.\n\nThe user will give you a screenshot and the texts on it. There may be some none-English characters (like Chinese) on it, indicating it's an non-English app.\n\nBased on the information you get, find one or more text elements on the page.\nHere is the description: download buttons on the page\n \nReturn in the following JSON format:\n{\n \"elements\": [ // Leave it an empty array when no element is found\n { \n \"id\": \"id of the element, like 123\", \n },\n // more ...\n ], \n errors?: [], // string[], error message if any\n}\n" - } -}, -{ - "logId": "27c2b890-5cc2-49f0-8432-7668c26ef2fe", - "sdkVersion": "0.1.1", - "logTime": 1721022621788, - "type": "extract", - "context": { - "screenshotBase64": "", - "size": { - "width": 1920, - "height": 1080 - }, - "timestamp": 1721022608034, - "content": [ - { - "content": "We use optional cookies to improve your experience on our websites, such as through social media connections, and to display personalized advertising based on your online activity. If you reject optional cookies, only cookies necessary to provide you the services will be used. You may change your selection by clicking “Manage Cookies” at the bottom of the page.", - "rect": { - "left": 144, - "top": 8, - "width": 1102, - "height": 49 - }, - "center": [ - 695, - 32 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='0']" - }, - { - "content": "Privacy Statement", - "rect": { - "left": 144, - "top": 25, - "width": 1060, - "height": 32 - }, - "center": [ - 674, - 41 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='1']" - }, - { - "content": "Third-Party Cookies", - "rect": { - "left": 210, - "top": 41, - "width": 121, - "height": 15 - }, - "center": [ - 270, - 48 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='2']" - }, - { - "content": "Accept", - "rect": { - "left": 1397, - "top": 26, - "width": 48, - "height": 15 - }, - "center": [ - 1421, - 33 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='3']" - }, - { - "content": "Reject", - "rect": { - "left": 1561, - "top": 26, - "width": 44, - "height": 16 - }, - "center": [ - 1583, - 34 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='4']" - }, - { - "content": "Manage cookies", - "rect": { - "left": 1689, - "top": 26, - "width": 110, - "height": 16 - }, - "center": [ - 1744, - 34 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='5']" - }, - { - "content": "Visual Studio Code", - "rect": { - "left": 376, - "top": 82, - "width": 170, - "height": 24 - }, - "center": [ - 461, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='6']" - }, - { - "content": "Docs", - "rect": { - "left": 573, - "top": 87, - "width": 38, - "height": 15 - }, - "center": [ - 592, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "logId": "e5875ae8-6aab-4df0-9d12-1cc36621252a", + "sdkVersion": "0.1.1", + "logTime": 1721022602461, + "type": "find", + "context": { + "screenshotBase64": "", + "size": { + "width": 1920, + "height": 1080 + }, + "timestamp": 1721022598776, + "content": [ + { + "content": "We use optional cookies to improve your experience on our websites, such as through social media connections, and to display personalized advertising based on your online activity. If you reject optional cookies, only cookies necessary to provide you the services will be used. You may change your selection by clicking “Manage Cookies” at the bottom of the page.", + "rect": { + "left": 144, + "top": 8, + "width": 1102, + "height": 49 + }, + "center": [695, 32], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='0']" }, - "locator": "[data-midscene-task-1='7']" - }, - { - "content": "Updates", - "rect": { - "left": 634, - "top": 87, - "width": 64, - "height": 17 - }, - "center": [ - 666, - 95 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Privacy Statement", + "rect": { + "left": 144, + "top": 25, + "width": 1060, + "height": 32 + }, + "center": [674, 41], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='1']" }, - "locator": "[data-midscene-task-1='8']" - }, - { - "content": "Blog", - "rect": { - "left": 721, - "top": 86, - "width": 34, - "height": 17 - }, - "center": [ - 738, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Third-Party Cookies", + "rect": { + "left": 210, + "top": 41, + "width": 121, + "height": 15 + }, + "center": [270, 48], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='2']" }, - "locator": "[data-midscene-task-1='9']" - }, - { - "content": "API", - "rect": { - "left": 779, - "top": 87, - "width": 24, - "height": 14 - }, - "center": [ - 791, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Accept", + "rect": { + "left": 1397, + "top": 26, + "width": 48, + "height": 15 + }, + "center": [1421, 33], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='3']" }, - "locator": "[data-midscene-task-1='10']" - }, - { - "content": "Extensions", - "rect": { - "left": 827, - "top": 87, - "width": 83, - "height": 15 - }, - "center": [ - 868, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Reject", + "rect": { + "left": 1561, + "top": 26, + "width": 44, + "height": 16 + }, + "center": [1583, 34], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='4']" }, - "locator": "[data-midscene-task-1='11']" - }, - { - "content": "FAQ", - "rect": { - "left": 932, - "top": 87, - "width": 31, - "height": 15 - }, - "center": [ - 947, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Manage cookies", + "rect": { + "left": 1689, + "top": 26, + "width": 110, + "height": 16 + }, + "center": [1744, 34], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='5']" }, - "locator": "[data-midscene-task-1='12']" - }, - { - "content": "Learn", - "rect": { - "left": 987, - "top": 88, - "width": 42, - "height": 15 - }, - "center": [ - 1008, - 95 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Visual Studio Code", + "rect": { + "left": 376, + "top": 82, + "width": 170, + "height": 24 + }, + "center": [461, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='6']" }, - "locator": "[data-midscene-task-1='13']" - }, - { - "content": "Search Docs", - "rect": { - "left": 1265, - "top": 78, - "width": 200, - "height": 32 + { + "content": "Docs", + "rect": { + "left": 573, + "top": 87, + "width": 38, + "height": 15 + }, + "center": [592, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='7']" }, - "center": [ - 1365, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Updates", + "rect": { + "left": 634, + "top": 87, + "width": 64, + "height": 17 + }, + "center": [666, 95], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='8']" }, - "locator": "[data-midscene-task-1='14']" - }, - { - "content": "Download", - "rect": { - "left": 1489, - "top": 85, - "width": 75, - "height": 19 + { + "content": "Blog", + "rect": { + "left": 721, + "top": 86, + "width": 34, + "height": 17 + }, + "center": [738, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='9']" }, - "center": [ - 1526, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "API", + "rect": { + "left": 779, + "top": 87, + "width": 24, + "height": 14 + }, + "center": [791, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='10']" }, - "locator": "[data-midscene-task-1='15']" - }, - { - "content": "Version 1.91", - "rect": { - "left": 684, - "top": 86, - "width": 70, - "height": 8 - }, - "center": [ - 719, - 90 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Extensions", + "rect": { + "left": 827, + "top": 87, + "width": 83, + "height": 15 + }, + "center": [868, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='11']" }, - "locator": "[data-midscene-task-1='16']" - }, - { - "content": "is now available! Read about the new features and fixes from June.", - "rect": { - "left": 339, - "top": 74, - "width": 1225, - "height": 22 - }, - "center": [ - 951, - 85 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "FAQ", + "rect": { + "left": 932, + "top": 87, + "width": 31, + "height": 15 + }, + "center": [947, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='12']" }, - "locator": "[data-midscene-task-1='17']" - }, - { - "content": "Free. Built on open source. Runs everywhere.", - "rect": { - "left": 340, - "top": 224, - "width": 331, - "height": 38 + { + "content": "Learn", + "rect": { + "left": 987, + "top": 88, + "width": 42, + "height": 15 + }, + "center": [1008, 95], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='13']" }, - "center": [ - 505, - 243 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Search Docs", + "rect": { + "left": 1265, + "top": 78, + "width": 200, + "height": 32 + }, + "center": [1365, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='14']" }, - "locator": "[data-midscene-task-1='18']" - }, - { - "content": "Code Editing. Redefined.", - "rect": { - "left": 342, - "top": 293, - "width": 326, - "height": 114 - }, - "center": [ - 505, - 350 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Download", + "rect": { + "left": 1489, + "top": 85, + "width": 75, + "height": 19 + }, + "center": [1526, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='15']" }, - "locator": "[data-midscene-task-1='19']" - }, - { - "content": "Download for macOS", - "rect": { - "left": 340, - "top": 452, - "width": 225, - "height": 50 + { + "content": "Version 1.91", + "rect": { + "left": 684, + "top": 86, + "width": 70, + "height": 8 + }, + "center": [719, 90], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='16']" }, - "center": [ - 452, - 477 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "is now available! Read about the new features and fixes from June.", + "rect": { + "left": 339, + "top": 74, + "width": 1225, + "height": 22 + }, + "center": [951, 85], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='17']" }, - "locator": "[data-midscene-task-1='20']" - }, - { - "content": "Web", - "rect": { - "left": 340, - "top": 520, - "width": 29, - "height": 17 + { + "content": "Free. Built on open source. Runs everywhere.", + "rect": { + "left": 340, + "top": 224, + "width": 331, + "height": 38 + }, + "center": [505, 243], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='18']" }, - "center": [ - 354, - 528 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Code Editing. Redefined.", + "rect": { + "left": 342, + "top": 293, + "width": 326, + "height": 114 + }, + "center": [505, 350], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='19']" }, - "locator": "[data-midscene-task-1='21']" - }, - { - "content": ",", - "rect": { - "left": 340, - "top": 518, - "width": 272, - "height": 22 + { + "content": "Download for macOS", + "rect": { + "left": 340, + "top": 452, + "width": 225, + "height": 50 + }, + "center": [452, 477], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='20']" }, - "center": [ - 476, - 529 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Web", + "rect": { + "left": 340, + "top": 520, + "width": 29, + "height": 17 + }, + "center": [354, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='21']" }, - "locator": "[data-midscene-task-1='22']" - }, - { - "content": "Insiders edition", - "rect": { - "left": 377, - "top": 520, - "width": 101, - "height": 17 - }, - "center": [ - 427, - 528 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": ",", + "rect": { + "left": 340, + "top": 518, + "width": 272, + "height": 22 + }, + "center": [476, 529], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='22']" }, - "locator": "[data-midscene-task-1='23']" - }, - { - "content": ", or", - "rect": { - "left": 340, - "top": 518, - "width": 272, - "height": 22 + { + "content": "Insiders edition", + "rect": { + "left": 377, + "top": 520, + "width": 101, + "height": 17 + }, + "center": [427, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='23']" }, - "center": [ - 476, - 529 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": ", or", + "rect": { + "left": 340, + "top": 518, + "width": 272, + "height": 22 + }, + "center": [476, 529], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='24']" }, - "locator": "[data-midscene-task-1='24']" - }, - { - "content": "other platforms", - "rect": { - "left": 504, - "top": 520, - "width": 102, - "height": 17 - }, - "center": [ - 555, - 528 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "other platforms", + "rect": { + "left": 504, + "top": 520, + "width": 102, + "height": 17 + }, + "center": [555, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='25']" }, - "locator": "[data-midscene-task-1='25']" - }, - { - "content": "By using VS Code, you agree to its", - "rect": { - "left": 340, - "top": 556, - "width": 344, - "height": 17 + { + "content": "By using VS Code, you agree to its", + "rect": { + "left": 340, + "top": 556, + "width": 344, + "height": 17 + }, + "center": [512, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='26']" }, - "center": [ - 512, - 564 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "license", + "rect": { + "left": 524, + "top": 557, + "width": 37, + "height": 13 + }, + "center": [542, 563], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='27']" }, - "locator": "[data-midscene-task-1='26']" - }, - { - "content": "license", - "rect": { - "left": 524, - "top": 557, - "width": 37, - "height": 13 - }, - "center": [ - 542, - 563 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "and", + "rect": { + "left": 524, + "top": 556, + "width": 160, + "height": 17 + }, + "center": [604, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='28']" }, - "locator": "[data-midscene-task-1='27']" - }, - { - "content": "and", - "rect": { - "left": 524, - "top": 556, - "width": 160, - "height": 17 - }, - "center": [ - 604, - 564 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "privacy statement", + "rect": { + "left": 587, + "top": 557, + "width": 94, + "height": 13 + }, + "center": [634, 563], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='29']" }, - "locator": "[data-midscene-task-1='28']" - }, - { - "content": "privacy statement", - "rect": { - "left": 587, - "top": 557, - "width": 94, - "height": 13 - }, - "center": [ - 634, - 563 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": ".", + "rect": { + "left": 524, + "top": 556, + "width": 160, + "height": 17 + }, + "center": [604, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='30']" }, - "locator": "[data-midscene-task-1='29']" - }, - { - "content": ".", - "rect": { - "left": 524, - "top": 556, - "width": 160, - "height": 17 - }, - "center": [ - 604, - 564 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Code in any language", + "rect": { + "left": 340, + "top": 856, + "width": 289, + "height": 29 + }, + "center": [484, 870], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='31']" }, - "locator": "[data-midscene-task-1='30']" - }, - { - "content": "Code in any language", - "rect": { - "left": 340, - "top": 856, - "width": 289, - "height": 29 + { + "content": "VS Code supports almost every major programming language. Several ship in the box, like JavaScript, TypeScript, CSS, and HTML, but extensions for others can be found in the VS Code Marketplace.", + "rect": { + "left": 340, + "top": 905, + "width": 355, + "height": 119 + }, + "center": [517, 964], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='32']" }, - "center": [ - 484, - 870 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "JavaScript", + "rect": { + "left": 832, + "top": 857, + "width": 95, + "height": 18 + }, + "center": [879, 866], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='33']" }, - "locator": "[data-midscene-task-1='31']" - }, - { - "content": "VS Code supports almost every major programming language. Several ship in the box, like JavaScript, TypeScript, CSS, and HTML, but extensions for others can be found in the VS Code Marketplace.", - "rect": { - "left": 340, - "top": 905, - "width": 355, - "height": 119 + { + "content": "TypeScript", + "rect": { + "left": 1111, + "top": 855, + "width": 101, + "height": 20 + }, + "center": [1161, 865], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='34']" }, - "center": [ - 517, - 964 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Python", + "rect": { + "left": 1387, + "top": 855, + "width": 66, + "height": 20 + }, + "center": [1420, 865], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='35']" }, - "locator": "[data-midscene-task-1='32']" - }, - { - "content": "JavaScript", - "rect": { - "left": 832, - "top": 857, - "width": 95, - "height": 18 - }, - "center": [ - 879, - 866 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "C#", + "rect": { + "left": 832, + "top": 903, + "width": 21, + "height": 17 + }, + "center": [842, 911], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='36']" }, - "locator": "[data-midscene-task-1='33']" - }, - { - "content": "TypeScript", - "rect": { - "left": 1111, - "top": 855, - "width": 101, - "height": 20 - }, - "center": [ - 1161, - 865 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "C++", + "rect": { + "left": 1111, + "top": 903, + "width": 29, + "height": 18 + }, + "center": [1125, 912], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='37']" }, - "locator": "[data-midscene-task-1='34']" - }, - { - "content": "Python", - "rect": { - "left": 1387, - "top": 855, - "width": 66, - "height": 20 - }, - "center": [ - 1420, - 865 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "HTML", + "rect": { + "left": 1390, + "top": 903, + "width": 38, + "height": 18 + }, + "center": [1409, 912], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='38']" }, - "locator": "[data-midscene-task-1='35']" - }, - { - "content": "C#", - "rect": { - "left": 832, - "top": 903, - "width": 21, - "height": 17 - }, - "center": [ - 842, - 911 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Java", + "rect": { + "left": 832, + "top": 955, + "width": 38, + "height": 13 + }, + "center": [851, 961], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='39']" }, - "locator": "[data-midscene-task-1='36']" - }, - { - "content": "C++", - "rect": { - "left": 1111, - "top": 903, - "width": 29, - "height": 18 - }, - "center": [ - 1125, - 912 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "JSON", + "rect": { + "left": 1111, + "top": 951, + "width": 41, + "height": 20 + }, + "center": [1131, 961], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='40']" }, - "locator": "[data-midscene-task-1='37']" - }, - { - "content": "HTML", - "rect": { - "left": 1390, - "top": 903, - "width": 38, - "height": 18 - }, - "center": [ - 1409, - 912 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "PHP", + "rect": { + "left": 1388, + "top": 951, + "width": 36, + "height": 18 + }, + "center": [1406, 960], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='41']" }, - "locator": "[data-midscene-task-1='38']" - }, - { - "content": "Java", - "rect": { - "left": 832, - "top": 955, - "width": 38, - "height": 13 - }, - "center": [ - 851, - 961 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Markdown", + "rect": { + "left": 829, + "top": 999, + "width": 84, + "height": 20 + }, + "center": [871, 1009], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='42']" }, - "locator": "[data-midscene-task-1='39']" - }, - { - "content": "JSON", - "rect": { - "left": 1111, - "top": 951, - "width": 41, - "height": 20 - }, - "center": [ - 1131, - 961 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Powershell", + "rect": { + "left": 1108, + "top": 999, + "width": 104, + "height": 20 + }, + "center": [1160, 1009], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='43']" }, - "locator": "[data-midscene-task-1='40']" - }, + { + "content": "YAML", + "rect": { + "left": 1387, + "top": 999, + "width": 47, + "height": 20 + }, + "center": [1410, 1009], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='44']" + } + ] + }, + "userQuery": { + "element": "download buttons on the page" + }, + "matchedSection": [], + "matchedElement": [ { - "content": "PHP", + "content": "Download", "rect": { - "left": 1388, - "top": 951, - "width": 36, - "height": 18 - }, - "center": [ - 1406, - 960 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + "left": 1489, + "top": 85, + "width": 75, + "height": 19 }, - "locator": "[data-midscene-task-1='41']" - }, - { - "content": "Markdown", - "rect": { - "left": 829, - "top": 999, - "width": 84, - "height": 20 - }, - "center": [ - 871, - 1009 - ], + "center": [1526, 94], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='42']" + "locator": "[data-midscene-task-1='15']" }, { - "content": "Powershell", + "content": "Download for macOS", "rect": { - "left": 1108, - "top": 999, - "width": 104, - "height": 20 - }, - "center": [ - 1160, - 1009 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + "left": 340, + "top": 452, + "width": 225, + "height": 50 }, - "locator": "[data-midscene-task-1='43']" - }, - { - "content": "YAML", - "rect": { - "left": 1387, - "top": 999, - "width": 47, - "height": 20 - }, - "center": [ - 1410, - 1009 - ], + "center": [452, 477], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='44']" + "locator": "[data-midscene-task-1='20']" } - ] - }, - "userQuery": { - "dataDemand": { - "cookiePrompt": "SECTION_MATCHER_FLAG/cookie prompt with its action buttons on the top of the page", - "navigation": "SECTION_MATCHER_FLAG/top navigation items besides logo", - "topRightWidgets": "SECTION_MATCHER_FLAG/widgets on the top right corner", - "downloadSection": "SECTION_MATCHER_FLAG/The main download section with a big button and links of other platforms", - "mainDownloadBtn": "follow LOCATE_ONE_ELEMENT: main download button", - "navNames": "string[], nav names in navigation section" + ], + "data": null, + "taskInfo": { + "url": "https://code.visualstudio.com/", + "name": "download buttons of vscode", + "durationMs": 3678, + "rawResponse": "{\"elements\":[{\"id\":\"15\"},{\"id\":\"20\"}]}", + "systemPrompt": "\nYou are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.\n\nThe user will give you a screenshot and the texts on it. There may be some none-English characters (like Chinese) on it, indicating it's an non-English app.\n\nBased on the information you get, find one or more text elements on the page.\nHere is the description: download buttons on the page\n \nReturn in the following JSON format:\n{\n \"elements\": [ // Leave it an empty array when no element is found\n { \n \"id\": \"id of the element, like 123\", \n },\n // more ...\n ], \n errors?: [], // string[], error message if any\n}\n" } }, - "matchedSection": [ - { - "name": "cookiePrompt", - "description": "This section contains the cookie prompt with its action buttons.", - "sectionCharacteristics": "It is at the top of the page, with a background color that usually stands out to inform users about cookies. The text and buttons for accepting, rejecting, and managing cookies are included.", + { + "logId": "27c2b890-5cc2-49f0-8432-7668c26ef2fe", + "sdkVersion": "0.1.1", + "logTime": 1721022621788, + "type": "extract", + "context": { + "screenshotBase64": "", + "size": { + "width": 1920, + "height": 1080 + }, + "timestamp": 1721022608034, "content": [ { "content": "We use optional cookies to improve your experience on our websites, such as through social media connections, and to display personalized advertising based on your online activity. If you reject optional cookies, only cookies necessary to provide you the services will be used. You may change your selection by clicking “Manage Cookies” at the bottom of the page.", @@ -1724,10 +755,7 @@ "width": 1102, "height": 49 }, - "center": [ - 695, - 32 - ], + "center": [695, 32], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1742,10 +770,7 @@ "width": 1060, "height": 32 }, - "center": [ - 674, - 41 - ], + "center": [674, 41], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1760,10 +785,7 @@ "width": 121, "height": 15 }, - "center": [ - 270, - 48 - ], + "center": [270, 48], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1778,10 +800,7 @@ "width": 48, "height": 15 }, - "center": [ - 1421, - 33 - ], + "center": [1421, 33], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1796,10 +815,7 @@ "width": 44, "height": 16 }, - "center": [ - 1583, - 34 - ], + "center": [1583, 34], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1814,78 +830,13 @@ "width": 110, "height": 16 }, - "center": [ - 1744, - 34 - ], + "center": [1744, 34], "page": { "_isDragging": false, "_timeoutSettings": {} }, "locator": "[data-midscene-task-1='5']" - } - ], - "rect": { - "left": 144, - "top": 8, - "width": 1655, - "height": 49 - } - }, - { - "name": "topRightWidgets", - "description": "This section contains the widgets on the top right corner such as Search and Download buttons.", - "sectionCharacteristics": "It is placed at the top right corner of the page, adjacent to the navigation section. This can include a search bar and other interactive elements like a download button.", - "content": [ - { - "content": "Search Docs", - "rect": { - "left": 1265, - "top": 78, - "width": 200, - "height": 32 - }, - "center": [ - 1365, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='14']" }, - { - "content": "Download", - "rect": { - "left": 1489, - "top": 85, - "width": 75, - "height": 19 - }, - "center": [ - 1526, - 94 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='15']" - } - ], - "rect": { - "left": 1265, - "top": 78, - "width": 299, - "height": 32 - } - }, - { - "name": "navigation", - "description": "This section includes the top navigation items, providing links to various sections of the site.", - "sectionCharacteristics": "Located below the cookie prompt, this section contains links such as Docs, Updates, Blog, API, Extensions, FAQ, and Learn, along with the Visual Studio Code logo. The navigation links are horizontally aligned.", - "content": [ { "content": "Visual Studio Code", "rect": { @@ -1894,10 +845,7 @@ "width": 170, "height": 24 }, - "center": [ - 461, - 94 - ], + "center": [461, 94], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1912,10 +860,7 @@ "width": 38, "height": 15 }, - "center": [ - 592, - 94 - ], + "center": [592, 94], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1930,10 +875,7 @@ "width": 64, "height": 17 }, - "center": [ - 666, - 95 - ], + "center": [666, 95], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1948,10 +890,7 @@ "width": 34, "height": 17 }, - "center": [ - 738, - 94 - ], + "center": [738, 94], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1966,10 +905,7 @@ "width": 24, "height": 14 }, - "center": [ - 791, - 94 - ], + "center": [791, 94], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -1984,65 +920,103 @@ "width": 83, "height": 15 }, - "center": [ - 868, - 94 - ], + "center": [868, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='11']" + }, + { + "content": "FAQ", + "rect": { + "left": 932, + "top": 87, + "width": 31, + "height": 15 + }, + "center": [947, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='12']" + }, + { + "content": "Learn", + "rect": { + "left": 987, + "top": 88, + "width": 42, + "height": 15 + }, + "center": [1008, 95], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='13']" + }, + { + "content": "Search Docs", + "rect": { + "left": 1265, + "top": 78, + "width": 200, + "height": 32 + }, + "center": [1365, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='14']" + }, + { + "content": "Download", + "rect": { + "left": 1489, + "top": 85, + "width": 75, + "height": 19 + }, + "center": [1526, 94], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='11']" + "locator": "[data-midscene-task-1='15']" }, { - "content": "FAQ", + "content": "Version 1.91", "rect": { - "left": 932, - "top": 87, - "width": 31, - "height": 15 + "left": 684, + "top": 86, + "width": 70, + "height": 8 }, - "center": [ - 947, - 94 - ], + "center": [719, 90], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='12']" + "locator": "[data-midscene-task-1='16']" }, { - "content": "Learn", + "content": "is now available! Read about the new features and fixes from June.", "rect": { - "left": 987, - "top": 88, - "width": 42, - "height": 15 + "left": 339, + "top": 74, + "width": 1225, + "height": 22 }, - "center": [ - 1008, - 95 - ], + "center": [951, 85], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='13']" - } - ], - "rect": { - "left": 376, - "top": 82, - "width": 653, - "height": 24 - } - }, - { - "name": "downloadSection", - "description": "The main download section with a big button and links for other platforms.", - "sectionCharacteristics": "Located prominently in the middle of the page, this section features a large download button for macOS and links for other platforms. It provides users immediate call-to-action for downloading the software.", - "content": [ + "locator": "[data-midscene-task-1='17']" + }, { "content": "Free. Built on open source. Runs everywhere.", "rect": { @@ -2051,10 +1025,7 @@ "width": 331, "height": 38 }, - "center": [ - 505, - 243 - ], + "center": [505, 243], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2069,10 +1040,7 @@ "width": 326, "height": 114 }, - "center": [ - 505, - 350 - ], + "center": [505, 350], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2087,10 +1055,7 @@ "width": 225, "height": 50 }, - "center": [ - 452, - 477 - ], + "center": [452, 477], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2105,10 +1070,7 @@ "width": 29, "height": 17 }, - "center": [ - 354, - 528 - ], + "center": [354, 528], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2123,10 +1085,7 @@ "width": 272, "height": 22 }, - "center": [ - 476, - 529 - ], + "center": [476, 529], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2141,10 +1100,7 @@ "width": 101, "height": 17 }, - "center": [ - 427, - 528 - ], + "center": [427, 528], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2159,10 +1115,7 @@ "width": 272, "height": 22 }, - "center": [ - 476, - 529 - ], + "center": [476, 529], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2177,10 +1130,7 @@ "width": 102, "height": 17 }, - "center": [ - 555, - 528 - ], + "center": [555, 528], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2195,10 +1145,7 @@ "width": 344, "height": 17 }, - "center": [ - 512, - 564 - ], + "center": [512, 564], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2213,10 +1160,7 @@ "width": 37, "height": 13 }, - "center": [ - 542, - 563 - ], + "center": [542, 563], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2231,10 +1175,7 @@ "width": 160, "height": 17 }, - "center": [ - 604, - 564 - ], + "center": [604, 564], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2249,10 +1190,7 @@ "width": 94, "height": 13 }, - "center": [ - 634, - 563 - ], + "center": [634, 563], "page": { "_isDragging": false, "_timeoutSettings": {} @@ -2267,1305 +1205,1809 @@ "width": 160, "height": 17 }, - "center": [ - 604, - 564 - ], + "center": [604, 564], "page": { "_isDragging": false, "_timeoutSettings": {} }, "locator": "[data-midscene-task-1='30']" - } - ], - "rect": { - "left": 340, - "top": 224, - "width": 344, - "height": 349 - } - } - ], - "matchedElement": [], - "data": { - "cookiePrompt": { - "name": "cookiePrompt", - "description": "This section contains the cookie prompt with its action buttons.", - "sectionCharacteristics": "It is at the top of the page, with a background color that usually stands out to inform users about cookies. The text and buttons for accepting, rejecting, and managing cookies are included.", - "content": [ + }, { - "content": "We use optional cookies to improve your experience on our websites, such as through social media connections, and to display personalized advertising based on your online activity. If you reject optional cookies, only cookies necessary to provide you the services will be used. You may change your selection by clicking “Manage Cookies” at the bottom of the page.", + "content": "Code in any language", "rect": { - "left": 144, - "top": 8, - "width": 1102, - "height": 49 + "left": 340, + "top": 856, + "width": 289, + "height": 29 }, - "center": [ - 695, - 32 - ], + "center": [484, 870], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='0']" + "locator": "[data-midscene-task-1='31']" }, { - "content": "Privacy Statement", + "content": "VS Code supports almost every major programming language. Several ship in the box, like JavaScript, TypeScript, CSS, and HTML, but extensions for others can be found in the VS Code Marketplace.", "rect": { - "left": 144, - "top": 25, - "width": 1060, - "height": 32 + "left": 340, + "top": 905, + "width": 355, + "height": 119 }, - "center": [ - 674, - 41 - ], + "center": [517, 964], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='1']" + "locator": "[data-midscene-task-1='32']" }, { - "content": "Third-Party Cookies", + "content": "JavaScript", "rect": { - "left": 210, - "top": 41, - "width": 121, - "height": 15 + "left": 832, + "top": 857, + "width": 95, + "height": 18 }, - "center": [ - 270, - 48 - ], + "center": [879, 866], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='2']" + "locator": "[data-midscene-task-1='33']" }, { - "content": "Accept", + "content": "TypeScript", "rect": { - "left": 1397, - "top": 26, - "width": 48, - "height": 15 + "left": 1111, + "top": 855, + "width": 101, + "height": 20 }, - "center": [ - 1421, - 33 - ], + "center": [1161, 865], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='3']" + "locator": "[data-midscene-task-1='34']" }, { - "content": "Reject", + "content": "Python", "rect": { - "left": 1561, - "top": 26, - "width": 44, - "height": 16 + "left": 1387, + "top": 855, + "width": 66, + "height": 20 }, - "center": [ - 1583, - 34 - ], + "center": [1420, 865], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='4']" + "locator": "[data-midscene-task-1='35']" }, { - "content": "Manage cookies", + "content": "C#", "rect": { - "left": 1689, - "top": 26, - "width": 110, - "height": 16 + "left": 832, + "top": 903, + "width": 21, + "height": 17 }, - "center": [ - 1744, - 34 - ], + "center": [842, 911], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='5']" - } - ], - "rect": { - "left": 144, - "top": 8, - "width": 1655, - "height": 49 - } - }, - "navigation": { - "name": "navigation", - "description": "This section includes the top navigation items, providing links to various sections of the site.", - "sectionCharacteristics": "Located below the cookie prompt, this section contains links such as Docs, Updates, Blog, API, Extensions, FAQ, and Learn, along with the Visual Studio Code logo. The navigation links are horizontally aligned.", - "content": [ + "locator": "[data-midscene-task-1='36']" + }, { - "content": "Visual Studio Code", + "content": "C++", "rect": { - "left": 376, - "top": 82, - "width": 170, - "height": 24 + "left": 1111, + "top": 903, + "width": 29, + "height": 18 }, - "center": [ - 461, - 94 - ], + "center": [1125, 912], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='6']" + "locator": "[data-midscene-task-1='37']" }, { - "content": "Docs", + "content": "HTML", "rect": { - "left": 573, - "top": 87, + "left": 1390, + "top": 903, "width": 38, - "height": 15 + "height": 18 }, - "center": [ - 592, - 94 - ], + "center": [1409, 912], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='7']" + "locator": "[data-midscene-task-1='38']" }, { - "content": "Updates", + "content": "Java", "rect": { - "left": 634, - "top": 87, - "width": 64, - "height": 17 + "left": 832, + "top": 955, + "width": 38, + "height": 13 }, - "center": [ - 666, - 95 - ], + "center": [851, 961], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='8']" + "locator": "[data-midscene-task-1='39']" }, { - "content": "Blog", + "content": "JSON", "rect": { - "left": 721, - "top": 86, - "width": 34, - "height": 17 + "left": 1111, + "top": 951, + "width": 41, + "height": 20 }, - "center": [ - 738, - 94 - ], + "center": [1131, 961], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='9']" + "locator": "[data-midscene-task-1='40']" }, { - "content": "API", + "content": "PHP", "rect": { - "left": 779, - "top": 87, - "width": 24, - "height": 14 + "left": 1388, + "top": 951, + "width": 36, + "height": 18 }, - "center": [ - 791, - 94 - ], + "center": [1406, 960], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='10']" + "locator": "[data-midscene-task-1='41']" }, { - "content": "Extensions", + "content": "Markdown", "rect": { - "left": 827, - "top": 87, - "width": 83, - "height": 15 + "left": 829, + "top": 999, + "width": 84, + "height": 20 }, - "center": [ - 868, - 94 - ], + "center": [871, 1009], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='11']" + "locator": "[data-midscene-task-1='42']" }, { - "content": "FAQ", + "content": "Powershell", "rect": { - "left": 932, - "top": 87, - "width": 31, - "height": 15 + "left": 1108, + "top": 999, + "width": 104, + "height": 20 }, - "center": [ - 947, - 94 - ], + "center": [1160, 1009], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='12']" + "locator": "[data-midscene-task-1='43']" }, { - "content": "Learn", + "content": "YAML", "rect": { - "left": 987, - "top": 88, - "width": 42, - "height": 15 + "left": 1387, + "top": 999, + "width": 47, + "height": 20 }, - "center": [ - 1008, - 95 - ], + "center": [1410, 1009], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='13']" + "locator": "[data-midscene-task-1='44']" + } + ] + }, + "userQuery": { + "dataDemand": { + "cookiePrompt": "SECTION_MATCHER_FLAG/cookie prompt with its action buttons on the top of the page", + "navigation": "SECTION_MATCHER_FLAG/top navigation items besides logo", + "topRightWidgets": "SECTION_MATCHER_FLAG/widgets on the top right corner", + "downloadSection": "SECTION_MATCHER_FLAG/The main download section with a big button and links of other platforms", + "mainDownloadBtn": "follow LOCATE_ONE_ELEMENT: main download button", + "navNames": "string[], nav names in navigation section" + } + }, + "matchedSection": [ + { + "name": "cookiePrompt", + "description": "This section contains the cookie prompt with its action buttons.", + "sectionCharacteristics": "It is at the top of the page, with a background color that usually stands out to inform users about cookies. The text and buttons for accepting, rejecting, and managing cookies are included.", + "content": [ + { + "content": "We use optional cookies to improve your experience on our websites, such as through social media connections, and to display personalized advertising based on your online activity. If you reject optional cookies, only cookies necessary to provide you the services will be used. You may change your selection by clicking “Manage Cookies” at the bottom of the page.", + "rect": { + "left": 144, + "top": 8, + "width": 1102, + "height": 49 + }, + "center": [695, 32], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='0']" + }, + { + "content": "Privacy Statement", + "rect": { + "left": 144, + "top": 25, + "width": 1060, + "height": 32 + }, + "center": [674, 41], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='1']" + }, + { + "content": "Third-Party Cookies", + "rect": { + "left": 210, + "top": 41, + "width": 121, + "height": 15 + }, + "center": [270, 48], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='2']" + }, + { + "content": "Accept", + "rect": { + "left": 1397, + "top": 26, + "width": 48, + "height": 15 + }, + "center": [1421, 33], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='3']" + }, + { + "content": "Reject", + "rect": { + "left": 1561, + "top": 26, + "width": 44, + "height": 16 + }, + "center": [1583, 34], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='4']" + }, + { + "content": "Manage cookies", + "rect": { + "left": 1689, + "top": 26, + "width": 110, + "height": 16 + }, + "center": [1744, 34], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='5']" + } + ], + "rect": { + "left": 144, + "top": 8, + "width": 1655, + "height": 49 + } + }, + { + "name": "topRightWidgets", + "description": "This section contains the widgets on the top right corner such as Search and Download buttons.", + "sectionCharacteristics": "It is placed at the top right corner of the page, adjacent to the navigation section. This can include a search bar and other interactive elements like a download button.", + "content": [ + { + "content": "Search Docs", + "rect": { + "left": 1265, + "top": 78, + "width": 200, + "height": 32 + }, + "center": [1365, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='14']" + }, + { + "content": "Download", + "rect": { + "left": 1489, + "top": 85, + "width": 75, + "height": 19 + }, + "center": [1526, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='15']" + } + ], + "rect": { + "left": 1265, + "top": 78, + "width": 299, + "height": 32 + } + }, + { + "name": "navigation", + "description": "This section includes the top navigation items, providing links to various sections of the site.", + "sectionCharacteristics": "Located below the cookie prompt, this section contains links such as Docs, Updates, Blog, API, Extensions, FAQ, and Learn, along with the Visual Studio Code logo. The navigation links are horizontally aligned.", + "content": [ + { + "content": "Visual Studio Code", + "rect": { + "left": 376, + "top": 82, + "width": 170, + "height": 24 + }, + "center": [461, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='6']" + }, + { + "content": "Docs", + "rect": { + "left": 573, + "top": 87, + "width": 38, + "height": 15 + }, + "center": [592, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='7']" + }, + { + "content": "Updates", + "rect": { + "left": 634, + "top": 87, + "width": 64, + "height": 17 + }, + "center": [666, 95], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='8']" + }, + { + "content": "Blog", + "rect": { + "left": 721, + "top": 86, + "width": 34, + "height": 17 + }, + "center": [738, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='9']" + }, + { + "content": "API", + "rect": { + "left": 779, + "top": 87, + "width": 24, + "height": 14 + }, + "center": [791, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='10']" + }, + { + "content": "Extensions", + "rect": { + "left": 827, + "top": 87, + "width": 83, + "height": 15 + }, + "center": [868, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='11']" + }, + { + "content": "FAQ", + "rect": { + "left": 932, + "top": 87, + "width": 31, + "height": 15 + }, + "center": [947, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='12']" + }, + { + "content": "Learn", + "rect": { + "left": 987, + "top": 88, + "width": 42, + "height": 15 + }, + "center": [1008, 95], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='13']" + } + ], + "rect": { + "left": 376, + "top": 82, + "width": 653, + "height": 24 + } + }, + { + "name": "downloadSection", + "description": "The main download section with a big button and links for other platforms.", + "sectionCharacteristics": "Located prominently in the middle of the page, this section features a large download button for macOS and links for other platforms. It provides users immediate call-to-action for downloading the software.", + "content": [ + { + "content": "Free. Built on open source. Runs everywhere.", + "rect": { + "left": 340, + "top": 224, + "width": 331, + "height": 38 + }, + "center": [505, 243], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='18']" + }, + { + "content": "Code Editing. Redefined.", + "rect": { + "left": 342, + "top": 293, + "width": 326, + "height": 114 + }, + "center": [505, 350], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='19']" + }, + { + "content": "Download for macOS", + "rect": { + "left": 340, + "top": 452, + "width": 225, + "height": 50 + }, + "center": [452, 477], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='20']" + }, + { + "content": "Web", + "rect": { + "left": 340, + "top": 520, + "width": 29, + "height": 17 + }, + "center": [354, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='21']" + }, + { + "content": ",", + "rect": { + "left": 340, + "top": 518, + "width": 272, + "height": 22 + }, + "center": [476, 529], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='22']" + }, + { + "content": "Insiders edition", + "rect": { + "left": 377, + "top": 520, + "width": 101, + "height": 17 + }, + "center": [427, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='23']" + }, + { + "content": ", or", + "rect": { + "left": 340, + "top": 518, + "width": 272, + "height": 22 + }, + "center": [476, 529], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='24']" + }, + { + "content": "other platforms", + "rect": { + "left": 504, + "top": 520, + "width": 102, + "height": 17 + }, + "center": [555, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='25']" + }, + { + "content": "By using VS Code, you agree to its", + "rect": { + "left": 340, + "top": 556, + "width": 344, + "height": 17 + }, + "center": [512, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='26']" + }, + { + "content": "license", + "rect": { + "left": 524, + "top": 557, + "width": 37, + "height": 13 + }, + "center": [542, 563], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='27']" + }, + { + "content": "and", + "rect": { + "left": 524, + "top": 556, + "width": 160, + "height": 17 + }, + "center": [604, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='28']" + }, + { + "content": "privacy statement", + "rect": { + "left": 587, + "top": 557, + "width": 94, + "height": 13 + }, + "center": [634, 563], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='29']" + }, + { + "content": ".", + "rect": { + "left": 524, + "top": 556, + "width": 160, + "height": 17 + }, + "center": [604, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='30']" + } + ], + "rect": { + "left": 340, + "top": 224, + "width": 344, + "height": 349 } - ], - "rect": { - "left": 376, - "top": 82, - "width": 653, - "height": 24 } + ], + "matchedElement": [], + "data": { + "cookiePrompt": { + "name": "cookiePrompt", + "description": "This section contains the cookie prompt with its action buttons.", + "sectionCharacteristics": "It is at the top of the page, with a background color that usually stands out to inform users about cookies. The text and buttons for accepting, rejecting, and managing cookies are included.", + "content": [ + { + "content": "We use optional cookies to improve your experience on our websites, such as through social media connections, and to display personalized advertising based on your online activity. If you reject optional cookies, only cookies necessary to provide you the services will be used. You may change your selection by clicking “Manage Cookies” at the bottom of the page.", + "rect": { + "left": 144, + "top": 8, + "width": 1102, + "height": 49 + }, + "center": [695, 32], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='0']" + }, + { + "content": "Privacy Statement", + "rect": { + "left": 144, + "top": 25, + "width": 1060, + "height": 32 + }, + "center": [674, 41], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='1']" + }, + { + "content": "Third-Party Cookies", + "rect": { + "left": 210, + "top": 41, + "width": 121, + "height": 15 + }, + "center": [270, 48], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='2']" + }, + { + "content": "Accept", + "rect": { + "left": 1397, + "top": 26, + "width": 48, + "height": 15 + }, + "center": [1421, 33], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='3']" + }, + { + "content": "Reject", + "rect": { + "left": 1561, + "top": 26, + "width": 44, + "height": 16 + }, + "center": [1583, 34], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='4']" + }, + { + "content": "Manage cookies", + "rect": { + "left": 1689, + "top": 26, + "width": 110, + "height": 16 + }, + "center": [1744, 34], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='5']" + } + ], + "rect": { + "left": 144, + "top": 8, + "width": 1655, + "height": 49 + } + }, + "navigation": { + "name": "navigation", + "description": "This section includes the top navigation items, providing links to various sections of the site.", + "sectionCharacteristics": "Located below the cookie prompt, this section contains links such as Docs, Updates, Blog, API, Extensions, FAQ, and Learn, along with the Visual Studio Code logo. The navigation links are horizontally aligned.", + "content": [ + { + "content": "Visual Studio Code", + "rect": { + "left": 376, + "top": 82, + "width": 170, + "height": 24 + }, + "center": [461, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='6']" + }, + { + "content": "Docs", + "rect": { + "left": 573, + "top": 87, + "width": 38, + "height": 15 + }, + "center": [592, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='7']" + }, + { + "content": "Updates", + "rect": { + "left": 634, + "top": 87, + "width": 64, + "height": 17 + }, + "center": [666, 95], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='8']" + }, + { + "content": "Blog", + "rect": { + "left": 721, + "top": 86, + "width": 34, + "height": 17 + }, + "center": [738, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='9']" + }, + { + "content": "API", + "rect": { + "left": 779, + "top": 87, + "width": 24, + "height": 14 + }, + "center": [791, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='10']" + }, + { + "content": "Extensions", + "rect": { + "left": 827, + "top": 87, + "width": 83, + "height": 15 + }, + "center": [868, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='11']" + }, + { + "content": "FAQ", + "rect": { + "left": 932, + "top": 87, + "width": 31, + "height": 15 + }, + "center": [947, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='12']" + }, + { + "content": "Learn", + "rect": { + "left": 987, + "top": 88, + "width": 42, + "height": 15 + }, + "center": [1008, 95], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='13']" + } + ], + "rect": { + "left": 376, + "top": 82, + "width": 653, + "height": 24 + } + }, + "topRightWidgets": { + "name": "topRightWidgets", + "description": "This section contains the widgets on the top right corner such as Search and Download buttons.", + "sectionCharacteristics": "It is placed at the top right corner of the page, adjacent to the navigation section. This can include a search bar and other interactive elements like a download button.", + "content": [ + { + "content": "Search Docs", + "rect": { + "left": 1265, + "top": 78, + "width": 200, + "height": 32 + }, + "center": [1365, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='14']" + }, + { + "content": "Download", + "rect": { + "left": 1489, + "top": 85, + "width": 75, + "height": 19 + }, + "center": [1526, 94], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='15']" + } + ], + "rect": { + "left": 1265, + "top": 78, + "width": 299, + "height": 32 + } + }, + "downloadSection": { + "name": "downloadSection", + "description": "The main download section with a big button and links for other platforms.", + "sectionCharacteristics": "Located prominently in the middle of the page, this section features a large download button for macOS and links for other platforms. It provides users immediate call-to-action for downloading the software.", + "content": [ + { + "content": "Free. Built on open source. Runs everywhere.", + "rect": { + "left": 340, + "top": 224, + "width": 331, + "height": 38 + }, + "center": [505, 243], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='18']" + }, + { + "content": "Code Editing. Redefined.", + "rect": { + "left": 342, + "top": 293, + "width": 326, + "height": 114 + }, + "center": [505, 350], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='19']" + }, + { + "content": "Download for macOS", + "rect": { + "left": 340, + "top": 452, + "width": 225, + "height": 50 + }, + "center": [452, 477], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='20']" + }, + { + "content": "Web", + "rect": { + "left": 340, + "top": 520, + "width": 29, + "height": 17 + }, + "center": [354, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='21']" + }, + { + "content": ",", + "rect": { + "left": 340, + "top": 518, + "width": 272, + "height": 22 + }, + "center": [476, 529], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='22']" + }, + { + "content": "Insiders edition", + "rect": { + "left": 377, + "top": 520, + "width": 101, + "height": 17 + }, + "center": [427, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='23']" + }, + { + "content": ", or", + "rect": { + "left": 340, + "top": 518, + "width": 272, + "height": 22 + }, + "center": [476, 529], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='24']" + }, + { + "content": "other platforms", + "rect": { + "left": 504, + "top": 520, + "width": 102, + "height": 17 + }, + "center": [555, 528], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='25']" + }, + { + "content": "By using VS Code, you agree to its", + "rect": { + "left": 340, + "top": 556, + "width": 344, + "height": 17 + }, + "center": [512, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='26']" + }, + { + "content": "license", + "rect": { + "left": 524, + "top": 557, + "width": 37, + "height": 13 + }, + "center": [542, 563], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='27']" + }, + { + "content": "and", + "rect": { + "left": 524, + "top": 556, + "width": 160, + "height": 17 + }, + "center": [604, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='28']" + }, + { + "content": "privacy statement", + "rect": { + "left": 587, + "top": 557, + "width": 94, + "height": 13 + }, + "center": [634, 563], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='29']" + }, + { + "content": ".", + "rect": { + "left": 524, + "top": 556, + "width": 160, + "height": 17 + }, + "center": [604, 564], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='30']" + } + ], + "rect": { + "left": 340, + "top": 224, + "width": 344, + "height": 349 + } + }, + "mainDownloadBtn": { + "content": "Download for macOS", + "rect": { + "left": 340, + "top": 452, + "width": 225, + "height": 50 + }, + "center": [452, 477], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='20']" + }, + "navNames": [ + "Docs", + "Updates", + "Blog", + "API", + "Extensions", + "FAQ", + "Learn" + ] }, - "topRightWidgets": { - "name": "topRightWidgets", - "description": "This section contains the widgets on the top right corner such as Search and Download buttons.", - "sectionCharacteristics": "It is placed at the top right corner of the page, adjacent to the navigation section. This can include a search bar and other interactive elements like a download button.", + "taskInfo": { + "url": "https://code.visualstudio.com/", + "name": "nav and features of online page", + "durationMs": 13740, + "rawResponse": "{\"language\":\"en\",\"sections\":[{\"name\":\"cookiePrompt\",\"description\":\"This section contains the cookie prompt with its action buttons.\",\"sectionCharacteristics\":\"It is at the top of the page, with a background color that usually stands out to inform users about cookies. The text and buttons for accepting, rejecting, and managing cookies are included.\",\"textIds\":[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\"]},{\"name\":\"navigation\",\"description\":\"This section includes the top navigation items, providing links to various sections of the site.\",\"sectionCharacteristics\":\"Located below the cookie prompt, this section contains links such as Docs, Updates, Blog, API, Extensions, FAQ, and Learn, along with the Visual Studio Code logo. The navigation links are horizontally aligned.\",\"textIds\":[\"6\",\"7\",\"8\",\"9\",\"10\",\"11\",\"12\",\"13\"]},{\"name\":\"topRightWidgets\",\"description\":\"This section contains the widgets on the top right corner such as Search and Download buttons.\",\"sectionCharacteristics\":\"It is placed at the top right corner of the page, adjacent to the navigation section. This can include a search bar and other interactive elements like a download button.\",\"textIds\":[\"14\",\"15\"]},{\"name\":\"downloadSection\",\"description\":\"The main download section with a big button and links for other platforms.\",\"sectionCharacteristics\":\"Located prominently in the middle of the page, this section features a large download button for macOS and links for other platforms. It provides users immediate call-to-action for downloading the software.\",\"textIds\":[\"18\",\"19\",\"20\",\"21\",\"22\",\"23\",\"24\",\"25\",\"26\",\"27\",\"28\",\"29\",\"30\"]}],\"data\":{\"cookiePrompt\":\"SECTION_MATCHER_FLAG/cookie prompt with its action buttons on the top of the page\",\"navigation\":\"SECTION_MATCHER_FLAG/top navigation items besides logo\",\"topRightWidgets\":\"SECTION_MATCHER_FLAG/widgets on the top right corner\",\"downloadSection\":\"SECTION_MATCHER_FLAG/The main download section with a big button and links of other platforms\",\"mainDownloadBtn\":\"LOCATE_ONE_ELEMENT/20\",\"navNames\":[\"Docs\",\"Updates\",\"Blog\",\"API\",\"Extensions\",\"FAQ\",\"Learn\"]}}", + "systemPrompt": "\nYou are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.\n\nThe user will give you a screenshot and the texts on it. There may be some none-English characters (like Chinese) on it, indicating it's an non-English app.\n\nYou have the following skills:\nskill name: segment_a_web_page \nskill content:\nBased on the functions and content of various elements on the page, segment the screenshot into different sections like navigation bar, product list, news area, etc. \nSome general rules for segmentation:\n* Each section should NOT overlap with each other.\n* Each text should only belong to one section.\n* [IMPORTANT] Whether the content visually appears to belong to different sections is a significant factor in segmenting the page.\n* Analyze the page in a top-to-bottom and left-to-right order.\n* The evidence indicates a separate section, for example \n - The background color of certain parts of the page changes.\n - A section of a page includes a title.\n* Provide the following data for each of the UI section you found.\n {\n \"name\": \"name of the section\",\n \"description\": \"briefly summarize the key content or usage of this section.\",\n \"sectionCharacteristics\": \"In view of the need to distinguish this section from the surrounding sections, explain the characteristics and how to define boundaries and what precautions to take.\",\n \"textIds\": [\"5\", \"6\", \"7\"], // ids of all text elements in this section\n }\n\nskill name: extract_data_from_UI\nrelated input: DATA_DEMAND\nskill content: \n* User will give you some data requirements in DATA_DEMAND. Consider the UI context, follow the user's instructions, and provide comprehensive data accordingly.\n* There may be some special commands in DATA_DEMAND, please pay extra attention\n - LOCATE_ONE_ELEMENT and LOCATE_ONE_OR_MORE_ELEMENTS: if you see a description that mentions the keyword LOCATE_ONE_ELEMENT or LOCATE_ONE_OR_MORE_ELEMENTS(e.g. follow LOCATE_ONE_ELEMENT : i want to find ...), it means user wants to locate a specific element meets the description. Return in this way: prefix + the id / comma-separated ids, for example: LOCATE_ONE_ELEMENT/1 , LOCATE_ONE_OR_MORE_ELEMENTS/1,2,3 . If not found, keep the prefix and leave the suffix empty, like LOCATE_ONE_ELEMENT/ .\n\nNow, do the following jobs:\nUse your segment_a_web_page skill to find the following section(s)\n* One section named `cookiePrompt`, usage or criteria : cookie prompt with its action buttons on the top of the page\n* One section named `navigation`, usage or criteria : top navigation items besides logo\n* One section named `topRightWidgets`, usage or criteria : widgets on the top right corner\n* One section named `downloadSection`, usage or criteria : The main download section with a big button and links of other platforms\nUse your extract_data_from_UI skill to find the following data, placing it in the `data` field\nDATA_DEMAND start:\nreturn in key-value style object, keys are cookiePrompt,navigation,topRightWidgets,downloadSection,mainDownloadBtn,navNames;\n{\n \"cookiePrompt\": \"SECTION_MATCHER_FLAG/cookie prompt with its action buttons on the top of the page\",\n \"navigation\": \"SECTION_MATCHER_FLAG/top navigation items besides logo\",\n \"topRightWidgets\": \"SECTION_MATCHER_FLAG/widgets on the top right corner\",\n \"downloadSection\": \"SECTION_MATCHER_FLAG/The main download section with a big button and links of other platforms\",\n \"mainDownloadBtn\": \"follow LOCATE_ONE_ELEMENT: main download button\",\n \"navNames\": \"string[], nav names in navigation section\"\n}\nDATA_DEMAND ends.\n\nReturn in the following JSON format:\n{\n language: \"en\", // \"en\" or \"zh\", the language of the page. Use the same language to describe section name, description, and similar fields.\n sections: [], // detailed information of each section from segment_a_web_page skill\n data: any, // the extracted data from extract_data_from_UI skill. Make sure both the value and scheme meet the DATA_DEMAND.\n errors?: [], // string[], error message if any\n}\n" + } + }, + { + "logId": "3b3c4824-165c-468c-b9da-936907d0df9f", + "sdkVersion": "0.1.1", + "logTime": 1721022638646, + "type": "extract", + "context": { + "screenshotBase64": "", + "size": { + "width": 1920, + "height": 1080 + }, + "timestamp": 1721022632511, "content": [ { - "content": "Search Docs", + "content": "Help", "rect": { - "left": 1265, - "top": 78, - "width": 200, - "height": 32 + "left": 24, + "top": 28, + "width": 34, + "height": 19 }, - "center": [ - 1365, - 94 - ], + "center": [41, 37], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='14']" + "locator": "[data-midscene-task-1='0']" }, { - "content": "Download", + "content": "Community", "rect": { - "left": 1489, - "top": 85, - "width": 75, - "height": 19 + "left": 86, + "top": 19, + "width": 83, + "height": 35 }, - "center": [ - 1526, - 94 - ], + "center": [127, 36], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='15']" - } - ], - "rect": { - "left": 1265, - "top": 78, - "width": 299, - "height": 32 - } - }, - "downloadSection": { - "name": "downloadSection", - "description": "The main download section with a big button and links for other platforms.", - "sectionCharacteristics": "Located prominently in the middle of the page, this section features a large download button for macOS and links for other platforms. It provides users immediate call-to-action for downloading the software.", - "content": [ + "locator": "[data-midscene-task-1='1']" + }, { - "content": "Free. Built on open source. Runs everywhere.", + "content": "Status", "rect": { - "left": 340, - "top": 224, - "width": 331, - "height": 38 + "left": 197, + "top": 29, + "width": 46, + "height": 14 }, - "center": [ - 505, - 243 - ], + "center": [220, 36], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='18']" + "locator": "[data-midscene-task-1='2']" }, { - "content": "Code Editing. Redefined.", + "content": "GitHub.com", "rect": { - "left": 342, - "top": 293, - "width": 326, - "height": 114 + "left": 1543, + "top": 23, + "width": 86, + "height": 23 }, - "center": [ - 505, - 350 - ], + "center": [1586, 34], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='19']" + "locator": "[data-midscene-task-1='3']" + }, + { + "content": "Twitter", + "rect": { + "left": 1658, + "top": 19, + "width": 50, + "height": 35 + }, + "center": [1683, 36], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='4']" + }, + { + "content": "All Systems Operational", + "rect": { + "left": 522, + "top": 415, + "width": 219, + "height": 24 + }, + "center": [631, 427], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='5']" }, { - "content": "Download for macOS", + "content": "Git Operations", "rect": { - "left": 340, - "top": 452, - "width": 225, - "height": 50 + "left": 492, + "top": 586, + "width": 107, + "height": 16 }, - "center": [ - 452, - 477 - ], + "center": [545, 594], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='20']" + "locator": "[data-midscene-task-1='6']" }, { - "content": "Web", + "content": "?", "rect": { - "left": 340, - "top": 520, - "width": 29, - "height": 17 + "left": 609, + "top": 582, + "width": 18, + "height": 18 }, - "center": [ - 354, - 528 - ], + "center": [618, 591], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='21']" + "locator": "[data-midscene-task-1='7']" }, { - "content": ",", + "content": "Normal", "rect": { - "left": 340, - "top": 518, - "width": 272, - "height": 22 + "left": 492, + "top": 605, + "width": 443, + "height": 20 }, - "center": [ - 476, - 529 - ], + "center": [713, 615], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='22']" + "locator": "[data-midscene-task-1='8']" }, { - "content": "Insiders edition", + "content": "API Requests", "rect": { - "left": 377, - "top": 520, - "width": 101, - "height": 17 + "left": 981, + "top": 587, + "width": 98, + "height": 15 }, - "center": [ - 427, - 528 - ], + "center": [1030, 594], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='23']" + "locator": "[data-midscene-task-1='9']" }, { - "content": ", or", + "content": "?", "rect": { - "left": 340, - "top": 518, - "width": 272, - "height": 22 + "left": 1088, + "top": 582, + "width": 18, + "height": 18 }, - "center": [ - 476, - 529 - ], + "center": [1097, 591], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='24']" + "locator": "[data-midscene-task-1='10']" }, { - "content": "other platforms", + "content": "Normal", "rect": { - "left": 504, - "top": 520, - "width": 102, - "height": 17 + "left": 981, + "top": 605, + "width": 441, + "height": 19 }, - "center": [ - 555, - 528 - ], + "center": [1201, 614], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='25']" + "locator": "[data-midscene-task-1='11']" }, { - "content": "By using VS Code, you agree to its", + "content": "Webhooks", "rect": { - "left": 340, - "top": 556, - "width": 344, - "height": 17 + "left": 492, + "top": 675, + "width": 78, + "height": 13 }, - "center": [ - 512, - 564 - ], + "center": [531, 681], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='26']" + "locator": "[data-midscene-task-1='12']" }, { - "content": "license", + "content": "?", "rect": { - "left": 524, - "top": 557, - "width": 37, - "height": 13 + "left": 579, + "top": 672, + "width": 18, + "height": 18 }, - "center": [ - 542, - 563 - ], + "center": [588, 681], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='27']" + "locator": "[data-midscene-task-1='13']" }, { - "content": "and", + "content": "Normal", "rect": { - "left": 524, - "top": 556, - "width": 160, - "height": 17 + "left": 492, + "top": 694, + "width": 444, + "height": 20 }, - "center": [ - 604, - 564 - ], + "center": [714, 704], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='28']" + "locator": "[data-midscene-task-1='14']" }, { - "content": "privacy statement", + "content": "Issues", "rect": { - "left": 587, - "top": 557, - "width": 94, + "left": 981, + "top": 676, + "width": 47, "height": 13 }, - "center": [ - 634, - 563 - ], + "center": [1004, 682], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='29']" + "locator": "[data-midscene-task-1='15']" }, { - "content": ".", + "content": "?", "rect": { - "left": 524, - "top": 556, - "width": 160, - "height": 17 + "left": 1037, + "top": 672, + "width": 18, + "height": 18 }, - "center": [ - 604, - 564 - ], + "center": [1046, 681], "page": { "_isDragging": false, "_timeoutSettings": {} }, - "locator": "[data-midscene-task-1='30']" - } - ], - "rect": { - "left": 340, - "top": 224, - "width": 344, - "height": 349 - } - }, - "mainDownloadBtn": { - "content": "Download for macOS", - "rect": { - "left": 340, - "top": 452, - "width": 225, - "height": 50 - }, - "center": [ - 452, - 477 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='20']" - }, - "navNames": [ - "Docs", - "Updates", - "Blog", - "API", - "Extensions", - "FAQ", - "Learn" - ] - }, - "taskInfo": { - "url": "https://code.visualstudio.com/", - "name": "nav and features of online page", - "durationMs": 13740, - "rawResponse": "{\"language\":\"en\",\"sections\":[{\"name\":\"cookiePrompt\",\"description\":\"This section contains the cookie prompt with its action buttons.\",\"sectionCharacteristics\":\"It is at the top of the page, with a background color that usually stands out to inform users about cookies. The text and buttons for accepting, rejecting, and managing cookies are included.\",\"textIds\":[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\"]},{\"name\":\"navigation\",\"description\":\"This section includes the top navigation items, providing links to various sections of the site.\",\"sectionCharacteristics\":\"Located below the cookie prompt, this section contains links such as Docs, Updates, Blog, API, Extensions, FAQ, and Learn, along with the Visual Studio Code logo. The navigation links are horizontally aligned.\",\"textIds\":[\"6\",\"7\",\"8\",\"9\",\"10\",\"11\",\"12\",\"13\"]},{\"name\":\"topRightWidgets\",\"description\":\"This section contains the widgets on the top right corner such as Search and Download buttons.\",\"sectionCharacteristics\":\"It is placed at the top right corner of the page, adjacent to the navigation section. This can include a search bar and other interactive elements like a download button.\",\"textIds\":[\"14\",\"15\"]},{\"name\":\"downloadSection\",\"description\":\"The main download section with a big button and links for other platforms.\",\"sectionCharacteristics\":\"Located prominently in the middle of the page, this section features a large download button for macOS and links for other platforms. It provides users immediate call-to-action for downloading the software.\",\"textIds\":[\"18\",\"19\",\"20\",\"21\",\"22\",\"23\",\"24\",\"25\",\"26\",\"27\",\"28\",\"29\",\"30\"]}],\"data\":{\"cookiePrompt\":\"SECTION_MATCHER_FLAG/cookie prompt with its action buttons on the top of the page\",\"navigation\":\"SECTION_MATCHER_FLAG/top navigation items besides logo\",\"topRightWidgets\":\"SECTION_MATCHER_FLAG/widgets on the top right corner\",\"downloadSection\":\"SECTION_MATCHER_FLAG/The main download section with a big button and links of other platforms\",\"mainDownloadBtn\":\"LOCATE_ONE_ELEMENT/20\",\"navNames\":[\"Docs\",\"Updates\",\"Blog\",\"API\",\"Extensions\",\"FAQ\",\"Learn\"]}}", - "systemPrompt": "\nYou are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.\n\nThe user will give you a screenshot and the texts on it. There may be some none-English characters (like Chinese) on it, indicating it's an non-English app.\n\nYou have the following skills:\nskill name: segment_a_web_page \nskill content:\nBased on the functions and content of various elements on the page, segment the screenshot into different sections like navigation bar, product list, news area, etc. \nSome general rules for segmentation:\n* Each section should NOT overlap with each other.\n* Each text should only belong to one section.\n* [IMPORTANT] Whether the content visually appears to belong to different sections is a significant factor in segmenting the page.\n* Analyze the page in a top-to-bottom and left-to-right order.\n* The evidence indicates a separate section, for example \n - The background color of certain parts of the page changes.\n - A section of a page includes a title.\n* Provide the following data for each of the UI section you found.\n {\n \"name\": \"name of the section\",\n \"description\": \"briefly summarize the key content or usage of this section.\",\n \"sectionCharacteristics\": \"In view of the need to distinguish this section from the surrounding sections, explain the characteristics and how to define boundaries and what precautions to take.\",\n \"textIds\": [\"5\", \"6\", \"7\"], // ids of all text elements in this section\n }\n\nskill name: extract_data_from_UI\nrelated input: DATA_DEMAND\nskill content: \n* User will give you some data requirements in DATA_DEMAND. Consider the UI context, follow the user's instructions, and provide comprehensive data accordingly.\n* There may be some special commands in DATA_DEMAND, please pay extra attention\n - LOCATE_ONE_ELEMENT and LOCATE_ONE_OR_MORE_ELEMENTS: if you see a description that mentions the keyword LOCATE_ONE_ELEMENT or LOCATE_ONE_OR_MORE_ELEMENTS(e.g. follow LOCATE_ONE_ELEMENT : i want to find ...), it means user wants to locate a specific element meets the description. Return in this way: prefix + the id / comma-separated ids, for example: LOCATE_ONE_ELEMENT/1 , LOCATE_ONE_OR_MORE_ELEMENTS/1,2,3 . If not found, keep the prefix and leave the suffix empty, like LOCATE_ONE_ELEMENT/ .\n\nNow, do the following jobs:\nUse your segment_a_web_page skill to find the following section(s)\n* One section named `cookiePrompt`, usage or criteria : cookie prompt with its action buttons on the top of the page\n* One section named `navigation`, usage or criteria : top navigation items besides logo\n* One section named `topRightWidgets`, usage or criteria : widgets on the top right corner\n* One section named `downloadSection`, usage or criteria : The main download section with a big button and links of other platforms\nUse your extract_data_from_UI skill to find the following data, placing it in the `data` field\nDATA_DEMAND start:\nreturn in key-value style object, keys are cookiePrompt,navigation,topRightWidgets,downloadSection,mainDownloadBtn,navNames;\n{\n \"cookiePrompt\": \"SECTION_MATCHER_FLAG/cookie prompt with its action buttons on the top of the page\",\n \"navigation\": \"SECTION_MATCHER_FLAG/top navigation items besides logo\",\n \"topRightWidgets\": \"SECTION_MATCHER_FLAG/widgets on the top right corner\",\n \"downloadSection\": \"SECTION_MATCHER_FLAG/The main download section with a big button and links of other platforms\",\n \"mainDownloadBtn\": \"follow LOCATE_ONE_ELEMENT: main download button\",\n \"navNames\": \"string[], nav names in navigation section\"\n}\nDATA_DEMAND ends.\n\nReturn in the following JSON format:\n{\n language: \"en\", // \"en\" or \"zh\", the language of the page. Use the same language to describe section name, description, and similar fields.\n sections: [], // detailed information of each section from segment_a_web_page skill\n data: any, // the extracted data from extract_data_from_UI skill. Make sure both the value and scheme meet the DATA_DEMAND.\n errors?: [], // string[], error message if any\n}\n" - } -}, -{ - "logId": "3b3c4824-165c-468c-b9da-936907d0df9f", - "sdkVersion": "0.1.1", - "logTime": 1721022638646, - "type": "extract", - "context": { - "screenshotBase64": "", - "size": { - "width": 1920, - "height": 1080 - }, - "timestamp": 1721022632511, - "content": [ - { - "content": "Help", - "rect": { - "left": 24, - "top": 28, - "width": 34, - "height": 19 - }, - "center": [ - 41, - 37 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='0']" - }, - { - "content": "Community", - "rect": { - "left": 86, - "top": 19, - "width": 83, - "height": 35 - }, - "center": [ - 127, - 36 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='1']" - }, - { - "content": "Status", - "rect": { - "left": 197, - "top": 29, - "width": 46, - "height": 14 - }, - "center": [ - 220, - 36 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='2']" - }, - { - "content": "GitHub.com", - "rect": { - "left": 1543, - "top": 23, - "width": 86, - "height": 23 - }, - "center": [ - 1586, - 34 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='3']" - }, - { - "content": "Twitter", - "rect": { - "left": 1658, - "top": 19, - "width": 50, - "height": 35 - }, - "center": [ - 1683, - 36 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + "locator": "[data-midscene-task-1='16']" }, - "locator": "[data-midscene-task-1='4']" - }, - { - "content": "All Systems Operational", - "rect": { - "left": 522, - "top": 415, - "width": 219, - "height": 24 - }, - "center": [ - 631, - 427 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='5']" - }, - { - "content": "Git Operations", - "rect": { - "left": 492, - "top": 586, - "width": 107, - "height": 16 - }, - "center": [ - 545, - 594 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='6']" - }, - { - "content": "?", - "rect": { - "left": 609, - "top": 582, - "width": 18, - "height": 18 - }, - "center": [ - 618, - 591 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='7']" - }, - { - "content": "Normal", - "rect": { - "left": 492, - "top": 605, - "width": 443, - "height": 20 - }, - "center": [ - 713, - 615 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='8']" - }, - { - "content": "API Requests", - "rect": { - "left": 981, - "top": 587, - "width": 98, - "height": 15 - }, - "center": [ - 1030, - 594 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='9']" - }, - { - "content": "?", - "rect": { - "left": 1088, - "top": 582, - "width": 18, - "height": 18 - }, - "center": [ - 1097, - 591 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='10']" - }, - { - "content": "Normal", - "rect": { - "left": 981, - "top": 605, - "width": 441, - "height": 19 - }, - "center": [ - 1201, - 614 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='11']" - }, - { - "content": "Webhooks", - "rect": { - "left": 492, - "top": 675, - "width": 78, - "height": 13 - }, - "center": [ - 531, - 681 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='12']" - }, - { - "content": "?", - "rect": { - "left": 579, - "top": 672, - "width": 18, - "height": 18 - }, - "center": [ - 588, - 681 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='13']" - }, - { - "content": "Normal", - "rect": { - "left": 492, - "top": 694, - "width": 444, - "height": 20 - }, - "center": [ - 714, - 704 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='14']" - }, - { - "content": "Issues", - "rect": { - "left": 981, - "top": 676, - "width": 47, - "height": 13 - }, - "center": [ - 1004, - 682 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='15']" - }, - { - "content": "?", - "rect": { - "left": 1037, - "top": 672, - "width": 18, - "height": 18 - }, - "center": [ - 1046, - 681 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='16']" - }, - { - "content": "Normal", - "rect": { - "left": 981, - "top": 694, - "width": 446, - "height": 21 - }, - "center": [ - 1204, - 704 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='17']" - }, - { - "content": "Pull Requests", - "rect": { - "left": 492, - "top": 765, - "width": 100, - "height": 17 - }, - "center": [ - 542, - 773 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} - }, - "locator": "[data-midscene-task-1='18']" - }, - { - "content": "?", - "rect": { - "left": 602, - "top": 761, - "width": 18, - "height": 18 - }, - "center": [ - 611, - 770 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Normal", + "rect": { + "left": 981, + "top": 694, + "width": 446, + "height": 21 + }, + "center": [1204, 704], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='17']" }, - "locator": "[data-midscene-task-1='19']" - }, - { - "content": "Normal", - "rect": { - "left": 492, - "top": 784, - "width": 438, - "height": 20 - }, - "center": [ - 711, - 794 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Pull Requests", + "rect": { + "left": 492, + "top": 765, + "width": 100, + "height": 17 + }, + "center": [542, 773], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='18']" }, - "locator": "[data-midscene-task-1='20']" - }, - { - "content": "Actions", - "rect": { - "left": 981, - "top": 765, - "width": 55, - "height": 14 - }, - "center": [ - 1008, - 772 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "?", + "rect": { + "left": 602, + "top": 761, + "width": 18, + "height": 18 + }, + "center": [611, 770], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='19']" }, - "locator": "[data-midscene-task-1='21']" - }, - { - "content": "?", - "rect": { - "left": 1046, - "top": 761, - "width": 18, - "height": 18 - }, - "center": [ - 1055, - 770 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Normal", + "rect": { + "left": 492, + "top": 784, + "width": 438, + "height": 20 + }, + "center": [711, 794], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='20']" }, - "locator": "[data-midscene-task-1='22']" - }, - { - "content": "Normal", - "rect": { - "left": 981, - "top": 784, - "width": 438, - "height": 20 - }, - "center": [ - 1200, - 794 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Actions", + "rect": { + "left": 981, + "top": 765, + "width": 55, + "height": 14 + }, + "center": [1008, 772], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='21']" }, - "locator": "[data-midscene-task-1='23']" - }, - { - "content": "Packages", - "rect": { - "left": 492, - "top": 854, - "width": 71, - "height": 17 - }, - "center": [ - 527, - 862 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "?", + "rect": { + "left": 1046, + "top": 761, + "width": 18, + "height": 18 + }, + "center": [1055, 770], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='22']" }, - "locator": "[data-midscene-task-1='24']" - }, - { - "content": "?", - "rect": { - "left": 572, - "top": 851, - "width": 18, - "height": 18 - }, - "center": [ - 581, - 860 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Normal", + "rect": { + "left": 981, + "top": 784, + "width": 438, + "height": 20 + }, + "center": [1200, 794], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='23']" }, - "locator": "[data-midscene-task-1='25']" - }, - { - "content": "Normal", - "rect": { - "left": 492, - "top": 873, - "width": 443, - "height": 20 - }, - "center": [ - 713, - 883 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Packages", + "rect": { + "left": 492, + "top": 854, + "width": 71, + "height": 17 + }, + "center": [527, 862], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='24']" }, - "locator": "[data-midscene-task-1='26']" - }, - { - "content": "Pages", - "rect": { - "left": 981, - "top": 854, - "width": 45, - "height": 17 - }, - "center": [ - 1003, - 862 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "?", + "rect": { + "left": 572, + "top": 851, + "width": 18, + "height": 18 + }, + "center": [581, 860], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='25']" }, - "locator": "[data-midscene-task-1='27']" - }, - { - "content": "?", - "rect": { - "left": 1036, - "top": 851, - "width": 18, - "height": 18 - }, - "center": [ - 1045, - 860 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Normal", + "rect": { + "left": 492, + "top": 873, + "width": 443, + "height": 20 + }, + "center": [713, 883], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='26']" }, - "locator": "[data-midscene-task-1='28']" - }, - { - "content": "Normal", - "rect": { - "left": 981, - "top": 873, - "width": 443, - "height": 20 - }, - "center": [ - 1202, - 883 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Pages", + "rect": { + "left": 981, + "top": 854, + "width": 45, + "height": 17 + }, + "center": [1003, 862], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='27']" }, - "locator": "[data-midscene-task-1='29']" - }, - { - "content": "Codespaces", - "rect": { - "left": 492, - "top": 944, - "width": 92, - "height": 16 - }, - "center": [ - 538, - 952 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "?", + "rect": { + "left": 1036, + "top": 851, + "width": 18, + "height": 18 + }, + "center": [1045, 860], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='28']" }, - "locator": "[data-midscene-task-1='30']" - }, - { - "content": "?", - "rect": { - "left": 593, - "top": 940, - "width": 18, - "height": 18 - }, - "center": [ - 602, - 949 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Normal", + "rect": { + "left": 981, + "top": 873, + "width": 443, + "height": 20 + }, + "center": [1202, 883], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='29']" }, - "locator": "[data-midscene-task-1='31']" - }, - { - "content": "Normal", - "rect": { - "left": 492, - "top": 962, - "width": 445, - "height": 21 - }, - "center": [ - 714, - 972 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Codespaces", + "rect": { + "left": 492, + "top": 944, + "width": 92, + "height": 16 + }, + "center": [538, 952], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='30']" }, - "locator": "[data-midscene-task-1='32']" - }, - { - "content": "Copilot", - "rect": { - "left": 981, - "top": 944, - "width": 53, - "height": 18 - }, - "center": [ - 1007, - 953 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "?", + "rect": { + "left": 593, + "top": 940, + "width": 18, + "height": 18 + }, + "center": [602, 949], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='31']" }, - "locator": "[data-midscene-task-1='33']" - }, - { - "content": "Normal", - "rect": { - "left": 981, - "top": 962, - "width": 443, - "height": 21 - }, - "center": [ - 1202, - 972 - ], - "page": { - "_isDragging": false, - "_timeoutSettings": {} + { + "content": "Normal", + "rect": { + "left": 492, + "top": 962, + "width": 445, + "height": 21 + }, + "center": [714, 972], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='32']" }, - "locator": "[data-midscene-task-1='34']" - } - ] - }, - "userQuery": { - "dataDemand": "this is a service status page. Extract all status data with this scheme: {[serviceName]: [statusText]}" - }, - "matchedSection": [], - "matchedElement": [], - "data": { - "Git Operations": "Normal", - "API Requests": "Normal", - "Webhooks": "Normal", - "Issues": "Normal", - "Pull Requests": "Normal", - "Actions": "Normal", - "Packages": "Normal", - "Pages": "Normal", - "Codespaces": "Normal", - "Copilot": "Normal" - }, - "taskInfo": { - "url": "https://www.githubstatus.com/", - "name": "split the Github status page", - "durationMs": 6127, - "rawResponse": "{\"language\":\"en\",\"data\":{\"Git Operations\":\"Normal\",\"API Requests\":\"Normal\",\"Webhooks\":\"Normal\",\"Issues\":\"Normal\",\"Pull Requests\":\"Normal\",\"Actions\":\"Normal\",\"Packages\":\"Normal\",\"Pages\":\"Normal\",\"Codespaces\":\"Normal\",\"Copilot\":\"Normal\"}}", - "systemPrompt": "\nYou are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.\n\nThe user will give you a screenshot and the texts on it. There may be some none-English characters (like Chinese) on it, indicating it's an non-English app.\n\nYou have the following skills:\n\nskill name: extract_data_from_UI\nrelated input: DATA_DEMAND\nskill content: \n* User will give you some data requirements in DATA_DEMAND. Consider the UI context, follow the user's instructions, and provide comprehensive data accordingly.\n* There may be some special commands in DATA_DEMAND, please pay extra attention\n - LOCATE_ONE_ELEMENT and LOCATE_ONE_OR_MORE_ELEMENTS: if you see a description that mentions the keyword LOCATE_ONE_ELEMENT or LOCATE_ONE_OR_MORE_ELEMENTS(e.g. follow LOCATE_ONE_ELEMENT : i want to find ...), it means user wants to locate a specific element meets the description. Return in this way: prefix + the id / comma-separated ids, for example: LOCATE_ONE_ELEMENT/1 , LOCATE_ONE_OR_MORE_ELEMENTS/1,2,3 . If not found, keep the prefix and leave the suffix empty, like LOCATE_ONE_ELEMENT/ .\n\nNow, do the following jobs:\n\nUse your extract_data_from_UI skill to find the following data, placing it in the `data` field\nDATA_DEMAND start:\n;\nthis is a service status page. Extract all status data with this scheme: {[serviceName]: [statusText]}\nDATA_DEMAND ends.\n\nReturn in the following JSON format:\n{\n language: \"en\", // \"en\" or \"zh\", the language of the page. Use the same language to describe section name, description, and similar fields.\n \n data: any, // the extracted data from extract_data_from_UI skill. Make sure both the value and scheme meet the DATA_DEMAND.\n errors?: [], // string[], error message if any\n}\n" + { + "content": "Copilot", + "rect": { + "left": 981, + "top": 944, + "width": 53, + "height": 18 + }, + "center": [1007, 953], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='33']" + }, + { + "content": "Normal", + "rect": { + "left": 981, + "top": 962, + "width": 443, + "height": 21 + }, + "center": [1202, 972], + "page": { + "_isDragging": false, + "_timeoutSettings": {} + }, + "locator": "[data-midscene-task-1='34']" + } + ] + }, + "userQuery": { + "dataDemand": "this is a service status page. Extract all status data with this scheme: {[serviceName]: [statusText]}" + }, + "matchedSection": [], + "matchedElement": [], + "data": { + "Git Operations": "Normal", + "API Requests": "Normal", + "Webhooks": "Normal", + "Issues": "Normal", + "Pull Requests": "Normal", + "Actions": "Normal", + "Packages": "Normal", + "Pages": "Normal", + "Codespaces": "Normal", + "Copilot": "Normal" + }, + "taskInfo": { + "url": "https://www.githubstatus.com/", + "name": "split the Github status page", + "durationMs": 6127, + "rawResponse": "{\"language\":\"en\",\"data\":{\"Git Operations\":\"Normal\",\"API Requests\":\"Normal\",\"Webhooks\":\"Normal\",\"Issues\":\"Normal\",\"Pull Requests\":\"Normal\",\"Actions\":\"Normal\",\"Packages\":\"Normal\",\"Pages\":\"Normal\",\"Codespaces\":\"Normal\",\"Copilot\":\"Normal\"}}", + "systemPrompt": "\nYou are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.\n\nThe user will give you a screenshot and the texts on it. There may be some none-English characters (like Chinese) on it, indicating it's an non-English app.\n\nYou have the following skills:\n\nskill name: extract_data_from_UI\nrelated input: DATA_DEMAND\nskill content: \n* User will give you some data requirements in DATA_DEMAND. Consider the UI context, follow the user's instructions, and provide comprehensive data accordingly.\n* There may be some special commands in DATA_DEMAND, please pay extra attention\n - LOCATE_ONE_ELEMENT and LOCATE_ONE_OR_MORE_ELEMENTS: if you see a description that mentions the keyword LOCATE_ONE_ELEMENT or LOCATE_ONE_OR_MORE_ELEMENTS(e.g. follow LOCATE_ONE_ELEMENT : i want to find ...), it means user wants to locate a specific element meets the description. Return in this way: prefix + the id / comma-separated ids, for example: LOCATE_ONE_ELEMENT/1 , LOCATE_ONE_OR_MORE_ELEMENTS/1,2,3 . If not found, keep the prefix and leave the suffix empty, like LOCATE_ONE_ELEMENT/ .\n\nNow, do the following jobs:\n\nUse your extract_data_from_UI skill to find the following data, placing it in the `data` field\nDATA_DEMAND start:\n;\nthis is a service status page. Extract all status data with this scheme: {[serviceName]: [statusText]}\nDATA_DEMAND ends.\n\nReturn in the following JSON format:\n{\n language: \"en\", // \"en\" or \"zh\", the language of the page. Use the same language to describe section name, description, and similar fields.\n \n data: any, // the extracted data from extract_data_from_UI skill. Make sure both the value and scheme meet the DATA_DEMAND.\n errors?: [], // string[], error message if any\n}\n" + } } -} -] \ No newline at end of file +] diff --git a/packages/midscene/demo_data/index.js b/packages/midscene/demo_data/index.js index 3190d3b4..ce1980d1 100644 --- a/packages/midscene/demo_data/index.js +++ b/packages/midscene/demo_data/index.js @@ -1,6 +1,5 @@ -import * as insightDemo from './demo.insight.json'; import * as actionDemo from './demo.actions.json'; +import * as insightDemo from './demo.insight.json'; export const insight = insightDemo.default; export const action = actionDemo.default; - diff --git a/packages/midscene/modern.config.ts b/packages/midscene/modern.config.ts index e6c037b8..c47fc085 100644 --- a/packages/midscene/modern.config.ts +++ b/packages/midscene/modern.config.ts @@ -1,4 +1,4 @@ -import { moduleTools, defineConfig } from '@modern-js/module-tools'; +import { defineConfig, moduleTools } from '@modern-js/module-tools'; export default defineConfig({ plugins: [moduleTools()], @@ -13,6 +13,6 @@ export default defineConfig({ }, // input: ['src/utils.ts', 'src/index.ts', 'src/image/index.ts'], externals: ['node:buffer'], - target: 'es2017' + target: 'es2017', }, }); diff --git a/packages/midscene/package.json b/packages/midscene/package.json index de7e024f..7cf1927e 100644 --- a/packages/midscene/package.json +++ b/packages/midscene/package.json @@ -6,10 +6,7 @@ "main": "./dist/lib/index.js", "module": "./dist/es/index.js", "types": "./dist/types/index.d.ts", - "files": [ - "dist", - "README.md" - ], + "files": ["dist", "README.md"], "exports": { ".": { "types": "./dist/types/index.d.ts", @@ -44,18 +41,10 @@ }, "typesVersions": { "*": { - ".": [ - "./dist/types/index.d.ts" - ], - "utils": [ - "./dist/types/utils.d.ts" - ], - "ai-model": [ - "./dist/types/ai-model.d.ts" - ], - "image": [ - "./dist/types/image.d.ts" - ] + ".": ["./dist/types/index.d.ts"], + "utils": ["./dist/types/utils.d.ts"], + "ai-model": ["./dist/types/ai-model.d.ts"], + "image": ["./dist/types/image.d.ts"] } }, "scripts": { diff --git a/packages/midscene/src/action/executor.ts b/packages/midscene/src/action/executor.ts index 5dd53322..629664b5 100644 --- a/packages/midscene/src/action/executor.ts +++ b/packages/midscene/src/action/executor.ts @@ -1,8 +1,8 @@ -import assert from 'assert'; -import { +import assert from 'node:assert'; +import type { + ExecutionDump, ExecutionTask, ExecutionTaskApply, - ExecutionDump, ExecutionTaskInsightLocateOutput, ExecutionTaskReturn, ExecutorContext, @@ -23,7 +23,11 @@ export class Executor { dumpFileName?: string; - constructor(name: string, description?: string, tasks?: ExecutionTaskApply[]) { + constructor( + name: string, + description?: string, + tasks?: ExecutionTaskApply[], + ) { this.status = tasks && tasks.length > 0 ? 'pending' : 'init'; this.name = name; this.description = description; @@ -38,7 +42,10 @@ export class Executor { } async append(task: ExecutionTaskApply[] | ExecutionTaskApply): Promise { - assert(this.status !== 'error', 'executor is in error state, cannot append task'); + assert( + this.status !== 'error', + 'executor is in error state, cannot append task', + ); if (Array.isArray(task)) { this.tasks.push(...task.map((item) => this.markTaskAsPending(item))); } else { @@ -51,14 +58,18 @@ export class Executor { async flush(): Promise { if (this.status === 'init' && this.tasks.length > 0) { - console.warn('illegal state for executor, status is init but tasks are not empty'); + console.warn( + 'illegal state for executor, status is init but tasks are not empty', + ); } assert(this.status !== 'running', 'executor is already running'); assert(this.status !== 'completed', 'executor is already completed'); assert(this.status !== 'error', 'executor is in error state'); - const nextPendingIndex = this.tasks.findIndex((task) => task.status === 'pending'); + const nextPendingIndex = this.tasks.findIndex( + (task) => task.status === 'pending', + ); if (nextPendingIndex < 0) { // all tasks are completed return; @@ -73,7 +84,10 @@ export class Executor { while (taskIndex < this.tasks.length) { const task = this.tasks[taskIndex]; - assert(task.status === 'pending', `task status should be pending, but got: ${task.status}`); + assert( + task.status === 'pending', + `task status should be pending, but got: ${task.status}`, + ); task.timing = { start: Date.now(), }; @@ -87,6 +101,7 @@ export class Executor { const { executor, param } = task; assert(executor, `executor is required for task type: ${task.type}`); + // biome-ignore lint/suspicious/noImplicitAnyLet: let returnValue; const executorContext: ExecutorContext = { task, @@ -99,13 +114,16 @@ export class Executor { ); returnValue = await task.executor(param, executorContext); if (task.subType === 'Locate') { - previousFindOutput = (returnValue as ExecutionTaskReturn) - ?.output; + previousFindOutput = ( + returnValue as ExecutionTaskReturn + )?.output; } } else if (task.type === 'Action' || task.type === 'Planning') { returnValue = await task.executor(param, executorContext); } else { - console.warn(`unsupported task type: ${task.type}, will try to execute it directly`); + console.warn( + `unsupported task type: ${task.type}, will try to execute it directly`, + ); returnValue = await task.executor(param, executorContext); } diff --git a/packages/midscene/src/ai-model/inspect.ts b/packages/midscene/src/ai-model/inspect.ts index 69984ab7..437ca35e 100644 --- a/packages/midscene/src/ai-model/inspect.ts +++ b/packages/midscene/src/ai-model/inspect.ts @@ -1,16 +1,28 @@ -import { ChatCompletionMessageParam } from 'openai/resources'; -import { systemPromptToFindElement } from './prompt/element_inspector'; +import type { + AIElementParseResponse, + AISectionParseResponse, + BaseElement, + UIContext, +} from '@/types'; +import type { ChatCompletionMessageParam } from 'openai/resources'; import { callToGetJSONObject } from './openai'; +import { systemPromptToFindElement } from './prompt/element_inspector'; import { describeUserPage, systemPromptToExtract } from './prompt/util'; -import { AIElementParseResponse, AISectionParseResponse, BaseElement, UIContext } from '@/types'; -export async function AiInspectElement(options: { +export async function AiInspectElement< + ElementType extends BaseElement = BaseElement, +>(options: { context: UIContext; multi: boolean; findElementDescription: string; callAI?: typeof callToGetJSONObject; }) { - const { context, multi, findElementDescription, callAI = callToGetJSONObject } = options; + const { + context, + multi, + findElementDescription, + callAI = callToGetJSONObject, + } = options; const { screenshotBase64 } = context; const { description, elementById } = await describeUserPage(context); @@ -43,7 +55,10 @@ export async function AiInspectElement(options: { +export async function AiExtractElementInfo< + T, + ElementType extends BaseElement = BaseElement, +>(options: { dataQuery: string | Record; sectionConstraints: { name: string; @@ -52,7 +67,12 @@ export async function AiExtractElementInfo; callAI?: typeof callToGetJSONObject; }) { - const { dataQuery, sectionConstraints, context, callAI = callToGetJSONObject } = options; + const { + dataQuery, + sectionConstraints, + context, + callAI = callToGetJSONObject, + } = options; const systemPrompt = systemPromptToExtract(dataQuery, sectionConstraints); const { screenshotBase64 } = context; diff --git a/packages/midscene/src/ai-model/openai.ts b/packages/midscene/src/ai-model/openai.ts index 3dbc158e..e4e6f7d1 100644 --- a/packages/midscene/src/ai-model/openai.ts +++ b/packages/midscene/src/ai-model/openai.ts @@ -1,15 +1,18 @@ -import assert from 'assert'; -import OpenAI, { ClientOptions } from 'openai'; -import { ChatCompletionMessageParam } from 'openai/resources'; -import { wrapOpenAI } from 'langsmith/wrappers'; +import assert from 'node:assert'; import { AIResponseFormat } from '@/types'; +import { wrapOpenAI } from 'langsmith/wrappers'; +import OpenAI, { type ClientOptions } from 'openai'; +import type { ChatCompletionMessageParam } from 'openai/resources'; const envConfigKey = 'MIDSCENE_OPENAI_INIT_CONFIG_JSON'; const envModelKey = 'MIDSCENE_MODEL_NAME'; const envSmithDebug = 'MIDSCENE_LANGSMITH_DEBUG'; let extraConfig: ClientOptions = {}; -if (typeof process.env[envConfigKey] === 'string' && process.env[envConfigKey]) { +if ( + typeof process.env[envConfigKey] === 'string' && + process.env[envConfigKey] +) { console.log('config for openai loaded'); extraConfig = JSON.parse(process.env[envConfigKey]); } @@ -48,7 +51,9 @@ export async function call( return content; } -export async function callToGetJSONObject(messages: ChatCompletionMessageParam[]): Promise { +export async function callToGetJSONObject( + messages: ChatCompletionMessageParam[], +): Promise { const response = await call(messages, AIResponseFormat.JSON); assert(response, 'empty response'); return JSON.parse(response); diff --git a/packages/midscene/src/ai-model/prompt/element_inspector.ts b/packages/midscene/src/ai-model/prompt/element_inspector.ts index e1de990b..9d728f03 100644 --- a/packages/midscene/src/ai-model/prompt/element_inspector.ts +++ b/packages/midscene/src/ai-model/prompt/element_inspector.ts @@ -1,4 +1,7 @@ -export function systemPromptToFindElement(description: string, multi?: boolean) { +export function systemPromptToFindElement( + description: string, + multi?: boolean, +) { return ` ## Role: You are an expert in software page image (2D) and page element text analysis. diff --git a/packages/midscene/src/ai-model/prompt/util.ts b/packages/midscene/src/ai-model/prompt/util.ts index bfaba849..d24d8238 100644 --- a/packages/midscene/src/ai-model/prompt/util.ts +++ b/packages/midscene/src/ai-model/prompt/util.ts @@ -1,6 +1,12 @@ -import assert from 'assert'; -import { Size, UISection, UIContext, BasicSectionQuery, BaseElement } from '@/types'; +import assert from 'node:assert'; import { imageInfoOfBase64 } from '@/image'; +import type { + BaseElement, + BasicSectionQuery, + Size, + UIContext, + UISection, +} from '@/types'; const characteristic = 'You are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.'; @@ -63,11 +69,14 @@ skill content: * There may be some special commands in DATA_DEMAND, please pay extra attention - ${ONE_ELEMENT_LOCATOR_PREFIX} and ${ELEMENTS_LOCATOR_PREFIX}: if you see a description that mentions the keyword ${ONE_ELEMENT_LOCATOR_PREFIX} or ${ELEMENTS_LOCATOR_PREFIX}(e.g. follow ${ONE_ELEMENT_LOCATOR_PREFIX} : i want to find ...), it means user wants to locate a specific element meets the description. Return in this way: prefix + the id / comma-separated ids, for example: ${ONE_ELEMENT_LOCATOR_PREFIX}/1 , ${ELEMENTS_LOCATOR_PREFIX}/1,2,3 . If not found, keep the prefix and leave the suffix empty, like ${ONE_ELEMENT_LOCATOR_PREFIX}/ .`; -export function promptsOfSectionQuery(constraints: BasicSectionQuery[]): string { +export function promptsOfSectionQuery( + constraints: BasicSectionQuery[], +): string { if (!constraints.length) { return ''; } - const instruction = 'Use your segment_a_web_page skill to find the following section(s)'; + const instruction = + 'Use your segment_a_web_page skill to find the following section(s)'; const singleSection = (c: BasicSectionQuery) => { assert( c.name || c.description, @@ -76,7 +85,9 @@ export function promptsOfSectionQuery(constraints: BasicSectionQuery[]): string const number = 'One section'; const name = c.name ? `named \`${c.name}\`` : ''; - const description = c.description ? `, usage or criteria : ${c.description}` : ''; + const description = c.description + ? `, usage or criteria : ${c.description}` + : ''; const basic = `* ${number} ${name}${description}`; return basic; @@ -88,7 +99,8 @@ export function systemPromptToExtract( dataQuery: Record | string, sections?: BasicSectionQuery[], ) { - const allSectionNames: string[] = sections?.filter((c) => c.name).map((c) => c.name || '') || []; + const allSectionNames: string[] = + sections?.filter((c) => c.name).map((c) => c.name || '') || []; const sectionFindingPrompt = promptsOfSectionQuery(sections || []); const sectionReturnFormat = allSectionNames.length ? ' sections: [], // detailed information of each section from segment_a_web_page skill' @@ -149,7 +161,9 @@ type PromptElementType = { content: BaseElement['content']; }; -export function describeElement(elements: (Pick & { id: string })[]) { +export function describeElement( + elements: (Pick & { id: string })[], +) { const sliceLength = 80; return elements .map((item) => @@ -159,7 +173,9 @@ export function describeElement(elements: (Pick item.rect.top, item.rect.left + item.rect.width, item.rect.top + item.rect.height, - item.content.length > sliceLength ? `${item.content.slice(0, sliceLength)}...` : item.content, + item.content.length > sliceLength + ? `${item.content.slice(0, sliceLength)}...` + : item.content, ].join(', '), ) .join('\n'); @@ -172,7 +188,10 @@ The sections are formatted in the following way: `; } -export function describeSections(sections: UISection[], colorOfSectionName: (name: string) => string) { +export function describeSections( + sections: UISection[], + colorOfSectionName: (name: string) => string, +) { return sections .map((item) => [ @@ -195,9 +214,9 @@ export function truncateText(text: string) { return text; } -export async function describeUserPage( - context: Omit, 'describer'>, -) { +export async function describeUserPage< + ElementType extends BaseElement = BaseElement, +>(context: Omit, 'describer'>) { const { screenshotBase64 } = context; let width: number; let height: number; @@ -236,29 +255,37 @@ export async function describeUserPage = elementsInfo.map((item) => { - const { id, attributes = {}, rect, content } = item; - const tailorContent = truncateText(content); - const tailorAttributes = Object.keys(attributes).reduce((res, currentKey: string) => { - const attributeVal = (attributes as any)[currentKey]; - res[currentKey] = truncateText(attributeVal); - return res; - }, {} as BaseElement['attributes']); - - return { - id, - attributes: tailorAttributes, - rect, - content: tailorContent, - }; - }); + const elementInfosDescription: Array = elementsInfo.map( + (item) => { + const { id, attributes = {}, rect, content } = item; + const tailorContent = truncateText(content); + const tailorAttributes = Object.keys(attributes).reduce( + (res, currentKey: string) => { + const attributeVal = (attributes as any)[currentKey]; + res[currentKey] = truncateText(attributeVal); + return res; + }, + {} as BaseElement['attributes'], + ); + + return { + id, + attributes: tailorAttributes, + rect, + content: tailorContent, + }; + }, + ); return JSON.stringify(elementInfosDescription); } /** * elements */ -export function retrieveElement(prompt: string, opt?: { multi: boolean }): string { +export function retrieveElement( + prompt: string, + opt?: { multi: boolean }, +): string { if (opt?.multi) { return `follow ${ELEMENTS_LOCATOR_PREFIX}: ${prompt}`; } @@ -269,10 +296,15 @@ export function ifElementTypeResponse(response: string): boolean { if (typeof response !== 'string') { return false; } - return response.startsWith(ONE_ELEMENT_LOCATOR_PREFIX) || response.startsWith(ELEMENTS_LOCATOR_PREFIX); + return ( + response.startsWith(ONE_ELEMENT_LOCATOR_PREFIX) || + response.startsWith(ELEMENTS_LOCATOR_PREFIX) + ); } -export function splitElementResponse(response: string): string | null | string[] { +export function splitElementResponse( + response: string, +): string | null | string[] { const oneElementSplitter = `${ONE_ELEMENT_LOCATOR_PREFIX}/`; if (response.startsWith(oneElementSplitter)) { const id = response.slice(oneElementSplitter.length); diff --git a/packages/midscene/src/automation/planning.ts b/packages/midscene/src/automation/planning.ts index c8ab48ab..3f164ee3 100644 --- a/packages/midscene/src/automation/planning.ts +++ b/packages/midscene/src/automation/planning.ts @@ -1,7 +1,7 @@ -import { ChatCompletionMessageParam } from 'openai/resources'; -import { PlanningAction, PlanningAIResponse, UIContext } from '@/types'; -import { callToGetJSONObject } from '@/ai-model/openai'; import { describeUserPage } from '@/ai-model'; +import { callToGetJSONObject } from '@/ai-model/openai'; +import type { PlanningAIResponse, PlanningAction, UIContext } from '@/types'; +import type { ChatCompletionMessageParam } from 'openai/resources'; const characteristic = 'You are a versatile professional in software UI design and testing. Your outstanding contributions will impact the user experience of billions of users.'; @@ -67,7 +67,8 @@ export async function plan( callAI?: typeof callToGetJSONObject; }, ): Promise<{ plans: PlanningAction[] }> { - const { callAI = callToGetJSONObject, context } = opts || {}; + const { callAI = callToGetJSONObject, context } = + opts || {}; const { screenshotBase64 } = context; const { description } = await describeUserPage(context); const systemPrompt = systemPromptToTaskPlanning(userPrompt); diff --git a/packages/midscene/src/image/info.ts b/packages/midscene/src/image/info.ts index 106e2063..9ef20501 100644 --- a/packages/midscene/src/image/info.ts +++ b/packages/midscene/src/image/info.ts @@ -2,7 +2,7 @@ import assert from 'node:assert'; import { Buffer } from 'node:buffer'; import { readFileSync } from 'node:fs'; import Sharp from 'sharp'; -import { Size } from '..'; +import type { Size } from '..'; /** * Retrieves the dimensions of an image asynchronously * @@ -47,7 +47,8 @@ export function base64Encoded(image: string, withHeader = true) { } if (image.endsWith('png')) { return `data:image/png;base64,${imageBuffer.toString('base64')}`; - } else if (image.endsWith('jpg') || image.endsWith('jpeg')) { + } + if (image.endsWith('jpg') || image.endsWith('jpeg')) { return `data:image/jpeg;base64,${imageBuffer.toString('base64')}`; } throw new Error('unsupported image type'); diff --git a/packages/midscene/src/image/transform.ts b/packages/midscene/src/image/transform.ts index 81096145..42e21b38 100644 --- a/packages/midscene/src/image/transform.ts +++ b/packages/midscene/src/image/transform.ts @@ -1,6 +1,6 @@ import { Buffer } from 'node:buffer'; +import type { Rect } from '@/types'; import Sharp from 'sharp'; -import { Rect } from '@/types'; /** * Saves a Base64-encoded image to a file @@ -10,7 +10,10 @@ import { Rect } from '@/types'; * @param options.outputPath - The path where the image will be saved * @throws Error if there is an error during the saving process */ -export async function saveBase64Image(options: { base64Data: string; outputPath: string }): Promise { +export async function saveBase64Image(options: { + base64Data: string; + outputPath: string; +}): Promise { const { base64Data, outputPath } = options; // Remove the base64 data prefix (if any) const base64Image = base64Data.split(';base64,').pop() || base64Data; @@ -84,7 +87,10 @@ export async function resizeImg(base64Data: string) { * @returns {Object} An object containing the new width and height. * @throws {Error} Throws an error if the width or height is not a positive number. */ -export function calculateNewDimensions(originalWidth: number, originalHeight: number) { +export function calculateNewDimensions( + originalWidth: number, + originalHeight: number, +) { // In low mode, the image is scaled to 512x512 pixels and 85 tokens are used to represent the image. // In high mode, the model looks at low-resolution images and then creates detailed crop images, using 170 tokens for each 512x512 pixel tile. In practical applications, it is recommended to control the image size within 2048x768 pixels const maxWidth = 768; // Maximum width @@ -128,7 +134,12 @@ export async function trimImage(image: string | Buffer): Promise<{ const imgInstance = Sharp(image); const instanceInfo = await imgInstance.metadata(); - if (!instanceInfo.width || instanceInfo.width <= 3 || !instanceInfo.height || instanceInfo.height <= 3) { + if ( + !instanceInfo.width || + instanceInfo.width <= 3 || + !instanceInfo.height || + instanceInfo.height <= 3 + ) { return null; } @@ -136,7 +147,10 @@ export async function trimImage(image: string | Buffer): Promise<{ resolveWithObject: true, }); - if (typeof info.trimOffsetLeft === 'undefined' || typeof info.trimOffsetTop === 'undefined') { + if ( + typeof info.trimOffsetLeft === 'undefined' || + typeof info.trimOffsetTop === 'undefined' + ) { return null; } @@ -164,9 +178,17 @@ export async function trimImage(image: string | Buffer): Promise<{ * @returns A Promise that resolves to a rectangle object representing the aligned coordinates * @throws Error if there is an error during image processing */ -export async function alignCoordByTrim(image: string | Buffer, centerRect: Rect): Promise { +export async function alignCoordByTrim( + image: string | Buffer, + centerRect: Rect, +): Promise { const imgInfo = await Sharp(image).metadata(); - if (!imgInfo?.width || !imgInfo.height || imgInfo.width <= 3 || imgInfo.height <= 3) { + if ( + !imgInfo?.width || + !imgInfo.height || + imgInfo.width <= 3 || + imgInfo.height <= 3 + ) { return centerRect; } try { diff --git a/packages/midscene/src/image/visualization.ts b/packages/midscene/src/image/visualization.ts index b3b80faa..e3171b4b 100644 --- a/packages/midscene/src/image/visualization.ts +++ b/packages/midscene/src/image/visualization.ts @@ -1,8 +1,8 @@ -import { Buffer } from 'buffer'; +import { Buffer } from 'node:buffer'; +import { getTmpFile } from '@/utils'; import Sharp from 'sharp'; -import { Color, UIContext, UISection } from '..'; +import type { Color, UIContext, UISection } from '..'; import { imageInfo } from './info'; -import { getTmpFile } from '@/utils'; const colors: Color[] = [ { @@ -97,11 +97,11 @@ export async function composeSectionDiagram( sectionNameColorMap[section.name] = color; return ` + height * ratio + }" fill="${color.hex}" /> + top * ratio + textFontSize + }" font-family="Arial" font-size="${textFontSize}" fill="black"> ${section.name} `; diff --git a/packages/midscene/src/index.ts b/packages/midscene/src/index.ts index 06d482af..a1d0da45 100644 --- a/packages/midscene/src/index.ts +++ b/packages/midscene/src/index.ts @@ -1,5 +1,5 @@ -import Insight from './insight'; import { Executor } from './action/executor'; +import Insight from './insight'; import { getElement, getSection } from './query'; import { setDumpDir } from './utils'; diff --git a/packages/midscene/src/insight/index.ts b/packages/midscene/src/insight/index.ts index 33e404d8..d46c6846 100644 --- a/packages/midscene/src/insight/index.ts +++ b/packages/midscene/src/insight/index.ts @@ -134,7 +134,6 @@ export default class Insight< } const elements: BaseElement[] = []; - // biome-ignore lint/complexity/noForEach: parseResult.elements.forEach((item) => { const element = elementById(item.id); diff --git a/packages/midscene/src/insight/utils.ts b/packages/midscene/src/insight/utils.ts index 95b959c8..2643b781 100644 --- a/packages/midscene/src/insight/utils.ts +++ b/packages/midscene/src/insight/utils.ts @@ -1,21 +1,26 @@ +import assert from 'node:assert'; +import { randomUUID } from 'node:crypto'; /* eslint-disable @typescript-eslint/ban-types */ -import { existsSync } from 'fs'; -import { join } from 'path'; -import { randomUUID } from 'crypto'; -import assert from 'assert'; -import { - UISection, - UIContext, - Rect, +import { existsSync } from 'node:fs'; +import { join } from 'node:path'; +import type { + BaseElement, + DumpMeta, + DumpSubscriber, + ElementById, InsightDump, LiteUISection, PartialInsightDumpFromSDK, - BaseElement, - ElementById, - DumpSubscriber, - DumpMeta, + Rect, + UIContext, + UISection, } from '@/types'; -import { getDumpDir, getPkgInfo, insightDumpFileExt, writeDumpFile } from '@/utils'; +import { + getDumpDir, + getPkgInfo, + insightDumpFileExt, + writeDumpFile, +} from '@/utils'; let logFileName = ''; const logContent: string[] = []; @@ -67,7 +72,10 @@ export function writeInsightDump( return id; } -export function idsIntoElements(ids: string[], elementById: ElementById): BaseElement[] { +export function idsIntoElements( + ids: string[], + elementById: ElementById, +): BaseElement[] { return ids.reduce((acc, id) => { const element = elementById(id); if (element) { @@ -79,6 +87,7 @@ export function idsIntoElements(ids: string[], elementById: ElementById): BaseEl }, []); } +// biome-ignore lint/complexity/noBannedTypes: export function shallowExpandIds( data: DataScheme, ifMeet: (id: string) => boolean, @@ -101,7 +110,10 @@ export function shallowExpandIds( return data; } -export function expandLiteSection(liteSection: LiteUISection, elementById: ElementById): UISection { +export function expandLiteSection( + liteSection: LiteUISection, + elementById: ElementById, +): UISection { const { textIds, ...remainingFields } = liteSection; const texts: BaseElement[] = idsIntoElements(textIds, elementById); @@ -111,7 +123,8 @@ export function expandLiteSection(liteSection: LiteUISection, elementById: Eleme let rightMost = -1; let bottomMost = -1; texts.forEach((text) => { - leftMost = leftMost === -1 ? text.rect.left : Math.min(leftMost, text.rect.left); + leftMost = + leftMost === -1 ? text.rect.left : Math.min(leftMost, text.rect.left); topMost = topMost === -1 ? text.rect.top : Math.min(topMost, text.rect.top); rightMost = Math.max(rightMost, text.rect.left + text.rect.width); bottomMost = Math.max(bottomMost, text.rect.top + text.rect.height); diff --git a/packages/midscene/src/query/fixture/script_get_all_texts.tmp.js b/packages/midscene/src/query/fixture/script_get_all_texts.tmp.js index d51cb605..12210f08 100644 --- a/packages/midscene/src/query/fixture/script_get_all_texts.tmp.js +++ b/packages/midscene/src/query/fixture/script_get_all_texts.tmp.js @@ -1,5 +1,5 @@ /* eslint-disable */ -(function () { +(() => { const TEXT_SIZE_THRESHOLD = 9; const taskIdKey = '_midscene_retrieve_task_id'; const nodeDataIdKey = 'data-midscene-task-'; @@ -45,7 +45,11 @@ // Check if the computed display property is "none" const style = window.getComputedStyle(el); - if (style.display === 'none' || style.visibility === 'hidden' || style.opacity === '0') { + if ( + style.display === 'none' || + style.visibility === 'hidden' || + style.opacity === '0' + ) { console.log('Element is hidden'); return false; } @@ -58,18 +62,29 @@ } // Check if the element is hidden via clipping or scrolling. - const scrollLeft = window.pageXOffset || document.documentElement.scrollLeft; + const scrollLeft = + window.pageXOffset || document.documentElement.scrollLeft; const scrollTop = window.pageYOffset || document.documentElement.scrollTop; const isInViewport = rect.top >= 0 + scrollTop && rect.left >= 0 + scrollLeft && - rect.bottom <= (window.innerHeight || document.documentElement.clientHeight) + scrollTop && - rect.right <= (window.innerWidth || document.documentElement.clientWidth) + scrollLeft; + rect.bottom <= + (window.innerHeight || document.documentElement.clientHeight) + + scrollTop && + rect.right <= + (window.innerWidth || document.documentElement.clientWidth) + + scrollLeft; if (!isInViewport) { console.log('Element is not in the viewport'); - console.log(rect, window.innerHeight, window.innerWidth, scrollTop, scrollLeft); + console.log( + rect, + window.innerHeight, + window.innerWidth, + scrollTop, + scrollLeft, + ); return false; } @@ -86,7 +101,12 @@ rect.bottom > parentRect.bottom + tolerance || rect.right > parentRect.right + tolerance ) { - console.log('Element is clipped by an ancestor', parent, rect, parentRect); + console.log( + 'Element is clipped by an ancestor', + parent, + rect, + parentRect, + ); return false; } } @@ -157,7 +177,10 @@ } const { rect } = answerRect; - if (rect.width < TEXT_SIZE_THRESHOLD || rect.height < TEXT_SIZE_THRESHOLD) { + if ( + rect.width < TEXT_SIZE_THRESHOLD || + rect.height < TEXT_SIZE_THRESHOLD + ) { console.log('Element is too small', text); return; } @@ -170,7 +193,10 @@ locator: selector, content: text, rect, - center: [Math.round(rect.left + rect.width / 2), Math.round(rect.top + rect.height / 2)], + center: [ + Math.round(rect.left + rect.width / 2), + Math.round(rect.top + rect.height / 2), + ], }); // should stop searching if the text is found @@ -192,6 +218,8 @@ window.extractTextWithPositionDFS = extractTextWithPositionDFS; window.ifNodeIsValid = validTextNodeContent; const container = - typeof window.get_all_text_container === 'undefined' ? document.body : window.get_all_text_container; + typeof window.get_all_text_container === 'undefined' + ? document.body + : window.get_all_text_container; return extractTextWithPositionDFS(container); })(); diff --git a/packages/midscene/src/query/index.ts b/packages/midscene/src/query/index.ts index eed0adbe..13b09bea 100644 --- a/packages/midscene/src/query/index.ts +++ b/packages/midscene/src/query/index.ts @@ -1,6 +1,9 @@ +import { + retrieveSection as promptOneSection, + retrieveElement, +} from '@/ai-model/prompt/util'; /** Everything you need to parse a query */ import getAllContentScript from './fixture/script_get_all_texts'; -import { retrieveElement, retrieveSection as promptOneSection } from '@/ai-model/prompt/util'; export function pageScriptToGetTexts(selector?: string) { let prependScript = ''; @@ -9,7 +12,9 @@ export function pageScriptToGetTexts(selector?: string) { const id = selector.slice(1); prependScript = `window.get_all_text_container = document.getElementById('${id}');`; } else { - throw new Error(`selector not supported yet: ${selector}. Only id selector (#id-name) is supported.`); + throw new Error( + `selector not supported yet: ${selector}. Only id selector (#id-name) is supported.`, + ); } } diff --git a/packages/midscene/src/types.ts b/packages/midscene/src/types.ts index 43259257..2f3e6dbf 100644 --- a/packages/midscene/src/types.ts +++ b/packages/midscene/src/types.ts @@ -15,17 +15,17 @@ export interface Size { export type Rect = Point & Size; enum NodeType { - 'INPUT' = 'INPUT Node', - 'BUTTON' = 'BUTTON Node', - 'IMG' = 'IMG Node', - 'TEXT' = 'TEXT Node', + INPUT = 'INPUT Node', + BUTTON = 'BUTTON Node', + IMG = 'IMG Node', + TEXT = 'TEXT Node', } export abstract class BaseElement { abstract id: string; abstract attributes: { - ['nodeType']: NodeType; + nodeType: NodeType; [key: string]: string; }; @@ -82,7 +82,9 @@ export abstract class UIContext { * insight */ -export type CallAIFn = (messages: ChatCompletionMessageParam[]) => Promise; +export type CallAIFn = ( + messages: ChatCompletionMessageParam[], +) => Promise; export interface InsightOptions { taskInfo?: Omit; @@ -133,7 +135,10 @@ export interface InsightDump extends DumpMeta { error?: string; } -export type PartialInsightDumpFromSDK = Omit; +export type PartialInsightDumpFromSDK = Omit< + InsightDump, + 'sdkVersion' | 'logTime' | 'logId' +>; export type DumpSubscriber = (dump: InsightDump) => Promise | void; @@ -154,7 +159,14 @@ export type ElementById = (id: string) => BaseElement | null; export interface PlanningAction { thought: string; - type: 'Locate' | 'Tap' | 'Hover' | 'Input' | 'KeyboardPress' | 'Scroll' | 'Error'; + type: + | 'Locate' + | 'Tap' + | 'Hover' + | 'Input' + | 'KeyboardPress' + | 'Scroll' + | 'Error'; param: ParamType; } @@ -170,7 +182,11 @@ export interface PlanningActionParamInputOrKeyPress { value: string; } export interface PlanningActionParamScroll { - scrollType: 'ScrollUntilBottom' | 'ScrollUntilTop' | 'ScrollDown' | 'ScrollUp'; + scrollType: + | 'ScrollUntilBottom' + | 'ScrollUntilTop' + | 'ScrollDown' + | 'ScrollUp'; } /** @@ -225,7 +241,10 @@ export interface ExecutionTaskApply< executor: ( param: TaskParam, context: ExecutorContext, - ) => Promise | void> | void; + ) => // biome-ignore lint/suspicious/noConfusingVoidType: + | Promise | undefined | void> + | undefined + | void; } export interface ExecutionTaskReturn { @@ -235,20 +254,29 @@ export interface ExecutionTaskReturn { cache?: TaskCacheInfo; } -export type ExecutionTask = ExecutionTaskApply> = - E & - ExecutionTaskReturn< - E extends ExecutionTaskApply ? TaskOutput : unknown, - E extends ExecutionTaskApply ? TaskLog : unknown - > & { - status: 'pending' | 'running' | 'success' | 'fail' | 'cancelled'; - error?: string; - timing?: { - start: number; - end?: number; - cost?: number; - }; +export type ExecutionTask< + E extends ExecutionTaskApply = ExecutionTaskApply< + any, + any, + any + >, +> = E & + ExecutionTaskReturn< + E extends ExecutionTaskApply + ? TaskOutput + : unknown, + E extends ExecutionTaskApply + ? TaskLog + : unknown + > & { + status: 'pending' | 'running' | 'success' | 'fail' | 'cancelled'; + error?: string; + timing?: { + start: number; + end?: number; + cost?: number; }; + }; export interface ExecutionDump extends DumpMeta { name: string; @@ -278,7 +306,8 @@ export type ExecutionTaskInsightLocateApply = ExecutionTaskApply< ExecutionTaskInsightLocateLog >; -export type ExecutionTaskInsightLocate = ExecutionTask; +export type ExecutionTaskInsightLocate = + ExecutionTask; /* task - insight-extract @@ -291,9 +320,13 @@ export interface ExecutionTaskInsightQueryOutput { data: any; } -export type ExecutionTaskInsightQueryApply = ExecutionTaskApply<'Insight', ExecutionTaskInsightQueryParam>; +export type ExecutionTaskInsightQueryApply = ExecutionTaskApply< + 'Insight', + ExecutionTaskInsightQueryParam +>; -export type ExecutionTaskInsightQuery = ExecutionTask; +export type ExecutionTaskInsightQuery = + ExecutionTask; // export type ExecutionTaskInsight = ExecutionTaskInsightLocate; // | ExecutionTaskInsightExtract; diff --git a/packages/midscene/src/utils.ts b/packages/midscene/src/utils.ts index c001bd26..aa37a7e8 100644 --- a/packages/midscene/src/utils.ts +++ b/packages/midscene/src/utils.ts @@ -1,9 +1,15 @@ -import { tmpdir } from 'os'; -import { basename, join } from 'path'; -import { copyFileSync, existsSync, mkdirSync, readFileSync, writeFileSync } from 'fs'; -import { randomUUID } from 'crypto'; -import assert from 'assert'; -import { Rect } from './types'; +import assert from 'node:assert'; +import { randomUUID } from 'node:crypto'; +import { + copyFileSync, + existsSync, + mkdirSync, + readFileSync, + writeFileSync, +} from 'node:fs'; +import { tmpdir } from 'node:os'; +import { basename, join } from 'node:path'; +import type { Rect } from './types'; interface PkgInfo { name: string; @@ -27,12 +33,11 @@ export function getPkgInfo(): PkgInfo { const { name, version } = JSON.parse(readFileSync(pkgJsonFile, 'utf-8')); pkg = { name, version }; return pkg; - } else { - return { - name: 'midscene-unknown-page-name', - version: '0.0.0', - }; } + return { + name: 'midscene-unknown-page-name', + version: '0.0.0', + }; } let logDir = join(process.cwd(), './midscene_run/'); diff --git a/packages/midscene/tests/ai-model/inspector/online_order_inspector.test.ts b/packages/midscene/tests/ai-model/inspector/online_order_inspector.test.ts index 09183d76..7eff213a 100644 --- a/packages/midscene/tests/ai-model/inspector/online_order_inspector.test.ts +++ b/packages/midscene/tests/ai-model/inspector/online_order_inspector.test.ts @@ -1,7 +1,12 @@ -import path from 'path'; -import { test, expect } from 'vitest'; -import { getPageTestData, repeat, runTestCases, writeFileSyncWithDir } from './util'; +import path from 'node:path'; import { AiInspectElement } from '@/ai-model'; +import { expect, test } from 'vitest'; +import { + getPageTestData, + repeat, + runTestCases, + writeFileSyncWithDir, +} from './util'; const testCases = [ { @@ -34,22 +39,32 @@ repeat(5, (repeatIndex) => { test( 'xicha: inspect element', async () => { - const { context } = await getPageTestData(path.join(__dirname, './test-data/online_order')); + const { context } = await getPageTestData( + path.join(__dirname, './test-data/online_order'), + ); - const { aiResponse, filterUnStableinf } = await runTestCases(testCases, async (testCase) => { - const { parseResult } = await AiInspectElement({ - context, - multi: testCase.multi, - findElementDescription: testCase.description, - }); - return parseResult; - }); + const { aiResponse, filterUnStableinf } = await runTestCases( + testCases, + async (testCase) => { + const { parseResult } = await AiInspectElement({ + context, + multi: testCase.multi, + findElementDescription: testCase.description, + }); + return parseResult; + }, + ); writeFileSyncWithDir( - path.join(__dirname, `__ai_responses__/online_order-inspector-element-${repeatIndex}.json`), + path.join( + __dirname, + `__ai_responses__/online_order-inspector-element-${repeatIndex}.json`, + ), JSON.stringify(aiResponse, null, 2), { encoding: 'utf-8' }, ); - expect(filterUnStableinf).toMatchFileSnapshot('./__snapshots__/online_order_inspector.test.ts.snap'); + expect(filterUnStableinf).toMatchFileSnapshot( + './__snapshots__/online_order_inspector.test.ts.snap', + ); }, { timeout: 99999, diff --git a/packages/midscene/tests/ai-model/inspector/test-data/githubstatus/element-snapshot.json b/packages/midscene/tests/ai-model/inspector/test-data/githubstatus/element-snapshot.json index 8620329f..4fccfe9e 100644 --- a/packages/midscene/tests/ai-model/inspector/test-data/githubstatus/element-snapshot.json +++ b/packages/midscene/tests/ai-model/inspector/test-data/githubstatus/element-snapshot.json @@ -8,10 +8,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='af212c0aa4']", - "center": [ - 960, - 248 - ], + "center": [960, 248], "content": "GitHub Octicon logo\n \n \n Help\n Community\n Status\n \n \n \n GitHub.com\n Twitter\n \n \n \n\n \n\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n x\n \n \n \n \n \n Get email notifications whenever GitHub creates, updates or resolves an incident.\n \n \n \n \n Email address:\n \n \n \n Enter OTP:\n \n Resend OTP in: seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get text message notifications whenever GitHub creates or resolves an incident.\n \n \n \n \n \n \n \n Country code:\n \n Afghanistan (+93)\nAlbania (+355)\nAlgeria (+213)\nAmerican Samoa (+1)\nAndorra (+376)\nAngola (+244)\nAnguilla (+1)\nAntigua and Barbuda (+1)\nArgentina (+54)\nArmenia (+374)\nAruba (+297)\nAustralia/Cocos/Christmas Island (+61)\nAustria (+43)\nAzerbaijan (+994)\nBahamas (+1)\nBahrain (+973)\nBangladesh (+880)\nBarbados (+1)\nBelarus (+375)\nBelgium (+32)\nBelize (+501)\nBenin (+229)\nBermuda (+1)\nBolivia (+591)\nBosnia and Herzegovina (+387)\nBotswana (+267)\nBrazil (+55)\nBrunei (+673)\nBulgaria (+359)\nBurkina Faso (+226)\nBurundi (+257)\nCambodia (+855)\nCameroon (+237)\nCanada (+1)\nCape Verde (+238)\nCayman Islands (+1)\nCentral Africa (+236)\nChad (+235)\nChile (+56)\nChina (+86)\nColombia (+57)\nComoros (+269)\nCongo (+242)\nCongo, Dem Rep (+243)\nCosta Rica (+506)\nCroatia (+385)\nCyprus (+357)\nCzech Republic (+420)\nDenmark (+45)\nDjibouti (+253)\nDominica (+1)\nDominican Republic (+1)\nEgypt (+20)\nEl Salvador (+503)\nEquatorial Guinea (+240)\nEstonia (+372)\nEthiopia (+251)\nFaroe Islands (+298)\nFiji (+679)\nFinland/Aland Islands (+358)\nFrance (+33)\nFrench Guiana (+594)\nFrench Polynesia (+689)\nGabon (+241)\nGambia (+220)\nGeorgia (+995)\nGermany (+49)\nGhana (+233)\nGibraltar (+350)\nGreece (+30)\nGreenland (+299)\nGrenada (+1)\nGuadeloupe (+590)\nGuam (+1)\nGuatemala (+502)\nGuinea (+224)\nGuyana (+592)\nHaiti (+509)\nHonduras (+504)\nHong Kong (+852)\nHungary (+36)\nIceland (+354)\nIndia (+91)\nIndonesia (+62)\nIraq (+964)\nIreland (+353)\nIsrael (+972)\nItaly (+39)\nJamaica (+1)\nJapan (+81)\nJordan (+962)\nKenya (+254)\nKorea, Republic of (+82)\nKosovo (+383)\nKuwait (+965)\nKyrgyzstan (+996)\nLaos (+856)\nLatvia (+371)\nLebanon (+961)\nLesotho (+266)\nLiberia (+231)\nLibya (+218)\nLiechtenstein (+423)\nLithuania (+370)\nLuxembourg (+352)\nMacao (+853)\nMacedonia (+389)\nMadagascar (+261)\nMalawi (+265)\nMalaysia (+60)\nMaldives (+960)\nMali (+223)\nMalta (+356)\nMartinique (+596)\nMauritania (+222)\nMauritius (+230)\nMexico (+52)\nMonaco (+377)\nMongolia (+976)\nMontenegro (+382)\nMontserrat (+1)\nMorocco/Western Sahara (+212)\nMozambique (+258)\nNamibia (+264)\nNepal (+977)\nNetherlands (+31)\nNew Zealand (+64)\nNicaragua (+505)\nNiger (+227)\nNigeria (+234)\nNorway (+47)\nOman (+968)\nPakistan (+92)\nPalestinian Territory (+970)\nPanama (+507)\nParaguay (+595)\nPeru (+51)\nPhilippines (+63)\nPoland (+48)\nPortugal (+351)\nPuerto Rico (+1)\nQatar (+974)\nReunion/Mayotte (+262)\nRomania (+40)\nRussia/Kazakhstan (+7)\nRwanda (+250)\nSamoa (+685)\nSan Marino (+378)\nSaudi Arabia (+966)\nSenegal (+221)\nSerbia (+381)\nSeychelles (+248)\nSierra Leone (+232)\nSingapore (+65)\nSlovakia (+421)\nSlovenia (+386)\nSouth Africa (+27)\nSpain (+34)\nSri Lanka (+94)\nSt Kitts and Nevis (+1)\nSt Lucia (+1)\nSt Vincent Grenadines (+1)\nSudan (+249)\nSuriname (+597)\nSwaziland (+268)\nSweden (+46)\nSwitzerland (+41)\nTaiwan (+886)\nTajikistan (+992)\nTanzania (+255)\nThailand (+66)\nTogo (+228)\nTonga (+676)\nTrinidad and Tobago (+1)\nTunisia (+216)\nTurkey (+90)\nTurks and Caicos Islands (+1)\nUganda (+256)\nUkraine (+380)\nUnited Arab Emirates (+971)\nUnited Kingdom (+44)\nUnited States (+1)\nUruguay (+598)\nUzbekistan (+998)\nVenezuela (+58)\nVietnam (+84)\nVirgin Islands, British (+1)\nVirgin Islands, U.S. (+1)\nYemen (+967)\nZambia (+260)\nZimbabwe (+263)\n \n Phone number:\n \n \n \n \n Change number\n Enter OTP:\n \n Resend OTP in: 30 seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n\n \n \n Message and data rates may apply. By subscribing you agree to our Privacy Policy, the Atlassian Terms of Service, and the Atlassian Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get incident updates and maintenance status messages in Slack.\n \n Subscribe via Slack\n By subscribing you acknowledge our Privacy Policy. In addition, you agree to the Atlassian Cloud Terms of Service and acknowledge Atlassian's Privacy Policy.\n \n\n\n \n \n Get webhook notifications whenever GitHub creates an incident, updates an incident, resolves an incident or changes a component status.\n \n \n \n \n Webhook URL:\n \n The URL we should send the webhooks to\n \n \n\n \n \n Email address:\n \n We'll send you email if your endpoint fails\n \n \n\n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n\n \n\n \n \n or \n view our profile.\n \n .twitter-follow-button {\n margin-bottom: -6px;\n }\n \n\n !function(d,s,id){var js,fjs=d.getElementsByTagName(s)[0],p=/^http:/.test(d.location)?'http':'https';if(!d.getElementById(id)){js=d.createElement(s);js.id=id;js.src=p+'://platform.twitter.com/widgets.js';fjs.parentNode.insertBefore(js,fjs);}}(document, 'script', 'twitter-wjs');\n \n\n \n Visit our support site.\n \n\n \n Get the Atom Feed or RSS Feed.\n \n \n \n \n \n \n\n\n\n\n\n \n \n\n\n $(function () {\n const phoneNumberInput = $('#phone-number');\n const errorDiv = $('#sms-atl-error')\n if(errorDiv.length){\n function checkSelectedCountry() {\n const selectedCountry = $('#phone-country').val();\n const isOtpEnabled = $('#phone-number-country-code').attr('data-otp-enabled') === 'true';\n const form = document.getElementById('subscribe-form-sms');\n form.action = '/subscriptions/new-sms';\n const isOtpFlow = document.getElementById('otp_verify_flow');\n document.getElementById('otp-container').style.display = \"none\";\n if(false && selectedCountry === 'sg') { // Replace 'SG' with the actual value representing Singapore in your select tag\n phoneNumberInput.prop('disabled', true);\n errorDiv.html(`Due to new Singapore government regulations, we're currently not supporting text subscriptions in Singapore. Learn more.
Select another method to subscribe.`);\n } else {\n phoneNumberInput.prop('readonly', false);\n errorDiv.html('');\n if(false){\n if(isOtpEnabled){\n document.getElementById('subscribe-btn-sms').value = \"Send OTP\";\n }\n else {\n isOtpFlow.value = false;\n document.getElementById('subscribe-btn-sms').value = \"Subscribe via Text Message\";\n }\n }\n }\n }\n\n $('#phone-country').on('change', checkSelectedCountry);\n checkSelectedCountry();\n }\n });\n\n document.addEventListener('DOMContentLoaded', function() {\n const dropdown = document.querySelector('#phone-number-country-code .phone-country');\n if (dropdown){\n const wrapperDiv = document.getElementById('phone-number-country-code');\n const selectedOption = dropdown.options[dropdown.selectedIndex];\n const otpEnabled = selectedOption.getAttribute('data-otp-enabled');\n\n wrapperDiv.setAttribute('data-otp-enabled', otpEnabled);\n\n dropdown.addEventListener('change', function() {\n const selectedOption = dropdown.options[dropdown.selectedIndex];\n const otpEnabled = selectedOption.getAttribute('data-otp-enabled');\n\n wrapperDiv.setAttribute('data-otp-enabled', otpEnabled);\n });\n }\n });\n\n var countdownTimer;\n var resendBtn = document.getElementById('resend');\n var timer = document.getElementById('timer');\n var form = document.getElementById('subscribe-form-sms');\n var RESEND_TIMER = 30;\n $(function() {\n $('#subscribe-form-sms').on('ajax:success', function(e, data, status, xhr){\n const form = this;\n const action = form.getAttribute('action');\n if (data.type === 'success' && data.otp_flow === true) {\n document.getElementById('subscriber_code').value = data.subscriber_code\n document.getElementById('otp-container').style.display = \"block\";\n $('#phone-number').prop('readonly', true);\n var display = document.getElementById('countdown');\n disableResend();\n startTimer(RESEND_TIMER, display)\n document.getElementById('subscribe-btn-sms').value = \"Verify OTP and Subscribe\";\n document.getElementById('otp_verify_flow').value = true;\n form.action = '/subscriptions/verify-otp';\n } else if (data.type === 'success' && action.includes('verify')){\n document.getElementById('otp-container').style.display = \"none\";\n $('#phone-number').val('').prop('readonly', false);\n $('#otp').val('');\n document.getElementById('subscribe-btn-sms').value = \"Send OTP\";\n document.getElementById('otp_verify_flow').value = false;\n form.action = '/subscriptions/new-sms';\n SP.currentPage.updatesDropdown.hide();\n }\n });\n $(\"#btn-subcriber-change-number\").on('click', () => {\n document.getElementById('otp-container').style.display = \"none\";\n $('#phone-number').prop('readonly', false);\n document.getElementById('subscribe-btn-sms').value = \"Send OTP\";\n form.action = '/subscriptions/new-sms';\n return false\n })\n $('#resend-otp-btn').on('click', function(e) {\n e.preventDefault();\n let phoneNumber = $('#phone-number').val();\n let countryCode = $('.phone-country').val();\n $.ajax({\n type: 'POST',\n url: \"/subscriptions/new-sms\",\n data: {\n phone_number: phoneNumber,\n phone_country: countryCode,\n type: 'resend'\n },\n }).done(function(data) {\n var messageOptions = (data.type !== undefined && data.type !== null) ? { cssClass: data.type } : {};\n HRB.utils.notify(data.text, messageOptions);\n var display = document.getElementById('countdown');\n disableResend();\n timer.style.display = \"none\"\n if (data.type === 'success') {\n startTimer(RESEND_TIMER, display);\n }\n })\n });\n })\n\n function startTimer(duration, display){\n var timer = duration, seconds;\n clearInterval(countdownTimer);\n countdownTimer = setInterval(function () {\n seconds = parseInt(timer % 60, 10);\n display.textContent = seconds;\n if(--timer < 0){\n enableResend();\n clearInterval(countdownTimer);\n }\n }, 1000);\n disableResend();\n }\n function enableResend(){\n resendBtn.style.display = \"block\";\n timer.style.display = \"none\"\n }\n function disableResend(){\n resendBtn.style.display = \"none\";\n timer.style.display = \"block\"\n }\n\n $(function() {\n $('#subscribe-form-email').on('submit', function() {\n var tokenField = document.getElementById('email-otp-token-field');\n let page_code = \"kctbh9vrtdwd\"\n let key = keyForEmailOtpToken($('#email').val(), page_code);\n tokenField.value = localStorage.getItem(key);\n });\n });\n\n var emailOtpCountdownTimer;\n var emailOtpResendBtn = document.getElementById('resend-email-otp');\n var emailOtpTimer = document.getElementById('email-otp-timer');\n var emailOtpForm = document.getElementById('subscribe-form-email');\n var EMAIL_OTP_RESEND_TIMER = 600;\n $(function() {\n $('#subscribe-form-email').on('ajax:success', function(e, data, status, xhr){\n const form = this;\n const action = form.getAttribute('action');\n if (data.type === 'success' && data.email_otp_verify_flow === true) {\n document.getElementById('email-otp-container').style.display = \"block\";\n var display = document.getElementById('email-otp-countdown');\n display.textContent = EMAIL_OTP_RESEND_TIMER;\n disableEmailOtpResend();\n startEmailOtpTimer(EMAIL_OTP_RESEND_TIMER, display)\n document.getElementById('subscribe-btn-email').value = \"Verify OTP and Subscribe\";\n document.getElementById('email_otp_verify_flow').value = true;\n form.action = '/subscriptions/verify-email-otp';\n } else if (data.type === 'success' && action.includes('verify')){\n let email = $('#email')\n let page_code = \"kctbh9vrtdwd\"\n let key = keyForEmailOtpToken(email.val(), page_code);\n localStorage.setItem(key, data.email_otp_auth_token);\n\n document.getElementById('email-otp-container').style.display = \"none\";\n email.val('').prop('readonly', false);\n $('#email-otp').val('');\n document.getElementById('subscribe-btn-email').value = \"Send OTP\";\n document.getElementById('email_otp_verify_flow').value = false;\n form.action = '/subscriptions/new-email';\n SP.currentPage.updatesDropdown.hide();\n }\n });\n $('#resend-email-otp-btn').on('click', function(e) {\n e.preventDefault();\n let email = $('#email').val();\n $.ajax({\n type: 'POST',\n url: \"/subscriptions/new-email\",\n data: {\n email: email\n },\n }).done(function(data) {\n var messageOptions = (data.type !== undefined && data.type !== null) ? { cssClass: data.type } : {};\n HRB.utils.notify(data.text, messageOptions);\n if (data.type === 'success') {\n var display = document.getElementById('email-otp-countdown');\n display.textContent = EMAIL_OTP_RESEND_TIMER;\n disableEmailOtpResend();\n emailOtpTimer.style.display = \"none\"\n startEmailOtpTimer(EMAIL_OTP_RESEND_TIMER, display);\n }\n })\n });\n })\n\n function startEmailOtpTimer(duration, display){\n var timer = duration, seconds;\n clearInterval(emailOtpCountdownTimer);\n emailOtpCountdownTimer = setInterval(function () {\n seconds = parseInt(timer, 10);\n display.textContent = seconds;\n if(--timer < 0){\n enableEmailOtpResend();\n clearInterval(emailOtpCountdownTimer);\n }\n }, 1000);\n disableEmailOtpResend();\n }\n\n function enableEmailOtpResend(){\n emailOtpResendBtn.style.display = \"block\";\n emailOtpTimer.style.display = \"none\"\n }\n function disableEmailOtpResend(){\n emailOtpResendBtn.style.display = \"none\";\n emailOtpTimer.style.display = \"block\"\n }\n function keyForEmailOtpToken(email, pageCode) {\n return email + '|' + pageCode+ '|SUBSCRIBE_VIA_EMAIL';\n }", "rect": { "left": 0, @@ -29,10 +26,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='2e61aa0fa8']", - "center": [ - 960, - 37 - ], + "center": [960, 37], "content": "GitHub Octicon logo\n \n \n Help\n Community\n Status\n \n \n \n GitHub.com\n Twitter\n \n \n \n\n \n\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n x\n \n \n \n \n \n Get email notifications whenever GitHub creates, updates or resolves an incident.\n \n \n \n \n Email address:\n \n \n \n Enter OTP:\n \n Resend OTP in: seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get text message notifications whenever GitHub creates or resolves an incident.\n \n \n \n \n \n \n \n Country code:\n \n Afghanistan (+93)\nAlbania (+355)\nAlgeria (+213)\nAmerican Samoa (+1)\nAndorra (+376)\nAngola (+244)\nAnguilla (+1)\nAntigua and Barbuda (+1)\nArgentina (+54)\nArmenia (+374)\nAruba (+297)\nAustralia/Cocos/Christmas Island (+61)\nAustria (+43)\nAzerbaijan (+994)\nBahamas (+1)\nBahrain (+973)\nBangladesh (+880)\nBarbados (+1)\nBelarus (+375)\nBelgium (+32)\nBelize (+501)\nBenin (+229)\nBermuda (+1)\nBolivia (+591)\nBosnia and Herzegovina (+387)\nBotswana (+267)\nBrazil (+55)\nBrunei (+673)\nBulgaria (+359)\nBurkina Faso (+226)\nBurundi (+257)\nCambodia (+855)\nCameroon (+237)\nCanada (+1)\nCape Verde (+238)\nCayman Islands (+1)\nCentral Africa (+236)\nChad (+235)\nChile (+56)\nChina (+86)\nColombia (+57)\nComoros (+269)\nCongo (+242)\nCongo, Dem Rep (+243)\nCosta Rica (+506)\nCroatia (+385)\nCyprus (+357)\nCzech Republic (+420)\nDenmark (+45)\nDjibouti (+253)\nDominica (+1)\nDominican Republic (+1)\nEgypt (+20)\nEl Salvador (+503)\nEquatorial Guinea (+240)\nEstonia (+372)\nEthiopia (+251)\nFaroe Islands (+298)\nFiji (+679)\nFinland/Aland Islands (+358)\nFrance (+33)\nFrench Guiana (+594)\nFrench Polynesia (+689)\nGabon (+241)\nGambia (+220)\nGeorgia (+995)\nGermany (+49)\nGhana (+233)\nGibraltar (+350)\nGreece (+30)\nGreenland (+299)\nGrenada (+1)\nGuadeloupe (+590)\nGuam (+1)\nGuatemala (+502)\nGuinea (+224)\nGuyana (+592)\nHaiti (+509)\nHonduras (+504)\nHong Kong (+852)\nHungary (+36)\nIceland (+354)\nIndia (+91)\nIndonesia (+62)\nIraq (+964)\nIreland (+353)\nIsrael (+972)\nItaly (+39)\nJamaica (+1)\nJapan (+81)\nJordan (+962)\nKenya (+254)\nKorea, Republic of (+82)\nKosovo (+383)\nKuwait (+965)\nKyrgyzstan (+996)\nLaos (+856)\nLatvia (+371)\nLebanon (+961)\nLesotho (+266)\nLiberia (+231)\nLibya (+218)\nLiechtenstein (+423)\nLithuania (+370)\nLuxembourg (+352)\nMacao (+853)\nMacedonia (+389)\nMadagascar (+261)\nMalawi (+265)\nMalaysia (+60)\nMaldives (+960)\nMali (+223)\nMalta (+356)\nMartinique (+596)\nMauritania (+222)\nMauritius (+230)\nMexico (+52)\nMonaco (+377)\nMongolia (+976)\nMontenegro (+382)\nMontserrat (+1)\nMorocco/Western Sahara (+212)\nMozambique (+258)\nNamibia (+264)\nNepal (+977)\nNetherlands (+31)\nNew Zealand (+64)\nNicaragua (+505)\nNiger (+227)\nNigeria (+234)\nNorway (+47)\nOman (+968)\nPakistan (+92)\nPalestinian Territory (+970)\nPanama (+507)\nParaguay (+595)\nPeru (+51)\nPhilippines (+63)\nPoland (+48)\nPortugal (+351)\nPuerto Rico (+1)\nQatar (+974)\nReunion/Mayotte (+262)\nRomania (+40)\nRussia/Kazakhstan (+7)\nRwanda (+250)\nSamoa (+685)\nSan Marino (+378)\nSaudi Arabia (+966)\nSenegal (+221)\nSerbia (+381)\nSeychelles (+248)\nSierra Leone (+232)\nSingapore (+65)\nSlovakia (+421)\nSlovenia (+386)\nSouth Africa (+27)\nSpain (+34)\nSri Lanka (+94)\nSt Kitts and Nevis (+1)\nSt Lucia (+1)\nSt Vincent Grenadines (+1)\nSudan (+249)\nSuriname (+597)\nSwaziland (+268)\nSweden (+46)\nSwitzerland (+41)\nTaiwan (+886)\nTajikistan (+992)\nTanzania (+255)\nThailand (+66)\nTogo (+228)\nTonga (+676)\nTrinidad and Tobago (+1)\nTunisia (+216)\nTurkey (+90)\nTurks and Caicos Islands (+1)\nUganda (+256)\nUkraine (+380)\nUnited Arab Emirates (+971)\nUnited Kingdom (+44)\nUnited States (+1)\nUruguay (+598)\nUzbekistan (+998)\nVenezuela (+58)\nVietnam (+84)\nVirgin Islands, British (+1)\nVirgin Islands, U.S. (+1)\nYemen (+967)\nZambia (+260)\nZimbabwe (+263)\n \n Phone number:\n \n \n \n \n Change number\n Enter OTP:\n \n Resend OTP in: 30 seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n\n \n \n Message and data rates may apply. By subscribing you agree to our Privacy Policy, the Atlassian Terms of Service, and the Atlassian Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get incident updates and maintenance status messages in Slack.\n \n Subscribe via Slack\n By subscribing you acknowledge our Privacy Policy. In addition, you agree to the Atlassian Cloud Terms of Service and acknowledge Atlassian's Privacy Policy.\n \n\n\n \n \n Get webhook notifications whenever GitHub creates an incident, updates an incident, resolves an incident or changes a component status.\n \n \n \n \n Webhook URL:\n \n The URL we should send the webhooks to\n \n \n\n \n \n Email address:\n \n We'll send you email if your endpoint fails\n \n \n\n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n\n \n\n \n \n or \n view our profile.\n \n .twitter-follow-button {\n margin-bottom: -6px;\n }\n \n\n !function(d,s,id){var js,fjs=d.getElementsByTagName(s)[0],p=/^http:/.test(d.location)?'http':'https';if(!d.getElementById(id)){js=d.createElement(s);js.id=id;js.src=p+'://platform.twitter.com/widgets.js';fjs.parentNode.insertBefore(js,fjs);}}(document, 'script', 'twitter-wjs');\n \n\n \n Visit our support site.\n \n\n \n Get the Atom Feed or RSS Feed.", "rect": { "left": 0, @@ -50,10 +44,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='5c2d0fc07e']", - "center": [ - 399, - 36 - ], + "center": [399, 36], "content": "Help\n Community\n Status", "rect": { "left": 24, @@ -72,10 +63,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='7480b80cb3']", - "center": [ - 41, - 37 - ], + "center": [41, 37], "content": "Help", "rect": { "left": 24, @@ -94,10 +82,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='f3a0bad871']", - "center": [ - 128, - 37 - ], + "center": [128, 37], "content": "Community", "rect": { "left": 86, @@ -116,10 +101,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='2b27b9d4f4']", - "center": [ - 220, - 37 - ], + "center": [220, 37], "content": "Status", "rect": { "left": 197, @@ -137,10 +119,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='40bf454063']", - "center": [ - 1522, - 36 - ], + "center": [1522, 36], "content": "GitHub.com\n Twitter\n \n \n \n\n \n\n\n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n \n x\n \n \n \n \n \n Get email notifications whenever GitHub creates, updates or resolves an incident.\n \n \n \n \n Email address:\n \n \n \n Enter OTP:\n \n Resend OTP in: seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get text message notifications whenever GitHub creates or resolves an incident.\n \n \n \n \n \n \n \n Country code:\n \n Afghanistan (+93)\nAlbania (+355)\nAlgeria (+213)\nAmerican Samoa (+1)\nAndorra (+376)\nAngola (+244)\nAnguilla (+1)\nAntigua and Barbuda (+1)\nArgentina (+54)\nArmenia (+374)\nAruba (+297)\nAustralia/Cocos/Christmas Island (+61)\nAustria (+43)\nAzerbaijan (+994)\nBahamas (+1)\nBahrain (+973)\nBangladesh (+880)\nBarbados (+1)\nBelarus (+375)\nBelgium (+32)\nBelize (+501)\nBenin (+229)\nBermuda (+1)\nBolivia (+591)\nBosnia and Herzegovina (+387)\nBotswana (+267)\nBrazil (+55)\nBrunei (+673)\nBulgaria (+359)\nBurkina Faso (+226)\nBurundi (+257)\nCambodia (+855)\nCameroon (+237)\nCanada (+1)\nCape Verde (+238)\nCayman Islands (+1)\nCentral Africa (+236)\nChad (+235)\nChile (+56)\nChina (+86)\nColombia (+57)\nComoros (+269)\nCongo (+242)\nCongo, Dem Rep (+243)\nCosta Rica (+506)\nCroatia (+385)\nCyprus (+357)\nCzech Republic (+420)\nDenmark (+45)\nDjibouti (+253)\nDominica (+1)\nDominican Republic (+1)\nEgypt (+20)\nEl Salvador (+503)\nEquatorial Guinea (+240)\nEstonia (+372)\nEthiopia (+251)\nFaroe Islands (+298)\nFiji (+679)\nFinland/Aland Islands (+358)\nFrance (+33)\nFrench Guiana (+594)\nFrench Polynesia (+689)\nGabon (+241)\nGambia (+220)\nGeorgia (+995)\nGermany (+49)\nGhana (+233)\nGibraltar (+350)\nGreece (+30)\nGreenland (+299)\nGrenada (+1)\nGuadeloupe (+590)\nGuam (+1)\nGuatemala (+502)\nGuinea (+224)\nGuyana (+592)\nHaiti (+509)\nHonduras (+504)\nHong Kong (+852)\nHungary (+36)\nIceland (+354)\nIndia (+91)\nIndonesia (+62)\nIraq (+964)\nIreland (+353)\nIsrael (+972)\nItaly (+39)\nJamaica (+1)\nJapan (+81)\nJordan (+962)\nKenya (+254)\nKorea, Republic of (+82)\nKosovo (+383)\nKuwait (+965)\nKyrgyzstan (+996)\nLaos (+856)\nLatvia (+371)\nLebanon (+961)\nLesotho (+266)\nLiberia (+231)\nLibya (+218)\nLiechtenstein (+423)\nLithuania (+370)\nLuxembourg (+352)\nMacao (+853)\nMacedonia (+389)\nMadagascar (+261)\nMalawi (+265)\nMalaysia (+60)\nMaldives (+960)\nMali (+223)\nMalta (+356)\nMartinique (+596)\nMauritania (+222)\nMauritius (+230)\nMexico (+52)\nMonaco (+377)\nMongolia (+976)\nMontenegro (+382)\nMontserrat (+1)\nMorocco/Western Sahara (+212)\nMozambique (+258)\nNamibia (+264)\nNepal (+977)\nNetherlands (+31)\nNew Zealand (+64)\nNicaragua (+505)\nNiger (+227)\nNigeria (+234)\nNorway (+47)\nOman (+968)\nPakistan (+92)\nPalestinian Territory (+970)\nPanama (+507)\nParaguay (+595)\nPeru (+51)\nPhilippines (+63)\nPoland (+48)\nPortugal (+351)\nPuerto Rico (+1)\nQatar (+974)\nReunion/Mayotte (+262)\nRomania (+40)\nRussia/Kazakhstan (+7)\nRwanda (+250)\nSamoa (+685)\nSan Marino (+378)\nSaudi Arabia (+966)\nSenegal (+221)\nSerbia (+381)\nSeychelles (+248)\nSierra Leone (+232)\nSingapore (+65)\nSlovakia (+421)\nSlovenia (+386)\nSouth Africa (+27)\nSpain (+34)\nSri Lanka (+94)\nSt Kitts and Nevis (+1)\nSt Lucia (+1)\nSt Vincent Grenadines (+1)\nSudan (+249)\nSuriname (+597)\nSwaziland (+268)\nSweden (+46)\nSwitzerland (+41)\nTaiwan (+886)\nTajikistan (+992)\nTanzania (+255)\nThailand (+66)\nTogo (+228)\nTonga (+676)\nTrinidad and Tobago (+1)\nTunisia (+216)\nTurkey (+90)\nTurks and Caicos Islands (+1)\nUganda (+256)\nUkraine (+380)\nUnited Arab Emirates (+971)\nUnited Kingdom (+44)\nUnited States (+1)\nUruguay (+598)\nUzbekistan (+998)\nVenezuela (+58)\nVietnam (+84)\nVirgin Islands, British (+1)\nVirgin Islands, U.S. (+1)\nYemen (+967)\nZambia (+260)\nZimbabwe (+263)\n \n Phone number:\n \n \n \n \n Change number\n Enter OTP:\n \n Resend OTP in: 30 seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n\n \n \n Message and data rates may apply. By subscribing you agree to our Privacy Policy, the Atlassian Terms of Service, and the Atlassian Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get incident updates and maintenance status messages in Slack.\n \n Subscribe via Slack\n By subscribing you acknowledge our Privacy Policy. In addition, you agree to the Atlassian Cloud Terms of Service and acknowledge Atlassian's Privacy Policy.\n \n\n\n \n \n Get webhook notifications whenever GitHub creates an incident, updates an incident, resolves an incident or changes a component status.\n \n \n \n \n Webhook URL:\n \n The URL we should send the webhooks to\n \n \n\n \n \n Email address:\n \n We'll send you email if your endpoint fails\n \n \n\n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n\n \n\n \n \n or \n view our profile.\n \n .twitter-follow-button {\n margin-bottom: -6px;\n }\n \n\n !function(d,s,id){var js,fjs=d.getElementsByTagName(s)[0],p=/^http:/.test(d.location)?'http':'https';if(!d.getElementById(id)){js=d.createElement(s);js.id=id;js.src=p+'://platform.twitter.com/widgets.js';fjs.parentNode.insertBefore(js,fjs);}}(document, 'script', 'twitter-wjs');\n \n\n \n Visit our support site.\n \n\n \n Get the Atom Feed or RSS Feed.", "rect": { "left": 1147, @@ -159,10 +138,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='44c2ed902a']", - "center": [ - 1586, - 37 - ], + "center": [1586, 37], "content": "GitHub.com", "rect": { "left": 1543, @@ -181,10 +157,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='7d552ded8a']", - "center": [ - 1683, - 37 - ], + "center": [1683, 37], "content": "Twitter", "rect": { "left": 1658, @@ -202,10 +175,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='c6687feb44']", - "center": [ - 1816, - 37 - ], + "center": [1816, 37], "content": "x\n \n \n \n \n \n Get email notifications whenever GitHub creates, updates or resolves an incident.\n \n \n \n \n Email address:\n \n \n \n Enter OTP:\n \n Resend OTP in: seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get text message notifications whenever GitHub creates or resolves an incident.\n \n \n \n \n \n \n \n Country code:\n \n Afghanistan (+93)\nAlbania (+355)\nAlgeria (+213)\nAmerican Samoa (+1)\nAndorra (+376)\nAngola (+244)\nAnguilla (+1)\nAntigua and Barbuda (+1)\nArgentina (+54)\nArmenia (+374)\nAruba (+297)\nAustralia/Cocos/Christmas Island (+61)\nAustria (+43)\nAzerbaijan (+994)\nBahamas (+1)\nBahrain (+973)\nBangladesh (+880)\nBarbados (+1)\nBelarus (+375)\nBelgium (+32)\nBelize (+501)\nBenin (+229)\nBermuda (+1)\nBolivia (+591)\nBosnia and Herzegovina (+387)\nBotswana (+267)\nBrazil (+55)\nBrunei (+673)\nBulgaria (+359)\nBurkina Faso (+226)\nBurundi (+257)\nCambodia (+855)\nCameroon (+237)\nCanada (+1)\nCape Verde (+238)\nCayman Islands (+1)\nCentral Africa (+236)\nChad (+235)\nChile (+56)\nChina (+86)\nColombia (+57)\nComoros (+269)\nCongo (+242)\nCongo, Dem Rep (+243)\nCosta Rica (+506)\nCroatia (+385)\nCyprus (+357)\nCzech Republic (+420)\nDenmark (+45)\nDjibouti (+253)\nDominica (+1)\nDominican Republic (+1)\nEgypt (+20)\nEl Salvador (+503)\nEquatorial Guinea (+240)\nEstonia (+372)\nEthiopia (+251)\nFaroe Islands (+298)\nFiji (+679)\nFinland/Aland Islands (+358)\nFrance (+33)\nFrench Guiana (+594)\nFrench Polynesia (+689)\nGabon (+241)\nGambia (+220)\nGeorgia (+995)\nGermany (+49)\nGhana (+233)\nGibraltar (+350)\nGreece (+30)\nGreenland (+299)\nGrenada (+1)\nGuadeloupe (+590)\nGuam (+1)\nGuatemala (+502)\nGuinea (+224)\nGuyana (+592)\nHaiti (+509)\nHonduras (+504)\nHong Kong (+852)\nHungary (+36)\nIceland (+354)\nIndia (+91)\nIndonesia (+62)\nIraq (+964)\nIreland (+353)\nIsrael (+972)\nItaly (+39)\nJamaica (+1)\nJapan (+81)\nJordan (+962)\nKenya (+254)\nKorea, Republic of (+82)\nKosovo (+383)\nKuwait (+965)\nKyrgyzstan (+996)\nLaos (+856)\nLatvia (+371)\nLebanon (+961)\nLesotho (+266)\nLiberia (+231)\nLibya (+218)\nLiechtenstein (+423)\nLithuania (+370)\nLuxembourg (+352)\nMacao (+853)\nMacedonia (+389)\nMadagascar (+261)\nMalawi (+265)\nMalaysia (+60)\nMaldives (+960)\nMali (+223)\nMalta (+356)\nMartinique (+596)\nMauritania (+222)\nMauritius (+230)\nMexico (+52)\nMonaco (+377)\nMongolia (+976)\nMontenegro (+382)\nMontserrat (+1)\nMorocco/Western Sahara (+212)\nMozambique (+258)\nNamibia (+264)\nNepal (+977)\nNetherlands (+31)\nNew Zealand (+64)\nNicaragua (+505)\nNiger (+227)\nNigeria (+234)\nNorway (+47)\nOman (+968)\nPakistan (+92)\nPalestinian Territory (+970)\nPanama (+507)\nParaguay (+595)\nPeru (+51)\nPhilippines (+63)\nPoland (+48)\nPortugal (+351)\nPuerto Rico (+1)\nQatar (+974)\nReunion/Mayotte (+262)\nRomania (+40)\nRussia/Kazakhstan (+7)\nRwanda (+250)\nSamoa (+685)\nSan Marino (+378)\nSaudi Arabia (+966)\nSenegal (+221)\nSerbia (+381)\nSeychelles (+248)\nSierra Leone (+232)\nSingapore (+65)\nSlovakia (+421)\nSlovenia (+386)\nSouth Africa (+27)\nSpain (+34)\nSri Lanka (+94)\nSt Kitts and Nevis (+1)\nSt Lucia (+1)\nSt Vincent Grenadines (+1)\nSudan (+249)\nSuriname (+597)\nSwaziland (+268)\nSweden (+46)\nSwitzerland (+41)\nTaiwan (+886)\nTajikistan (+992)\nTanzania (+255)\nThailand (+66)\nTogo (+228)\nTonga (+676)\nTrinidad and Tobago (+1)\nTunisia (+216)\nTurkey (+90)\nTurks and Caicos Islands (+1)\nUganda (+256)\nUkraine (+380)\nUnited Arab Emirates (+971)\nUnited Kingdom (+44)\nUnited States (+1)\nUruguay (+598)\nUzbekistan (+998)\nVenezuela (+58)\nVietnam (+84)\nVirgin Islands, British (+1)\nVirgin Islands, U.S. (+1)\nYemen (+967)\nZambia (+260)\nZimbabwe (+263)\n \n Phone number:\n \n \n \n \n Change number\n Enter OTP:\n \n Resend OTP in: 30 seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n\n \n \n Message and data rates may apply. By subscribing you agree to our Privacy Policy, the Atlassian Terms of Service, and the Atlassian Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get incident updates and maintenance status messages in Slack.\n \n Subscribe via Slack\n By subscribing you acknowledge our Privacy Policy. In addition, you agree to the Atlassian Cloud Terms of Service and acknowledge Atlassian's Privacy Policy.\n \n\n\n \n \n Get webhook notifications whenever GitHub creates an incident, updates an incident, resolves an incident or changes a component status.\n \n \n \n \n Webhook URL:\n \n The URL we should send the webhooks to\n \n \n\n \n \n Email address:\n \n We'll send you email if your endpoint fails\n \n \n\n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n\n \n\n \n \n or \n view our profile.\n \n .twitter-follow-button {\n margin-bottom: -6px;\n }\n \n\n !function(d,s,id){var js,fjs=d.getElementsByTagName(s)[0],p=/^http:/.test(d.location)?'http':'https';if(!d.getElementById(id)){js=d.createElement(s);js.id=id;js.src=p+'://platform.twitter.com/widgets.js';fjs.parentNode.insertBefore(js,fjs);}}(document, 'script', 'twitter-wjs');\n \n\n \n Visit our support site.\n \n\n \n Get the Atom Feed or RSS Feed.", "rect": { "left": 1736, @@ -225,10 +195,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='9bf8cb5fec']", - "center": [ - 1816, - 36 - ], + "center": [1816, 36], "content": "x\n \n \n \n \n \n Get email notifications whenever GitHub creates, updates or resolves an incident.\n \n \n \n \n Email address:\n \n \n \n Enter OTP:\n \n Resend OTP in: seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get text message notifications whenever GitHub creates or resolves an incident.\n \n \n \n \n \n \n \n Country code:\n \n Afghanistan (+93)\nAlbania (+355)\nAlgeria (+213)\nAmerican Samoa (+1)\nAndorra (+376)\nAngola (+244)\nAnguilla (+1)\nAntigua and Barbuda (+1)\nArgentina (+54)\nArmenia (+374)\nAruba (+297)\nAustralia/Cocos/Christmas Island (+61)\nAustria (+43)\nAzerbaijan (+994)\nBahamas (+1)\nBahrain (+973)\nBangladesh (+880)\nBarbados (+1)\nBelarus (+375)\nBelgium (+32)\nBelize (+501)\nBenin (+229)\nBermuda (+1)\nBolivia (+591)\nBosnia and Herzegovina (+387)\nBotswana (+267)\nBrazil (+55)\nBrunei (+673)\nBulgaria (+359)\nBurkina Faso (+226)\nBurundi (+257)\nCambodia (+855)\nCameroon (+237)\nCanada (+1)\nCape Verde (+238)\nCayman Islands (+1)\nCentral Africa (+236)\nChad (+235)\nChile (+56)\nChina (+86)\nColombia (+57)\nComoros (+269)\nCongo (+242)\nCongo, Dem Rep (+243)\nCosta Rica (+506)\nCroatia (+385)\nCyprus (+357)\nCzech Republic (+420)\nDenmark (+45)\nDjibouti (+253)\nDominica (+1)\nDominican Republic (+1)\nEgypt (+20)\nEl Salvador (+503)\nEquatorial Guinea (+240)\nEstonia (+372)\nEthiopia (+251)\nFaroe Islands (+298)\nFiji (+679)\nFinland/Aland Islands (+358)\nFrance (+33)\nFrench Guiana (+594)\nFrench Polynesia (+689)\nGabon (+241)\nGambia (+220)\nGeorgia (+995)\nGermany (+49)\nGhana (+233)\nGibraltar (+350)\nGreece (+30)\nGreenland (+299)\nGrenada (+1)\nGuadeloupe (+590)\nGuam (+1)\nGuatemala (+502)\nGuinea (+224)\nGuyana (+592)\nHaiti (+509)\nHonduras (+504)\nHong Kong (+852)\nHungary (+36)\nIceland (+354)\nIndia (+91)\nIndonesia (+62)\nIraq (+964)\nIreland (+353)\nIsrael (+972)\nItaly (+39)\nJamaica (+1)\nJapan (+81)\nJordan (+962)\nKenya (+254)\nKorea, Republic of (+82)\nKosovo (+383)\nKuwait (+965)\nKyrgyzstan (+996)\nLaos (+856)\nLatvia (+371)\nLebanon (+961)\nLesotho (+266)\nLiberia (+231)\nLibya (+218)\nLiechtenstein (+423)\nLithuania (+370)\nLuxembourg (+352)\nMacao (+853)\nMacedonia (+389)\nMadagascar (+261)\nMalawi (+265)\nMalaysia (+60)\nMaldives (+960)\nMali (+223)\nMalta (+356)\nMartinique (+596)\nMauritania (+222)\nMauritius (+230)\nMexico (+52)\nMonaco (+377)\nMongolia (+976)\nMontenegro (+382)\nMontserrat (+1)\nMorocco/Western Sahara (+212)\nMozambique (+258)\nNamibia (+264)\nNepal (+977)\nNetherlands (+31)\nNew Zealand (+64)\nNicaragua (+505)\nNiger (+227)\nNigeria (+234)\nNorway (+47)\nOman (+968)\nPakistan (+92)\nPalestinian Territory (+970)\nPanama (+507)\nParaguay (+595)\nPeru (+51)\nPhilippines (+63)\nPoland (+48)\nPortugal (+351)\nPuerto Rico (+1)\nQatar (+974)\nReunion/Mayotte (+262)\nRomania (+40)\nRussia/Kazakhstan (+7)\nRwanda (+250)\nSamoa (+685)\nSan Marino (+378)\nSaudi Arabia (+966)\nSenegal (+221)\nSerbia (+381)\nSeychelles (+248)\nSierra Leone (+232)\nSingapore (+65)\nSlovakia (+421)\nSlovenia (+386)\nSouth Africa (+27)\nSpain (+34)\nSri Lanka (+94)\nSt Kitts and Nevis (+1)\nSt Lucia (+1)\nSt Vincent Grenadines (+1)\nSudan (+249)\nSuriname (+597)\nSwaziland (+268)\nSweden (+46)\nSwitzerland (+41)\nTaiwan (+886)\nTajikistan (+992)\nTanzania (+255)\nThailand (+66)\nTogo (+228)\nTonga (+676)\nTrinidad and Tobago (+1)\nTunisia (+216)\nTurkey (+90)\nTurks and Caicos Islands (+1)\nUganda (+256)\nUkraine (+380)\nUnited Arab Emirates (+971)\nUnited Kingdom (+44)\nUnited States (+1)\nUruguay (+598)\nUzbekistan (+998)\nVenezuela (+58)\nVietnam (+84)\nVirgin Islands, British (+1)\nVirgin Islands, U.S. (+1)\nYemen (+967)\nZambia (+260)\nZimbabwe (+263)\n \n Phone number:\n \n \n \n \n Change number\n Enter OTP:\n \n Resend OTP in: 30 seconds \n \n Didn't receive the OTP?\n Resend OTP \n \n \n \n \n\n \n \n Message and data rates may apply. By subscribing you agree to our Privacy Policy, the Atlassian Terms of Service, and the Atlassian Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n \n\n \n \n Get incident updates and maintenance status messages in Slack.\n \n Subscribe via Slack\n By subscribing you acknowledge our Privacy Policy. In addition, you agree to the Atlassian Cloud Terms of Service and acknowledge Atlassian's Privacy Policy.\n \n\n\n \n \n Get webhook notifications whenever GitHub creates an incident, updates an incident, resolves an incident or changes a component status.\n \n \n \n \n Webhook URL:\n \n The URL we should send the webhooks to\n \n \n\n \n \n Email address:\n \n We'll send you email if your endpoint fails\n \n \n\n \n \n By subscribing you agree to our Privacy Policy. This site is protected by reCAPTCHA and the Google Privacy Policy and Terms of Service apply.\n\n \n\n \n \n or \n view our profile.\n \n .twitter-follow-button {\n margin-bottom: -6px;\n }\n \n\n !function(d,s,id){var js,fjs=d.getElementsByTagName(s)[0],p=/^http:/.test(d.location)?'http':'https';if(!d.getElementById(id)){js=d.createElement(s);js.id=id;js.src=p+'://platform.twitter.com/widgets.js';fjs.parentNode.insertBefore(js,fjs);}}(document, 'script', 'twitter-wjs');\n \n\n \n Visit our support site.\n \n\n \n Get the Atom Feed or RSS Feed.", "rect": { "left": 1736, @@ -256,10 +223,7 @@ "width": 1920, "height": 422 }, - "center": [ - 960, - 284 - ] + "center": [960, 284] }, { "id": "33bccce59f", @@ -270,10 +234,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='33bccce59f']", - "center": [ - 960, - 428 - ], + "center": [960, 428], "content": "All Systems Operational", "rect": { "left": 470, @@ -291,10 +252,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='221b0092a0']", - "center": [ - 632, - 427 - ], + "center": [632, 427], "content": "All Systems Operational", "rect": { "left": 522, @@ -312,10 +270,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='9cfcb84f47']", - "center": [ - 960, - 756 - ], + "center": [960, 756], "content": "Git Operations\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n API Requests\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Webhooks\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Visit www.githubstatus.com for more information\n \n\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Issues\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Pull Requests\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Actions\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Packages\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Pages\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Codespaces\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Copilot\n \n\n\n \n\n \n\nNormal", "rect": { "left": 470, @@ -333,10 +288,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='1d2f8b067e']", - "center": [ - 960, - 783 - ], + "center": [960, 783], "content": "Git Operations\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n API Requests\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Webhooks\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Visit www.githubstatus.com for more information\n \n\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Issues\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Pull Requests\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Actions\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Packages\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Pages\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Codespaces\n \n\n ?\n\n \n\n \n\nNormal\n\n \n \n \n\n\n \n Copilot\n \n\n\n \n\n \n\nNormal", "rect": { "left": 470, @@ -354,10 +306,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='1775a8e5ac']", - "center": [ - 717, - 605 - ], + "center": [717, 605], "content": "Git Operations\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 472, @@ -377,10 +326,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='9b8133c565']", - "center": [ - 716, - 606 - ], + "center": [716, 606], "content": "Git Operations\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 492, @@ -398,10 +344,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='786dafff55']", - "center": [ - 546, - 593 - ], + "center": [546, 593], "content": "Git Operations", "rect": { "left": 492, @@ -419,10 +362,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='9fd504c355']", - "center": [ - 618, - 591 - ], + "center": [618, 591], "content": "?", "rect": { "left": 609, @@ -440,10 +380,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='15865efd31']", - "center": [ - 716, - 619 - ], + "center": [716, 619], "content": "Normal", "rect": { "left": 492, @@ -461,10 +398,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='cac86a3a9a']", - "center": [ - 1206, - 605 - ], + "center": [1206, 605], "content": "API Requests\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 961, @@ -484,10 +418,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='bfd2275c7b']", - "center": [ - 1205, - 606 - ], + "center": [1205, 606], "content": "API Requests\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 981, @@ -505,10 +436,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='244ab8c4c3']", - "center": [ - 1030, - 593 - ], + "center": [1030, 593], "content": "API Requests", "rect": { "left": 981, @@ -526,10 +454,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='b0bdb14a2c']", - "center": [ - 1097, - 591 - ], + "center": [1097, 591], "content": "?", "rect": { "left": 1088, @@ -547,10 +472,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='e4932f3f94']", - "center": [ - 1205, - 619 - ], + "center": [1205, 619], "content": "Normal", "rect": { "left": 981, @@ -568,10 +490,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='6ec8080ca6']", - "center": [ - 717, - 695 - ], + "center": [717, 695], "content": "Webhooks\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 472, @@ -591,10 +510,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='5d616d114f']", - "center": [ - 716, - 695 - ], + "center": [716, 695], "content": "Webhooks\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 492, @@ -612,10 +528,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='675456f129']", - "center": [ - 531, - 682 - ], + "center": [531, 682], "content": "Webhooks", "rect": { "left": 492, @@ -633,10 +546,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='cd6731b4d0']", - "center": [ - 588, - 681 - ], + "center": [588, 681], "content": "?", "rect": { "left": 579, @@ -654,10 +564,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='4abfbbde8a']", - "center": [ - 716, - 708 - ], + "center": [716, 708], "content": "Normal", "rect": { "left": 492, @@ -675,10 +582,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='646e1d48fb']", - "center": [ - 1206, - 695 - ], + "center": [1206, 695], "content": "Issues\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 961, @@ -698,10 +602,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='8b54cbe825']", - "center": [ - 1205, - 695 - ], + "center": [1205, 695], "content": "Issues\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 981, @@ -719,10 +620,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='49b1a5e3e1']", - "center": [ - 1005, - 682 - ], + "center": [1005, 682], "content": "Issues", "rect": { "left": 981, @@ -740,10 +638,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='ce72a7bb29']", - "center": [ - 1046, - 681 - ], + "center": [1046, 681], "content": "?", "rect": { "left": 1037, @@ -761,10 +656,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='44c6cea361']", - "center": [ - 1205, - 708 - ], + "center": [1205, 708], "content": "Normal", "rect": { "left": 981, @@ -782,10 +674,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='1a7bbd910b']", - "center": [ - 717, - 784 - ], + "center": [717, 784], "content": "Pull Requests\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 472, @@ -805,10 +694,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='d04547b004']", - "center": [ - 716, - 785 - ], + "center": [716, 785], "content": "Pull Requests\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 492, @@ -826,10 +712,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='0ef54614d8']", - "center": [ - 542, - 772 - ], + "center": [542, 772], "content": "Pull Requests", "rect": { "left": 492, @@ -847,10 +730,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='00228c1c08']", - "center": [ - 611, - 770 - ], + "center": [611, 770], "content": "?", "rect": { "left": 602, @@ -868,10 +748,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='206fe45dc0']", - "center": [ - 716, - 798 - ], + "center": [716, 798], "content": "Normal", "rect": { "left": 492, @@ -889,10 +766,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='075070b70a']", - "center": [ - 1206, - 784 - ], + "center": [1206, 784], "content": "Actions\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 961, @@ -912,10 +786,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='ebbf5e2217']", - "center": [ - 1205, - 785 - ], + "center": [1205, 785], "content": "Actions\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 981, @@ -933,10 +804,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='89c422b218']", - "center": [ - 1009, - 772 - ], + "center": [1009, 772], "content": "Actions", "rect": { "left": 981, @@ -954,10 +822,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='e923998486']", - "center": [ - 1055, - 770 - ], + "center": [1055, 770], "content": "?", "rect": { "left": 1046, @@ -975,10 +840,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='967447a1c6']", - "center": [ - 1205, - 798 - ], + "center": [1205, 798], "content": "Normal", "rect": { "left": 981, @@ -996,10 +858,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='11a7c1ed98']", - "center": [ - 717, - 874 - ], + "center": [717, 874], "content": "Packages\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 472, @@ -1019,10 +878,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='1c9544f01a']", - "center": [ - 716, - 874 - ], + "center": [716, 874], "content": "Packages\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 492, @@ -1040,10 +896,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='41a6c2ed19']", - "center": [ - 528, - 861 - ], + "center": [528, 861], "content": "Packages", "rect": { "left": 492, @@ -1061,10 +914,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='b8f8f86b4c']", - "center": [ - 581, - 860 - ], + "center": [581, 860], "content": "?", "rect": { "left": 572, @@ -1082,10 +932,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='4d10b3b116']", - "center": [ - 716, - 887 - ], + "center": [716, 887], "content": "Normal", "rect": { "left": 492, @@ -1103,10 +950,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='7d79505914']", - "center": [ - 1206, - 874 - ], + "center": [1206, 874], "content": "Pages\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 961, @@ -1126,10 +970,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='e502c0d9c5']", - "center": [ - 1205, - 874 - ], + "center": [1205, 874], "content": "Pages\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 981, @@ -1147,10 +988,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='cae85d0269']", - "center": [ - 1004, - 861 - ], + "center": [1004, 861], "content": "Pages", "rect": { "left": 981, @@ -1168,10 +1006,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='85a6924c98']", - "center": [ - 1045, - 860 - ], + "center": [1045, 860], "content": "?", "rect": { "left": 1036, @@ -1189,10 +1024,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='352bb8a0c7']", - "center": [ - 1205, - 887 - ], + "center": [1205, 887], "content": "Normal", "rect": { "left": 981, @@ -1210,10 +1042,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='7ebaccd865']", - "center": [ - 717, - 963 - ], + "center": [717, 963], "content": "Codespaces\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 472, @@ -1233,10 +1062,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='318eb23523']", - "center": [ - 716, - 963 - ], + "center": [716, 963], "content": "Codespaces\n \n\n ?\n\n \n\n \n\nNormal", "rect": { "left": 492, @@ -1254,10 +1080,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='f397787172']", - "center": [ - 538, - 950 - ], + "center": [538, 950], "content": "Codespaces", "rect": { "left": 492, @@ -1275,10 +1098,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='d7f5fbb4d0']", - "center": [ - 602, - 949 - ], + "center": [602, 949], "content": "?", "rect": { "left": 593, @@ -1296,10 +1116,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='aafa155f98']", - "center": [ - 716, - 976 - ], + "center": [716, 976], "content": "Normal", "rect": { "left": 492, @@ -1317,10 +1134,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='aa396a5b1d']", - "center": [ - 1206, - 963 - ], + "center": [1206, 963], "content": "Copilot\n \n\n\n \n\n \n\nNormal", "rect": { "left": 961, @@ -1340,10 +1154,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='ecfafeca6d']", - "center": [ - 1205, - 963 - ], + "center": [1205, 963], "content": "Copilot\n \n\n\n \n\n \n\nNormal", "rect": { "left": 981, @@ -1361,10 +1172,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='18dce47ff0']", - "center": [ - 1008, - 950 - ], + "center": [1008, 950], "content": "Copilot", "rect": { "left": 981, @@ -1382,10 +1190,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='3ac8478628']", - "center": [ - 1205, - 976 - ], + "center": [1205, 976], "content": "Normal", "rect": { "left": 981, @@ -1394,4 +1199,4 @@ "height": 28 } } -] \ No newline at end of file +] diff --git a/packages/midscene/tests/ai-model/inspector/test-data/online_order/element-snapshot.json b/packages/midscene/tests/ai-model/inspector/test-data/online_order/element-snapshot.json index 75c10845..d7933ca6 100644 --- a/packages/midscene/tests/ai-model/inspector/test-data/online_order/element-snapshot.json +++ b/packages/midscene/tests/ai-model/inspector/test-data/online_order/element-snapshot.json @@ -16,10 +16,7 @@ "width": 22, "height": 22 }, - "center": [ - 26, - 28 - ] + "center": [26, 28] }, { "id": "ba59909699", @@ -30,10 +27,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='ba59909699']", - "center": [ - 81, - 28 - ], + "center": [81, 28], "content": "中文", "rect": { "left": 59, @@ -59,10 +53,7 @@ "width": 44, "height": 44 }, - "center": [ - 225, - 30 - ] + "center": [225, 30] }, { "id": "f775c69cb4", @@ -81,10 +72,7 @@ "width": 25, "height": 25 }, - "center": [ - 363, - 29 - ] + "center": [363, 29] }, { "id": "67bd3fba57", @@ -94,10 +82,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='67bd3fba57']", - "center": [ - 200, - 80 - ], + "center": [200, 80], "content": "商家暂未开通外卖和自取业务", "rect": { "left": 75, @@ -114,10 +99,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='c900f4b633']", - "center": [ - 209, - 98 - ], + "center": [209, 98], "content": "喜茶 HEYTEA (Vivocity)", "rect": { "left": 22, @@ -134,10 +116,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='6f761c984e']", - "center": [ - 209, - 133 - ], + "center": [209, 133], "content": "全岛派送,起送价 S$ 11", "rect": { "left": 22, @@ -162,10 +141,7 @@ "width": 368, "height": 180 }, - "center": [ - 200, - 350 - ] + "center": [200, 350] }, { "id": "31c3a4bba6", @@ -175,10 +151,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='31c3a4bba6']", - "center": [ - 200, - 482 - ], + "center": [200, 482], "content": "喜茶为芝士现泡茶的原创者。自创立以来,喜茶专注于呈现来自世界各地的优质茶香, 让茶饮这一古老文化焕发出新的生命力。", "rect": { "left": 16, @@ -195,10 +168,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='079f969064']", - "center": [ - 128, - 569 - ], + "center": [128, 569], "content": "菜单", "rect": { "left": 112, @@ -215,10 +185,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='b0931353bd']", - "center": [ - 200, - 569 - ], + "center": [200, 569], "content": "评价", "rect": { "left": 184, @@ -235,10 +202,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='afab9cd02f']", - "center": [ - 272, - 569 - ], + "center": [272, 569], "content": "商家", "rect": { "left": 256, @@ -255,10 +219,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='88e7fb5aed']", - "center": [ - 29, - 623 - ], + "center": [29, 623], "content": "时令鲜果", "rect": { "left": 5, @@ -275,10 +236,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='cee3c6676f']", - "center": [ - 35, - 683 - ], + "center": [35, 683], "content": "清爽不喝腻", "rect": { "left": 5, @@ -295,10 +253,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='512ab699a2']", - "center": [ - 35, - 743 - ], + "center": [35, 743], "content": "经典不踩雷", "rect": { "left": 5, @@ -315,10 +270,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='3948acb8ad']", - "center": [ - 23, - 803 - ], + "center": [23, 803], "content": "要浓郁", "rect": { "left": 5, @@ -335,10 +287,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='985ed5922f']", - "center": [ - 23, - 863 - ], + "center": [23, 863], "content": "要简单", "rect": { "left": 5, @@ -355,10 +304,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='e54bc06de4']", - "center": [ - 126, - 633 - ], + "center": [126, 633], "content": "时令鲜果 (1)", "rect": { "left": 88, @@ -384,10 +330,7 @@ "width": 90, "height": 90 }, - "center": [ - 133, - 704 - ] + "center": [133, 704] }, { "id": "c93bedf4d3", @@ -397,10 +340,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='c93bedf4d3']", - "center": [ - 288, - 670 - ], + "center": [288, 670], "content": "多肉大橘(首创)", "rect": { "left": 188, @@ -418,10 +358,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='1b673db570']", - "center": [ - 288, - 691 - ], + "center": [288, 691], "content": "果肉丰盈,粒粒饱满多汁;精心调配果汁与绿妍茶底的比例打制成冰沙;搭配清甜的无香精桂花冻与Q谈脆波波,再用香水柠檬片作点缀,口感丰富,香气有层次。默认标准杯500ml,可根据个人喜好选择加大杯650ml。", "rect": { "left": 188, @@ -439,10 +376,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='14103376fb']", - "center": [ - 211, - 736 - ], + "center": [211, 736], "content": "6.82", "rect": { "left": 192, @@ -460,10 +394,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='580cfae23c']", - "center": [ - 355, - 732 - ], + "center": [355, 732], "content": "选规格", "rect": { "left": 322, @@ -480,10 +411,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='7f10fa6626']", - "center": [ - 134, - 776 - ], + "center": [134, 776], "content": "清爽不喝腻 (5)", "rect": { "left": 88, @@ -509,10 +437,7 @@ "width": 90, "height": 90 }, - "center": [ - 133, - 847 - ] + "center": [133, 847] }, { "id": "2e297d0b4a", @@ -522,10 +447,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='2e297d0b4a']", - "center": [ - 288, - 813 - ], + "center": [288, 813], "content": "轻芒芒甘露", "rect": { "left": 188, @@ -543,10 +465,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='9f0e2a5328']", - "center": [ - 288, - 834 - ], + "center": [288, 834], "content": "更清爽的芒芒甘露,芒着清爽,芒着解腻。当季芒果,新鲜手切,香甜浓郁;加入红柚果粒与Q弹的脆波波,轻盈不轻量。清爽的芒果绿妍冰沙平衡椰浆的比例,清爽不简单。", "rect": { "left": 188, @@ -564,10 +483,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='0250e12e67']", - "center": [ - 211, - 879 - ], + "center": [211, 879], "content": "6.54", "rect": { "left": 192, @@ -585,10 +501,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='925c254744']", - "center": [ - 355, - 875 - ], + "center": [355, 875], "content": "选规格", "rect": { "left": 322, @@ -613,9 +526,6 @@ "width": 42, "height": 42 }, - "center": [ - 379, - 834 - ] + "center": [379, 834] } -] \ No newline at end of file +] diff --git a/packages/midscene/tests/ai-model/inspector/test-data/todo/element-snapshot.json b/packages/midscene/tests/ai-model/inspector/test-data/todo/element-snapshot.json index 503a1554..8534442e 100644 --- a/packages/midscene/tests/ai-model/inspector/test-data/todo/element-snapshot.json +++ b/packages/midscene/tests/ai-model/inspector/test-data/todo/element-snapshot.json @@ -7,10 +7,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='ab22d01377']", - "center": [ - 640, - 66 - ], + "center": [640, 66], "content": "todos", "rect": { "left": 365, @@ -39,10 +36,7 @@ "width": 550, "height": 65 }, - "center": [ - 640, - 163 - ] + "center": [640, 163] }, { "id": "eb02ad0e19", @@ -62,10 +56,7 @@ "width": 40, "height": 60 }, - "center": [ - 389, - 163 - ] + "center": [389, 163] }, { "id": "22625b5e51", @@ -77,10 +68,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='22625b5e51']", - "center": [ - 388, - 164 - ], + "center": [388, 164], "content": "Toggle All Input", "rect": { "left": 365, @@ -107,10 +95,7 @@ "width": 40, "height": 40 }, - "center": [ - 385, - 225 - ] + "center": [385, 225] }, { "id": "46449cb0ef", @@ -121,10 +106,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='46449cb0ef']", - "center": [ - 640, - 226 - ], + "center": [640, 226], "content": "Learn Python", "rect": { "left": 365, @@ -151,10 +133,7 @@ "width": 40, "height": 40 }, - "center": [ - 385, - 285 - ] + "center": [385, 285] }, { "id": "b5bacc879a", @@ -165,10 +144,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='b5bacc879a']", - "center": [ - 640, - 286 - ], + "center": [640, 286], "content": "Learn Rust", "rect": { "left": 365, @@ -194,10 +170,7 @@ "width": 40, "height": 40 }, - "center": [ - 885, - 285 - ] + "center": [885, 285] }, { "id": "eb987bf616", @@ -217,10 +190,7 @@ "width": 40, "height": 40 }, - "center": [ - 385, - 345 - ] + "center": [385, 345] }, { "id": "6c30e37d29", @@ -231,10 +201,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='6c30e37d29']", - "center": [ - 640, - 346 - ], + "center": [640, 346], "content": "Learn AI", "rect": { "left": 365, @@ -252,10 +219,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='12fb207e82']", - "center": [ - 416, - 395 - ], + "center": [416, 395], "content": "3 items left!", "rect": { "left": 380, @@ -274,10 +238,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='944c1fae15']", - "center": [ - 565, - 395 - ], + "center": [565, 395], "content": "All", "rect": { "left": 550, @@ -296,10 +257,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='fc1a3e34a0']", - "center": [ - 613, - 395 - ], + "center": [613, 395], "content": "Active", "rect": { "left": 586, @@ -318,10 +276,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='0f8f471e06']", - "center": [ - 688, - 395 - ], + "center": [688, 395], "content": "Completed", "rect": { "left": 645, @@ -346,10 +301,7 @@ "width": 103, "height": 19 }, - "center": [ - 849, - 395 - ] + "center": [849, 395] }, { "id": "586415981c", @@ -359,10 +311,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='586415981c']", - "center": [ - 640, - 486 - ], + "center": [640, 486], "content": "Double-click to edit a todo", "rect": { "left": 365, @@ -379,10 +328,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='d501ec8b0c']", - "center": [ - 640, - 508 - ], + "center": [640, 508], "content": "Created by the TodoMVC Team", "rect": { "left": 365, @@ -399,10 +345,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='332bc0052f']", - "center": [ - 640, - 530 - ], + "center": [640, 530], "content": "Part of TodoMVC", "rect": { "left": 365, @@ -420,10 +363,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='70ba39c5c6']", - "center": [ - 656, - 530 - ], + "center": [656, 530], "content": "TodoMVC", "rect": { "left": 632, @@ -432,4 +372,4 @@ "height": 13 } } -] \ No newline at end of file +] diff --git a/packages/midscene/tests/ai-model/inspector/test-data/visualstudio/element-snapshot.json b/packages/midscene/tests/ai-model/inspector/test-data/visualstudio/element-snapshot.json index 826ee887..b3ff1ec5 100644 --- a/packages/midscene/tests/ai-model/inspector/test-data/visualstudio/element-snapshot.json +++ b/packages/midscene/tests/ai-model/inspector/test-data/visualstudio/element-snapshot.json @@ -8,10 +8,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='bd82c77920']", - "center": [ - 640, - 29 - ], + "center": [640, 29], "content": "Visual Studio Code\n \t\t\t\t\t\n \t \n \t \n \t \n \t\t \t\t\n \t \n \t \n \t \n \t Docs\n \t Updates\n \t Blog\n \t API\n Extensions\n \t FAQ\n Learn\n \t Search\n \t \n \n \t \n \t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \t\t\t\t\t\t\n \t \n \t \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\t\n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \t \n \t \n \t \n \t\t\t\t\t\tDownload", "rect": { "left": 0, @@ -31,10 +28,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='bd82c77920']", - "center": [ - 640, - 29 - ], + "center": [640, 29], "content": "Visual Studio Code\n \t\t\t\t\t\n \t \n \t \n \t \n \t\t \t\t\n \t \n \t \n \t \n \t Docs\n \t Updates\n \t Blog\n \t API\n Extensions\n \t FAQ\n Learn\n \t Search\n \t \n \n \t \n \t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \t\t\t\t\t\t\n \t \n \t \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\t\n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \t \n \t \n \t \n \t\t\t\t\t\tDownload", "rect": { "left": 0, @@ -53,10 +47,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='bd82c77920']", - "center": [ - 640, - 29 - ], + "center": [640, 29], "content": "Visual Studio Code\n \t\t\t\t\t\n \t \n \t \n \t \n \t\t \t\t\n \t \n \t \n \t \n \t Docs\n \t Updates\n \t Blog\n \t API\n Extensions\n \t FAQ\n Learn\n \t Search\n \t \n \n \t \n \t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \t\t\t\t\t\t\n \t \n \t \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\t\n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \t \n \t \n \t \n \t\t\t\t\t\tDownload", "rect": { "left": 0, @@ -74,10 +65,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='4a1d2260db']", - "center": [ - 640, - 29 - ], + "center": [640, 29], "content": "Visual Studio Code\n \t\t\t\t\t\n \t \n \t \n \t \n \t\t \t\t\n \t \n \t \n \t \n \t Docs\n \t Updates\n \t Blog\n \t API\n Extensions\n \t FAQ\n Learn\n \t Search\n \t \n \n \t \n \t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \t\t\t\t\t\t\n \t \n \t \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\t\n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \t \n \t \n \t \n \t\t\t\t\t\tDownload", "rect": { "left": 40, @@ -95,10 +83,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='ae5debb6cf']", - "center": [ - 191, - 29 - ], + "center": [191, 29], "content": "Visual Studio Code", "rect": { "left": 80, @@ -115,10 +100,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='a482926fdd']", - "center": [ - 201, - 29 - ], + "center": [201, 29], "content": "Visual Studio Code", "rect": { "left": 116, @@ -137,10 +119,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='894271f682']", - "center": [ - 640, - 29 - ], + "center": [640, 29], "content": "Docs\n \t Updates\n \t Blog\n \t API\n Extensions\n \t FAQ\n Learn\n \t Search\n \t \n \n \t \n \t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\n \t\t\t\t\t\t\n \t \n \t \n \t\t\t\t\t\t\t\t\n \t\t\t\t\t\t\t\t\n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \n \t \t \n \t \n \t \n \t\t\t\t\t\tDownload", "rect": { "left": 80, @@ -158,10 +137,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='38057e67de']", - "center": [ - 541, - 29 - ], + "center": [541, 29], "content": "Docs\n \t Updates\n \t Blog\n \t API\n Extensions\n \t FAQ\n Learn\n \t Search", "rect": { "left": 301, @@ -180,10 +156,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='8b9d1a6179']", - "center": [ - 332, - 29 - ], + "center": [332, 29], "content": "Docs", "rect": { "left": 301, @@ -202,10 +175,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='42af1c2281']", - "center": [ - 407, - 29 - ], + "center": [407, 29], "content": "Updates", "rect": { "left": 363, @@ -224,10 +194,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='2b762b1507']", - "center": [ - 478, - 29 - ], + "center": [478, 29], "content": "Blog", "rect": { "left": 449, @@ -246,10 +213,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='fbd4626b77']", - "center": [ - 532, - 29 - ], + "center": [532, 29], "content": "API", "rect": { "left": 507, @@ -270,10 +234,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='839a7fdd30']", - "center": [ - 609, - 29 - ], + "center": [609, 29], "content": "Extensions", "rect": { "left": 556, @@ -292,10 +253,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='c292124789']", - "center": [ - 688, - 29 - ], + "center": [688, 29], "content": "FAQ", "rect": { "left": 660, @@ -314,10 +272,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='606008ee28']", - "center": [ - 748, - 29 - ], + "center": [748, 29], "content": "Learn", "rect": { "left": 715, @@ -336,10 +291,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='226cfbfa70']", - "center": [ - 1025, - 29 - ], + "center": [1025, 29], "content": "Download", "rect": { "left": 849, @@ -366,10 +318,7 @@ "width": 32, "height": 32 }, - "center": [ - 865, - 29 - ] + "center": [865, 29] }, { "id": "ffbf775768", @@ -389,10 +338,7 @@ "width": 20, "height": 20 }, - "center": [ - 865, - 29 - ] + "center": [865, 29] }, { "id": "0c513c2cab", @@ -414,10 +360,7 @@ "width": 200, "height": 32 }, - "center": [ - 985, - 29 - ] + "center": [985, 29] }, { "id": "d76390b753", @@ -438,10 +381,7 @@ "width": 28, "height": 20 }, - "center": [ - 906, - 29 - ] + "center": [906, 29] }, { "id": "eec7cedece", @@ -461,10 +401,7 @@ "width": 16, "height": 16 }, - "center": [ - 906, - 30 - ] + "center": [906, 30] }, { "id": "23914e93fd", @@ -477,10 +414,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='23914e93fd']", - "center": [ - 1147, - 29 - ], + "center": [1147, 29], "content": "Download", "rect": { "left": 1093, @@ -497,10 +431,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='7a1a2d8f0e']", - "center": [ - 1147, - 30 - ], + "center": [1147, 30], "content": "Download", "rect": { "left": 1109, @@ -518,10 +449,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='398138147f']", - "center": [ - 640, - 85 - ], + "center": [640, 85], "content": "Version 1.91 is now available! Read about the new features and fixes from June.\n \n Dismiss this update", "rect": { "left": 0, @@ -539,10 +467,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='c82d77bcb3']", - "center": [ - 624, - 85 - ], + "center": [624, 85], "content": "Version 1.91 is now available! Read about the new features and fixes from June.", "rect": { "left": 24, @@ -560,10 +485,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='553bf30386']", - "center": [ - 624, - 85 - ], + "center": [624, 85], "content": "Version 1.91 is now available! Read about the new features and fixes from June.", "rect": { "left": 64, @@ -582,10 +504,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='c4a6236bc9']", - "center": [ - 404, - 86 - ], + "center": [404, 86], "content": "Version 1.91", "rect": { "left": 364, @@ -603,10 +522,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='79cb132d74']", - "center": [ - 640, - 394 - ], + "center": [640, 394], "content": "Free. Built on open source. Runs everywhere.\n Code Editing. Redefined.\n \n \n \n \n Download for macOS\n Download for Windows\n \n \n .debDebian, Ubuntu...\n .rpmRed Hat, Fedora...\n \n \n \n .deb (x86)\n .rpm (x86)\n .tar.gz (x86)\n \n \n \n DownloadStable Build\n \n \n \n Web, Insiders edition, or other platforms\n By using VS Code, you agree to its license and privacy statement.", "rect": { "left": 40, @@ -624,10 +540,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='40e6911724']", - "center": [ - 240, - 394 - ], + "center": [240, 394], "content": "Free. Built on open source. Runs everywhere.\n Code Editing. Redefined.\n \n \n \n \n Download for macOS\n Download for Windows\n \n \n .debDebian, Ubuntu...\n .rpmRed Hat, Fedora...\n \n \n \n .deb (x86)\n .rpm (x86)\n .tar.gz (x86)\n \n \n \n DownloadStable Build\n \n \n \n Web, Insiders edition, or other platforms\n By using VS Code, you agree to its license and privacy statement.", "rect": { "left": 40, @@ -645,10 +558,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='aed37e3385']", - "center": [ - 246, - 195 - ], + "center": [246, 195], "content": "Free. Built on open source. Runs everywhere.", "rect": { "left": 80, @@ -665,10 +575,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='fad8ef3f4d']", - "center": [ - 240, - 337 - ], + "center": [240, 337], "content": "Code Editing. Redefined.", "rect": { "left": 80, @@ -686,10 +593,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='2f9412a521']", - "center": [ - 240, - 543 - ], + "center": [240, 543], "content": "Download for macOS\n Download for Windows\n \n \n .debDebian, Ubuntu...\n .rpmRed Hat, Fedora...\n \n \n \n .deb (x86)\n .rpm (x86)\n .tar.gz (x86)\n \n \n \n DownloadStable Build\n \n \n \n Web, Insiders edition, or other platforms\n By using VS Code, you agree to its license and privacy statement.", "rect": { "left": 80, @@ -708,10 +612,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='d79e1a9930']", - "center": [ - 240, - 499 - ], + "center": [240, 499], "content": "Download for macOS\n Download for Windows\n \n \n .debDebian, Ubuntu...\n .rpmRed Hat, Fedora...\n \n \n \n .deb (x86)\n .rpm (x86)\n .tar.gz (x86)\n \n \n \n DownloadStable Build", "rect": { "left": 80, @@ -729,10 +630,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='d79e1a9930']", - "center": [ - 240, - 499 - ], + "center": [240, 499], "content": "Download for macOS\n Download for Windows\n \n \n .debDebian, Ubuntu...\n .rpmRed Hat, Fedora...\n \n \n \n .deb (x86)\n .rpm (x86)\n .tar.gz (x86)\n \n \n \n DownloadStable Build", "rect": { "left": 80, @@ -759,10 +657,7 @@ "width": 241, "height": 50 }, - "center": [ - 201, - 499 - ] + "center": [201, 499] }, { "id": "c755a574e2", @@ -774,10 +669,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='c755a574e2']", - "center": [ - 240, - 551 - ], + "center": [240, 551], "content": "Web, Insiders edition, or other platforms", "rect": { "left": 80, @@ -799,10 +691,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='ea2585add6']", - "center": [ - 95, - 552 - ], + "center": [95, 552], "content": "Web", "rect": { "left": 80, @@ -821,10 +710,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='34a6664272']", - "center": [ - 168, - 552 - ], + "center": [168, 552], "content": "Insiders edition", "rect": { "left": 117, @@ -842,10 +728,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='1ef8370ece']", - "center": [ - 295, - 552 - ], + "center": [295, 552], "content": "other platforms", "rect": { "left": 244, @@ -863,10 +746,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='fc5a82d77a']", - "center": [ - 240, - 596 - ], + "center": [240, 596], "content": "By using VS Code, you agree to its license and privacy statement.", "rect": { "left": 80, @@ -884,10 +764,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='62187837c8']", - "center": [ - 160, - 604 - ], + "center": [160, 604], "content": "license and privacy statement.", "rect": { "left": 80, @@ -908,10 +785,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='deba00f856']", - "center": [ - 99, - 604 - ], + "center": [99, 604], "content": "license", "rect": { "left": 80, @@ -932,10 +806,7 @@ "nodeType": "TEXT Node" }, "locator": "[_midscene_retrieve_task_id='324717c5ff']", - "center": [ - 190, - 604 - ], + "center": [190, 604], "content": "privacy statement", "rect": { "left": 143, @@ -962,9 +833,6 @@ "width": 720, "height": 457 }, - "center": [ - 840, - 395 - ] + "center": [840, 395] } -] \ No newline at end of file +] diff --git a/packages/midscene/tests/ai-model/inspector/todo_inspector.test.ts b/packages/midscene/tests/ai-model/inspector/todo_inspector.test.ts index fde5a6ab..196c8766 100644 --- a/packages/midscene/tests/ai-model/inspector/todo_inspector.test.ts +++ b/packages/midscene/tests/ai-model/inspector/todo_inspector.test.ts @@ -1,7 +1,12 @@ -import path from 'path'; -import { it, expect } from 'vitest'; -import { getPageTestData, repeat, runTestCases, writeFileSyncWithDir } from './util'; +import path from 'node:path'; import { AiInspectElement } from '@/ai-model'; +import { expect, it } from 'vitest'; +import { + getPageTestData, + repeat, + runTestCases, + writeFileSyncWithDir, +} from './util'; const testTodoCases = [ { @@ -30,22 +35,32 @@ repeat(2, (repeatIndex) => { it( 'todo: inspect element', async () => { - const { context } = await getPageTestData(path.join(__dirname, './test-data/todo')); + const { context } = await getPageTestData( + path.join(__dirname, './test-data/todo'), + ); - const { aiResponse, filterUnStableinf } = await runTestCases(testTodoCases, async (testCase) => { - const { parseResult } = await AiInspectElement({ - context, - multi: testCase.multi, - findElementDescription: testCase.description, - }); - return parseResult; - }); + const { aiResponse, filterUnStableinf } = await runTestCases( + testTodoCases, + async (testCase) => { + const { parseResult } = await AiInspectElement({ + context, + multi: testCase.multi, + findElementDescription: testCase.description, + }); + return parseResult; + }, + ); writeFileSyncWithDir( - path.join(__dirname, `__ai_responses__/todo-inspector-element-${repeatIndex}.json`), + path.join( + __dirname, + `__ai_responses__/todo-inspector-element-${repeatIndex}.json`, + ), JSON.stringify(aiResponse, null, 2), { encoding: 'utf-8' }, ); - expect(filterUnStableinf).toMatchFileSnapshot('./__snapshots__/todo_inspector.test.ts.snap'); + expect(filterUnStableinf).toMatchFileSnapshot( + './__snapshots__/todo_inspector.test.ts.snap', + ); }, { timeout: 99999, diff --git a/packages/midscene/tests/ai-model/inspector/util.ts b/packages/midscene/tests/ai-model/inspector/util.ts index 30d8ca5c..da180131 100644 --- a/packages/midscene/tests/ai-model/inspector/util.ts +++ b/packages/midscene/tests/ai-model/inspector/util.ts @@ -1,137 +1,135 @@ -import path from 'path'; -import { writeFileSync, existsSync, mkdirSync, readFileSync } from 'fs'; -import { base64Encoded, imageInfoOfBase64 } from '@/image'; +import { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs'; +import path from 'node:path'; import { describeUserPage } from '@/ai-model'; +import { base64Encoded, imageInfoOfBase64 } from '@/image'; type TestCase = { - description: string; - multi: boolean + description: string; + multi: boolean; }; - export interface AiElementsResponse { - "elements": Array<{ - "id": string, - "reason": string, - "text": string, - }> + elements: Array<{ + id: string; + reason: string; + text: string; + }>; } export interface TextAiElementResponse extends AiElementsResponse { - // for test - "caseIndex"?: number; - "prompt": string; - "error"?: string, - "spendTime": string, + // for test + caseIndex?: number; + prompt: string; + error?: string; + spendTime: string; } export async function runTestCases( - testCases: Array, - getAiResponse: (options: { - description: string; - multi: boolean; - }) => (Promise) + testCases: Array, + getAiResponse: (options: { + description: string; + multi: boolean; + }) => Promise, ) { - let aiResponse: Array = []; - - const aiReq = testCases.map(async (testCase, caseIndex)=>{ - const startTime = Date.now(); - const msg = await getAiResponse(testCase); - const endTime = Date.now(); - const spendTime = (endTime - startTime)/1000; - if (msg.elements) { - aiResponse.push({ - ...msg, - prompt: testCase.description, - caseIndex, - spendTime: `${spendTime}s` - }); - } else { - aiResponse.push({ - error: `can't find element with description: ${testCase.description}` - } as any); - } - }); - await Promise.all(aiReq); - aiResponse = aiResponse.sort((a,b)=> { - if (a.caseIndex !== undefined && b.caseIndex !== undefined) { - return a.caseIndex - b.caseIndex; - } else { - return -1; - } - }); - - aiResponse.forEach((item)=>{ - if ('caseIndex' in item) { - delete item.caseIndex - } - }); + let aiResponse: Array = []; + + const aiReq = testCases.map(async (testCase, caseIndex) => { + const startTime = Date.now(); + const msg = await getAiResponse(testCase); + const endTime = Date.now(); + const spendTime = (endTime - startTime) / 1000; + if (msg.elements) { + aiResponse.push({ + ...msg, + prompt: testCase.description, + caseIndex, + spendTime: `${spendTime}s`, + }); + } else { + aiResponse.push({ + error: `can't find element with description: ${testCase.description}`, + } as any); + } + }); + await Promise.all(aiReq); + aiResponse = aiResponse.sort((a, b) => { + if (a.caseIndex !== undefined && b.caseIndex !== undefined) { + return a.caseIndex - b.caseIndex; + } + return -1; + }); - const filterUnStableinf = aiResponse.map((aiInfo)=>{ - const { elements = [] , prompt, error = []} = aiInfo; - return { - elements: elements.map((element)=> { - return { - id: element.id.toString(), - }; - }), - prompt, - error, - } - }); + aiResponse.forEach((item) => { + if ('caseIndex' in item) { + item.caseIndex = undefined; + } + }); + const filterUnStableinf = aiResponse.map((aiInfo) => { + const { elements = [], prompt, error = [] } = aiInfo; return { - aiResponse, - filterUnStableinf + elements: elements.map((element) => { + return { + id: element.id.toString(), + }; + }), + prompt, + error, }; -} - + }); + return { + aiResponse, + filterUnStableinf, + }; +} export const repeat = (times: number, fn: (index: number) => void) => { - for (let i = 1; i <= times; i++) { - fn(i); - } + for (let i = 1; i <= times; i++) { + fn(i); + } }; - function ensureDirectoryExistence(filePath: string) { - const dirname = path.dirname(filePath); - if (existsSync(dirname)) { - return true; - } - ensureDirectoryExistence(dirname); - mkdirSync(dirname); + const dirname = path.dirname(filePath); + if (existsSync(dirname)) { + return true; + } + ensureDirectoryExistence(dirname); + mkdirSync(dirname); } type WriteFileSyncParams = Parameters; -export function writeFileSyncWithDir(filePath: string, content: WriteFileSyncParams[1], options: WriteFileSyncParams[2] = {}) { - ensureDirectoryExistence(filePath); - writeFileSync(filePath, content, options); +export function writeFileSyncWithDir( + filePath: string, + content: WriteFileSyncParams[1], + options: WriteFileSyncParams[2] = {}, +) { + ensureDirectoryExistence(filePath); + writeFileSync(filePath, content, options); } -export async function getPageTestData(targetDir: string){ - const resizeOutputImgP = path.join(targetDir, 'input.png'); - const snapshotJsonPath = path.join(targetDir, 'element-snapshot.json'); - const snapshotJson = readFileSync(snapshotJsonPath, { encoding: 'utf-8'}); - const screenshotBase64 = base64Encoded(resizeOutputImgP); - const size = await imageInfoOfBase64(screenshotBase64); - const baseContext = { - size, - content: JSON.parse(snapshotJson), - screenshotBase64: base64Encoded(resizeOutputImgP), - }; - - return { - context: { - ...baseContext, - describer: async () => { - return describeUserPage(baseContext); - }, - }, - snapshotJson, - screenshotBase64: base64Encoded(resizeOutputImgP) - }; +export async function getPageTestData(targetDir: string) { + const resizeOutputImgP = path.join(targetDir, 'input.png'); + const snapshotJsonPath = path.join(targetDir, 'element-snapshot.json'); + const snapshotJson = readFileSync(snapshotJsonPath, { encoding: 'utf-8' }); + const screenshotBase64 = base64Encoded(resizeOutputImgP); + const size = await imageInfoOfBase64(screenshotBase64); + const baseContext = { + size, + content: JSON.parse(snapshotJson), + screenshotBase64: base64Encoded(resizeOutputImgP), + }; + + return { + context: { + ...baseContext, + describer: async () => { + return describeUserPage(baseContext); + }, + }, + snapshotJson, + screenshotBase64: base64Encoded(resizeOutputImgP), + }; } - diff --git a/packages/midscene/tests/ai-model/openai.test.ts b/packages/midscene/tests/ai-model/openai.test.ts index 71ea968a..a1b1ab11 100644 --- a/packages/midscene/tests/ai-model/openai.test.ts +++ b/packages/midscene/tests/ai-model/openai.test.ts @@ -1,35 +1,41 @@ import { call, callToGetJSONObject } from '@/ai-model/openai'; import { AIResponseFormat } from '@/types'; -import { describe, it, expect, vi } from 'vitest'; +import { describe, expect, it, vi } from 'vitest'; vi.setConfig({ testTimeout: 20 * 1000, }); describe('openai', () => { it('basic', async () => { - const result = await call([ - { - role: 'system', - content: 'Answer the question', - }, { - role: 'user', - content: '鲁迅认识周树人吗?回答我:1. 分析原因 2.回答:是/否/无效问题', - }, - ], AIResponseFormat.TEXT); + const result = await call( + [ + { + role: 'system', + content: 'Answer the question', + }, + { + role: 'user', + content: + '鲁迅认识周树人吗?回答我:1. 分析原因 2.回答:是/否/无效问题', + }, + ], + AIResponseFormat.TEXT, + ); expect(result.length).toBeGreaterThan(1); }); - + it('call to get json result', async () => { - const result = await callToGetJSONObject<{answer: number}>([ + const result = await callToGetJSONObject<{ answer: number }>([ { role: 'system', content: 'Answer the question with JSON: {answer: number}', - }, { + }, + { role: 'user', content: '3 x 5 = ?', }, - ]); + ]); expect(result.answer).toBe(15); }); -}); \ No newline at end of file +}); diff --git a/packages/midscene/tests/automation/planning.test.ts b/packages/midscene/tests/automation/planning.test.ts index 16475589..af338e23 100644 --- a/packages/midscene/tests/automation/planning.test.ts +++ b/packages/midscene/tests/automation/planning.test.ts @@ -14,7 +14,7 @@ // const localPage = `file://${getFixture('simple.html')}`; // describe('automation - planning', () => { // let browser: Browser; -// beforeEach(() => +// beforeEach(() => // async () => { // await browser?.close(); // }, diff --git a/packages/midscene/tests/executor/index.test.ts b/packages/midscene/tests/executor/index.test.ts index 459434cb..12334719 100644 --- a/packages/midscene/tests/executor/index.test.ts +++ b/packages/midscene/tests/executor/index.test.ts @@ -1,13 +1,13 @@ -import { it, describe, expect, vi } from 'vitest'; -import { fakeInsight } from 'tests/utils'; -import { +import { Executor } from '@/action/executor'; +import type { DumpSubscriber, ExecutionTaskActionApply, ExecutionTaskInsightLocate, ExecutionTaskInsightLocateApply, InsightDump, } from '@/index'; -import { Executor } from '@/action/executor'; +import { fakeInsight } from 'tests/utils'; +import { describe, expect, it, vi } from 'vitest'; const insightFindTask = (shouldThrow?: boolean) => { let insightDump: InsightDump | undefined; @@ -90,16 +90,20 @@ describe('executor', () => { const inputTasks = [insightTask1, actionTask]; - const executor = new Executor('test', 'hello, this is a test', inputTasks); + const executor = new Executor( + 'test', + 'hello, this is a test', + inputTasks, + ); await executor.flush(); const tasks = executor.tasks as ExecutionTaskInsightLocate[]; - const { element } = tasks[0].output!; + const { element } = tasks[0].output || {}; expect(element).toBeTruthy(); expect(tasks.length).toBe(inputTasks.length); expect(tasks[0].status).toBe('success'); expect(tasks[0].output).toMatchSnapshot(); - expect(tasks[0].log!.dump).toBeTruthy(); + expect(tasks[0].log?.dump).toBeTruthy(); expect(tasks[0].timing?.end).toBeTruthy(); expect(tasks[0].cache).toBeTruthy(); expect(tasks[0].cache?.hit).toEqual(false); diff --git a/packages/midscene/tests/fixtures/dump.json b/packages/midscene/tests/fixtures/dump.json index cab43a1e..ded55651 100644 --- a/packages/midscene/tests/fixtures/dump.json +++ b/packages/midscene/tests/fixtures/dump.json @@ -1387,4 +1387,4 @@ } ] } -} \ No newline at end of file +} diff --git a/packages/midscene/tests/image/index.test.ts b/packages/midscene/tests/image/index.test.ts index 474e8ea0..76be2e18 100644 --- a/packages/midscene/tests/image/index.test.ts +++ b/packages/midscene/tests/image/index.test.ts @@ -1,6 +1,12 @@ -import { alignCoordByTrim, base64Encoded, imageInfo, imageInfoOfBase64, trimImage } from '@/image'; +import { + alignCoordByTrim, + base64Encoded, + imageInfo, + imageInfoOfBase64, + trimImage, +} from '@/image'; import { getFixture } from 'tests/utils'; -import { describe, it, expect } from 'vitest'; +import { describe, expect, it } from 'vitest'; describe('image utils', () => { const image = getFixture('icon.png'); @@ -49,10 +55,14 @@ describe('image utils', () => { expect(small).toBeNull(); }); - it('align a sub-image', async () => { const file = getFixture('long-text.png'); - const rect = await alignCoordByTrim(file, { left: 140, top: 50, width: 200, height: 80 }); + const rect = await alignCoordByTrim(file, { + left: 140, + top: 50, + width: 200, + height: 80, + }); expect(rect).toMatchSnapshot(); }); -}); \ No newline at end of file +}); diff --git a/packages/midscene/tests/insight/index.test.ts b/packages/midscene/tests/insight/index.test.ts index 60b24a1f..51207054 100644 --- a/packages/midscene/tests/insight/index.test.ts +++ b/packages/midscene/tests/insight/index.test.ts @@ -27,7 +27,7 @@ // const localPage = `file://${getFixture('simple.html')}`; // describe('insight - basic', () => { // let browser: Browser; -// beforeEach(() => +// beforeEach(() => // async () => { // await browser?.close(); // }, @@ -77,12 +77,11 @@ // await insight.locate('another find'); // expect(dumpCollector.mock.calls.length).toBe(2); // only be called once - // const dumpCollector2 = vi.fn(); // insight.onceDumpUpdatedFn = dumpCollector2; // await insight.extract('should fail'); // expect(dumpCollector2.mock.calls.length).toBe(2); - + // await insight.extract('should fail'); // expect(dumpCollector2.mock.calls.length).toBe(2); // }); @@ -109,7 +108,7 @@ // try { // await insight.extract('something'); // } catch(e) { -// // +// // // } // const logContent2 = readFileSync(join(getDumpDir(), './latest.insight.json'), 'utf-8'); @@ -184,7 +183,7 @@ // describe('find', () => { // const vscodeSite = 'https://code.visualstudio.com/'; // let browser: Browser; -// beforeEach(() => +// beforeEach(() => // async () => { // await browser?.close(); // }, @@ -196,7 +195,7 @@ // const btn = await insight.locate('the main download button on the page'); // expect(btn).toBeTruthy(); // expect(/download/i.test(btn!.content)).toBeTruthy(); - + // const btnWithProperties = await insight.locate('all the download buttons on the page', {multi: true}); // expect(btnWithProperties.length).toBeGreaterThanOrEqual(2); // }); @@ -204,7 +203,7 @@ // describe('extract', () => { // let browser: Browser; -// beforeEach(() => +// beforeEach(() => // async () => { // await browser?.close(); // }, @@ -228,7 +227,7 @@ // // types mismatch, should raise an error // await insight.extract<{dataItem: number}>({ -// foo: 'abcde', // should be error +// foo: 'abcde', // should be error // }); // [ // result, // should be: any diff --git a/packages/midscene/tests/insight/prompts.test.ts b/packages/midscene/tests/insight/prompts.test.ts index 748ec08c..3e7e6355 100644 --- a/packages/midscene/tests/insight/prompts.test.ts +++ b/packages/midscene/tests/insight/prompts.test.ts @@ -41,11 +41,11 @@ // it('system prompt for extract, wo/ section', () => { // const prompt = systemPromptToExtract('help me to find out'); // expect(prompt).toMatchSnapshot(); - + // const prompt2 = systemPromptToExtract({foo: 'tell me the color of sea'}); // expect(prompt2).toMatchSnapshot(); // }); - + // it('system prompt for extract, w/ section', () => { // const prompt = systemPromptToExtract('find something', sectionQuery1); // expect(prompt).toMatchSnapshot(); diff --git a/packages/midscene/tests/insight/utils.test.ts b/packages/midscene/tests/insight/utils.test.ts index dd8639dc..336e933a 100644 --- a/packages/midscene/tests/insight/utils.test.ts +++ b/packages/midscene/tests/insight/utils.test.ts @@ -31,7 +31,7 @@ // rect: { top: 0, left: 0, width: 100, height: 100 }, // center: [0,0] as [number, number], // })); - + // const elementIds = ['1', '2', '3']; // const elements = idsIntoElements(elementIds, elementById); // expect(elements).toMatchSnapshot(); @@ -60,7 +60,7 @@ // } // return null; // }; - + // const data1 = { // title: 'title', // element: '9', @@ -117,7 +117,6 @@ // expect(aligned).toMatchSnapshot(); // }); - // it('get context of online context', async () => { // const browser = await launch('https://www.baidu.com'); // const context = await parseContextFromPuppeteerBrowser(browser); @@ -130,7 +129,7 @@ // // append // const logId = writeInsightDump(mockDumpData, tmpDir); // expect(typeof logId).toBe('string'); - + // const getLatestContent = () => { // const file = join(tmpDir, 'latest.insight.json'); // const content = JSON.parse(readFileSync(file, 'utf-8')); @@ -157,4 +156,3 @@ // }); // }); - diff --git a/packages/midscene/tests/query.test.ts b/packages/midscene/tests/query.test.ts index 38ea372e..5a925112 100644 --- a/packages/midscene/tests/query.test.ts +++ b/packages/midscene/tests/query.test.ts @@ -1,6 +1,6 @@ +import { pageScriptToGetTexts } from '@/query'; /* eslint-disable @typescript-eslint/no-implied-eval */ import { describe, it } from 'vitest'; -import { pageScriptToGetTexts } from '@/query'; describe('query', () => { it('make sure scripts are valid', () => { diff --git a/packages/midscene/tests/utils.test.ts b/packages/midscene/tests/utils.test.ts index 1f444f70..fd135a67 100644 --- a/packages/midscene/tests/utils.test.ts +++ b/packages/midscene/tests/utils.test.ts @@ -1,12 +1,18 @@ -import { getDumpDir, getTmpDir, getTmpFile, overlapped, setDumpDir } from '@/utils'; -import { tmpdir } from 'os'; -import { describe, it, expect } from 'vitest'; +import { tmpdir } from 'node:os'; +import { + getDumpDir, + getTmpDir, + getTmpFile, + overlapped, + setDumpDir, +} from '@/utils'; +import { describe, expect, it } from 'vitest'; describe('utils', () => { it('tmpDir', () => { const testDir = getTmpDir(); expect(typeof testDir).toBe('string'); - + const testFile = getTmpFile('txt'); expect(testFile.endsWith('.txt')).toBe(true); }); @@ -28,5 +34,4 @@ describe('utils', () => { const target2 = { left: 200, top: 200, width: 100, height: 100 }; expect(overlapped(container, target2)).toBeFalsy(); }); - -}); \ No newline at end of file +}); diff --git a/packages/midscene/tests/utils.ts b/packages/midscene/tests/utils.ts index 7823825f..459be7c5 100644 --- a/packages/midscene/tests/utils.ts +++ b/packages/midscene/tests/utils.ts @@ -1,10 +1,14 @@ +import { readFileSync, writeFileSync } from 'node:fs'; /* eslint-disable @typescript-eslint/no-magic-numbers */ -import path, { join } from 'path'; -import { readFileSync, writeFileSync } from 'fs'; -import { base64Encoded, imageInfoOfBase64, transformImgPathToBase64 } from '@/image'; -import { vi } from 'vitest'; -import { BaseElement, UIContext } from '@/types'; +import path, { join } from 'node:path'; +import { + base64Encoded, + imageInfoOfBase64, + transformImgPathToBase64, +} from '@/image'; import Insight from '@/insight'; +import type { BaseElement, UIContext } from '@/types'; +import { vi } from 'vitest'; export function getFixture(name: string) { return join(__dirname, 'fixtures', name); @@ -29,19 +33,19 @@ export function fakeInsight(content: string) { screenshotBase64: base64Encoded(screenshot), size: { width: 1920, height: 1080 }, content: [ - { - id: '0', - content, - rect: { - width: 100, - height: 100, - top: 200, - left: 200, - }, - center: [250, 250], - tap: vi.fn() as unknown, + { + id: '0', + content, + rect: { + width: 100, + height: 100, + top: 200, + left: 200, }, - // describer: basicPa + center: [250, 250], + tap: vi.fn() as unknown, + }, + // describer: basicPa ] as unknown as BaseElement[], }; const context: UIContext = { @@ -53,23 +57,24 @@ export function fakeInsight(content: string) { errors: [], }); - const insight = new Insight( - context, - { - aiVendorFn: aiVendor as any, - }, - ); + const insight = new Insight(context, { + aiVendorFn: aiVendor as any, + }); return insight; } - export function generateUIContext(testDataPath: string) { - return async ()=> { - const screenshotBase64 = await transformImgPathToBase64(path.join(testDataPath, 'input.png')); + return async () => { + const screenshotBase64 = await transformImgPathToBase64( + path.join(testDataPath, 'input.png'), + ); const size = await imageInfoOfBase64(screenshotBase64); - const captureElementSnapshot = readFileSync(path.join(testDataPath, 'element-snapshot.json'), 'utf-8'); + const captureElementSnapshot = readFileSync( + path.join(testDataPath, 'element-snapshot.json'), + 'utf-8', + ); // align element const elementsInfo = JSON.parse(captureElementSnapshot) as BaseElement[]; @@ -82,6 +87,6 @@ export function generateUIContext(testDataPath: string) { return { ...baseContext, - } + }; }; } diff --git a/packages/midscene/vitest.config.ts b/packages/midscene/vitest.config.ts index 16ee4691..bfec2c6c 100644 --- a/packages/midscene/vitest.config.ts +++ b/packages/midscene/vitest.config.ts @@ -1,16 +1,21 @@ +import path from 'node:path'; import { defineConfig } from 'vitest/config'; -import path from 'path'; const enableTest = process.env.AITEST; -const aiModelTest = enableTest !== 'true' ? ['tests/ai-model/**/*.test.ts']: []; +const aiModelTest = + enableTest !== 'true' ? ['tests/ai-model/**/*.test.ts'] : []; export default defineConfig({ test: { // include: ['tests/inspector/*.test.ts'], include: ['tests/**/*.test.ts'], // Need to improve the corresponding testing - exclude: ['tests/insight/*.test.ts', 'tests/automation/planning.test.ts', ...aiModelTest] + exclude: [ + 'tests/insight/*.test.ts', + 'tests/automation/planning.test.ts', + ...aiModelTest, + ], }, resolve: { alias: { diff --git a/packages/playwright-demo/e2e/ai-xicha.spec.ts b/packages/playwright-demo/e2e/ai-xicha.spec.ts index b9dacee9..c1873e31 100644 --- a/packages/playwright-demo/e2e/ai-xicha.spec.ts +++ b/packages/playwright-demo/e2e/ai-xicha.spec.ts @@ -1,7 +1,7 @@ import { test } from './fixture'; test.beforeEach(async ({ page }) => { - page.setViewportSize({ width: 400, height: 905}); + page.setViewportSize({ width: 400, height: 905 }); await page.goto('https://heyteavivocity.meuu.online/home'); await page.waitForLoadState('networkidle'); }); diff --git a/packages/playwright-demo/e2e/fixture.ts b/packages/playwright-demo/e2e/fixture.ts index 010e3714..94b4748e 100644 --- a/packages/playwright-demo/e2e/fixture.ts +++ b/packages/playwright-demo/e2e/fixture.ts @@ -1,5 +1,5 @@ -import { test as base } from '@playwright/test'; import type { PlayWrightAiFixtureType } from '@midscene/web'; import { PlaywrightAiFixture } from '@midscene/web'; +import { test as base } from '@playwright/test'; export const test = base.extend(PlaywrightAiFixture()); diff --git a/packages/playwright-demo/modern.config.ts b/packages/playwright-demo/modern.config.ts index c981cedf..93b3e670 100644 --- a/packages/playwright-demo/modern.config.ts +++ b/packages/playwright-demo/modern.config.ts @@ -1,5 +1,5 @@ -import path from 'path'; -import { moduleTools, defineConfig } from '@modern-js/module-tools'; +import path from 'node:path'; +import { defineConfig, moduleTools } from '@modern-js/module-tools'; import { testingPlugin } from '@modern-js/plugin-testing'; export default defineConfig({ @@ -7,7 +7,10 @@ export default defineConfig({ buildPreset: 'npm-library-es2019', buildConfig: { alias: { - '@playwright/test': path.resolve(__dirname, 'node_modules/@playwright/test'), + '@playwright/test': path.resolve( + __dirname, + 'node_modules/@playwright/test', + ), }, }, testing: { diff --git a/packages/playwright-demo/tsconfig.json b/packages/playwright-demo/tsconfig.json index 45049427..b3e6762a 100644 --- a/packages/playwright-demo/tsconfig.json +++ b/packages/playwright-demo/tsconfig.json @@ -18,6 +18,6 @@ "skipLibCheck": true, "strict": true }, - "exclude": [ "node_modules"], + "exclude": ["node_modules"], "include": ["src", "e2e", "./playwright.config.ts", "./modern.config.ts"] } diff --git a/packages/visualizer-report/config/public/test-data-list.json b/packages/visualizer-report/config/public/test-data-list.json index c40011c3..08665e7b 100644 --- a/packages/visualizer-report/config/public/test-data-list.json +++ b/packages/visualizer-report/config/public/test-data-list.json @@ -1,3 +1,40 @@ { - "test-list":[{"testId":"45161835cecba6378a04-b2821fd5751102caa08c","title":"ai todo","status":"passed","duration":22204,"location":{"file":"/Users/bytedance/github/midscene/packages/web-integration/tests/e2e/ai-auto-todo.spec.ts","line":8,"column":5},"dumpPath":"/Users/bytedance/github/midscene/packages/web-integration/midscene_run/playwright-45161835cecba6378a04-b2821fd5751102caa08c.web-dump.json"},{"testId":"31de72c0afc13db9dc09-50c9ddc9a1d0c466547f","title":"ai order2","status":"passed","duration":40848,"location":{"file":"/Users/bytedance/github/midscene/packages/web-integration/tests/e2e/ai-xicha.spec.ts","line":36,"column":5},"dumpPath":"/Users/bytedance/github/midscene/packages/web-integration/midscene_run/playwright-31de72c0afc13db9dc09-50c9ddc9a1d0c466547f.web-dump.json"},{"testId":"31de72c0afc13db9dc09-00e11f768b63da0c779a","title":"ai order","status":"passed","duration":51045,"location":{"file":"/Users/bytedance/github/midscene/packages/web-integration/tests/e2e/ai-xicha.spec.ts","line":9,"column":5},"dumpPath":"/Users/bytedance/github/midscene/packages/web-integration/midscene_run/playwright-31de72c0afc13db9dc09-00e11f768b63da0c779a.web-dump.json"}] -} \ No newline at end of file + "test-list": [ + { + "testId": "45161835cecba6378a04-b2821fd5751102caa08c", + "title": "ai todo", + "status": "passed", + "duration": 22204, + "location": { + "file": "/Users/bytedance/github/midscene/packages/web-integration/tests/e2e/ai-auto-todo.spec.ts", + "line": 8, + "column": 5 + }, + "dumpPath": "/Users/bytedance/github/midscene/packages/web-integration/midscene_run/playwright-45161835cecba6378a04-b2821fd5751102caa08c.web-dump.json" + }, + { + "testId": "31de72c0afc13db9dc09-50c9ddc9a1d0c466547f", + "title": "ai order2", + "status": "passed", + "duration": 40848, + "location": { + "file": "/Users/bytedance/github/midscene/packages/web-integration/tests/e2e/ai-xicha.spec.ts", + "line": 36, + "column": 5 + }, + "dumpPath": "/Users/bytedance/github/midscene/packages/web-integration/midscene_run/playwright-31de72c0afc13db9dc09-50c9ddc9a1d0c466547f.web-dump.json" + }, + { + "testId": "31de72c0afc13db9dc09-00e11f768b63da0c779a", + "title": "ai order", + "status": "passed", + "duration": 51045, + "location": { + "file": "/Users/bytedance/github/midscene/packages/web-integration/tests/e2e/ai-xicha.spec.ts", + "line": 9, + "column": 5 + }, + "dumpPath": "/Users/bytedance/github/midscene/packages/web-integration/midscene_run/playwright-31de72c0afc13db9dc09-00e11f768b63da0c779a.web-dump.json" + } + ] +} diff --git a/packages/visualizer-report/modern.config.ts b/packages/visualizer-report/modern.config.ts index 32646a9f..e2591b02 100644 --- a/packages/visualizer-report/modern.config.ts +++ b/packages/visualizer-report/modern.config.ts @@ -1,4 +1,4 @@ -import path from 'path'; +import path from 'node:path'; import { appTools, defineConfig } from '@modern-js/app-tools'; // https://modernjs.dev/en/configure/app/usage diff --git a/packages/visualizer-report/package.json b/packages/visualizer-report/package.json index 0bdfdc34..05ac9099 100644 --- a/packages/visualizer-report/package.json +++ b/packages/visualizer-report/package.json @@ -11,10 +11,7 @@ "lint": "modern lint", "upgrade": "modern upgrade" }, - "files": [ - "dist", - "README.md" - ], + "files": ["dist", "README.md"], "engines": { "node": ">=16.18.1" }, @@ -23,10 +20,7 @@ "node --max_old_space_size=8192 ./node_modules/eslint/bin/eslint.js --fix --color --cache --quiet" ] }, - "eslintIgnore": [ - "node_modules/", - "dist/" - ], + "eslintIgnore": ["node_modules/", "dist/"], "dependencies": { "@modern-js/runtime": "^2.56.2", "@midscene/visualizer": "workspace:*", diff --git a/packages/visualizer-report/src/App.tsx b/packages/visualizer-report/src/App.tsx index 0e6c2393..57252441 100644 --- a/packages/visualizer-report/src/App.tsx +++ b/packages/visualizer-report/src/App.tsx @@ -1,6 +1,6 @@ import { BrowserRouter, Route, Routes } from '@modern-js/runtime/router'; -import { Report } from './pages/Report'; import { Home } from './pages/Home'; +import { Report } from './pages/Report'; export default () => { return ( diff --git a/packages/visualizer-report/src/pages/Home.tsx b/packages/visualizer-report/src/pages/Home.tsx index 538a83a4..807c13b0 100644 --- a/packages/visualizer-report/src/pages/Home.tsx +++ b/packages/visualizer-report/src/pages/Home.tsx @@ -1,7 +1,8 @@ import { useNavigate } from '@modern-js/runtime/router'; -import React, { useEffect, useState } from 'react'; -import { Menu, Collapse } from 'antd'; -import type { MenuProps, CollapseProps } from 'antd'; +import { Collapse, Menu } from 'antd'; +import type { CollapseProps, MenuProps } from 'antd'; +import type React from 'react'; +import { useEffect, useState } from 'react'; import styeld from './Home.module.css'; import './TestResult.css'; @@ -73,7 +74,9 @@ const TestResult = (props: { }, ) || {}; - const items: CollapseProps['items'] = Object.keys(groupTestDataWithFileName).map((fileName, key) => { + const items: CollapseProps['items'] = Object.keys( + groupTestDataWithFileName, + ).map((fileName, key) => { return { key, label: fileName, @@ -81,8 +84,10 @@ const TestResult = (props: { const timeMinutes = Math.floor(testData.duration / 1000 / 60); const timeSeconds = Math.floor((testData.duration / 1000) % 60); return ( + // biome-ignore lint/a11y/useKeyWithClickEvents:
key={key} onClick={() => { navigator(`/report?dumpId=${testData.dumpPath.split('/').pop()}`); @@ -94,7 +99,8 @@ const TestResult = (props: { {testData.title} - duration: {timeMinutes !== 0 && `${timeMinutes}m`} {timeSeconds && `${timeSeconds}s`} + duration: {timeMinutes !== 0 && `${timeMinutes}m`}{' '} + {timeSeconds && `${timeSeconds}s`}
@@ -161,7 +167,12 @@ export const StatsNavView: React.FC<{ return ( <>
- +
@@ -210,7 +221,10 @@ export function Home() { return (
- +
); } diff --git a/packages/visualizer-report/src/pages/Report.tsx b/packages/visualizer-report/src/pages/Report.tsx index ca03aac6..0898bbdd 100644 --- a/packages/visualizer-report/src/pages/Report.tsx +++ b/packages/visualizer-report/src/pages/Report.tsx @@ -1,6 +1,6 @@ import { Visualizer } from '@midscene/visualizer'; -import React, { useEffect, useState } from 'react'; import { useNavigate } from '@modern-js/runtime/router'; +import React, { useEffect, useState } from 'react'; declare module '@midscene/visualizer' { export function Visualizer(dumpInfo: any): any; @@ -30,7 +30,7 @@ export function Report() { .finally(() => { setLoading(false); }); - }, []); + }, [dumpId]); return (
@@ -44,7 +44,7 @@ export function Report() { /> )} -
+
); diff --git a/packages/visualizer-report/tsconfig.json b/packages/visualizer-report/tsconfig.json index 7fc0dc1c..1e035ca4 100644 --- a/packages/visualizer-report/tsconfig.json +++ b/packages/visualizer-report/tsconfig.json @@ -8,7 +8,7 @@ "@/*": ["./src/*"], "@shared/*": ["./shared/*"] }, - "types": ["react"], + "types": ["react"] }, "include": ["src", "shared", "config", "modern.config.ts"], "exclude": ["**/node_modules"] diff --git a/packages/visualizer/docs/index.tsx b/packages/visualizer/docs/index.tsx index 1d1f8019..40d014de 100644 --- a/packages/visualizer/docs/index.tsx +++ b/packages/visualizer/docs/index.tsx @@ -1,5 +1,5 @@ -import React from 'react'; import Tool from '@/index'; +import React from 'react'; export default () => { // return ; diff --git a/packages/visualizer/modern.config.ts b/packages/visualizer/modern.config.ts index a9df6e8a..19fa5b9c 100644 --- a/packages/visualizer/modern.config.ts +++ b/packages/visualizer/modern.config.ts @@ -1,4 +1,4 @@ -import { moduleTools, defineConfig } from '@modern-js/module-tools'; +import { defineConfig, moduleTools } from '@modern-js/module-tools'; import { modulePluginDoc } from '@modern-js/plugin-module-doc'; export default defineConfig({ diff --git a/packages/visualizer/package.json b/packages/visualizer/package.json index a4a64f69..9e031f84 100644 --- a/packages/visualizer/package.json +++ b/packages/visualizer/package.json @@ -5,10 +5,7 @@ "jsnext:source": "./src/index.ts", "main": "./dist/lib/index.js", "module": "./dist/es/index.js", - "files": [ - "dist", - "README.md" - ], + "files": ["dist", "README.md"], "scripts": { "dev": "modern dev", "build": "modern build", @@ -39,12 +36,7 @@ "rimraf": "~3.0.2", "typescript": "~5.0.4" }, - "sideEffects": [ - "**/*.css", - "**/*.less", - "**/*.sass", - "**/*.scss" - ], + "sideEffects": ["**/*.css", "**/*.less", "**/*.sass", "**/*.scss"], "publishConfig": { "access": "public" } diff --git a/packages/visualizer/src/component/blackboard.tsx b/packages/visualizer/src/component/blackboard.tsx index 4592796c..11bf6d42 100644 --- a/packages/visualizer/src/component/blackboard.tsx +++ b/packages/visualizer/src/component/blackboard.tsx @@ -1,10 +1,10 @@ 'use client'; -import * as PIXI from 'pixi.js'; -import { ReactElement, useEffect, useMemo, useRef, useState } from 'react'; import { Checkbox } from 'antd'; import type { CheckboxProps } from 'antd'; -import { Rect } from '../../../midscene/dist/types'; -import { highlightColorForType, colorForName } from './color'; +import * as PIXI from 'pixi.js'; +import { type ReactElement, useEffect, useMemo, useRef, useState } from 'react'; +import type { Rect } from '../../../midscene/dist/types'; +import { colorForName, highlightColorForType } from './color'; import './blackboard.less'; import { useBlackboardPreference, useInsightDump } from './store'; @@ -17,9 +17,15 @@ const noop = () => { const BlackBoard = (): JSX.Element => { const dump = useInsightDump((store) => store.data); - const setHighlightSectionNames = useInsightDump((store) => store.setHighlightSectionNames); - const setHighlightElements = useInsightDump((store) => store.setHighlightElements); - const highlightSectionNames = useInsightDump((store) => store.highlightSectionNames); + const setHighlightSectionNames = useInsightDump( + (store) => store.setHighlightSectionNames, + ); + const setHighlightElements = useInsightDump( + (store) => store.setHighlightElements, + ); + const highlightSectionNames = useInsightDump( + (store) => store.highlightSectionNames, + ); const highlightElements = useInsightDump((store) => store.highlightElements); const { context, matchedSection: sections, matchedElement: elements } = dump!; @@ -37,7 +43,8 @@ const BlackBoard = (): JSX.Element => { // key overlays const pixiBgRef = useRef(); - const { bgVisible, setBgVisible, textsVisible, setTextsVisible } = useBlackboardPreference(); + const { bgVisible, setBgVisible, textsVisible, setTextsVisible } = + useBlackboardPreference(); useEffect(() => { Promise.resolve( @@ -45,7 +52,11 @@ const BlackBoard = (): JSX.Element => { if (!domRef.current || !screenWidth) { return; } - await app.init({ width: screenWidth, height: screenHeight, background: 0xffffff }); + await app.init({ + width: screenWidth, + height: screenHeight, + background: 0xffffff, + }); const canvasEl = domRef.current; domRef.current.appendChild(app.canvas); // Ensure app.view is appended const { clientWidth } = domRef.current.parentElement!; @@ -141,7 +152,9 @@ const BlackBoard = (): JSX.Element => { const [graphics, texts] = rectMarkForItem( section.rect, section.name, - ifHighlight ? highlightColorForType('section') : colorForName('section', section.name), + ifHighlight + ? highlightColorForType('section') + : colorForName('section', section.name), ifHighlight ? 1 : itemFillAlpha, () => { setHighlightSectionNames([section.name]); @@ -161,7 +174,14 @@ const BlackBoard = (): JSX.Element => { if (elements.includes(element)) { return; } - const [graphics, texts] = rectMarkForItem(rect, '', highlightColorForType('element'), 1, noop, noop); + const [graphics, texts] = rectMarkForItem( + rect, + '', + highlightColorForType('element'), + 1, + noop, + noop, + ); itemMarkContainer.addChild(graphics); textContainer.addChild(texts); }); @@ -173,7 +193,9 @@ const BlackBoard = (): JSX.Element => { const [graphics, texts] = rectMarkForItem( rect, content, - ifHighlight ? highlightColorForType('element') : colorForName('element', content), + ifHighlight + ? highlightColorForType('element') + : colorForName('element', content), ifHighlight ? 1 : itemFillAlpha, () => { setHighlightElements([element]); @@ -190,7 +212,9 @@ const BlackBoard = (): JSX.Element => { const { content } = section; const ifHighlight = highlightSectionNames.includes(section.name); - const sectionTheme = ifHighlight ? '#FFFFFF' : colorForName('section', section.name); + const sectionTheme = ifHighlight + ? '#FFFFFF' + : colorForName('section', section.name); content.forEach((text) => { const { content, rect } = text; @@ -245,13 +269,17 @@ const BlackBoard = (): JSX.Element => { if (highlightElementRects.length === 1) { bottomTipA = (
-
Element: {JSON.stringify(highlightElementRects[0])}
+
+ Element: {JSON.stringify(highlightElementRects[0])} +
); } else if (highlightElementRects.length > 1) { bottomTipA = (
-
Element: {JSON.stringify(highlightElementRects)}
+
+ Element: {JSON.stringify(highlightElementRects)} +
); } @@ -260,20 +288,28 @@ const BlackBoard = (): JSX.Element => { if (highlightSectionRects.length === 1) { bottomTipB = (
-
Section: {JSON.stringify(highlightSectionRects[0])}
+
+ Section: {JSON.stringify(highlightSectionRects[0])} +
); } else if (highlightSectionRects.length > 1) { bottomTipB = (
-
Sections: {JSON.stringify(highlightSectionRects)}
+
+ Sections: {JSON.stringify(highlightSectionRects)} +
); } return (
-
+
diff --git a/packages/visualizer/src/component/color.tsx b/packages/visualizer/src/component/color.tsx index 736bda7d..e334d0e4 100644 --- a/packages/visualizer/src/component/color.tsx +++ b/packages/visualizer/src/component/color.tsx @@ -16,19 +16,20 @@ function djb2Hash(str?: string): number { return hash >>> 0; // Convert to unsigned 32 } -export function colorForName(type: 'section' | 'element', name: string): string { +export function colorForName( + type: 'section' | 'element', + name: string, +): string { const hashNumber = djb2Hash(name); if (type === 'section') { return sectionColor[hashNumber % sectionColor.length]; - } else { - return elementColor[hashNumber % elementColor.length]; } + return elementColor[hashNumber % elementColor.length]; } export function highlightColorForType(type: 'section' | 'element'): string { if (type === 'section') { return highlightColorForSection; - } else { - return highlightColorForElement; } + return highlightColorForElement; } diff --git a/packages/visualizer/src/component/detail-panel.tsx b/packages/visualizer/src/component/detail-panel.tsx index f68533f6..9a8e31f5 100644 --- a/packages/visualizer/src/component/detail-panel.tsx +++ b/packages/visualizer/src/component/detail-panel.tsx @@ -1,11 +1,15 @@ 'use client'; import './detail-panel.less'; -import { Segmented, ConfigProvider } from 'antd'; -import { CameraOutlined, FileTextOutlined, ScheduleOutlined } from '@ant-design/icons'; +import { useExecutionDump, useInsightDump } from '@/component/store'; +import { filterBase64Value, timeStr } from '@/utils'; +import { + CameraOutlined, + FileTextOutlined, + ScheduleOutlined, +} from '@ant-design/icons'; +import { ConfigProvider, Segmented } from 'antd'; import { useEffect, useState } from 'react'; import BlackBoard from './blackboard'; -import { useExecutionDump, useInsightDump } from '@/component/store'; -import { timeStr, filterBase64Value } from '@/utils'; const ScreenshotItem = (props: { time: string; img: string }) => { return ( @@ -29,14 +33,18 @@ const DetailPanel = (): JSX.Element => { const [preferredViewType, setViewType] = useState(VIEW_TYPE_BLACKBOARD); const viewType = - preferredViewType === VIEW_TYPE_BLACKBOARD && !dumpId ? VIEW_TYPE_SCREENSHOT : preferredViewType; + preferredViewType === VIEW_TYPE_BLACKBOARD && !dumpId + ? VIEW_TYPE_SCREENSHOT + : preferredViewType; let content; if (!activeTask) { content =
please select a task
; } else if (viewType === VIEW_TYPE_JSON) { content = ( -
{filterBase64Value(JSON.stringify(activeTask, undefined, 2))}
+
+ {filterBase64Value(JSON.stringify(activeTask, undefined, 2))} +
); } else if (viewType === VIEW_TYPE_BLACKBOARD) { if (dumpId) { @@ -52,8 +60,12 @@ const DetailPanel = (): JSX.Element => { .filter((item) => item.screenshot) .map((item, index) => { const fullTime = timeStr(item.ts); - const str = item.timing ? `${fullTime} / ${item.timing}` : fullTime; - return ; + const str = item.timing + ? `${fullTime} / ${item.timing}` + : fullTime; + return ( + + ); })}
); @@ -71,7 +83,11 @@ const DetailPanel = (): JSX.Element => { } else if (viewType === VIEW_TYPE_SCREENSHOT) { setViewType(VIEW_TYPE_JSON); } else { - setViewType(blackboardViewAvailable ? VIEW_TYPE_BLACKBOARD : VIEW_TYPE_SCREENSHOT); + setViewType( + blackboardViewAvailable + ? VIEW_TYPE_BLACKBOARD + : VIEW_TYPE_SCREENSHOT, + ); } e.preventDefault(); } @@ -85,11 +101,19 @@ const DetailPanel = (): JSX.Element => { }); const options = [ - { label: 'Screenshots', value: VIEW_TYPE_SCREENSHOT, icon: }, + { + label: 'Screenshots', + value: VIEW_TYPE_SCREENSHOT, + icon: , + }, { label: 'JSON View', value: VIEW_TYPE_JSON, icon: }, ]; if (blackboardViewAvailable) { - options.unshift({ label: 'Visualization', value: VIEW_TYPE_BLACKBOARD, icon: }); + options.unshift({ + label: 'Visualization', + value: VIEW_TYPE_BLACKBOARD, + icon: , + }); } return (
diff --git a/packages/visualizer/src/component/detail-side.tsx b/packages/visualizer/src/component/detail-side.tsx index 2645b903..85d1c345 100644 --- a/packages/visualizer/src/component/detail-side.tsx +++ b/packages/visualizer/src/component/detail-side.tsx @@ -1,9 +1,9 @@ /* eslint-disable max-lines */ 'use client'; import './detail-side.less'; +import { timeStr, typeStr } from '@/utils'; import { RadiusSettingOutlined } from '@ant-design/icons'; -import { Tooltip, Tag, Timeline, TimelineItemProps } from 'antd'; -import { +import type { BaseElement, ExecutionTaskAction, ExecutionTaskInsightLocate, @@ -11,11 +11,11 @@ import { ExecutionTaskPlanning, UISection, } from '@midscene/core'; +import { Tag, Timeline, type TimelineItemProps, Tooltip } from 'antd'; import { highlightColorForType } from './color'; -import { useExecutionDump, useInsightDump } from './store'; -import PanelTitle from './panel-title'; import { timeCostStrElement } from './misc'; -import { timeStr, typeStr } from '@/utils'; +import PanelTitle from './panel-title'; +import { useExecutionDump, useInsightDump } from './store'; const noop = () => {}; const Card = (props: { @@ -28,11 +28,23 @@ const Card = (props: { onMouseLeave?: () => void; content: any; }) => { - const { highlightWithColor, title, subtitle, onMouseEnter, onMouseLeave, content, characteristic } = props; + const { + highlightWithColor, + title, + subtitle, + onMouseEnter, + onMouseLeave, + content, + characteristic, + } = props; const titleTag = props.characteristic ? (
- + @@ -41,9 +53,13 @@ const Card = (props: {
) : null; - const titleRightPaddingClass = props.characteristic ? 'title-right-padding' : ''; + const titleRightPaddingClass = props.characteristic + ? 'title-right-padding' + : ''; const modeClass = props.liteMode ? 'item-lite' : ''; - const highlightStyle = highlightWithColor ? { backgroundColor: highlightWithColor } : {}; + const highlightStyle = highlightWithColor + ? { backgroundColor: highlightWithColor } + : {}; return (
{/* {extraSection} */} -
+
{title} {titleTag}
-
+
{subtitle}
-
+
{content}
); }; -const MetaKV = (props: { data: { key: string; content: string | JSX.Element }[] }) => { +const MetaKV = (props: { + data: { key: string; content: string | JSX.Element }[]; +}) => { return (
{props.data.map((item, index) => { @@ -94,27 +121,33 @@ const DetailSide = (): JSX.Element => { const task = useExecutionDump((store) => store.activeTask); const dump = useInsightDump((store) => store.data); const { matchedSection: sections, matchedElement: elements } = dump || {}; - const highlightSectionNames = useInsightDump((store) => store.highlightSectionNames); + const highlightSectionNames = useInsightDump( + (store) => store.highlightSectionNames, + ); const highlightElements = useInsightDump((store) => store.highlightElements); - const setHighlightSectionNames = useInsightDump((store) => store.setHighlightSectionNames); - const setHighlightElements = useInsightDump((store) => store.setHighlightElements); + const setHighlightSectionNames = useInsightDump( + (store) => store.setHighlightSectionNames, + ); + const setHighlightElements = useInsightDump( + (store) => store.setHighlightElements, + ); - const setHighlightSectionName = function (name: string) { + const setHighlightSectionName = (name: string) => { setHighlightSectionNames([name]); }; - const setHighlightElement = function (element: BaseElement) { + const setHighlightElement = (element: BaseElement) => { setHighlightElements([element]); }; - const unhighlightSection = function () { + const unhighlightSection = () => { setHighlightSectionNames([]); }; - const unhighlightElement = function () { + const unhighlightElement = () => { setHighlightElements([]); }; - const kv = function (data: Record) { + const kv = (data: Record) => { const isElementItem = (value: unknown): value is BaseElement => Boolean(value) && typeof value === 'object' && @@ -154,7 +187,11 @@ const DetailSide = (): JSX.Element => { ); if (Array.isArray(data) || typeof data !== 'object') { - return
{JSON.stringify(data, undefined, 2)}
; + return ( +
+          {JSON.stringify(data, undefined, 2)}
+        
+ ); } return Object.keys(data).map((key) => { @@ -162,14 +199,27 @@ const DetailSide = (): JSX.Element => { let content; if (typeof value === 'object' && isElementItem(value)) { content = elementEl(value); - } else if (Array.isArray(value) && value.some((item) => isElementItem(item))) { - content = value.map((item, index) => {elementEl(item)}); + } else if ( + Array.isArray(value) && + value.some((item) => isElementItem(item)) + ) { + content = value.map((item, index) => ( + {elementEl(item)} + )); } else if (typeof value === 'object' && isSectionItem(value)) { content = sectionEl(value); - } else if (Array.isArray(value) && value.some((item) => isSectionItem(item))) { - content = value.map((item, index) => {sectionEl(item)}); + } else if ( + Array.isArray(value) && + value.some((item) => isSectionItem(item)) + ) { + content = value.map((item, index) => ( + {sectionEl(item)} + )); } else { - content = typeof value === 'string' ? value : JSON.stringify(value, undefined, 2); + content = + typeof value === 'string' + ? value + : JSON.stringify(value, undefined, 2); } return ( @@ -210,7 +260,10 @@ const DetailSide = (): JSX.Element => { taskParam = MetaKV({ data: [ { key: 'type', content: (task && typeStr(task)) || '' }, - { key: 'param', content: (task as ExecutionTaskPlanning)?.param?.userPrompt }, + { + key: 'param', + content: (task as ExecutionTaskPlanning)?.param?.userPrompt, + }, ], }); } else if (task?.type === 'Insight') { @@ -232,7 +285,11 @@ const DetailSide = (): JSX.Element => { { key: 'type', content: (task && typeStr(task)) || '' }, { key: 'value', - content: JSON.stringify((task as ExecutionTaskAction)?.param?.value, undefined, 2), + content: JSON.stringify( + (task as ExecutionTaskAction)?.param?.value, + undefined, + 2, + ), }, ], }); @@ -251,7 +308,9 @@ const DetailSide = (): JSX.Element => { 'sectionCharacteristics', ]); const sectionKV = Object.keys(kvToShow).length ? kv(kvToShow) : null; - const highlightColor = ifHighlight ? highlightColorForType('section') : undefined; + const highlightColor = ifHighlight + ? highlightColorForType('section') + : undefined; return ( { const matchedElementsEl = elements?.length ? elements.map((element, idx) => { const ifHighlight = highlightElements.includes(element); - const highlightColor = ifHighlight ? highlightColorForType('element') : undefined; + const highlightColor = ifHighlight + ? highlightColorForType('element') + : undefined; const elementKV = kv( objectWithoutKeys(element as any, [ @@ -322,7 +383,7 @@ const DetailSide = (): JSX.Element => { onMouseEnter={noop} onMouseLeave={noop} content={
{JSON.stringify(dump.data, undefined, 2)}
} - >
+ /> ) : null; console.log('dump is', dump); @@ -339,7 +400,11 @@ const DetailSide = (): JSX.Element => { {typeStr(item as any)}

{item.thought}

-

{item.param ? JSON.stringify(item.param || {}, undefined, 2) : null}

+

+ {item.param + ? JSON.stringify(item.param || {}, undefined, 2) + : null} +

), }; diff --git a/packages/visualizer/src/component/global-hover-preview.tsx b/packages/visualizer/src/component/global-hover-preview.tsx index a0e39cf7..0021d01a 100644 --- a/packages/visualizer/src/component/global-hover-preview.tsx +++ b/packages/visualizer/src/component/global-hover-preview.tsx @@ -7,7 +7,9 @@ const GlobalHoverPreview = () => { const wrapperRef = useRef(null); const hoverTask = useExecutionDump((store) => store.hoverTask); const hoverTimestamp = useExecutionDump((store) => store.hoverTimestamp); - const hoverPreviewConfig = useExecutionDump((store) => store.hoverPreviewConfig); + const hoverPreviewConfig = useExecutionDump( + (store) => store.hoverPreviewConfig, + ); const [imageW, setImageW] = useState(size); const [imageH, setImageH] = useState(size); @@ -25,19 +27,27 @@ const GlobalHoverPreview = () => { let left = 0; let top = 0; - const shouldShow = images?.length && typeof x !== 'undefined' && typeof y !== 'undefined'; + const shouldShow = + images?.length && typeof x !== 'undefined' && typeof y !== 'undefined'; if (shouldShow) { const { clientWidth, clientHeight } = document.body; const widthInPractice = imageW >= imageH ? size : size * (imageW / imageH); const heightInPractice = imageW >= imageH ? size * (imageH / imageW) : size; - left = x + widthInPractice > clientWidth ? clientWidth - widthInPractice : x; - top = y + heightInPractice > clientHeight ? clientHeight - heightInPractice : y; + left = + x + widthInPractice > clientWidth ? clientWidth - widthInPractice : x; + top = + y + heightInPractice > clientHeight ? clientHeight - heightInPractice : y; } // if x + size exceed the screen width, use (screenWidth - size) instead return shouldShow ? ( -
+
{images?.length ? ( + // biome-ignore lint/a11y/useAltText: { diff --git a/packages/visualizer/src/component/sidebar.tsx b/packages/visualizer/src/component/sidebar.tsx index 7bf79ffc..c177a340 100644 --- a/packages/visualizer/src/component/sidebar.tsx +++ b/packages/visualizer/src/component/sidebar.tsx @@ -1,5 +1,6 @@ import './sidebar.less'; -import { useEffect } from 'react'; +import { useAllCurrentTasks, useExecutionDump } from '@/component/store'; +import { typeStr } from '@/utils'; import { ArrowRightOutlined, CheckOutlined, @@ -8,13 +9,12 @@ import { LogoutOutlined, MinusOutlined, } from '@ant-design/icons'; -import { ExecutionTask, ExecutionTaskInsightQuery } from '@midscene/core'; +import type { ExecutionTask, ExecutionTaskInsightQuery } from '@midscene/core'; import { Button } from 'antd'; -import PanelTitle from './panel-title'; -import { timeCostStrElement } from './misc'; +import { useEffect } from 'react'; import Logo from './assets/logo-plain2.svg'; -import { useAllCurrentTasks, useExecutionDump } from '@/component/store'; -import { typeStr } from '@/utils'; +import { timeCostStrElement } from './misc'; +import PanelTitle from './panel-title'; const SideItem = (props: { task: ExecutionTask; @@ -45,11 +45,14 @@ const SideItem = (props: { let contentRow: JSX.Element | undefined; if (task.type === 'Planning') { - contentRow =
{task.param?.userPrompt}
; + contentRow = ( +
{task.param?.userPrompt}
+ ); } else if (task.type === 'Insight' && task.subType === 'Query') { // debugger; const demand = (task as ExecutionTaskInsightQuery).param?.dataDemand; - const contentToShow = typeof demand === 'string' ? demand : JSON.stringify(demand); + const contentToShow = + typeof demand === 'string' ? demand : JSON.stringify(demand); contentRow =
{contentToShow}
; } else { // debugger; @@ -71,8 +74,10 @@ const SideItem = (props: { }} > {' '} -
- {statusIcon} +
+ + {statusIcon} +
{typeStr(task)}
{statusText}
@@ -81,18 +86,25 @@ const SideItem = (props: { ); }; -const Sidebar = (props: { hideLogo?: boolean; logoAction?: () => void }): JSX.Element => { +const Sidebar = (props: { + hideLogo?: boolean; + logoAction?: () => void; +}): JSX.Element => { const groupedDumps = useExecutionDump((store) => store.dump); const setActiveTask = useExecutionDump((store) => store.setActiveTask); const activeTask = useExecutionDump((store) => store.activeTask); const setHoverTask = useExecutionDump((store) => store.setHoverTask); - const setHoverPreviewConfig = useExecutionDump((store) => store.setHoverPreviewConfig); + const setHoverPreviewConfig = useExecutionDump( + (store) => store.setHoverPreviewConfig, + ); // const selectedTaskIndex = useExecutionDump((store) => store.selectedTaskIndex); // const setSelectedTaskIndex = useExecutionDump((store) => store.setSelectedTaskIndex); const reset = useExecutionDump((store) => store.reset); const allTasks = useAllCurrentTasks(); - const currentSelectedIndex = allTasks?.findIndex((task) => task === activeTask); + const currentSelectedIndex = allTasks?.findIndex( + (task) => task === activeTask, + ); useEffect(() => { // all tasks @@ -153,7 +165,9 @@ const Sidebar = (props: { hideLogo?: boolean; logoAction?: () => void }): JSX.El let seperator: JSX.Element; switch (indexOfExecution) { case 0: - seperator =
; + seperator = ( +
+ ); break; // case group.executions.length - 1: // seperator =
; @@ -186,7 +200,11 @@ const Sidebar = (props: { hideLogo?: boolean; logoAction?: () => void }): JSX.El return (
-
+
{ diff --git a/packages/visualizer/src/component/store.tsx b/packages/visualizer/src/component/store.tsx index 9be156bf..e478b397 100644 --- a/packages/visualizer/src/component/store.tsx +++ b/packages/visualizer/src/component/store.tsx @@ -1,10 +1,10 @@ import { create } from 'zustand'; -import { - InsightDump, +import type { BaseElement, - ExecutionTaskInsightLocate, ExecutionTask, + ExecutionTaskInsightLocate, GroupedActionDump, + InsightDump, } from '../../../midscene/dist/types'; export const useBlackboardPreference = create<{ @@ -73,7 +73,7 @@ export const useExecutionDump = create<{ set({ activeTask: task }); console.log('task set', task); if (task.type === 'Insight') { - syncToInsightDump((task as ExecutionTaskInsightLocate).log!.dump!); + syncToInsightDump((task as ExecutionTaskInsightLocate).log?.dump!); } else { resetInsightDump(); } @@ -127,7 +127,12 @@ export const useInsightDump = create<{ reset: () => void; }>((set) => { let loadId = 0; - const initData = { _loadId: 0, highlightSectionNames: [], highlightElements: [], data: null }; + const initData = { + _loadId: 0, + highlightSectionNames: [], + highlightElements: [], + data: null, + }; return { ...initData, diff --git a/packages/visualizer/src/component/timeline.tsx b/packages/visualizer/src/component/timeline.tsx index dd4961fa..0b2a218a 100644 --- a/packages/visualizer/src/component/timeline.tsx +++ b/packages/visualizer/src/component/timeline.tsx @@ -1,9 +1,9 @@ +import * as PIXI from 'pixi.js'; /* eslint-disable max-lines */ import { useEffect, useMemo, useRef } from 'react'; -import * as PIXI from 'pixi.js'; import './timeline.less'; -import { ExecutionRecorderItem, ExecutionTask } from '@midscene/core'; +import type { ExecutionRecorderItem, ExecutionTask } from '@midscene/core'; import { useAllCurrentTasks, useExecutionDump } from './store'; interface TimelineItem { @@ -57,7 +57,12 @@ const TimelineWidget = (props: { const highlightMaskContainer = useMemo(() => new PIXI.Container(), []); const containerUpdaterRef = useRef( // eslint-disable-next-line @typescript-eslint/no-empty-function - (_s: number | undefined, _e: number | undefined, _hs: number | undefined, _he: number | undefined) => {}, + ( + _s: number | undefined, + _e: number | undefined, + _hs: number | undefined, + _he: number | undefined, + ) => {}, ); const indicatorContainer = useMemo(() => new PIXI.Container(), []); @@ -126,8 +131,8 @@ const TimelineWidget = (props: { let singleGridWidth = 100 * sizeRatio; let gridCount = Math.floor(canvasWidth / singleGridWidth); const stepCandidate = [ - 50, 100, 200, 300, 500, 1000, 2000, 3000, 5000, 6000, 8000, 9000, 10000, 20000, 30000, 40000, 60000, - 90000, 12000, 300000, + 50, 100, 200, 300, 500, 1000, 2000, 3000, 5000, 6000, 8000, 9000, + 10000, 20000, 30000, 40000, 60000, 90000, 12000, 300000, ]; let timeStep = stepCandidate[0]; for (let i = stepCandidate.length - 1; i >= 0; i--) { @@ -212,7 +217,8 @@ const TimelineWidget = (props: { // draw all screenshots screenshotsContainer.removeChildren(); const screenshotTop = timeTitleBottom + commonPadding * 1.5; - const screenshotMaxHeight = canvasHeight - screenshotTop - commonPadding * 1.5; + const screenshotMaxHeight = + canvasHeight - screenshotTop - commonPadding * 1.5; allScreenshots.forEach((screenshot, index) => { const container = new PIXI.Container(); shotContainers.push(container); @@ -228,7 +234,9 @@ const TimelineWidget = (props: { const originalHeight = img.height; const screenshotHeight = screenshotMaxHeight; - const screenshotWidth = Math.floor((screenshotHeight / originalHeight) * originalWidth); + const screenshotWidth = Math.floor( + (screenshotHeight / originalHeight) * originalWidth, + ); const screenshotX = leftForTimeOffset(screenshot.timeOffset); allScreenshots[index].x = screenshotX; @@ -238,7 +246,12 @@ const TimelineWidget = (props: { const border = new PIXI.Graphics(); border.lineStyle(sizeRatio, shotBorderColor, 1); - border.drawRect(screenshotX, screenshotTop, screenshotWidth, screenshotMaxHeight); + border.drawRect( + screenshotX, + screenshotTop, + screenshotWidth, + screenshotMaxHeight, + ); border.endFill(); container.addChild(border); @@ -258,19 +271,38 @@ const TimelineWidget = (props: { ) => { highlightMaskContainer.removeChildren(); - const mask = (start: number | undefined, end: number | undefined, color: number, alpha: number) => { - if (typeof start === 'undefined' || typeof end === 'undefined' || end === 0) { + const mask = ( + start: number | undefined, + end: number | undefined, + color: number, + alpha: number, + ) => { + if ( + typeof start === 'undefined' || + typeof end === 'undefined' || + end === 0 + ) { return; } const leftBorder = new PIXI.Graphics(); leftBorder.beginFill(gridHighlightColor, 1); - leftBorder.drawRect(leftForTimeOffset(start), 0, sizeRatio, canvasHeight); + leftBorder.drawRect( + leftForTimeOffset(start), + 0, + sizeRatio, + canvasHeight, + ); leftBorder.endFill(); highlightMaskContainer.addChild(leftBorder); const rightBorder = new PIXI.Graphics(); rightBorder.beginFill(gridHighlightColor, 1); - rightBorder.drawRect(leftForTimeOffset(end), 0, sizeRatio, canvasHeight); + rightBorder.drawRect( + leftForTimeOffset(end), + 0, + sizeRatio, + canvasHeight, + ); rightBorder.endFill(); highlightMaskContainer.addChild(rightBorder); @@ -289,7 +321,12 @@ const TimelineWidget = (props: { mask(start, end, gridHighlightColor, highlightMaskAlpha); mask(hoverStart, hoverEnd, hoverMaskColor, hoverMaskAlpha); }; - highlightMaskUpdater(props.highlightMask?.startMs, props.highlightMask?.endMs, 0, 0); + highlightMaskUpdater( + props.highlightMask?.startMs, + props.highlightMask?.endMs, + 0, + 0, + ); containerUpdaterRef.current = highlightMaskUpdater; // keep tracking the position of the mouse moving above the canvas @@ -300,7 +337,10 @@ const TimelineWidget = (props: { indicatorContainer.removeChildren(); // find out the screenshot that is closest to the mouse on the left - const { closestScreenshot, closestIndex } = closestScreenshotItemOnXY(x, y); + const { closestScreenshot, closestIndex } = closestScreenshotItemOnXY( + x, + y, + ); if (closestIndex < 0) { props.onUnhighlight?.(); return; @@ -315,9 +355,9 @@ const TimelineWidget = (props: { newSpirit.lineStyle(2, gridHighlightColor, 1); newSpirit.drawRect( x, // follow mouse - closestScreenshot!.y!, - closestScreenshot!.width!, - closestScreenshot!.height!, + closestScreenshot?.y!, + closestScreenshot?.width!, + closestScreenshot?.height!, ); newSpirit.endFill(); indicatorContainer.addChild(newSpirit); @@ -381,7 +421,7 @@ const TimelineWidget = (props: { ); }, []); - return
; + return
; }; const Timeline = () => { @@ -391,7 +431,9 @@ const Timeline = () => { const activeTask = useExecutionDump((store) => store.activeTask); const hoverTask = useExecutionDump((store) => store.hoverTask); const setHoverTask = useExecutionDump((store) => store.setHoverTask); - const setHoverPreviewConfig = useExecutionDump((store) => store.setHoverPreviewConfig); + const setHoverPreviewConfig = useExecutionDump( + (store) => store.setHoverPreviewConfig, + ); // should be first task time ? let startingTime = -1; @@ -405,7 +447,10 @@ const Timeline = () => { startingTime = item.ts; } }); - if (current.timing?.start && (startingTime === -1 || startingTime > current.timing.start)) { + if ( + current.timing?.start && + (startingTime === -1 || startingTime > current.timing.start) + ) { startingTime = current.timing.start; } const recorderItemWithId = recorders.map((item) => { @@ -460,7 +505,9 @@ const Timeline = () => { // overall left of wrapper - const maskConfigForTask = (task?: ExecutionTask | null): HighlightMask | undefined => { + const maskConfigForTask = ( + task?: ExecutionTask | null, + ): HighlightMask | undefined => { if (!task) { return undefined; } diff --git a/packages/visualizer/src/index.tsx b/packages/visualizer/src/index.tsx index 629569f4..1232a4ce 100644 --- a/packages/visualizer/src/index.tsx +++ b/packages/visualizer/src/index.tsx @@ -1,18 +1,18 @@ import './index.less'; -import { ConfigProvider, message, Upload, Button } from 'antd'; +import DetailSide from '@/component/detail-side'; +import Sidebar from '@/component/sidebar'; +import { useExecutionDump } from '@/component/store'; +import type { GroupedActionDump } from '@midscene/core'; +import { Helmet } from '@modern-js/runtime/head'; +import { Button, ConfigProvider, Upload, message } from 'antd'; import type { UploadProps } from 'antd'; import { useEffect, useRef, useState } from 'react'; -import { Helmet } from '@modern-js/runtime/head'; import { Panel, PanelGroup, PanelResizeHandle } from 'react-resizable-panels'; -import { GroupedActionDump } from '@midscene/core'; -import Timeline from './component/timeline'; -import DetailPanel from './component/detail-panel'; import Logo from './component/assets/logo-plain.svg'; +import DetailPanel from './component/detail-panel'; import GlobalHoverPreview from './component/global-hover-preview'; +import Timeline from './component/timeline'; import demoDump from './demo-dump.json'; -import { useExecutionDump } from '@/component/store'; -import DetailSide from '@/component/detail-side'; -import Sidebar from '@/component/sidebar'; const { Dragger } = Upload; @@ -121,7 +121,8 @@ export function Visualizer(props: {

- All data will be processed locally by the browser. No data will be sent to the server. + All data will be processed locally by the browser. No data will be + sent to the server.

@@ -160,7 +161,10 @@ export function Visualizer(props: {
- +
@@ -187,7 +191,9 @@ export function Visualizer(props: { // modify rspress theme const navHeightKey = '--rp-nav-height'; - const originalNavHeight = getComputedStyle(document.documentElement).getPropertyValue(navHeightKey); + const originalNavHeight = getComputedStyle( + document.documentElement, + ).getPropertyValue(navHeightKey); if (ifInRspressPage) { const newNavHeight = '42px'; @@ -198,7 +204,10 @@ export function Visualizer(props: { // Cleanup function to revert the change return () => { if (ifInRspressPage) { - document.documentElement.style.setProperty(navHeightKey, originalNavHeight); + document.documentElement.style.setProperty( + navHeightKey, + originalNavHeight, + ); } }; }, []); @@ -225,7 +234,11 @@ export function Visualizer(props: { MidScene.js - Visualization Tool -
+
{mainContent}
diff --git a/packages/visualizer/src/utils.ts b/packages/visualizer/src/utils.ts index 12c925c2..9ae0465c 100644 --- a/packages/visualizer/src/utils.ts +++ b/packages/visualizer/src/utils.ts @@ -1,16 +1,29 @@ +import type { + ExecutionDump, + ExecutionTask, + ExecutionTaskInsightLocate, + InsightDump, +} from '@midscene/core'; /* eslint-disable @typescript-eslint/no-empty-function */ import dayjs from 'dayjs'; -import type { ExecutionDump, ExecutionTaskInsightLocate, InsightDump, ExecutionTask } from '@midscene/core'; -export function insightDumpToExecutionDump(insightDump: InsightDump | InsightDump[]): ExecutionDump { - const insightToTask = (insightDump: InsightDump): ExecutionTaskInsightLocate => { +export function insightDumpToExecutionDump( + insightDump: InsightDump | InsightDump[], +): ExecutionDump { + const insightToTask = ( + insightDump: InsightDump, + ): ExecutionTaskInsightLocate => { const task: ExecutionTaskInsightLocate = { type: 'Insight', subType: insightDump.type === 'locate' ? 'Locate' : 'Query', status: insightDump.error ? 'fail' : 'success', param: { - ...(insightDump.userQuery.element ? { query: insightDump.userQuery } : {}), - ...(insightDump.userQuery.dataDemand ? { dataDemand: insightDump.userQuery.dataDemand } : {}), + ...(insightDump.userQuery.element + ? { query: insightDump.userQuery } + : {}), + ...(insightDump.userQuery.dataDemand + ? { dataDemand: insightDump.userQuery.dataDemand } + : {}), insight: {} as any, } as any, log: { @@ -34,15 +47,14 @@ export function insightDumpToExecutionDump(insightDump: InsightDump | InsightDum tasks: [insightToTask(insightDump)], }; return result; - } else { - const result: ExecutionDump = { - sdkVersion: insightDump[0].sdkVersion, - logTime: insightDump[0].logTime, - name: 'Insight', - tasks: insightDump.map(insightToTask), - }; - return result; } + const result: ExecutionDump = { + sdkVersion: insightDump[0].sdkVersion, + logTime: insightDump[0].logTime, + name: 'Insight', + tasks: insightDump.map(insightToTask), + }; + return result; } export function timeStr(timestamp?: number) { diff --git a/packages/visualizer/tsconfig.json b/packages/visualizer/tsconfig.json index 74dd05cf..6ae4d71c 100644 --- a/packages/visualizer/tsconfig.json +++ b/packages/visualizer/tsconfig.json @@ -16,9 +16,8 @@ "rootDir": "src", "skipLibCheck": true, "strict": true, - "types": ["react"], + "types": ["react"] }, "exclude": ["**/node_modules"], "include": ["src"] } - diff --git a/packages/web-integration/modern.config.ts b/packages/web-integration/modern.config.ts index b05ea334..e2a04767 100644 --- a/packages/web-integration/modern.config.ts +++ b/packages/web-integration/modern.config.ts @@ -1,4 +1,4 @@ -import { moduleTools, defineConfig } from '@modern-js/module-tools'; +import { defineConfig, moduleTools } from '@modern-js/module-tools'; export default defineConfig({ plugins: [moduleTools()], diff --git a/packages/web-integration/modern.inspect.config.ts b/packages/web-integration/modern.inspect.config.ts index 2e56939b..8abd46ec 100644 --- a/packages/web-integration/modern.inspect.config.ts +++ b/packages/web-integration/modern.inspect.config.ts @@ -1,4 +1,4 @@ -import { moduleTools, defineConfig } from '@modern-js/module-tools'; +import { defineConfig, moduleTools } from '@modern-js/module-tools'; // It was split into two configuration files because of a bug in the build config array export default defineConfig({ @@ -8,13 +8,13 @@ export default defineConfig({ buildType: 'bundle', format: 'iife', input: { - htmlElement:'src/extractor/index.ts', + htmlElement: 'src/extractor/index.ts', }, outDir: 'dist/script', - esbuildOptions: options => { - options.globalName = 'midscene_element_inspector' + esbuildOptions: (options) => { + options.globalName = 'midscene_element_inspector'; return options; }, target: 'es2017', - } + }, }); diff --git a/packages/web-integration/package.json b/packages/web-integration/package.json index aefa99d4..71c41bf0 100644 --- a/packages/web-integration/package.json +++ b/packages/web-integration/package.json @@ -30,18 +30,10 @@ }, "typesVersions": { "*": { - ".": [ - "./dist/types/index.d.ts" - ], - "playwright-report": [ - "./dist/types/playwright-report.d.ts" - ], - "constants": [ - "./dist/types/constants.d.ts" - ], - "html-element": [ - "./dist/types/html-element/index.d.ts" - ] + ".": ["./dist/types/index.d.ts"], + "playwright-report": ["./dist/types/playwright-report.d.ts"], + "constants": ["./dist/types/constants.d.ts"], + "html-element": ["./dist/types/html-element/index.d.ts"] } }, "scripts": { @@ -60,10 +52,7 @@ "e2e:ui": "playwright test --config=playwright.config.ts --ui", "e2e:ui-cache": "MIDSCENE_CACHE=true playwright test --config=playwright.config.ts --ui" }, - "files": [ - "dist", - "README.md" - ], + "files": ["dist", "README.md"], "dependencies": { "openai": "4.47.1", "sharp": "0.33.3", diff --git a/packages/web-integration/report-script.mjs b/packages/web-integration/report-script.mjs index c9692eaa..b2ba29f5 100644 --- a/packages/web-integration/report-script.mjs +++ b/packages/web-integration/report-script.mjs @@ -1,11 +1,14 @@ +import os from 'node:os'; import path from 'node:path'; -import os from 'os'; import fsExtra from 'fs-extra'; const projectDir = process.cwd(); -const reportHtmlDir = path.join(projectDir, `node_modules/@midscene/visualizer-report/dist`); -const distPath = path.join(projectDir, `dist/visualizer-report`); -const distPublicPath = path.join(projectDir, `dist/visualizer-report/public`); +const reportHtmlDir = path.join( + projectDir, + 'node_modules/@midscene/visualizer-report/dist', +); +const distPath = path.join(projectDir, 'dist/visualizer-report'); +const distPublicPath = path.join(projectDir, 'dist/visualizer-report/public'); const tempDir = path.join(os.tmpdir(), 'temp-folder'); diff --git a/packages/web-integration/src/common/agent.ts b/packages/web-integration/src/common/agent.ts index d9b072b2..a114dd10 100644 --- a/packages/web-integration/src/common/agent.ts +++ b/packages/web-integration/src/common/agent.ts @@ -1,8 +1,8 @@ -import { ExecutionDump, GroupedActionDump } from '@midscene/core'; +import type { WebPage } from '@/common/page'; +import type { ExecutionDump, GroupedActionDump } from '@midscene/core'; import { groupedActionDumpFileExt, writeDumpFile } from '@midscene/core/utils'; import { PageTaskExecutor } from '../common/tasks'; -import { AiTaskCache } from './task-cache'; -import { WebPage } from '@/common/page'; +import type { AiTaskCache } from './task-cache'; export class PageAgent { page: WebPage; @@ -15,7 +15,10 @@ export class PageAgent { actionAgent: PageTaskExecutor; - constructor(page: WebPage, opts?: { testId?: string; taskFile?: string; cache?: AiTaskCache }) { + constructor( + page: WebPage, + opts?: { testId?: string; taskFile?: string; cache?: AiTaskCache }, + ) { this.page = page; this.dumps = [ { @@ -85,9 +88,12 @@ export class PageAgent { async ai(taskPrompt: string, type = 'action') { if (type === 'action') { return this.aiAction(taskPrompt); - } else if (type === 'query') { + } + if (type === 'query') { return this.aiQuery(taskPrompt); } - throw new Error(`Unknown or Unsupported task type: ${type}, only support 'action' or 'query'`); + throw new Error( + `Unknown or Unsupported task type: ${type}, only support 'action' or 'query'`, + ); } } diff --git a/packages/web-integration/src/common/cdp.ts b/packages/web-integration/src/common/cdp.ts deleted file mode 100644 index 108b2f52..00000000 --- a/packages/web-integration/src/common/cdp.ts +++ /dev/null @@ -1,322 +0,0 @@ -// fork from https://github.com/zerostep-ai/zerostep/blob/5ba8ca5282879f444e5dfefbcc1f03c76591469e/packages/playwright/src/cdp.ts -import type { Page } from 'playwright'; - -const cdpSessionByPage = new Map(); - -export const WEBDRIVER_ELEMENT_KEY = 'element-6066-11e4-a52e-4f735466cecf'; - -export type ScrollType = 'up' | 'down' | 'bottom' | 'top'; - -/** - * Closes the cdp session and clears the global shared reference. This - * happens automatically when a page closes in a playwright test, so - * should generally not be necessary. - */ -export const detachCPDSession = async (page: Page) => { - if (cdpSessionByPage.has(page)) { - await cdpSessionByPage.get(page)!.detach(); - cdpSessionByPage.delete(page); - } -}; - -/** - * Returns a stable reference to a CDP session. - */ -export const getCDPSession = async (page: Page): Promise => { - if (!cdpSessionByPage.has(page)) { - try { - const session = await page.context().newCDPSession(page); - cdpSessionByPage.set(page, session); - } catch (e) { - throw Error('The ai() function can only be run against Chromium browsers.'); - } - } - - return cdpSessionByPage.get(page)!; -}; - -export const getScreenshot = async (page: Page) => { - const cdpSession = await getCDPSession(page); - const screenshot = await cdpSession.send('Page.captureScreenshot'); - return screenshot.data; // Base64-encoded image data -}; - -export const scrollIntoView = async (page: Page, args: { id: string }) => { - const cdpSession = await getCDPSession(page); - - await cdpSession.send('DOM.scrollIntoViewIfNeeded', { - backendNodeId: parseInt(args.id, 10), - }); -}; - -export const getTitle = async (page: Page) => { - const cdpSession = await getCDPSession(page); - const returnedValue = await cdpSession.send('Runtime.evaluate', { - expression: 'document.title', - returnByValue: true, - }); - - return returnedValue.result.value; -}; - -export const get = async (page: Page, args: { url: string }) => { - const cdpSession = await getCDPSession(page); - await cdpSession.send('Page.navigate', { - url: args.url, - }); -}; - -export const scrollElement = async (page: Page, args: { id: string; target: ScrollType }) => { - await runFunctionOn(page, { - functionDeclaration: `function() { - let element = this - let elementHeight = 0 - - switch (element.tagName) { - case 'BODY': - case 'HTML': - element = document.scrollingElement || document.body - elementHeight = window.visualViewport?.height ?? 720 - break - default: - elementHeight = element.clientHeight ?? 720 - break - } - - const relativeScrollDistance = 0.75 * elementHeight - - switch ("${args.target}") { - case 'top': - return element.scrollTo({ top: 0 }) - case 'bottom': - return element.scrollTo({ top: element.scrollHeight }) - case 'up': - return element.scrollBy({ top: -relativeScrollDistance }) - case 'down': - return element.scrollBy({ top: relativeScrollDistance }) - default: - throw Error('Unsupported scroll target ${args.target}') - } - }`, - backendNodeId: parseInt(args.id, 10), - }); -}; - -export const runFunctionOn = async ( - page: Page, - args: { functionDeclaration: string; backendNodeId: number }, -) => { - const cdpSession = await getCDPSession(page); - const { - object: { objectId }, - } = await cdpSession.send('DOM.resolveNode', { backendNodeId: args.backendNodeId }); - await cdpSession.send('Runtime.callFunctionOn', { - functionDeclaration: args.functionDeclaration, - objectId, - }); -}; - -export const clearElement = async (page: Page, args: { id: string }) => { - return await runFunctionOn(page, { - functionDeclaration: `function() {this.value=''}`, - backendNodeId: parseInt(args.id, 10), - }); -}; - -export const sendKeysToElement = async (page: Page, args: { id: string; value: string[] }) => { - const cdpSession = await getCDPSession(page); - const value = args.value[0]; - - const { nodeId } = await cdpSession.send('DOM.requestNode', { objectId: args.id }); - await cdpSession.send('DOM.focus', { nodeId }); - - // eslint-disable-next-line @typescript-eslint/prefer-for-of - for (let i = 0; i < value.length; i++) { - await cdpSession.send('Input.dispatchKeyEvent', { - type: 'char', - text: value[i], - }); - } - - return true; -}; - -export const getElementAttribute = async (page: Page, args: { id: string; name: string }) => { - const cdpSession = await getCDPSession(page); - - const { nodeId } = await cdpSession.send('DOM.requestNode', { objectId: args.id }); - const { attributes } = await cdpSession.send('DOM.getAttributes', { nodeId }); - - for (let i = 0; i < attributes.length; i++) { - if (attributes[i] === args.name) { - return attributes[i + 1]; - } - } - return attributes; -}; - -export const getElementTagName = async (page: Page, args: { id: string }) => { - const cdpSession = await getCDPSession(page); - const returnedValue = await cdpSession.send('Runtime.callFunctionOn', { - functionDeclaration: `function() {return this.tagName}`, - objectId: args.id, - returnByValue: true, - }); - - return returnedValue.result.value; -}; - -export const clickElement = async (page: Page, args: { id: string }) => { - const cdpSession = await getCDPSession(page); - const { centerX, centerY } = await getContentQuads(page, { backendNodeId: parseInt(args.id, 10) }); - - await cdpSession.send('Input.dispatchMouseEvent', { - type: 'mousePressed', - x: centerX, - y: centerY, - button: 'left', - clickCount: 1, - buttons: 1, - }); - - await cdpSession.send('Input.dispatchMouseEvent', { - type: 'mouseReleased', - x: centerX, - y: centerY, - button: 'left', - clickCount: 1, - buttons: 1, - }); - - return true; -}; - -export const getContentQuads = async (page: Page, args: { backendNodeId: number }) => { - const cdpSession = await getCDPSession(page); - const quadsResponse = await cdpSession.send('DOM.getContentQuads', args); - - const [topLeftX, topLeftY, topRightX, topRightY, bottomRightX, bottomRightY, bottomLeftX, bottomLeftY] = - quadsResponse.quads[0]; - - const width = topRightX - topLeftX; - const height = bottomRightY - topRightY; - const centerX = topLeftX + width / 2; - const centerY = topRightY + height / 2; - - console.log('getContentQuads', args, { - width, - height, - topLeftX, - topLeftY, - centerX, - centerY, - }); - return { - topLeftX, - topLeftY, - topRightX, - topRightY, - bottomRightX, - bottomRightY, - bottomLeftX, - bottomLeftY, - width, - height, - centerX, - centerY, - }; -}; - -export const focusElement = async (page: Page, args: { backendNodeId: number }) => { - const cdpSession = await getCDPSession(page); - await cdpSession.send('DOM.focus', args); -}; - -export const getElementRect = async (page: Page, args: { id: string }) => { - const cdpSession = await getCDPSession(page); - const returnedValue = await cdpSession.send('Runtime.callFunctionOn', { - functionDeclaration: `function() {return JSON.parse(JSON.stringify(this.getBoundingClientRect()))}`, - objectId: args.id, - returnByValue: true, - }); - - return returnedValue.result.value; -}; - -export const findElements = async (page: Page, args: { using: string; value: string }) => { - switch (args.using) { - case 'css selector': - case 'tag name': - return await querySelectorAll(page, { selector: args.value }); - default: - throw Error(`Unsupported findElements strategy ${args.using}`); - } -}; - -export const querySelectorAll = async (page: Page, args: { selector: string }) => { - const cdpSession = await getCDPSession(page); - const rootDocumentNode = await cdpSession.send('DOM.getDocument', { depth: -1 }); - const returned = await cdpSession.send('DOM.querySelectorAll', { - nodeId: rootDocumentNode.root.nodeId, - selector: args.selector, - }); - const resolvedNodesPromises = returned.nodeIds.map( - async (nodeId) => await cdpSession.send('DOM.resolveNode', { nodeId }), - ); - const resolvedNodes = await Promise.all(resolvedNodesPromises); - const returnValue = resolvedNodes.map((node) => ({ [WEBDRIVER_ELEMENT_KEY]: node.object.objectId })); - return returnValue; -}; - -export const getCurrentUrl = async (page: Page) => { - const cdpSession = await getCDPSession(page); - const returned = await cdpSession.send('Page.getNavigationHistory'); - const returnValue = returned.entries[returned.currentIndex].url; - return returnValue; -}; - -export const executeScript = async (page: Page, args: { script: string; args: any[] }) => { - const functionDeclaration = `function() { ${args.script} }`; - const functionArgs = args.args.map((arg) => { - if (typeof arg === 'boolean' || typeof arg === 'string' || typeof arg === 'number') { - return { value: arg }; - } else if (arg && typeof arg === 'object' && Reflect.has(arg, WEBDRIVER_ELEMENT_KEY)) { - return { objectId: arg[WEBDRIVER_ELEMENT_KEY] }; - } else { - return { value: undefined }; - } - }); - - const cdpSession = await getCDPSession(page); - await cdpSession.send('Runtime.enable'); - const window = await cdpSession.send('Runtime.evaluate', { expression: 'window' }); - - const returnedRef = await cdpSession.send('Runtime.callFunctionOn', { - objectId: window.result.objectId, - functionDeclaration, - arguments: functionArgs, - }); - - if (returnedRef.result.className === 'NodeList') { - const nodeProperties = await cdpSession.send('Runtime.getProperties', { - objectId: returnedRef.result.objectId!, - ownProperties: true, - }); - return nodeProperties.result - .map((e: any) => (!isNaN(parseInt(e.name, 10)) ? { [WEBDRIVER_ELEMENT_KEY]: e.value?.objectId } : null)) - .filter((e: any) => e); - } else if (returnedRef.result.className === 'HTMLHtmlElement') { - return { [WEBDRIVER_ELEMENT_KEY]: returnedRef.result.objectId }; - } else { - const returnedValue = await cdpSession.send('Runtime.callFunctionOn', { - objectId: window.result.objectId, - functionDeclaration, - arguments: functionArgs, - returnByValue: true, - }); - - return returnedValue.result.value; - } -}; - -type CDPSession = Awaited['newCDPSession']>>; diff --git a/packages/web-integration/src/common/page.d.ts b/packages/web-integration/src/common/page.d.ts index 442939ff..72d1b0fa 100644 --- a/packages/web-integration/src/common/page.d.ts +++ b/packages/web-integration/src/common/page.d.ts @@ -1,5 +1,5 @@ import type { Page as PlaywrightPage } from 'playwright'; -import type { Page as PuppeteerPage, KeyInput } from 'puppeteer'; +import type { KeyInput, Page as PuppeteerPage } from 'puppeteer'; export type WebPage = PlaywrightPage | PuppeteerPage; export type WebKeyInput = KeyInput; diff --git a/packages/web-integration/src/common/task-cache.ts b/packages/web-integration/src/common/task-cache.ts index aded39a2..ed99e2d1 100644 --- a/packages/web-integration/src/common/task-cache.ts +++ b/packages/web-integration/src/common/task-cache.ts @@ -1,5 +1,5 @@ -import { AIElementParseResponse, PlanningAction } from '@midscene/core'; -import { WebUIContext } from './utils'; +import type { AIElementParseResponse, PlanningAction } from '@midscene/core'; +import type { WebUIContext } from './utils'; export type PlanTask = { type: 'plan'; @@ -61,8 +61,16 @@ export class TaskCache { * @param userPrompt String type, representing user prompt information * @return Returns a Promise object that resolves to a boolean or object */ - readCache(pageContext: WebUIContext, type: 'plan', userPrompt: string): PlanTask['response']; - readCache(pageContext: WebUIContext, type: 'locate', userPrompt: string): LocateTask['response']; + readCache( + pageContext: WebUIContext, + type: 'plan', + userPrompt: string, + ): PlanTask['response']; + readCache( + pageContext: WebUIContext, + type: 'locate', + userPrompt: string, + ): LocateTask['response']; readCache( pageContext: WebUIContext, type: 'plan' | 'locate', @@ -111,7 +119,10 @@ export class TaskCache { } } - pageContextEqual(taskPageContext: LocateTask['pageContext'], pageContext: WebUIContext) { + pageContextEqual( + taskPageContext: LocateTask['pageContext'], + pageContext: WebUIContext, + ) { return ( taskPageContext.size.width === pageContext.size.width && taskPageContext.size.height === pageContext.size.height diff --git a/packages/web-integration/src/common/tasks.ts b/packages/web-integration/src/common/tasks.ts index ec816805..66dad219 100644 --- a/packages/web-integration/src/common/tasks.ts +++ b/packages/web-integration/src/common/tasks.ts @@ -1,32 +1,32 @@ -import assert from 'assert'; +import assert from 'node:assert'; +import type { WebPage } from '@/common/page'; import Insight, { - AIElementParseResponse, - DumpSubscriber, - ExecutionDump, - ExecutionRecorderItem, - ExecutionTaskActionApply, - ExecutionTaskApply, - ExecutionTaskInsightLocateApply, - ExecutionTaskInsightQueryApply, - ExecutionTaskPlanningApply, + type AIElementParseResponse, + type DumpSubscriber, + type ExecutionDump, + type ExecutionRecorderItem, + type ExecutionTaskActionApply, + type ExecutionTaskApply, + type ExecutionTaskInsightLocateApply, + type ExecutionTaskInsightQueryApply, + type ExecutionTaskPlanningApply, Executor, - InsightDump, - InsightExtractParam, + type InsightDump, + type InsightExtractParam, plan, - PlanningAction, - PlanningActionParamHover, - PlanningActionParamInputOrKeyPress, - PlanningActionParamScroll, - PlanningActionParamTap, + type PlanningAction, + type PlanningActionParamHover, + type PlanningActionParamInputOrKeyPress, + type PlanningActionParamScroll, + type PlanningActionParamTap, } from '@midscene/core'; -import { commonScreenshotParam, getTmpFile, sleep } from '@midscene/core/utils'; import { base64Encoded } from '@midscene/core/image'; +import { commonScreenshotParam, getTmpFile, sleep } from '@midscene/core/utils'; +import type { ChatCompletionMessageParam } from 'openai/resources'; import type { KeyInput, Page as PuppeteerPage } from 'puppeteer'; -import { ChatCompletionMessageParam } from 'openai/resources'; -import { WebElementInfo } from '../web-element'; -import { parseContextFromWebPage, WebUIContext } from './utils'; -import { AiTaskCache, TaskCache } from './task-cache'; -import { WebPage } from '@/common/page'; +import type { WebElementInfo } from '../web-element'; +import { type AiTaskCache, TaskCache } from './task-cache'; +import { type WebUIContext, parseContextFromWebPage } from './utils'; export class PageTaskExecutor { page: WebPage; @@ -60,7 +60,9 @@ export class PageTaskExecutor { return item; } - private wrapExecutorWithScreenshot(taskApply: ExecutionTaskApply): ExecutionTaskApply { + private wrapExecutorWithScreenshot( + taskApply: ExecutionTaskApply, + ): ExecutionTaskApply { const taskWithScreenshot: ExecutionTaskApply = { ...taskApply, executor: async (param, context, ...args) => { @@ -97,7 +99,11 @@ export class PageTaskExecutor { }; this.insight.onceDumpUpdatedFn = dumpCollector; const pageContext = await this.insight.contextRetrieverFn(); - const locateCache = this.taskCache.readCache(pageContext, 'locate', param.prompt); + const locateCache = this.taskCache.readCache( + pageContext, + 'locate', + param.prompt, + ); let locateResult: AIElementParseResponse | undefined; const callAI = this.insight.aiVendorFn; const element = await this.insight.locate(param.prompt, { @@ -137,85 +143,109 @@ export class PageTaskExecutor { }, }; return taskFind; - } else if (plan.type === 'Input') { - const taskActionInput: ExecutionTaskActionApply = { - type: 'Action', - subType: 'Input', - param: plan.param, - executor: async (taskParam, { element }) => { - if (element) { - await this.page.mouse.click(element.center[0], element.center[1]); - } - assert(taskParam.value, 'No value to input'); - await this.page.keyboard.type(taskParam.value); - }, - }; + } + if (plan.type === 'Input') { + const taskActionInput: ExecutionTaskActionApply = + { + type: 'Action', + subType: 'Input', + param: plan.param, + executor: async (taskParam, { element }) => { + if (element) { + await this.page.mouse.click( + element.center[0], + element.center[1], + ); + } + assert(taskParam.value, 'No value to input'); + await this.page.keyboard.type(taskParam.value); + }, + }; return taskActionInput; - } else if (plan.type === 'KeyboardPress') { - const taskActionKeyboardPress: ExecutionTaskActionApply = { - type: 'Action', - subType: 'KeyboardPress', - param: plan.param, - executor: async (taskParam) => { - assert(taskParam.value, 'No key to press'); - await this.page.keyboard.press(taskParam.value as KeyInput); - }, - }; + } + if (plan.type === 'KeyboardPress') { + const taskActionKeyboardPress: ExecutionTaskActionApply = + { + type: 'Action', + subType: 'KeyboardPress', + param: plan.param, + executor: async (taskParam) => { + assert(taskParam.value, 'No key to press'); + await this.page.keyboard.press(taskParam.value as KeyInput); + }, + }; return taskActionKeyboardPress; - } else if (plan.type === 'Tap') { - const taskActionTap: ExecutionTaskActionApply = { - type: 'Action', - subType: 'Tap', - executor: async (param, { element }) => { - assert(element, 'Element not found, cannot tap'); - await this.page.mouse.click(element.center[0], element.center[1]); - }, - }; + } + if (plan.type === 'Tap') { + const taskActionTap: ExecutionTaskActionApply = + { + type: 'Action', + subType: 'Tap', + executor: async (param, { element }) => { + assert(element, 'Element not found, cannot tap'); + await this.page.mouse.click( + element.center[0], + element.center[1], + ); + }, + }; return taskActionTap; - } else if (plan.type === 'Hover') { - const taskActionHover: ExecutionTaskActionApply = { - type: 'Action', - subType: 'Hover', - executor: async (param, { element }) => { - // console.log('executor args', param, element); - assert(element, 'Element not found, cannot hover'); - await this.page.mouse.move(element.center[0], element.center[1]); - }, - }; + } + if (plan.type === 'Hover') { + const taskActionHover: ExecutionTaskActionApply = + { + type: 'Action', + subType: 'Hover', + executor: async (param, { element }) => { + // console.log('executor args', param, element); + assert(element, 'Element not found, cannot hover'); + await this.page.mouse.move( + element.center[0], + element.center[1], + ); + }, + }; return taskActionHover; - } else if (plan.type === 'Scroll') { - const taskActionScroll: ExecutionTaskActionApply = { - type: 'Action', - subType: 'Scroll', - param: plan.param, - executor: async (taskParam) => { - const scrollToEventName = taskParam.scrollType; - const innerHeight = await (this.page as PuppeteerPage).evaluate(() => window.innerHeight); + } + if (plan.type === 'Scroll') { + const taskActionScroll: ExecutionTaskActionApply = + { + type: 'Action', + subType: 'Scroll', + param: plan.param, + executor: async (taskParam) => { + const scrollToEventName = taskParam.scrollType; + const innerHeight = await (this.page as PuppeteerPage).evaluate( + () => window.innerHeight, + ); - switch (scrollToEventName) { - case 'ScrollUntilTop': - await this.page.mouse.wheel(0, -9999999); - break; - case 'ScrollUntilBottom': - await this.page.mouse.wheel(0, 9999999); - break; - case 'ScrollUp': - await this.page.mouse.wheel(0, -innerHeight); - break; - case 'ScrollDown': - await this.page.mouse.wheel(0, innerHeight); - break; - default: - console.error('Unknown scroll event type:', scrollToEventName); - } - }, - }; + switch (scrollToEventName) { + case 'ScrollUntilTop': + await this.page.mouse.wheel(0, -9999999); + break; + case 'ScrollUntilBottom': + await this.page.mouse.wheel(0, 9999999); + break; + case 'ScrollUp': + await this.page.mouse.wheel(0, -innerHeight); + break; + case 'ScrollDown': + await this.page.mouse.wheel(0, innerHeight); + break; + default: + console.error( + 'Unknown scroll event type:', + scrollToEventName, + ); + } + }, + }; return taskActionScroll; - } else if (plan.type === 'Error') { + } + if (plan.type === 'Error') { throw new Error(`Got a task plan with type Error: ${plan.thought}`); - } else { - throw new Error(`Unknown or Unsupported task type: ${plan.type}`); } + throw new Error(`Unknown or Unsupported task type: ${plan.type}`); }) .map((task: ExecutionTaskApply) => { return this.wrapExecutorWithScreenshot(task); @@ -224,7 +254,9 @@ export class PageTaskExecutor { return tasks; } - async action(userPrompt: string /* , actionInfo?: { actionType?: EventActions[number]['action'] } */) { + async action( + userPrompt: string /* , actionInfo?: { actionType?: EventActions[number]['action'] } */, + ) { const taskExecutor = new Executor(userPrompt); taskExecutor.description = userPrompt; @@ -237,7 +269,11 @@ export class PageTaskExecutor { executor: async (param) => { const pageContext = await this.insight.contextRetrieverFn(); let planResult: { plans: PlanningAction[] }; - const planCache = this.taskCache.readCache(pageContext, 'plan', userPrompt); + const planCache = this.taskCache.readCache( + pageContext, + 'plan', + userPrompt, + ); if (planCache) { planResult = planCache; } else { diff --git a/packages/web-integration/src/common/utils.ts b/packages/web-integration/src/common/utils.ts index 2ac4a021..6f769eba 100644 --- a/packages/web-integration/src/common/utils.ts +++ b/packages/web-integration/src/common/utils.ts @@ -1,12 +1,16 @@ -import fs, { readFileSync } from 'fs'; -import assert from 'assert'; -import { Buffer } from 'buffer'; -import path from 'path'; -import { UIContext, PlaywrightParserOpt } from '@midscene/core'; -import { alignCoordByTrim, base64Encoded, imageInfoOfBase64 } from '@midscene/core/image'; +import assert from 'node:assert'; +import type { Buffer } from 'node:buffer'; +import fs, { readFileSync } from 'node:fs'; +import path from 'node:path'; +import type { PlaywrightParserOpt, UIContext } from '@midscene/core'; +import { + alignCoordByTrim, + base64Encoded, + imageInfoOfBase64, +} from '@midscene/core/image'; import { getTmpFile } from '@midscene/core/utils'; -import { WebElementInfo, WebElementInfoType } from '../web-element'; -import { WebPage } from './page'; +import { WebElementInfo, type WebElementInfoType } from '../web-element'; +import type { WebPage } from './page'; export type WebUIContext = UIContext & { url: string; @@ -25,7 +29,11 @@ export async function parseContextFromWebPage( const screenshotBase64 = base64Encoded(file); const captureElementSnapshot = await getElementInfosFromPage(page); // align element - const elementsInfo = await alignElements(screenshotBuffer, captureElementSnapshot, page); + const elementsInfo = await alignElements( + screenshotBuffer, + captureElementSnapshot, + page, + ); const size = await imageInfoOfBase64(screenshotBase64); return { diff --git a/packages/web-integration/src/extractor/constants.ts b/packages/web-integration/src/extractor/constants.ts index 07c1b69c..85cde422 100644 --- a/packages/web-integration/src/extractor/constants.ts +++ b/packages/web-integration/src/extractor/constants.ts @@ -3,8 +3,8 @@ export const TEXT_SIZE_THRESHOLD = 9; export const TEXT_MAX_SIZE = 40; export enum NodeType { - 'INPUT' = 'INPUT Node', - 'BUTTON' = 'BUTTON Node', - 'IMG' = 'IMG Node', - 'TEXT' = 'TEXT Node', + INPUT = 'INPUT Node', + BUTTON = 'BUTTON Node', + IMG = 'IMG Node', + TEXT = 'TEXT Node', } diff --git a/packages/web-integration/src/extractor/extractor.ts b/packages/web-integration/src/extractor/extractor.ts index 824dcb77..b2d66e14 100644 --- a/packages/web-integration/src/extractor/extractor.ts +++ b/packages/web-integration/src/extractor/extractor.ts @@ -1,3 +1,5 @@ +import { NodeType, TEXT_SIZE_THRESHOLD } from './constants'; +import { isButtonElement, isImgElement, isInputElement } from './dom-util'; import { generateHash, getNodeAttributes, @@ -7,8 +9,6 @@ import { validTextNodeContent, visibleRect, } from './util'; -import { NodeType, TEXT_SIZE_THRESHOLD } from './constants'; -import { isButtonElement, isImgElement, isInputElement } from './dom-util'; interface NodeDescriptor { node: Node; @@ -19,7 +19,7 @@ export interface ElementInfo { id: string; indexId: string; nodeHashId: string; - locator: string | void; + locator: string | undefined; attributes: { nodeType: NodeType; [key: string]: string; @@ -40,7 +40,9 @@ function generateId(numberId: number) { return `${numberId}`; } -export function extractTextWithPositionDFS(initNode: Node = container): ElementInfo[] { +export function extractTextWithPositionDFS( + initNode: Node = container, +): ElementInfo[] { const elementInfoArray: ElementInfo[] = []; const nodeMapTree: NodeDescriptor = { node: initNode, children: [] }; let nodeIndex = 1; @@ -86,7 +88,10 @@ export function extractTextWithPositionDFS(initNode: Node = container): ElementI }, content: attributes.placeholder || '', rect, - center: [Math.round(rect.left + rect.width / 2), Math.round(rect.top + rect.height / 2)], + center: [ + Math.round(rect.left + rect.width / 2), + Math.round(rect.top + rect.height / 2), + ], }); return; } @@ -108,7 +113,10 @@ export function extractTextWithPositionDFS(initNode: Node = container): ElementI }, content, rect, - center: [Math.round(rect.left + rect.width / 2), Math.round(rect.top + rect.height / 2)], + center: [ + Math.round(rect.left + rect.width / 2), + Math.round(rect.top + rect.height / 2), + ], }); return; } @@ -128,7 +136,10 @@ export function extractTextWithPositionDFS(initNode: Node = container): ElementI }, content: '', rect, - center: [Math.round(rect.left + rect.width / 2), Math.round(rect.top + rect.height / 2)], + center: [ + Math.round(rect.left + rect.width / 2), + Math.round(rect.top + rect.height / 2), + ], }); return; } @@ -155,7 +166,10 @@ export function extractTextWithPositionDFS(initNode: Node = container): ElementI const text = validTextNodeContent(node); if (text) { - if (rect.width < TEXT_SIZE_THRESHOLD || rect.height < TEXT_SIZE_THRESHOLD) { + if ( + rect.width < TEXT_SIZE_THRESHOLD || + rect.height < TEXT_SIZE_THRESHOLD + ) { logger('Element is too small', text); return; } @@ -171,7 +185,10 @@ export function extractTextWithPositionDFS(initNode: Node = container): ElementI nodeType: NodeType.TEXT, }, locator: selector, - center: [Math.round(rect.left + rect.width / 2), Math.round(rect.top + rect.height / 2)], + center: [ + Math.round(rect.left + rect.width / 2), + Math.round(rect.top + rect.height / 2), + ], // attributes, content: text, rect, diff --git a/packages/web-integration/src/extractor/util.ts b/packages/web-integration/src/extractor/util.ts index dbcb736a..ccb3ed94 100644 --- a/packages/web-integration/src/extractor/util.ts +++ b/packages/web-integration/src/extractor/util.ts @@ -15,7 +15,10 @@ function selectorForValue(val: number | string): string { return `[${taskIdKey}='${val}']`; } -export function setDataForNode(node: HTMLElement | Node, nodeHash: string): string { +export function setDataForNode( + node: HTMLElement | Node, + nodeHash: string, +): string { const taskId = taskIdKey; if (!(node instanceof HTMLElement)) { return ''; @@ -30,12 +33,19 @@ export function setDataForNode(node: HTMLElement | Node, nodeHash: string): stri return selector; } -export function getPseudoElementContent(element: Node): { before: string; after: string } { +export function getPseudoElementContent(element: Node): { + before: string; + after: string; +} { if (!(element instanceof HTMLElement)) { return { before: '', after: '' }; } - const beforeContent = window.getComputedStyle(element, '::before').getPropertyValue('content'); - const afterContent = window.getComputedStyle(element, '::after').getPropertyValue('content'); + const beforeContent = window + .getComputedStyle(element, '::before') + .getPropertyValue('content'); + const afterContent = window + .getComputedStyle(element, '::after') + .getPropertyValue('content'); return { before: beforeContent === 'none' ? '' : beforeContent.replace(/"/g, ''), after: afterContent === 'none' ? '' : afterContent.replace(/"/g, ''), @@ -44,7 +54,11 @@ export function getPseudoElementContent(element: Node): { before: string; after: export function hasOverflowY(element: HTMLElement): boolean { const style = window.getComputedStyle(element); - return style.overflowY === 'scroll' || style.overflowY === 'auto' || style.overflowY === 'hidden'; + return ( + style.overflowY === 'scroll' || + style.overflowY === 'auto' || + style.overflowY === 'hidden' + ); } export function visibleRect( @@ -81,8 +95,11 @@ export function visibleRect( const isInViewport = rect.top >= 0 + scrollTop && rect.left >= 0 + scrollLeft && - rect.bottom <= (window.innerHeight || document.documentElement.clientHeight) + scrollTop && - rect.right <= (window.innerWidth || document.documentElement.clientWidth) + scrollLeft; + rect.bottom <= + (window.innerHeight || document.documentElement.clientHeight) + + scrollTop && + rect.right <= + (window.innerWidth || document.documentElement.clientWidth) + scrollLeft; if (!isInViewport) { logger('Element is not in the viewport'); @@ -142,7 +159,9 @@ export function validTextNodeContent(node: Node): string | false { return false; } -export function getNodeAttributes(node: HTMLElement | Node): Record { +export function getNodeAttributes( + node: HTMLElement | Node, +): Record { if (!node || !(node instanceof HTMLElement) || !node.attributes) { return {}; } diff --git a/packages/web-integration/src/img/img.ts b/packages/web-integration/src/img/img.ts index f53ba5c1..9661145e 100644 --- a/packages/web-integration/src/img/img.ts +++ b/packages/web-integration/src/img/img.ts @@ -1,7 +1,7 @@ -import assert from 'assert'; +import assert from 'node:assert'; import { Buffer } from 'node:buffer'; +import type { NodeType } from '@/extractor/constants'; import sharp from 'sharp'; -import { NodeType } from '@/extractor/constants'; // Define picture path type ElementType = { @@ -16,7 +16,11 @@ type ElementType = { }; }; -const createSvgOverlay = (elements: Array, imageWidth: number, imageHeight: number) => { +const createSvgOverlay = ( + elements: Array, + imageWidth: number, + imageHeight: number, +) => { let svgContent = ``; // Define color array @@ -26,7 +30,7 @@ const createSvgOverlay = (elements: Array, imageWidth: number, imag ]; // Define clipping path - svgContent += ``; + svgContent += ''; elements.forEach((element, index) => { svgContent += ` @@ -34,7 +38,7 @@ const createSvgOverlay = (elements: Array, imageWidth: number, imag `; }); - svgContent += ``; + svgContent += ''; elements.forEach((element, index) => { // Calculate the width and height of the text @@ -74,7 +78,7 @@ const createSvgOverlay = (elements: Array, imageWidth: number, imag `; }); - svgContent += ``; + svgContent += ''; return Buffer.from(svgContent); }; @@ -93,8 +97,16 @@ export const processImageElementInfo = async (options: { if (width && height) { // Create svg overlay - const svgOverlay = createSvgOverlay(options.elementsPositionInfo, width, height); - const svgOverlayWithoutText = createSvgOverlay(options.elementsPositionInfoWithoutText, width, height); + const svgOverlay = createSvgOverlay( + options.elementsPositionInfo, + width, + height, + ); + const svgOverlayWithoutText = createSvgOverlay( + options.elementsPositionInfoWithoutText, + width, + height, + ); // Composite picture const compositeElementInfoImgBase64 = await sharp(imageBuffer) @@ -126,7 +138,6 @@ export const processImageElementInfo = async (options: { compositeElementInfoImgBase64, compositeElementInfoImgWithoutTextBase64, }; - } else { - throw Error('Image processing failed because width or height is undefined'); } + throw Error('Image processing failed because width or height is undefined'); }; diff --git a/packages/web-integration/src/img/util.ts b/packages/web-integration/src/img/util.ts index 9a114a2a..461c62e7 100644 --- a/packages/web-integration/src/img/util.ts +++ b/packages/web-integration/src/img/util.ts @@ -1,9 +1,10 @@ -import { getElementInfosFromPage } from '../common/utils'; import { NodeType } from '@/extractor/constants'; -import { ElementInfo } from '@/extractor/extractor'; +import type { ElementInfo } from '@/extractor/extractor'; +import { getElementInfosFromPage } from '../common/utils'; export async function getElementInfos(page: any) { - const captureElementSnapshot: Array = await getElementInfosFromPage(page); + const captureElementSnapshot: Array = + await getElementInfosFromPage(page); const elementsPositionInfo = captureElementSnapshot.map((elementInfo) => { return { label: elementInfo.indexId.toString(), @@ -14,12 +15,14 @@ export async function getElementInfos(page: any) { attributes: elementInfo.attributes, }; }); - const elementsPositionInfoWithoutText = elementsPositionInfo.filter((elementInfo) => { - if (elementInfo.attributes.nodeType === NodeType.TEXT) { - return false; - } - return true; - }); + const elementsPositionInfoWithoutText = elementsPositionInfo.filter( + (elementInfo) => { + if (elementInfo.attributes.nodeType === NodeType.TEXT) { + return false; + } + return true; + }, + ); return { elementsPositionInfo, captureElementSnapshot, diff --git a/packages/web-integration/src/playwright/cache.ts b/packages/web-integration/src/playwright/cache.ts index 46838308..d27058f5 100644 --- a/packages/web-integration/src/playwright/cache.ts +++ b/packages/web-integration/src/playwright/cache.ts @@ -1,10 +1,14 @@ -import path, { join } from 'path'; -import fs from 'fs'; -import { writeDumpFile, getDumpDirPath } from '@midscene/core/utils'; -import { AiTaskCache } from '@/common/task-cache'; +import fs from 'node:fs'; +import path, { join } from 'node:path'; +import type { AiTaskCache } from '@/common/task-cache'; import { findNearestPackageJson } from '@/common/utils'; +import { getDumpDirPath, writeDumpFile } from '@midscene/core/utils'; -export function writeTestCache(taskFile: string, taskTitle: string, taskCacheJson: AiTaskCache) { +export function writeTestCache( + taskFile: string, + taskTitle: string, + taskCacheJson: AiTaskCache, +) { const packageJson = getPkgInfo(); writeDumpFile({ fileName: `${taskFile}(${taskTitle})`, @@ -25,13 +29,19 @@ export function writeTestCache(taskFile: string, taskTitle: string, taskCacheJso } export function readTestCache(taskFile: string, taskTitle: string) { - const cacheFile = join(getDumpDirPath('cache'), `${taskFile}(${taskTitle}).json`); + const cacheFile = join( + getDumpDirPath('cache'), + `${taskFile}(${taskTitle}).json`, + ); const pkgInfo = getPkgInfo(); if (process.env.MIDSCENE_CACHE === 'true' && fs.existsSync(cacheFile)) { try { const data = fs.readFileSync(cacheFile, 'utf8'); const jsonData = JSON.parse(data); - if (jsonData.pkgName !== pkgInfo.name || jsonData.pkgVersion !== pkgInfo.version) { + if ( + jsonData.pkgName !== pkgInfo.name || + jsonData.pkgVersion !== pkgInfo.version + ) { return undefined; } return jsonData as AiTaskCache; diff --git a/packages/web-integration/src/playwright/index.ts b/packages/web-integration/src/playwright/index.ts index 4fa73f48..14c9d127 100644 --- a/packages/web-integration/src/playwright/index.ts +++ b/packages/web-integration/src/playwright/index.ts @@ -1,10 +1,10 @@ -import { randomUUID } from 'crypto'; +import { randomUUID } from 'node:crypto'; +import { PageAgent } from '@/common/agent'; +import type { WebPage } from '@/common/page'; +import type { TestInfo, TestType } from '@playwright/test'; import type { Page as PlaywrightPage } from 'playwright'; -import { TestInfo, TestType } from '@playwright/test'; -import { PageTaskExecutor } from '../common/tasks'; +import type { PageTaskExecutor } from '../common/tasks'; import { readTestCache, writeTestCache } from './cache'; -import { WebPage } from '@/common/page'; -import { PageAgent } from '@/common/agent'; export type APITestType = Pick, 'step'>; @@ -14,7 +14,7 @@ const groupAndCaseForTest = (testInfo: TestInfo) => { const titlePath = [...testInfo.titlePath]; if (titlePath.length > 1) { - taskTitle = titlePath.pop()!; + taskTitle = titlePath.pop() || 'unnamed'; taskFile = `${titlePath.join(' > ')}:${testInfo.line}`; } else if (titlePath.length === 1) { taskTitle = titlePath[0]; @@ -29,12 +29,17 @@ const groupAndCaseForTest = (testInfo: TestInfo) => { const midSceneAgentKeyId = '_midSceneAgentId'; export const PlaywrightAiFixture = () => { const pageAgentMap: Record = {}; - const agentForPage = (page: WebPage, opts: { testId: string; taskFile: string; taskTitle: string }) => { + const agentForPage = ( + page: WebPage, + opts: { testId: string; taskFile: string; taskTitle: string }, + ) => { let idForPage = (page as any)[midSceneAgentKeyId]; if (!idForPage) { idForPage = randomUUID(); (page as any)[midSceneAgentKeyId] = idForPage; - const testCase = readTestCache(opts.taskFile, opts.taskTitle) || { aiTasks: [] }; + const testCase = readTestCache(opts.taskFile, opts.taskTitle) || { + aiTasks: [], + }; pageAgentMap[idForPage] = new PageAgent(page, { testId: `${opts.testId}-${idForPage}`, taskFile: opts.taskFile, @@ -45,15 +50,25 @@ export const PlaywrightAiFixture = () => { }; return { - ai: async ({ page }: { page: PlaywrightPage }, use: any, testInfo: TestInfo) => { + ai: async ( + { page }: { page: PlaywrightPage }, + use: any, + testInfo: TestInfo, + ) => { const { taskFile, taskTitle } = groupAndCaseForTest(testInfo); - const agent = agentForPage(page, { testId: testInfo.testId, taskFile, taskTitle }); - await use(async (taskPrompt: string, opts?: { type?: 'action' | 'query' }) => { - await page.waitForLoadState('networkidle'); - const actionType = opts?.type || 'action'; - const result = await agent.ai(taskPrompt, actionType); - return result; + const agent = agentForPage(page, { + testId: testInfo.testId, + taskFile, + taskTitle, }); + await use( + async (taskPrompt: string, opts?: { type?: 'action' | 'query' }) => { + await page.waitForLoadState('networkidle'); + const actionType = opts?.type || 'action'; + const result = await agent.ai(taskPrompt, actionType); + return result; + }, + ); const taskCacheJson = agent.actionAgent.taskCache.generateTaskCache(); writeTestCache(taskFile, taskTitle, taskCacheJson); if (agent.dumpFile) { @@ -66,9 +81,17 @@ export const PlaywrightAiFixture = () => { }); } }, - aiAction: async ({ page }: { page: PlaywrightPage }, use: any, testInfo: TestInfo) => { + aiAction: async ( + { page }: { page: PlaywrightPage }, + use: any, + testInfo: TestInfo, + ) => { const { taskFile, taskTitle } = groupAndCaseForTest(testInfo); - const agent = agentForPage(page, { testId: testInfo.testId, taskFile, taskTitle }); + const agent = agentForPage(page, { + testId: testInfo.testId, + taskFile, + taskTitle, + }); await use(async (taskPrompt: string) => { await page.waitForLoadState('networkidle'); await agent.aiAction(taskPrompt); @@ -83,10 +106,18 @@ export const PlaywrightAiFixture = () => { }); } }, - aiQuery: async ({ page }: { page: PlaywrightPage }, use: any, testInfo: TestInfo) => { + aiQuery: async ( + { page }: { page: PlaywrightPage }, + use: any, + testInfo: TestInfo, + ) => { const { taskFile, taskTitle } = groupAndCaseForTest(testInfo); - const agent = agentForPage(page, { testId: testInfo.testId, taskFile, taskTitle }); - await use(async function (demand: any) { + const agent = agentForPage(page, { + testId: testInfo.testId, + taskFile, + taskTitle, + }); + await use(async (demand: any) => { await page.waitForLoadState('networkidle'); const result = await agent.aiQuery(demand); return result; @@ -105,7 +136,10 @@ export const PlaywrightAiFixture = () => { }; export type PlayWrightAiFixtureType = { - ai: (prompt: string, opts?: { type?: 'action' | 'query' }) => Promise; + ai: ( + prompt: string, + opts?: { type?: 'action' | 'query' }, + ) => Promise; aiAction: (taskPrompt: string) => ReturnType; aiQuery: (demand: any) => Promise; }; diff --git a/packages/web-integration/src/playwright/reporter/index.ts b/packages/web-integration/src/playwright/reporter/index.ts index b16d9293..5cbaf44c 100644 --- a/packages/web-integration/src/playwright/reporter/index.ts +++ b/packages/web-integration/src/playwright/reporter/index.ts @@ -6,7 +6,7 @@ import type { TestCase, TestResult, } from '@playwright/test/reporter'; -import { TestData } from './type'; +import type { TestData } from './type'; import { generateTestData } from './util'; const testDataList: Array = []; @@ -35,8 +35,11 @@ class MidSceneReporter implements Reporter { return false; }); aiActionTestData.forEach((testData) => { - const parseData = JSON.parse(testData.description!); - if (parseData.testId === test.id && !testDataList.find((item) => item.testId === test.id)) { + const parseData = JSON.parse(testData?.description || '{}'); + if ( + parseData.testId === test.id && + !testDataList.find((item) => item.testId === test.id) + ) { testDataList.push({ testId: test.id, title: test.title, diff --git a/packages/web-integration/src/playwright/reporter/select-cache-file.ts b/packages/web-integration/src/playwright/reporter/select-cache-file.ts index 94b4fa84..bd7ffce4 100644 --- a/packages/web-integration/src/playwright/reporter/select-cache-file.ts +++ b/packages/web-integration/src/playwright/reporter/select-cache-file.ts @@ -1,5 +1,5 @@ -import fs from 'fs'; -import path from 'path'; +import fs from 'node:fs'; +import path from 'node:path'; import inquirer from 'inquirer'; interface Task { @@ -103,7 +103,8 @@ export const getTask = async (): Promise => { const { selectedTasks } = await inquirer.prompt([ { type: 'checkbox', - message: action === 'select' ? 'Select tasks to run' : 'Select tasks to exclude', + message: + action === 'select' ? 'Select tasks to run' : 'Select tasks to exclude', name: 'selectedTasks', choices: taskChoices, }, @@ -111,10 +112,14 @@ export const getTask = async (): Promise => { if (action === 'select') { // Retain tasks based on user selection - tasksFile.aiTasks = tasksFile.aiTasks.filter((_, index) => selectedTasks.includes(index)); + tasksFile.aiTasks = tasksFile.aiTasks.filter((_, index) => + selectedTasks.includes(index), + ); } else if (action === 'exclude') { // Exclude tasks based on user selection - tasksFile.aiTasks = tasksFile.aiTasks.filter((_, index) => !selectedTasks.includes(index)); + tasksFile.aiTasks = tasksFile.aiTasks.filter( + (_, index) => !selectedTasks.includes(index), + ); } // Write the updated tasks back to the JSON file diff --git a/packages/web-integration/src/playwright/reporter/util.ts b/packages/web-integration/src/playwright/reporter/util.ts index 27c56490..dff35b53 100644 --- a/packages/web-integration/src/playwright/reporter/util.ts +++ b/packages/web-integration/src/playwright/reporter/util.ts @@ -1,25 +1,28 @@ -import path from 'path'; -import assert from 'assert'; -import os from 'os'; -import fsExtra from 'fs-extra'; -import { TestData } from './type'; +import assert from 'node:assert'; +import os from 'node:os'; +import path from 'node:path'; import { findNearestPackageJson } from '@/common/utils'; +import fsExtra from 'fs-extra'; +import type { TestData } from './type'; export function generateTestData(testDataList: Array) { - const filterDataList = testDataList.reduce((res, testData) => { - if (res.find((item) => item.testId === testData.testId)) { - return res; - } else { + const filterDataList = testDataList.reduce( + (res, testData) => { + if (res.find((item) => item.testId === testData.testId)) { + return res; + } + // biome-ignore lint/performance/noAccumulatingSpread: return [...res, testData]; - } - }, [] as Array); + }, + [] as Array, + ); const reportDir = findNearestPackageJson(__dirname); assert(reportDir, `can't get reportDir from ${__dirname}`); const targetReportDir = path.join(process.cwd(), 'midscene_run', 'report'); // Copy the contents of the report html folder to the report folder - const reportHtmlDir = path.join(reportDir, `dist/visualizer-report`); + const reportHtmlDir = path.join(reportDir, 'dist/visualizer-report'); const tempDir = path.join(os.tmpdir(), 'temp-folder'); try { // First copy to the temporary directory diff --git a/packages/web-integration/src/web-element.ts b/packages/web-integration/src/web-element.ts index 2f4e3807..fa1de4ea 100644 --- a/packages/web-integration/src/web-element.ts +++ b/packages/web-integration/src/web-element.ts @@ -1,12 +1,12 @@ -import { BaseElement, Rect } from '@midscene/core'; -import { NodeType } from './extractor/constants'; -import { WebPage } from './common/page'; +import type { BaseElement, Rect } from '@midscene/core'; +import type { WebPage } from './common/page'; +import type { NodeType } from './extractor/constants'; export interface WebElementInfoType extends BaseElement { id: string; locator: string; attributes: { - ['nodeType']: NodeType; + nodeType: NodeType; [key: string]: string; }; } @@ -25,7 +25,7 @@ export class WebElementInfo implements BaseElement { id: string; attributes: { - ['nodeType']: NodeType; + nodeType: NodeType; [key: string]: string; }; @@ -43,13 +43,16 @@ export class WebElementInfo implements BaseElement { locator: string; id: string; attributes: { - ['nodeType']: NodeType; + nodeType: NodeType; [key: string]: string; }; }) { this.content = content; this.rect = rect; - this.center = [Math.floor(rect.left + rect.width / 2), Math.floor(rect.top + rect.height / 2)]; + this.center = [ + Math.floor(rect.left + rect.width / 2), + Math.floor(rect.top + rect.height / 2), + ]; this.page = page; this.locator = locator; this.id = id; diff --git a/packages/web-integration/tests/e2e/ai-auto-todo.spec.ts b/packages/web-integration/tests/e2e/ai-auto-todo.spec.ts index 0ea58dd1..e8784f70 100644 --- a/packages/web-integration/tests/e2e/ai-auto-todo.spec.ts +++ b/packages/web-integration/tests/e2e/ai-auto-todo.spec.ts @@ -6,9 +6,15 @@ test.beforeEach(async ({ page }) => { }); test('ai todo', async ({ ai, aiQuery }) => { - await ai('Enter "Learn JS today" in the task box, then press Enter to create'); - await ai('Enter "Learn Rust tomorrow" in the task box, then press Enter to create'); - await ai('Enter "Learning AI the day after tomorrow" in the task box, then press Enter to create'); + await ai( + 'Enter "Learn JS today" in the task box, then press Enter to create', + ); + await ai( + 'Enter "Learn Rust tomorrow" in the task box, then press Enter to create', + ); + await ai( + 'Enter "Learning AI the day after tomorrow" in the task box, then press Enter to create', + ); await ai( 'Move your mouse over the second item in the task list and click the Delete button to the right of the second task', ); @@ -19,6 +25,9 @@ test('ai todo', async ({ ai, aiQuery }) => { expect(taskList.length).toBe(1); expect(taskList[0]).toBe('Learning AI the day after tomorrow'); - const placeholder = await ai('string, return the placeholder text in the input box', { type: 'query' }); + const placeholder = await ai( + 'string, return the placeholder text in the input box', + { type: 'query' }, + ); expect(placeholder).toBe('What needs to be done?'); }); diff --git a/packages/web-integration/tests/e2e/tool.ts b/packages/web-integration/tests/e2e/tool.ts index 74582b18..37865ae6 100644 --- a/packages/web-integration/tests/e2e/tool.ts +++ b/packages/web-integration/tests/e2e/tool.ts @@ -1,43 +1,72 @@ -import { existsSync, mkdirSync, writeFileSync } from 'fs'; -import path from 'path'; -import { Page as PlaywrightPage } from '@playwright/test'; -import { resizeImg, saveBase64Image } from '@midscene/core/image'; -import { getElementInfos } from '@/img/util'; +import { existsSync, mkdirSync, writeFileSync } from 'node:fs'; +import path from 'node:path'; import { processImageElementInfo } from '@/img/img'; +import { getElementInfos } from '@/img/util'; +import { resizeImg, saveBase64Image } from '@midscene/core/image'; +import type { Page as PlaywrightPage } from '@playwright/test'; -export async function generateTestData(page: PlaywrightPage, targetDir: string, inputImgBase64: string) { - const { elementsPositionInfo, captureElementSnapshot, elementsPositionInfoWithoutText } = - await getElementInfos(page); +export async function generateTestData( + page: PlaywrightPage, + targetDir: string, + inputImgBase64: string, +) { + const { + elementsPositionInfo, + captureElementSnapshot, + elementsPositionInfoWithoutText, + } = await getElementInfos(page); const inputImagePath = path.join(targetDir, 'input.png'); const outputImagePath = path.join(targetDir, 'output.png'); - const outputWithoutTextImgPath = path.join(targetDir, 'output_without_text.png'); + const outputWithoutTextImgPath = path.join( + targetDir, + 'output_without_text.png', + ); const resizeOutputImgPath = path.join(targetDir, 'resize-output.png'); const snapshotJsonPath = path.join(targetDir, 'element-snapshot.json'); - const { compositeElementInfoImgBase64, compositeElementInfoImgWithoutTextBase64 } = - await processImageElementInfo({ - elementsPositionInfo, - elementsPositionInfoWithoutText, - inputImgBase64, - }); + const { + compositeElementInfoImgBase64, + compositeElementInfoImgWithoutTextBase64, + } = await processImageElementInfo({ + elementsPositionInfo, + elementsPositionInfoWithoutText, + inputImgBase64, + }); const resizeImgBase64 = await resizeImg(inputImgBase64); - writeFileSyncWithDir(snapshotJsonPath, JSON.stringify(captureElementSnapshot, null, 2)); - await saveBase64Image({ base64Data: inputImgBase64, outputPath: inputImagePath }); - await saveBase64Image({ base64Data: compositeElementInfoImgBase64, outputPath: outputImagePath }); + writeFileSyncWithDir( + snapshotJsonPath, + JSON.stringify(captureElementSnapshot, null, 2), + ); + await saveBase64Image({ + base64Data: inputImgBase64, + outputPath: inputImagePath, + }); + await saveBase64Image({ + base64Data: compositeElementInfoImgBase64, + outputPath: outputImagePath, + }); await saveBase64Image({ base64Data: compositeElementInfoImgWithoutTextBase64, outputPath: outputWithoutTextImgPath, }); - await saveBase64Image({ base64Data: resizeImgBase64, outputPath: resizeOutputImgPath }); + await saveBase64Image({ + base64Data: resizeImgBase64, + outputPath: resizeOutputImgPath, + }); } export function generateTestDataPath(testDataName: string) { // `dist/lib/index.js` Is the default export path - const modulePath = require.resolve('@midscene/core').replace('dist/lib/index.js', ''); - const midsceneTestDataPath = path.join(modulePath, `tests/ai-model/inspector/test-data/${testDataName}`); + const modulePath = require + .resolve('@midscene/core') + .replace('dist/lib/index.js', ''); + const midsceneTestDataPath = path.join( + modulePath, + `tests/ai-model/inspector/test-data/${testDataName}`, + ); return midsceneTestDataPath; } diff --git a/packages/web-integration/tests/puppeteer/showcase.spec.ts b/packages/web-integration/tests/puppeteer/showcase.spec.ts index 9d87b7fd..bf4aa959 100644 --- a/packages/web-integration/tests/puppeteer/showcase.spec.ts +++ b/packages/web-integration/tests/puppeteer/showcase.spec.ts @@ -1,7 +1,7 @@ -import { it, describe, expect, vi } from 'vitest'; +import { PuppeteerAgent } from '@/puppeteer'; import { sleep } from '@midscene/core/utils'; +import { describe, expect, it, vi } from 'vitest'; import { launchPage } from './utils'; -import { PuppeteerAgent } from '@/puppeteer'; vi.setConfig({ testTimeout: 60 * 1000, diff --git a/packages/web-integration/tests/puppeteer/utils.ts b/packages/web-integration/tests/puppeteer/utils.ts index 40743b42..4572c3e8 100644 --- a/packages/web-integration/tests/puppeteer/utils.ts +++ b/packages/web-integration/tests/puppeteer/utils.ts @@ -1,5 +1,5 @@ -import assert from 'assert'; -import puppeteer, { Viewport } from 'puppeteer'; +import assert from 'node:assert'; +import puppeteer, { type Viewport } from 'puppeteer'; export async function launchPage( url: string, @@ -26,7 +26,10 @@ export async function launchPage( (async () => { const response = await page.goto(url); if (response?.status) { - assert(response.status() <= 399, `Page load failed: ${response.status()}`); + assert( + response.status() <= 399, + `Page load failed: ${response.status()}`, + ); } })(), ]); diff --git a/packages/web-integration/tests/task-cache.test.ts b/packages/web-integration/tests/task-cache.test.ts index c9b1c03d..f7030054 100644 --- a/packages/web-integration/tests/task-cache.test.ts +++ b/packages/web-integration/tests/task-cache.test.ts @@ -1,8 +1,8 @@ -import { describe, it, expect, beforeEach } from 'vitest'; -import { AIElementParseResponse } from '@midscene/core'; -import { LocateTask, PlanTask, TaskCache } from '@/common/task-cache'; -import { WebElementInfo } from '@/web-element'; -import { WebUIContext } from '@/common/utils'; +import { type LocateTask, type PlanTask, TaskCache } from '@/common/task-cache'; +import type { WebUIContext } from '@/common/utils'; +import type { WebElementInfo } from '@/web-element'; +import type { AIElementParseResponse } from '@midscene/core'; +import { beforeEach, describe, expect, it } from 'vitest'; describe('TaskCache', () => { let taskCache: TaskCache; @@ -23,15 +23,30 @@ describe('TaskCache', () => { }); it('should return false if no cache is available', async () => { - const result = taskCache.readCache(formalPageContext, 'plan', 'test prompt'); + const result = taskCache.readCache( + formalPageContext, + 'plan', + 'test prompt', + ); expect(result).toBe(false); }); it('should return false if the prompt does not match', async () => { taskCache.cache = { - aiTasks: [{ type: 'plan', prompt: 'different prompt', pageContext, response: { plans: [] } }], + aiTasks: [ + { + type: 'plan', + prompt: 'different prompt', + pageContext, + response: { plans: [] }, + }, + ], }; - const result = taskCache.readCache(formalPageContext, 'plan', 'test prompt'); + const result = taskCache.readCache( + formalPageContext, + 'plan', + 'test prompt', + ); expect(result).toBe(false); }); @@ -42,20 +57,39 @@ describe('TaskCache', () => { type: 'locate', prompt: 'test prompt', pageContext, - response: { elements: [{ id: 'element3' }] } as AIElementParseResponse, + response: { + elements: [{ id: 'element3' }], + } as AIElementParseResponse, }, ], }; - const result = taskCache.readCache(formalPageContext, 'locate', 'test prompt'); + const result = taskCache.readCache( + formalPageContext, + 'locate', + 'test prompt', + ); expect(result).toBe(false); }); it('should return cached response if the conditions match', async () => { - const cachedResponse = { plans: [{ type: 'Locate', thought: '', param: {} }] } as PlanTask['response']; + const cachedResponse = { + plans: [{ type: 'Locate', thought: '', param: {} }], + } as PlanTask['response']; taskCache.cache = { - aiTasks: [{ type: 'plan', prompt: 'test prompt', pageContext, response: cachedResponse }], + aiTasks: [ + { + type: 'plan', + prompt: 'test prompt', + pageContext, + response: cachedResponse, + }, + ], }; - const result = taskCache.readCache(formalPageContext, 'plan', 'test prompt'); + const result = taskCache.readCache( + formalPageContext, + 'plan', + 'test prompt', + ); expect(result).toEqual(cachedResponse); }); @@ -74,8 +108,14 @@ describe('TaskCache', () => { const isEqual = taskCache.pageContextEqual(pageContext, formalPageContext); expect(isEqual).toBe(true); - const differentContext = { ...formalPageContext, size: { width: 800, height: 600 } }; - const isNotEqual = taskCache.pageContextEqual(pageContext, differentContext); + const differentContext = { + ...formalPageContext, + size: { width: 800, height: 600 }, + }; + const isNotEqual = taskCache.pageContextEqual( + pageContext, + differentContext, + ); expect(isNotEqual).toBe(false); }); diff --git a/packages/web-integration/tsconfig.json b/packages/web-integration/tsconfig.json index 1839cf7c..7a1f4977 100644 --- a/packages/web-integration/tsconfig.json +++ b/packages/web-integration/tsconfig.json @@ -18,6 +18,6 @@ "skipLibCheck": true, "strict": true }, - "exclude": [ "node_modules"], + "exclude": ["node_modules"], "include": ["src", "tests", "./playwright.config.ts", "./vitest.config"] } diff --git a/packages/web-integration/vitest.config.ts b/packages/web-integration/vitest.config.ts index f728b386..51a2c047 100644 --- a/packages/web-integration/vitest.config.ts +++ b/packages/web-integration/vitest.config.ts @@ -1,9 +1,10 @@ +import path from 'node:path'; import { defineConfig } from 'vitest/config'; -import path from 'path'; const enableTest = process.env.AITEST; -const aiModelTest = enableTest !== 'true' ? ['tests/puppeteer/bing.test.ts']: []; +const aiModelTest = + enableTest !== 'true' ? ['tests/puppeteer/bing.test.ts'] : []; export default defineConfig({ resolve: { @@ -13,6 +14,6 @@ export default defineConfig({ }, test: { include: ['./tests/**/*.test.ts'], - exclude: [...aiModelTest] + exclude: [...aiModelTest], }, }); diff --git a/scripts/release.js b/scripts/release.js index 4f1cf009..6c34f9da 100644 --- a/scripts/release.js +++ b/scripts/release.js @@ -1,4 +1,4 @@ -const fs = require('fs'); +const fs = require('node:fs'); const semver = require('semver'); const dayjs = require('dayjs'); const args = require('minimist')(process.argv.slice(2)); @@ -96,7 +96,7 @@ async function bumpVersion() { let version = args.version; if (version && actionPublishCanary) { const hash = dayjs().format('YYYYMMDDHHmmss'); - version = semver.inc(currentVersion, version, 'beta-' + hash); + version = semver.inc(currentVersion, version, `beta-${hash}`); } return await bumpPrompt({