@rpascene/core 0.30.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (200) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +9 -0
  3. package/dist/es/agent/agent.mjs +636 -0
  4. package/dist/es/agent/agent.mjs.map +1 -0
  5. package/dist/es/agent/common.mjs +0 -0
  6. package/dist/es/agent/index.mjs +6 -0
  7. package/dist/es/agent/task-cache.mjs +184 -0
  8. package/dist/es/agent/task-cache.mjs.map +1 -0
  9. package/dist/es/agent/tasks.mjs +666 -0
  10. package/dist/es/agent/tasks.mjs.map +1 -0
  11. package/dist/es/agent/ui-utils.mjs +72 -0
  12. package/dist/es/agent/ui-utils.mjs.map +1 -0
  13. package/dist/es/agent/utils.mjs +162 -0
  14. package/dist/es/agent/utils.mjs.map +1 -0
  15. package/dist/es/ai-model/action-executor.mjs +129 -0
  16. package/dist/es/ai-model/action-executor.mjs.map +1 -0
  17. package/dist/es/ai-model/common.mjs +355 -0
  18. package/dist/es/ai-model/common.mjs.map +1 -0
  19. package/dist/es/ai-model/conversation-history.mjs +58 -0
  20. package/dist/es/ai-model/conversation-history.mjs.map +1 -0
  21. package/dist/es/ai-model/index.mjs +11 -0
  22. package/dist/es/ai-model/inspect.mjs +286 -0
  23. package/dist/es/ai-model/inspect.mjs.map +1 -0
  24. package/dist/es/ai-model/llm-planning.mjs +140 -0
  25. package/dist/es/ai-model/llm-planning.mjs.map +1 -0
  26. package/dist/es/ai-model/prompt/assertion.mjs +31 -0
  27. package/dist/es/ai-model/prompt/assertion.mjs.map +1 -0
  28. package/dist/es/ai-model/prompt/common.mjs +7 -0
  29. package/dist/es/ai-model/prompt/common.mjs.map +1 -0
  30. package/dist/es/ai-model/prompt/describe.mjs +44 -0
  31. package/dist/es/ai-model/prompt/describe.mjs.map +1 -0
  32. package/dist/es/ai-model/prompt/extraction.mjs +140 -0
  33. package/dist/es/ai-model/prompt/extraction.mjs.map +1 -0
  34. package/dist/es/ai-model/prompt/llm-locator.mjs +275 -0
  35. package/dist/es/ai-model/prompt/llm-locator.mjs.map +1 -0
  36. package/dist/es/ai-model/prompt/llm-planning.mjs +367 -0
  37. package/dist/es/ai-model/prompt/llm-planning.mjs.map +1 -0
  38. package/dist/es/ai-model/prompt/llm-section-locator.mjs +47 -0
  39. package/dist/es/ai-model/prompt/llm-section-locator.mjs.map +1 -0
  40. package/dist/es/ai-model/prompt/playwright-generator.mjs +117 -0
  41. package/dist/es/ai-model/prompt/playwright-generator.mjs.map +1 -0
  42. package/dist/es/ai-model/prompt/ui-tars-locator.mjs +34 -0
  43. package/dist/es/ai-model/prompt/ui-tars-locator.mjs.map +1 -0
  44. package/dist/es/ai-model/prompt/ui-tars-planning.mjs +36 -0
  45. package/dist/es/ai-model/prompt/ui-tars-planning.mjs.map +1 -0
  46. package/dist/es/ai-model/prompt/util.mjs +124 -0
  47. package/dist/es/ai-model/prompt/util.mjs.map +1 -0
  48. package/dist/es/ai-model/prompt/yaml-generator.mjs +219 -0
  49. package/dist/es/ai-model/prompt/yaml-generator.mjs.map +1 -0
  50. package/dist/es/ai-model/service-caller/index.mjs +537 -0
  51. package/dist/es/ai-model/service-caller/index.mjs.map +1 -0
  52. package/dist/es/ai-model/ui-tars-planning.mjs +201 -0
  53. package/dist/es/ai-model/ui-tars-planning.mjs.map +1 -0
  54. package/dist/es/device/index.mjs +152 -0
  55. package/dist/es/device/index.mjs.map +1 -0
  56. package/dist/es/image/index.mjs +2 -0
  57. package/dist/es/index.mjs +11 -0
  58. package/dist/es/index.mjs.map +1 -0
  59. package/dist/es/insight/index.mjs +233 -0
  60. package/dist/es/insight/index.mjs.map +1 -0
  61. package/dist/es/insight/utils.mjs +15 -0
  62. package/dist/es/insight/utils.mjs.map +1 -0
  63. package/dist/es/report.mjs +88 -0
  64. package/dist/es/report.mjs.map +1 -0
  65. package/dist/es/tree.mjs +2 -0
  66. package/dist/es/types.mjs +11 -0
  67. package/dist/es/types.mjs.map +1 -0
  68. package/dist/es/utils.mjs +204 -0
  69. package/dist/es/utils.mjs.map +1 -0
  70. package/dist/es/yaml/builder.mjs +13 -0
  71. package/dist/es/yaml/builder.mjs.map +1 -0
  72. package/dist/es/yaml/index.mjs +3 -0
  73. package/dist/es/yaml/player.mjs +372 -0
  74. package/dist/es/yaml/player.mjs.map +1 -0
  75. package/dist/es/yaml/utils.mjs +73 -0
  76. package/dist/es/yaml/utils.mjs.map +1 -0
  77. package/dist/es/yaml.mjs +0 -0
  78. package/dist/lib/agent/agent.js +683 -0
  79. package/dist/lib/agent/agent.js.map +1 -0
  80. package/dist/lib/agent/common.js +5 -0
  81. package/dist/lib/agent/index.js +81 -0
  82. package/dist/lib/agent/index.js.map +1 -0
  83. package/dist/lib/agent/task-cache.js +236 -0
  84. package/dist/lib/agent/task-cache.js.map +1 -0
  85. package/dist/lib/agent/tasks.js +703 -0
  86. package/dist/lib/agent/tasks.js.map +1 -0
  87. package/dist/lib/agent/ui-utils.js +121 -0
  88. package/dist/lib/agent/ui-utils.js.map +1 -0
  89. package/dist/lib/agent/utils.js +233 -0
  90. package/dist/lib/agent/utils.js.map +1 -0
  91. package/dist/lib/ai-model/action-executor.js +163 -0
  92. package/dist/lib/ai-model/action-executor.js.map +1 -0
  93. package/dist/lib/ai-model/common.js +461 -0
  94. package/dist/lib/ai-model/common.js.map +1 -0
  95. package/dist/lib/ai-model/conversation-history.js +92 -0
  96. package/dist/lib/ai-model/conversation-history.js.map +1 -0
  97. package/dist/lib/ai-model/index.js +131 -0
  98. package/dist/lib/ai-model/index.js.map +1 -0
  99. package/dist/lib/ai-model/inspect.js +326 -0
  100. package/dist/lib/ai-model/inspect.js.map +1 -0
  101. package/dist/lib/ai-model/llm-planning.js +174 -0
  102. package/dist/lib/ai-model/llm-planning.js.map +1 -0
  103. package/dist/lib/ai-model/prompt/assertion.js +65 -0
  104. package/dist/lib/ai-model/prompt/assertion.js.map +1 -0
  105. package/dist/lib/ai-model/prompt/common.js +41 -0
  106. package/dist/lib/ai-model/prompt/common.js.map +1 -0
  107. package/dist/lib/ai-model/prompt/describe.js +78 -0
  108. package/dist/lib/ai-model/prompt/describe.js.map +1 -0
  109. package/dist/lib/ai-model/prompt/extraction.js +180 -0
  110. package/dist/lib/ai-model/prompt/extraction.js.map +1 -0
  111. package/dist/lib/ai-model/prompt/llm-locator.js +315 -0
  112. package/dist/lib/ai-model/prompt/llm-locator.js.map +1 -0
  113. package/dist/lib/ai-model/prompt/llm-planning.js +407 -0
  114. package/dist/lib/ai-model/prompt/llm-planning.js.map +1 -0
  115. package/dist/lib/ai-model/prompt/llm-section-locator.js +84 -0
  116. package/dist/lib/ai-model/prompt/llm-section-locator.js.map +1 -0
  117. package/dist/lib/ai-model/prompt/playwright-generator.js +178 -0
  118. package/dist/lib/ai-model/prompt/playwright-generator.js.map +1 -0
  119. package/dist/lib/ai-model/prompt/ui-tars-locator.js +68 -0
  120. package/dist/lib/ai-model/prompt/ui-tars-locator.js.map +1 -0
  121. package/dist/lib/ai-model/prompt/ui-tars-planning.js +73 -0
  122. package/dist/lib/ai-model/prompt/ui-tars-planning.js.map +1 -0
  123. package/dist/lib/ai-model/prompt/util.js +176 -0
  124. package/dist/lib/ai-model/prompt/util.js.map +1 -0
  125. package/dist/lib/ai-model/prompt/yaml-generator.js +280 -0
  126. package/dist/lib/ai-model/prompt/yaml-generator.js.map +1 -0
  127. package/dist/lib/ai-model/service-caller/index.js +623 -0
  128. package/dist/lib/ai-model/service-caller/index.js.map +1 -0
  129. package/dist/lib/ai-model/ui-tars-planning.js +238 -0
  130. package/dist/lib/ai-model/ui-tars-planning.js.map +1 -0
  131. package/dist/lib/device/index.js +255 -0
  132. package/dist/lib/device/index.js.map +1 -0
  133. package/dist/lib/image/index.js +56 -0
  134. package/dist/lib/image/index.js.map +1 -0
  135. package/dist/lib/index.js +103 -0
  136. package/dist/lib/index.js.map +1 -0
  137. package/dist/lib/insight/index.js +267 -0
  138. package/dist/lib/insight/index.js.map +1 -0
  139. package/dist/lib/insight/utils.js +49 -0
  140. package/dist/lib/insight/utils.js.map +1 -0
  141. package/dist/lib/report.js +122 -0
  142. package/dist/lib/report.js.map +1 -0
  143. package/dist/lib/tree.js +44 -0
  144. package/dist/lib/tree.js.map +1 -0
  145. package/dist/lib/types.js +82 -0
  146. package/dist/lib/types.js.map +1 -0
  147. package/dist/lib/utils.js +283 -0
  148. package/dist/lib/utils.js.map +1 -0
  149. package/dist/lib/yaml/builder.js +57 -0
  150. package/dist/lib/yaml/builder.js.map +1 -0
  151. package/dist/lib/yaml/index.js +80 -0
  152. package/dist/lib/yaml/index.js.map +1 -0
  153. package/dist/lib/yaml/player.js +406 -0
  154. package/dist/lib/yaml/player.js.map +1 -0
  155. package/dist/lib/yaml/utils.js +126 -0
  156. package/dist/lib/yaml/utils.js.map +1 -0
  157. package/dist/lib/yaml.js +20 -0
  158. package/dist/lib/yaml.js.map +1 -0
  159. package/dist/types/agent/agent.d.ts +156 -0
  160. package/dist/types/agent/common.d.ts +0 -0
  161. package/dist/types/agent/index.d.ts +9 -0
  162. package/dist/types/agent/task-cache.d.ts +48 -0
  163. package/dist/types/agent/tasks.d.ts +48 -0
  164. package/dist/types/agent/ui-utils.d.ts +7 -0
  165. package/dist/types/agent/utils.d.ts +52 -0
  166. package/dist/types/ai-model/action-executor.d.ts +19 -0
  167. package/dist/types/ai-model/common.d.ts +569 -0
  168. package/dist/types/ai-model/conversation-history.d.ts +18 -0
  169. package/dist/types/ai-model/index.d.ts +13 -0
  170. package/dist/types/ai-model/inspect.d.ts +46 -0
  171. package/dist/types/ai-model/llm-planning.d.ts +11 -0
  172. package/dist/types/ai-model/prompt/assertion.d.ts +2 -0
  173. package/dist/types/ai-model/prompt/common.d.ts +2 -0
  174. package/dist/types/ai-model/prompt/describe.d.ts +1 -0
  175. package/dist/types/ai-model/prompt/extraction.d.ts +4 -0
  176. package/dist/types/ai-model/prompt/llm-locator.d.ts +9 -0
  177. package/dist/types/ai-model/prompt/llm-planning.d.ts +9 -0
  178. package/dist/types/ai-model/prompt/llm-section-locator.d.ts +6 -0
  179. package/dist/types/ai-model/prompt/playwright-generator.d.ts +26 -0
  180. package/dist/types/ai-model/prompt/ui-tars-locator.d.ts +1 -0
  181. package/dist/types/ai-model/prompt/ui-tars-planning.d.ts +2 -0
  182. package/dist/types/ai-model/prompt/util.d.ts +47 -0
  183. package/dist/types/ai-model/prompt/yaml-generator.d.ts +100 -0
  184. package/dist/types/ai-model/service-caller/index.d.ts +48 -0
  185. package/dist/types/ai-model/ui-tars-planning.d.ts +59 -0
  186. package/dist/types/device/index.d.ts +2158 -0
  187. package/dist/types/image/index.d.ts +1 -0
  188. package/dist/types/index.d.ts +12 -0
  189. package/dist/types/insight/index.d.ts +31 -0
  190. package/dist/types/insight/utils.d.ts +2 -0
  191. package/dist/types/report.d.ts +12 -0
  192. package/dist/types/tree.d.ts +1 -0
  193. package/dist/types/types.d.ts +414 -0
  194. package/dist/types/utils.d.ts +40 -0
  195. package/dist/types/yaml/builder.d.ts +2 -0
  196. package/dist/types/yaml/index.d.ts +3 -0
  197. package/dist/types/yaml/player.d.ts +34 -0
  198. package/dist/types/yaml/utils.d.ts +9 -0
  199. package/dist/types/yaml.d.ts +178 -0
  200. package/package.json +108 -0
@@ -0,0 +1,204 @@
1
+ import { execSync } from "node:child_process";
2
+ import { appendFileSync, closeSync, existsSync, mkdirSync, openSync, readFileSync, readSync, statSync, truncateSync, writeFileSync } from "node:fs";
3
+ import { tmpdir } from "node:os";
4
+ import { join } from "node:path";
5
+ import { defaultRunDirName, getRpasceneRunSubDir } from "@rpascene/shared/common";
6
+ import { RPASCENE_CACHE, RPASCENE_DEBUG_MODE, globalConfigManager } from "@rpascene/shared/env";
7
+ import { getRunningPkgInfo } from "@rpascene/shared/node";
8
+ import { assert, escapeScriptTag, ifInBrowser, ifInWorker, logMsg, uuid } from "@rpascene/shared/utils";
9
+ let logEnvReady = false;
10
+ const groupedActionDumpFileExt = 'web-dump.json';
11
+ function processCacheConfig(cache, cacheId) {
12
+ if (void 0 !== cache) {
13
+ if (false === cache) return;
14
+ if (true === cache) return {
15
+ id: cacheId
16
+ };
17
+ if ('object' == typeof cache && null !== cache) {
18
+ if (!cache.id) return {
19
+ ...cache,
20
+ id: cacheId
21
+ };
22
+ return cache;
23
+ }
24
+ }
25
+ const envEnabled = globalConfigManager.getEnvConfigInBoolean(RPASCENE_CACHE);
26
+ if (envEnabled && cacheId) return {
27
+ id: cacheId
28
+ };
29
+ }
30
+ const reportInitializedMap = new Map();
31
+ function getReportTpl() {
32
+ const reportTpl = 'REPLACE_ME_WITH_REPORT_HTML';
33
+ return reportTpl;
34
+ }
35
+ function insertScriptBeforeClosingHtml(filePath, scriptContent) {
36
+ const htmlEndTag = '</html>';
37
+ const stat = statSync(filePath);
38
+ const readSize = Math.min(stat.size, 4096);
39
+ const start = Math.max(0, stat.size - readSize);
40
+ const buffer = Buffer.alloc(stat.size - start);
41
+ const fd = openSync(filePath, 'r');
42
+ readSync(fd, buffer, 0, buffer.length, start);
43
+ closeSync(fd);
44
+ const tailStr = buffer.toString('utf8');
45
+ const htmlEndIdx = tailStr.lastIndexOf(htmlEndTag);
46
+ if (-1 === htmlEndIdx) throw new Error(`No </html> found in file\u{FF1A}${filePath}`);
47
+ const beforeHtmlInTail = tailStr.slice(0, htmlEndIdx);
48
+ const htmlEndPos = start + Buffer.byteLength(beforeHtmlInTail, 'utf8');
49
+ truncateSync(filePath, htmlEndPos);
50
+ appendFileSync(filePath, `${scriptContent}\n${htmlEndTag}\n`);
51
+ }
52
+ function reportHTMLContent(dumpData, reportPath, appendReport, withTpl = true) {
53
+ let tpl = '';
54
+ if (withTpl) {
55
+ tpl = getReportTpl();
56
+ if (!tpl) {
57
+ console.warn('reportTpl is not set, will not write report');
58
+ return '';
59
+ }
60
+ }
61
+ const writeToFile = reportPath && !ifInBrowser;
62
+ let dumpContent = '';
63
+ if ('string' == typeof dumpData) dumpContent = '<script type="rpascene_web_dump" type="application/json">\n' + escapeScriptTag(dumpData) + "\n<\/script>";
64
+ else {
65
+ const { dumpString, attributes } = dumpData;
66
+ const attributesArr = Object.keys(attributes || {}).map((key)=>`${key}="${encodeURIComponent(attributes[key])}"`);
67
+ dumpContent = '<script type="rpascene_web_dump" type="application/json" ' + attributesArr.join(' ') + '>\n' + escapeScriptTag(dumpString) + "\n<\/script>";
68
+ }
69
+ if (writeToFile) {
70
+ if (!appendReport) {
71
+ writeFileSync(reportPath, tpl + dumpContent, {
72
+ flag: 'w'
73
+ });
74
+ return reportPath;
75
+ }
76
+ if (!reportInitializedMap.get(reportPath)) {
77
+ writeFileSync(reportPath, tpl, {
78
+ flag: 'w'
79
+ });
80
+ reportInitializedMap.set(reportPath, true);
81
+ }
82
+ insertScriptBeforeClosingHtml(reportPath, dumpContent);
83
+ return reportPath;
84
+ }
85
+ return tpl + dumpContent;
86
+ }
87
+ function writeDumpReport(fileName, dumpData, appendReport) {
88
+ if (ifInBrowser || ifInWorker) {
89
+ console.log('will not write report in browser');
90
+ return null;
91
+ }
92
+ const reportPath = join(getRpasceneRunSubDir('report'), `${fileName}.html`);
93
+ reportHTMLContent(dumpData, reportPath, appendReport);
94
+ if (process.env.RPASCENE_DEBUG_LOG_JSON) {
95
+ const jsonPath = `${reportPath}.json`;
96
+ let data;
97
+ data = 'string' == typeof dumpData ? JSON.parse(dumpData) : dumpData;
98
+ writeFileSync(jsonPath, JSON.stringify(data, null, 2), {
99
+ flag: appendReport ? 'a' : 'w'
100
+ });
101
+ logMsg(`Rpascene - dump file written: ${jsonPath}`);
102
+ }
103
+ return reportPath;
104
+ }
105
+ function writeLogFile(opts) {
106
+ if (ifInBrowser || ifInWorker) return '/mock/report.html';
107
+ const { fileName, fileExt, fileContent, type = 'dump' } = opts;
108
+ const targetDir = getRpasceneRunSubDir(type);
109
+ if (!logEnvReady) {
110
+ assert(targetDir, 'logDir should be set before writing dump file');
111
+ const gitIgnorePath = join(targetDir, '../../.gitignore');
112
+ const gitPath = join(targetDir, '../../.git');
113
+ let gitIgnoreContent = '';
114
+ if (existsSync(gitPath)) {
115
+ if (existsSync(gitIgnorePath)) gitIgnoreContent = readFileSync(gitIgnorePath, 'utf-8');
116
+ if (!gitIgnoreContent.includes(`${defaultRunDirName}/`)) writeFileSync(gitIgnorePath, `${gitIgnoreContent}
117
+ # RPA\u{667A}\u{80FD}\u{52A9}\u{624B} dump files
118
+ ${defaultRunDirName}/dump\n${defaultRunDirName}/report\n${defaultRunDirName}/tmp\n${defaultRunDirName}/log\n`, 'utf-8');
119
+ }
120
+ logEnvReady = true;
121
+ }
122
+ const filePath = join(targetDir, `${fileName}.${fileExt}`);
123
+ if ('dump' !== type) writeFileSync(filePath, JSON.stringify(fileContent));
124
+ if (null == opts ? void 0 : opts.generateReport) return writeDumpReport(fileName, fileContent, opts.appendReport);
125
+ return filePath;
126
+ }
127
+ function getTmpDir() {
128
+ try {
129
+ const runningPkgInfo = getRunningPkgInfo();
130
+ if (!runningPkgInfo) return null;
131
+ const { name } = runningPkgInfo;
132
+ const tmpPath = join(tmpdir(), name);
133
+ mkdirSync(tmpPath, {
134
+ recursive: true
135
+ });
136
+ return tmpPath;
137
+ } catch (e) {
138
+ return null;
139
+ }
140
+ }
141
+ function getTmpFile(fileExtWithoutDot) {
142
+ if (ifInBrowser || ifInWorker) return null;
143
+ const tmpDir = getTmpDir();
144
+ const filename = `${uuid()}.${fileExtWithoutDot}`;
145
+ return join(tmpDir, filename);
146
+ }
147
+ function overlapped(container, target) {
148
+ return container.left < target.left + target.width && container.left + container.width > target.left && container.top < target.top + target.height && container.top + container.height > target.top;
149
+ }
150
+ async function sleep(ms) {
151
+ return new Promise((resolve)=>setTimeout(resolve, ms));
152
+ }
153
+ function replacerForPageObject(_key, value) {
154
+ var _value_constructor, _value_constructor1;
155
+ if (value && (null == (_value_constructor = value.constructor) ? void 0 : _value_constructor.name) === 'Page') return '[Page object]';
156
+ if (value && (null == (_value_constructor1 = value.constructor) ? void 0 : _value_constructor1.name) === 'Browser') return '[Browser object]';
157
+ return value;
158
+ }
159
+ function stringifyDumpData(data, indents) {
160
+ return JSON.stringify(data, replacerForPageObject, indents);
161
+ }
162
+ function getVersion() {
163
+ return "0.30.8";
164
+ }
165
+ function debugLog(...message) {
166
+ const debugMode = process.env[RPASCENE_DEBUG_MODE];
167
+ if (debugMode) console.log('[Rpascene]', ...message);
168
+ }
169
+ let lastReportedRepoUrl = '';
170
+ function uploadTestInfoToServer({ testUrl, serverUrl }) {
171
+ let repoUrl = '';
172
+ let userEmail = '';
173
+ try {
174
+ repoUrl = execSync('git config --get remote.origin.url').toString().trim();
175
+ userEmail = execSync('git config --get user.email').toString().trim();
176
+ } catch (error) {
177
+ debugLog('Failed to get git info:', error);
178
+ }
179
+ if (serverUrl && (repoUrl && repoUrl !== lastReportedRepoUrl || !repoUrl && testUrl)) {
180
+ debugLog('Uploading test info to server', {
181
+ serverUrl,
182
+ repoUrl,
183
+ testUrl,
184
+ userEmail
185
+ });
186
+ fetch(serverUrl, {
187
+ method: 'POST',
188
+ headers: {
189
+ 'Content-Type': 'application/json'
190
+ },
191
+ body: JSON.stringify({
192
+ repo_url: repoUrl,
193
+ test_url: testUrl,
194
+ user_email: userEmail
195
+ })
196
+ }).then((response)=>response.json()).then((data)=>{
197
+ debugLog('Successfully uploaded test info to server:', data);
198
+ }).catch((error)=>debugLog('Failed to upload test info to server:', error));
199
+ lastReportedRepoUrl = repoUrl;
200
+ }
201
+ }
202
+ export { appendFileSync, getReportTpl, getTmpDir, getTmpFile, getVersion, groupedActionDumpFileExt, insertScriptBeforeClosingHtml, overlapped, processCacheConfig, replacerForPageObject, reportHTMLContent, sleep, stringifyDumpData, uploadTestInfoToServer, writeDumpReport, writeLogFile };
203
+
204
+ //# sourceMappingURL=utils.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.mjs","sources":["webpack://@rpascene/core/./src/utils.ts"],"sourcesContent":["import { execSync } from 'node:child_process';\nimport * as fs from 'node:fs';\nimport { existsSync, mkdirSync, readFileSync, writeFileSync } from 'node:fs';\nimport { tmpdir } from 'node:os';\nimport * as path from 'node:path';\nimport {\n defaultRunDirName,\n getRpasceneRunSubDir,\n} from '@rpascene/shared/common';\nimport {\n RPASCENE_CACHE,\n RPASCENE_DEBUG_MODE,\n globalConfigManager,\n} from '@rpascene/shared/env';\nimport { getRunningPkgInfo } from '@rpascene/shared/node';\nimport { assert, logMsg } from '@rpascene/shared/utils';\nimport {\n escapeScriptTag,\n ifInBrowser,\n ifInWorker,\n uuid,\n} from '@rpascene/shared/utils';\nimport type { Cache, Rect, ReportDumpWithAttributes } from './types';\n\nlet logEnvReady = false;\n\nexport { appendFileSync } from 'node:fs';\n\nexport const groupedActionDumpFileExt = 'web-dump.json';\n\n/**\n * Process cache configuration with environment variable support and backward compatibility.\n *\n * @param cache - The original cache configuration\n * @param cacheId - The cache ID to use as:\n * 1. Fallback ID when cache is true or cache object has no ID\n * 2. Legacy cacheId when cache is undefined (requires RPASCENE_CACHE env var)\n * @returns Processed cache configuration\n */\nexport function processCacheConfig(\n cache: Cache | undefined,\n cacheId: string,\n): Cache | undefined {\n // 1. New cache object configuration (highest priority)\n if (cache !== undefined) {\n if (cache === false) {\n return undefined; // Completely disable cache\n }\n\n if (cache === true) {\n // Auto-generate ID using cacheId for CLI/YAML scenarios\n // Agent will validate and reject this later if needed\n return { id: cacheId };\n }\n\n // cache is object configuration\n if (typeof cache === 'object' && cache !== null) {\n // Auto-generate ID using cacheId when missing (for CLI/YAML scenarios)\n if (!cache.id) {\n return { ...cache, id: cacheId };\n }\n return cache;\n }\n }\n\n // 2. Backward compatibility: support old cacheId (requires environment variable)\n // When cache is undefined, check if legacy cacheId mode is enabled via env var\n const envEnabled = globalConfigManager.getEnvConfigInBoolean(RPASCENE_CACHE);\n\n if (envEnabled && cacheId) {\n return { id: cacheId };\n }\n\n // 3. No cache configuration\n return undefined;\n}\n\nconst reportInitializedMap = new Map<string, boolean>();\n\ndeclare const __DEV_REPORT_PATH__: string;\n\nexport function getReportTpl() {\n if (typeof __DEV_REPORT_PATH__ === 'string' && __DEV_REPORT_PATH__) {\n return fs.readFileSync(__DEV_REPORT_PATH__, 'utf-8');\n }\n const reportTpl = 'REPLACE_ME_WITH_REPORT_HTML';\n\n return reportTpl;\n}\n\n/**\n * high performance, insert script before </html> in HTML file\n * only truncate and append, no temporary file\n */\nexport function insertScriptBeforeClosingHtml(\n filePath: string,\n scriptContent: string,\n): void {\n const htmlEndTag = '</html>';\n const stat = fs.statSync(filePath);\n\n const readSize = Math.min(stat.size, 4096);\n const start = Math.max(0, stat.size - readSize);\n const buffer = Buffer.alloc(stat.size - start);\n const fd = fs.openSync(filePath, 'r');\n fs.readSync(fd, buffer, 0, buffer.length, start);\n fs.closeSync(fd);\n\n const tailStr = buffer.toString('utf8');\n const htmlEndIdx = tailStr.lastIndexOf(htmlEndTag);\n if (htmlEndIdx === -1) {\n throw new Error(`No </html> found in file:${filePath}`);\n }\n\n // calculate the correct byte position: char position to byte position\n const beforeHtmlInTail = tailStr.slice(0, htmlEndIdx);\n const htmlEndPos = start + Buffer.byteLength(beforeHtmlInTail, 'utf8');\n\n // truncate to </html> before\n fs.truncateSync(filePath, htmlEndPos);\n // append script and </html>\n fs.appendFileSync(filePath, `${scriptContent}\\n${htmlEndTag}\\n`);\n}\n\nexport function reportHTMLContent(\n dumpData: string | ReportDumpWithAttributes,\n reportPath?: string,\n appendReport?: boolean,\n withTpl = true, // whether return with report template, default = true\n): string {\n let tpl = '';\n if (withTpl) {\n tpl = getReportTpl();\n\n if (!tpl) {\n console.warn('reportTpl is not set, will not write report');\n return '';\n }\n }\n // if reportPath is set, it means we are in write to file mode\n const writeToFile = reportPath && !ifInBrowser;\n let dumpContent = '';\n\n if (typeof dumpData === 'string') {\n // do not use template string here, will cause bundle error\n dumpContent =\n // biome-ignore lint/style/useTemplate: <explanation>\n '<script type=\"rpascene_web_dump\" type=\"application/json\">\\n' +\n escapeScriptTag(dumpData) +\n '\\n</script>';\n } else {\n const { dumpString, attributes } = dumpData;\n const attributesArr = Object.keys(attributes || {}).map((key) => {\n return `${key}=\"${encodeURIComponent(attributes![key])}\"`;\n });\n\n dumpContent =\n // do not use template string here, will cause bundle error\n // biome-ignore lint/style/useTemplate: <explanation>\n '<script type=\"rpascene_web_dump\" type=\"application/json\" ' +\n attributesArr.join(' ') +\n '>\\n' +\n escapeScriptTag(dumpString) +\n '\\n</script>';\n }\n\n if (writeToFile) {\n if (!appendReport) {\n writeFileSync(reportPath!, tpl + dumpContent, { flag: 'w' });\n return reportPath!;\n }\n\n if (!reportInitializedMap.get(reportPath!)) {\n writeFileSync(reportPath!, tpl, { flag: 'w' });\n reportInitializedMap.set(reportPath!, true);\n }\n\n insertScriptBeforeClosingHtml(reportPath!, dumpContent);\n return reportPath!;\n }\n\n return tpl + dumpContent;\n}\n\nexport function writeDumpReport(\n fileName: string,\n dumpData: string | ReportDumpWithAttributes,\n appendReport?: boolean,\n): string | null {\n if (ifInBrowser || ifInWorker) {\n console.log('will not write report in browser');\n return null;\n }\n\n const reportPath = path.join(\n getRpasceneRunSubDir('report'),\n `${fileName}.html`,\n );\n\n reportHTMLContent(dumpData, reportPath, appendReport);\n\n if (process.env.RPASCENE_DEBUG_LOG_JSON) {\n const jsonPath = `${reportPath}.json`;\n let data;\n\n if (typeof dumpData === 'string') {\n data = JSON.parse(dumpData) as ReportDumpWithAttributes;\n } else {\n data = dumpData;\n }\n\n writeFileSync(jsonPath, JSON.stringify(data, null, 2), {\n flag: appendReport ? 'a' : 'w',\n });\n\n logMsg(`Rpascene - dump file written: ${jsonPath}`);\n }\n\n return reportPath;\n}\n\nexport function writeLogFile(opts: {\n fileName: string;\n fileExt: string;\n fileContent: string | ReportDumpWithAttributes;\n type: 'dump' | 'cache' | 'report' | 'tmp';\n generateReport?: boolean;\n appendReport?: boolean;\n}) {\n if (ifInBrowser || ifInWorker) {\n return '/mock/report.html';\n }\n const { fileName, fileExt, fileContent, type = 'dump' } = opts;\n const targetDir = getRpasceneRunSubDir(type);\n // Ensure directory exists\n if (!logEnvReady) {\n assert(targetDir, 'logDir should be set before writing dump file');\n\n // gitIgnore in the parent directory\n const gitIgnorePath = path.join(targetDir, '../../.gitignore');\n const gitPath = path.join(targetDir, '../../.git');\n let gitIgnoreContent = '';\n\n if (existsSync(gitPath)) {\n // if the git path exists, we need to add the log folder to the git ignore file\n if (existsSync(gitIgnorePath)) {\n gitIgnoreContent = readFileSync(gitIgnorePath, 'utf-8');\n }\n\n // ignore the log folder\n if (!gitIgnoreContent.includes(`${defaultRunDirName}/`)) {\n writeFileSync(\n gitIgnorePath,\n `${gitIgnoreContent}\\n# RPA智能助手 dump files\\n${defaultRunDirName}/dump\\n${defaultRunDirName}/report\\n${defaultRunDirName}/tmp\\n${defaultRunDirName}/log\\n`,\n 'utf-8',\n );\n }\n }\n\n logEnvReady = true;\n }\n\n const filePath = path.join(targetDir, `${fileName}.${fileExt}`);\n\n if (type !== 'dump') {\n // do not write dump file any more\n writeFileSync(filePath, JSON.stringify(fileContent));\n }\n\n if (opts?.generateReport) {\n return writeDumpReport(fileName, fileContent, opts.appendReport);\n }\n\n return filePath;\n}\n\nexport function getTmpDir(): string | null {\n try {\n const runningPkgInfo = getRunningPkgInfo();\n if (!runningPkgInfo) {\n return null;\n }\n const { name } = runningPkgInfo;\n const tmpPath = path.join(tmpdir(), name);\n mkdirSync(tmpPath, { recursive: true });\n return tmpPath;\n } catch (e) {\n return null;\n }\n}\n\nexport function getTmpFile(fileExtWithoutDot: string): string | null {\n if (ifInBrowser || ifInWorker) {\n return null;\n }\n const tmpDir = getTmpDir();\n const filename = `${uuid()}.${fileExtWithoutDot}`;\n return path.join(tmpDir!, filename);\n}\n\nexport function overlapped(container: Rect, target: Rect) {\n // container and the target have some part overlapped\n return (\n container.left < target.left + target.width &&\n container.left + container.width > target.left &&\n container.top < target.top + target.height &&\n container.top + container.height > target.top\n );\n}\n\nexport async function sleep(ms: number) {\n return new Promise((resolve) => setTimeout(resolve, ms));\n}\n\nexport function replacerForPageObject(_key: string, value: any) {\n if (value && value.constructor?.name === 'Page') {\n return '[Page object]';\n }\n if (value && value.constructor?.name === 'Browser') {\n return '[Browser object]';\n }\n return value;\n}\n\nexport function stringifyDumpData(data: any, indents?: number) {\n return JSON.stringify(data, replacerForPageObject, indents);\n}\n\ndeclare const __VERSION__: string;\n\nexport function getVersion() {\n return __VERSION__;\n}\n\nfunction debugLog(...message: any[]) {\n // always read from process.env, and cannot be override by modelConfig, overrideAIConfig, etc.\n // also avoid circular dependency\n const debugMode = process.env[RPASCENE_DEBUG_MODE];\n if (debugMode) {\n console.log('[Rpascene]', ...message);\n }\n}\n\nlet lastReportedRepoUrl = '';\nexport function uploadTestInfoToServer({\n testUrl,\n serverUrl,\n}: { testUrl: string; serverUrl?: string }) {\n let repoUrl = '';\n let userEmail = '';\n\n try {\n repoUrl = execSync('git config --get remote.origin.url').toString().trim();\n userEmail = execSync('git config --get user.email').toString().trim();\n } catch (error) {\n debugLog('Failed to get git info:', error);\n }\n\n // Only upload test info if:\n // 1. Server URL is configured AND\n // 2. Either:\n // - We have a repo URL that's different from last reported one (to avoid duplicate reports)\n // - OR we don't have a repo URL but have a test URL (for non-git environments)\n if (\n serverUrl &&\n ((repoUrl && repoUrl !== lastReportedRepoUrl) || (!repoUrl && testUrl))\n ) {\n debugLog('Uploading test info to server', {\n serverUrl,\n repoUrl,\n testUrl,\n userEmail,\n });\n\n fetch(serverUrl, {\n method: 'POST',\n headers: {\n 'Content-Type': 'application/json',\n },\n body: JSON.stringify({\n repo_url: repoUrl,\n test_url: testUrl,\n user_email: userEmail,\n }),\n })\n .then((response) => response.json())\n .then((data) => {\n debugLog('Successfully uploaded test info to server:', data);\n })\n .catch((error) =>\n debugLog('Failed to upload test info to server:', error),\n );\n lastReportedRepoUrl = repoUrl;\n }\n}\n"],"names":["logEnvReady","groupedActionDumpFileExt","processCacheConfig","cache","cacheId","undefined","envEnabled","globalConfigManager","RPASCENE_CACHE","reportInitializedMap","Map","getReportTpl","reportTpl","insertScriptBeforeClosingHtml","filePath","scriptContent","htmlEndTag","stat","fs","readSize","Math","start","buffer","Buffer","fd","tailStr","htmlEndIdx","Error","beforeHtmlInTail","htmlEndPos","reportHTMLContent","dumpData","reportPath","appendReport","withTpl","tpl","console","writeToFile","ifInBrowser","dumpContent","escapeScriptTag","dumpString","attributes","attributesArr","Object","key","encodeURIComponent","writeFileSync","writeDumpReport","fileName","ifInWorker","path","getRpasceneRunSubDir","process","jsonPath","data","JSON","logMsg","writeLogFile","opts","fileExt","fileContent","type","targetDir","assert","gitIgnorePath","gitPath","gitIgnoreContent","existsSync","readFileSync","defaultRunDirName","getTmpDir","runningPkgInfo","getRunningPkgInfo","name","tmpPath","tmpdir","mkdirSync","e","getTmpFile","fileExtWithoutDot","tmpDir","filename","uuid","overlapped","container","target","sleep","ms","Promise","resolve","setTimeout","replacerForPageObject","_key","value","_value_constructor","_value_constructor1","stringifyDumpData","indents","getVersion","__VERSION__","debugLog","message","debugMode","RPASCENE_DEBUG_MODE","lastReportedRepoUrl","uploadTestInfoToServer","testUrl","serverUrl","repoUrl","userEmail","execSync","error","fetch","response"],"mappings":";;;;;;;;AAwBA,IAAIA,cAAc;AAIX,MAAMC,2BAA2B;AAWjC,SAASC,mBACdC,KAAwB,EACxBC,OAAe;IAGf,IAAID,AAAUE,WAAVF,OAAqB;QACvB,IAAIA,AAAU,UAAVA,OACF;QAGF,IAAIA,AAAU,SAAVA,OAGF,OAAO;YAAE,IAAIC;QAAQ;QAIvB,IAAI,AAAiB,YAAjB,OAAOD,SAAsBA,AAAU,SAAVA,OAAgB;YAE/C,IAAI,CAACA,MAAM,EAAE,EACX,OAAO;gBAAE,GAAGA,KAAK;gBAAE,IAAIC;YAAQ;YAEjC,OAAOD;QACT;IACF;IAIA,MAAMG,aAAaC,oBAAoB,qBAAqB,CAACC;IAE7D,IAAIF,cAAcF,SAChB,OAAO;QAAE,IAAIA;IAAQ;AAKzB;AAEA,MAAMK,uBAAuB,IAAIC;AAI1B,SAASC;IAId,MAAMC,YAAY;IAElB,OAAOA;AACT;AAMO,SAASC,8BACdC,QAAgB,EAChBC,aAAqB;IAErB,MAAMC,aAAa;IACnB,MAAMC,OAAOC,SAAYJ;IAEzB,MAAMK,WAAWC,KAAK,GAAG,CAACH,KAAK,IAAI,EAAE;IACrC,MAAMI,QAAQD,KAAK,GAAG,CAAC,GAAGH,KAAK,IAAI,GAAGE;IACtC,MAAMG,SAASC,OAAO,KAAK,CAACN,KAAK,IAAI,GAAGI;IACxC,MAAMG,KAAKN,SAAYJ,UAAU;IACjCI,SAAYM,IAAIF,QAAQ,GAAGA,OAAO,MAAM,EAAED;IAC1CH,UAAaM;IAEb,MAAMC,UAAUH,OAAO,QAAQ,CAAC;IAChC,MAAMI,aAAaD,QAAQ,WAAW,CAACT;IACvC,IAAIU,AAAe,OAAfA,YACF,MAAM,IAAIC,MAAM,CAAC,gCAAyB,EAAEb,UAAU;IAIxD,MAAMc,mBAAmBH,QAAQ,KAAK,CAAC,GAAGC;IAC1C,MAAMG,aAAaR,QAAQE,OAAO,UAAU,CAACK,kBAAkB;IAG/DV,aAAgBJ,UAAUe;IAE1BX,eAAkBJ,UAAU,GAAGC,cAAc,EAAE,EAAEC,WAAW,EAAE,CAAC;AACjE;AAEO,SAASc,kBACdC,QAA2C,EAC3CC,UAAmB,EACnBC,YAAsB,EACtBC,UAAU,IAAI;IAEd,IAAIC,MAAM;IACV,IAAID,SAAS;QACXC,MAAMxB;QAEN,IAAI,CAACwB,KAAK;YACRC,QAAQ,IAAI,CAAC;YACb,OAAO;QACT;IACF;IAEA,MAAMC,cAAcL,cAAc,CAACM;IACnC,IAAIC,cAAc;IAElB,IAAI,AAAoB,YAApB,OAAOR,UAETQ,cAEE,gEACAC,gBAAgBT,YAChB;SACG;QACL,MAAM,EAAEU,UAAU,EAAEC,UAAU,EAAE,GAAGX;QACnC,MAAMY,gBAAgBC,OAAO,IAAI,CAACF,cAAc,CAAC,GAAG,GAAG,CAAC,CAACG,MAChD,GAAGA,IAAI,EAAE,EAAEC,mBAAmBJ,UAAW,CAACG,IAAI,EAAE,CAAC,CAAC;QAG3DN,cAGE,8DACAI,cAAc,IAAI,CAAC,OACnB,QACAH,gBAAgBC,cAChB;IACJ;IAEA,IAAIJ,aAAa;QACf,IAAI,CAACJ,cAAc;YACjBc,cAAcf,YAAaG,MAAMI,aAAa;gBAAE,MAAM;YAAI;YAC1D,OAAOP;QACT;QAEA,IAAI,CAACvB,qBAAqB,GAAG,CAACuB,aAAc;YAC1Ce,cAAcf,YAAaG,KAAK;gBAAE,MAAM;YAAI;YAC5C1B,qBAAqB,GAAG,CAACuB,YAAa;QACxC;QAEAnB,8BAA8BmB,YAAaO;QAC3C,OAAOP;IACT;IAEA,OAAOG,MAAMI;AACf;AAEO,SAASS,gBACdC,QAAgB,EAChBlB,QAA2C,EAC3CE,YAAsB;IAEtB,IAAIK,eAAeY,YAAY;QAC7Bd,QAAQ,GAAG,CAAC;QACZ,OAAO;IACT;IAEA,MAAMJ,aAAamB,KACjBC,qBAAqB,WACrB,GAAGH,SAAS,KAAK,CAAC;IAGpBnB,kBAAkBC,UAAUC,YAAYC;IAExC,IAAIoB,QAAQ,GAAG,CAAC,uBAAuB,EAAE;QACvC,MAAMC,WAAW,GAAGtB,WAAW,KAAK,CAAC;QACrC,IAAIuB;QAGFA,OADE,AAAoB,YAApB,OAAOxB,WACFyB,KAAK,KAAK,CAACzB,YAEXA;QAGTgB,cAAcO,UAAUE,KAAK,SAAS,CAACD,MAAM,MAAM,IAAI;YACrD,MAAMtB,eAAe,MAAM;QAC7B;QAEAwB,OAAO,CAAC,8BAA8B,EAAEH,UAAU;IACpD;IAEA,OAAOtB;AACT;AAEO,SAAS0B,aAAaC,IAO5B;IACC,IAAIrB,eAAeY,YACjB,OAAO;IAET,MAAM,EAAED,QAAQ,EAAEW,OAAO,EAAEC,WAAW,EAAEC,OAAO,MAAM,EAAE,GAAGH;IAC1D,MAAMI,YAAYX,qBAAqBU;IAEvC,IAAI,CAAC9D,aAAa;QAChBgE,OAAOD,WAAW;QAGlB,MAAME,gBAAgBd,KAAUY,WAAW;QAC3C,MAAMG,UAAUf,KAAUY,WAAW;QACrC,IAAII,mBAAmB;QAEvB,IAAIC,WAAWF,UAAU;YAEvB,IAAIE,WAAWH,gBACbE,mBAAmBE,aAAaJ,eAAe;YAIjD,IAAI,CAACE,iBAAiB,QAAQ,CAAC,GAAGG,kBAAkB,CAAC,CAAC,GACpDvB,cACEkB,eACA,GAAGE,iBAAiB;;AAAwB,EAAEG,kBAAkB,OAAO,EAAEA,kBAAkB,SAAS,EAAEA,kBAAkB,MAAM,EAAEA,kBAAkB,MAAM,CAAC,EACzJ;QAGN;QAEAtE,cAAc;IAChB;IAEA,MAAMc,WAAWqC,KAAUY,WAAW,GAAGd,SAAS,CAAC,EAAEW,SAAS;IAE9D,IAAIE,AAAS,WAATA,MAEFf,cAAcjC,UAAU0C,KAAK,SAAS,CAACK;IAGzC,IAAIF,QAAAA,OAAAA,KAAAA,IAAAA,KAAM,cAAc,EACtB,OAAOX,gBAAgBC,UAAUY,aAAaF,KAAK,YAAY;IAGjE,OAAO7C;AACT;AAEO,SAASyD;IACd,IAAI;QACF,MAAMC,iBAAiBC;QACvB,IAAI,CAACD,gBACH,OAAO;QAET,MAAM,EAAEE,IAAI,EAAE,GAAGF;QACjB,MAAMG,UAAUxB,KAAUyB,UAAUF;QACpCG,UAAUF,SAAS;YAAE,WAAW;QAAK;QACrC,OAAOA;IACT,EAAE,OAAOG,GAAG;QACV,OAAO;IACT;AACF;AAEO,SAASC,WAAWC,iBAAyB;IAClD,IAAI1C,eAAeY,YACjB,OAAO;IAET,MAAM+B,SAASV;IACf,MAAMW,WAAW,GAAGC,OAAO,CAAC,EAAEH,mBAAmB;IACjD,OAAO7B,KAAU8B,QAASC;AAC5B;AAEO,SAASE,WAAWC,SAAe,EAAEC,MAAY;IAEtD,OACED,UAAU,IAAI,GAAGC,OAAO,IAAI,GAAGA,OAAO,KAAK,IAC3CD,UAAU,IAAI,GAAGA,UAAU,KAAK,GAAGC,OAAO,IAAI,IAC9CD,UAAU,GAAG,GAAGC,OAAO,GAAG,GAAGA,OAAO,MAAM,IAC1CD,UAAU,GAAG,GAAGA,UAAU,MAAM,GAAGC,OAAO,GAAG;AAEjD;AAEO,eAAeC,MAAMC,EAAU;IACpC,OAAO,IAAIC,QAAQ,CAACC,UAAYC,WAAWD,SAASF;AACtD;AAEO,SAASI,sBAAsBC,IAAY,EAAEC,KAAU;QAC/CC,oBAGAC;IAHb,IAAIF,SAASC,AAAAA,SAAAA,CAAAA,qBAAAA,MAAM,WAAW,AAAD,IAAhBA,KAAAA,IAAAA,mBAAmB,IAAI,AAAD,MAAM,QACvC,OAAO;IAET,IAAID,SAASE,AAAAA,SAAAA,CAAAA,sBAAAA,MAAM,WAAW,AAAD,IAAhBA,KAAAA,IAAAA,oBAAmB,IAAI,AAAD,MAAM,WACvC,OAAO;IAET,OAAOF;AACT;AAEO,SAASG,kBAAkB1C,IAAS,EAAE2C,OAAgB;IAC3D,OAAO1C,KAAK,SAAS,CAACD,MAAMqC,uBAAuBM;AACrD;AAIO,SAASC;IACd,OAAOC;AACT;AAEA,SAASC,SAAS,GAAGC,OAAc;IAGjC,MAAMC,YAAYlD,QAAQ,GAAG,CAACmD,oBAAoB;IAClD,IAAID,WACFnE,QAAQ,GAAG,CAAC,iBAAiBkE;AAEjC;AAEA,IAAIG,sBAAsB;AACnB,SAASC,uBAAuB,EACrCC,OAAO,EACPC,SAAS,EAC+B;IACxC,IAAIC,UAAU;IACd,IAAIC,YAAY;IAEhB,IAAI;QACFD,UAAUE,SAAS,sCAAsC,QAAQ,GAAG,IAAI;QACxED,YAAYC,SAAS,+BAA+B,QAAQ,GAAG,IAAI;IACrE,EAAE,OAAOC,OAAO;QACdX,SAAS,2BAA2BW;IACtC;IAOA,IACEJ,aACEC,CAAAA,WAAWA,YAAYJ,uBAAyB,CAACI,WAAWF,OAAM,GACpE;QACAN,SAAS,iCAAiC;YACxCO;YACAC;YACAF;YACAG;QACF;QAEAG,MAAML,WAAW;YACf,QAAQ;YACR,SAAS;gBACP,gBAAgB;YAClB;YACA,MAAMpD,KAAK,SAAS,CAAC;gBACnB,UAAUqD;gBACV,UAAUF;gBACV,YAAYG;YACd;QACF,GACG,IAAI,CAAC,CAACI,WAAaA,SAAS,IAAI,IAChC,IAAI,CAAC,CAAC3D;YACL8C,SAAS,8CAA8C9C;QACzD,GACC,KAAK,CAAC,CAACyD,QACNX,SAAS,yCAAyCW;QAEtDP,sBAAsBI;IACxB;AACF"}
@@ -0,0 +1,13 @@
1
+ import js_yaml from "js-yaml";
2
+ function buildYaml(env, tasks) {
3
+ const result = {
4
+ target: env,
5
+ tasks
6
+ };
7
+ return js_yaml.dump(result, {
8
+ indent: 2
9
+ });
10
+ }
11
+ export { buildYaml };
12
+
13
+ //# sourceMappingURL=builder.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"yaml\\builder.mjs","sources":["webpack://@rpascene/core/./src/yaml/builder.ts"],"sourcesContent":["import yaml from 'js-yaml';\nimport type {\n RpasceneYamlScript,\n RpasceneYamlScriptWebEnv,\n RpasceneYamlTask,\n} from '../types';\n\nexport function buildYaml(\n env: RpasceneYamlScriptWebEnv,\n tasks: RpasceneYamlTask[],\n) {\n const result: RpasceneYamlScript = {\n target: env,\n tasks,\n };\n\n return yaml.dump(result, {\n indent: 2,\n });\n}\n"],"names":["buildYaml","env","tasks","result","yaml"],"mappings":";AAOO,SAASA,UACdC,GAA6B,EAC7BC,KAAyB;IAEzB,MAAMC,SAA6B;QACjC,QAAQF;QACRC;IACF;IAEA,OAAOE,QAAAA,IAAS,CAACD,QAAQ;QACvB,QAAQ;IACV;AACF"}
@@ -0,0 +1,3 @@
1
+ export * from "./player.mjs";
2
+ export * from "./builder.mjs";
3
+ export * from "./utils.mjs";
@@ -0,0 +1,372 @@
1
+ import { existsSync, mkdirSync, writeFileSync } from "node:fs";
2
+ import { basename, dirname, join, resolve as external_node_path_resolve } from "node:path";
3
+ import { assert, ifInBrowser, ifInWorker } from "@rpascene/shared/utils";
4
+ import { getRpasceneRunSubDir } from "@rpascene/shared/common";
5
+ import { getDebug } from "@rpascene/shared/logger";
6
+ import { buildDetailedLocateParam, buildDetailedLocateParamAndRestParams } from "./utils.mjs";
7
+ function _define_property(obj, key, value) {
8
+ if (key in obj) Object.defineProperty(obj, key, {
9
+ value: value,
10
+ enumerable: true,
11
+ configurable: true,
12
+ writable: true
13
+ });
14
+ else obj[key] = value;
15
+ return obj;
16
+ }
17
+ const debug = getDebug('yaml-player');
18
+ class ScriptPlayer {
19
+ setResult(key, value) {
20
+ const keyToUse = key || this.unnamedResultIndex++;
21
+ if (this.result[keyToUse]) console.warn(`result key ${keyToUse} already exists, will overwrite`);
22
+ this.result[keyToUse] = value;
23
+ return this.flushResult();
24
+ }
25
+ setPlayerStatus(status, error) {
26
+ this.status = status;
27
+ this.errorInSetup = error;
28
+ }
29
+ notifyCurrentTaskStatusChange(taskIndex) {
30
+ const taskIndexToNotify = 'number' == typeof taskIndex ? taskIndex : this.currentTaskIndex;
31
+ if ('number' != typeof taskIndexToNotify) return;
32
+ const taskStatus = this.taskStatusList[taskIndexToNotify];
33
+ if (this.onTaskStatusChange) this.onTaskStatusChange(taskStatus);
34
+ }
35
+ async setTaskStatus(index, statusValue, error) {
36
+ this.taskStatusList[index].status = statusValue;
37
+ if (error) this.taskStatusList[index].error = error;
38
+ this.notifyCurrentTaskStatusChange(index);
39
+ }
40
+ setTaskIndex(taskIndex) {
41
+ this.currentTaskIndex = taskIndex;
42
+ }
43
+ flushResult() {
44
+ if (this.output) {
45
+ const output = external_node_path_resolve(process.cwd(), this.output);
46
+ const outputDir = dirname(output);
47
+ if (!existsSync(outputDir)) mkdirSync(outputDir, {
48
+ recursive: true
49
+ });
50
+ writeFileSync(output, JSON.stringify(this.result || {}, void 0, 2));
51
+ }
52
+ }
53
+ flushUnstableLogContent() {
54
+ if (this.unstableLogContent) {
55
+ var _this_interfaceAgent;
56
+ const content = null == (_this_interfaceAgent = this.interfaceAgent) ? void 0 : _this_interfaceAgent._unstableLogContent();
57
+ const filePath = external_node_path_resolve(process.cwd(), this.unstableLogContent);
58
+ const outputDir = dirname(filePath);
59
+ if (!existsSync(outputDir)) mkdirSync(outputDir, {
60
+ recursive: true
61
+ });
62
+ writeFileSync(filePath, JSON.stringify(content, null, 2));
63
+ }
64
+ }
65
+ async playTask(taskStatus, agent) {
66
+ const { flow } = taskStatus;
67
+ assert(flow, 'missing flow in task');
68
+ for(const flowItemIndex in flow){
69
+ const currentStep = Number.parseInt(flowItemIndex, 10);
70
+ taskStatus.currentStep = currentStep;
71
+ const flowItem = flow[flowItemIndex];
72
+ debug(`playing step ${flowItemIndex}, flowItem=${JSON.stringify(flowItem)}`);
73
+ if ('aiAction' in flowItem || 'ai' in flowItem) {
74
+ const actionTask = flowItem;
75
+ const prompt = actionTask.aiAction || actionTask.ai;
76
+ assert(prompt, 'missing prompt for ai (aiAction)');
77
+ await agent.aiAction(prompt, {
78
+ cacheable: actionTask.cacheable
79
+ });
80
+ } else if ('aiAssert' in flowItem) {
81
+ const assertTask = flowItem;
82
+ const prompt = assertTask.aiAssert;
83
+ const msg = assertTask.errorMessage;
84
+ assert(prompt, 'missing prompt for aiAssert');
85
+ const { pass, thought, message } = await agent.aiAssert(prompt, msg, {
86
+ keepRawResponse: true
87
+ }) || {};
88
+ this.setResult(assertTask.name, {
89
+ pass,
90
+ thought,
91
+ message
92
+ });
93
+ if (!pass) throw new Error(message);
94
+ } else if ('aiQuery' in flowItem) {
95
+ const queryTask = flowItem;
96
+ const prompt = queryTask.aiQuery;
97
+ const options = {
98
+ domIncluded: queryTask.domIncluded,
99
+ screenshotIncluded: queryTask.screenshotIncluded
100
+ };
101
+ assert(prompt, 'missing prompt for aiQuery');
102
+ const queryResult = await agent.aiQuery(prompt, options);
103
+ this.setResult(queryTask.name, queryResult);
104
+ } else if ('aiNumber' in flowItem) {
105
+ const numberTask = flowItem;
106
+ const prompt = numberTask.aiNumber;
107
+ const options = {
108
+ domIncluded: numberTask.domIncluded,
109
+ screenshotIncluded: numberTask.screenshotIncluded
110
+ };
111
+ assert(prompt, 'missing prompt for aiNumber');
112
+ const numberResult = await agent.aiNumber(prompt, options);
113
+ this.setResult(numberTask.name, numberResult);
114
+ } else if ('aiString' in flowItem) {
115
+ const stringTask = flowItem;
116
+ const prompt = stringTask.aiString;
117
+ const options = {
118
+ domIncluded: stringTask.domIncluded,
119
+ screenshotIncluded: stringTask.screenshotIncluded
120
+ };
121
+ assert(prompt, 'missing prompt for aiString');
122
+ const stringResult = await agent.aiString(prompt, options);
123
+ this.setResult(stringTask.name, stringResult);
124
+ } else if ('aiBoolean' in flowItem) {
125
+ const booleanTask = flowItem;
126
+ const prompt = booleanTask.aiBoolean;
127
+ const options = {
128
+ domIncluded: booleanTask.domIncluded,
129
+ screenshotIncluded: booleanTask.screenshotIncluded
130
+ };
131
+ assert(prompt, 'missing prompt for aiBoolean');
132
+ const booleanResult = await agent.aiBoolean(prompt, options);
133
+ this.setResult(booleanTask.name, booleanResult);
134
+ } else if ('aiAsk' in flowItem) {
135
+ const askTask = flowItem;
136
+ const prompt = askTask.aiAsk;
137
+ assert(prompt, 'missing prompt for aiAsk');
138
+ const askResult = await agent.aiAsk(prompt);
139
+ this.setResult(askTask.name, askResult);
140
+ } else if ('aiLocate' in flowItem) {
141
+ const locateTask = flowItem;
142
+ const prompt = locateTask.aiLocate;
143
+ assert(prompt, 'missing prompt for aiLocate');
144
+ const locateResult = await agent.aiLocate(prompt, locateTask);
145
+ this.setResult(locateTask.name, locateResult);
146
+ } else if ('aiWaitFor' in flowItem) {
147
+ const waitForTask = flowItem;
148
+ const prompt = waitForTask.aiWaitFor;
149
+ assert(prompt, 'missing prompt for aiWaitFor');
150
+ const timeout = waitForTask.timeout;
151
+ await agent.aiWaitFor(prompt, {
152
+ timeoutMs: timeout
153
+ });
154
+ } else if ('sleep' in flowItem) {
155
+ const sleepTask = flowItem;
156
+ const ms = sleepTask.sleep;
157
+ let msNumber = ms;
158
+ if ('string' == typeof ms) msNumber = Number.parseInt(ms, 10);
159
+ assert(msNumber && msNumber > 0, `ms for sleep must be greater than 0, but got ${ms}`);
160
+ await new Promise((resolve)=>setTimeout(resolve, msNumber));
161
+ } else if ("javascript" in flowItem) {
162
+ const evaluateJavaScriptTask = flowItem;
163
+ const result = await agent.evaluateJavaScript(evaluateJavaScriptTask.javascript);
164
+ this.setResult(evaluateJavaScriptTask.name, result);
165
+ } else if ('logScreenshot' in flowItem) {
166
+ const logScreenshotTask = flowItem;
167
+ await agent.logScreenshot(logScreenshotTask.logScreenshot, {
168
+ content: logScreenshotTask.content || ''
169
+ });
170
+ } else if ('aiInput' in flowItem) {
171
+ const { aiInput, ...inputTask } = flowItem;
172
+ let locatePrompt;
173
+ let value;
174
+ if (inputTask.locate) {
175
+ value = aiInput || inputTask.value;
176
+ locatePrompt = inputTask.locate;
177
+ } else {
178
+ locatePrompt = aiInput || '';
179
+ value = inputTask.value;
180
+ }
181
+ await agent.callActionInActionSpace('Input', {
182
+ ...inputTask,
183
+ ...void 0 !== value ? {
184
+ value: String(value)
185
+ } : {},
186
+ ...locatePrompt ? {
187
+ locate: buildDetailedLocateParam(locatePrompt, inputTask)
188
+ } : {}
189
+ });
190
+ } else if ('aiKeyboardPress' in flowItem) {
191
+ const { aiKeyboardPress, ...keyboardPressTask } = flowItem;
192
+ let locatePrompt;
193
+ let keyName;
194
+ if (keyboardPressTask.locate) {
195
+ keyName = aiKeyboardPress;
196
+ locatePrompt = keyboardPressTask.locate;
197
+ } else if (keyboardPressTask.keyName) {
198
+ keyName = keyboardPressTask.keyName;
199
+ locatePrompt = aiKeyboardPress;
200
+ } else keyName = aiKeyboardPress;
201
+ await agent.callActionInActionSpace('KeyboardPress', {
202
+ ...keyboardPressTask,
203
+ ...keyName ? {
204
+ keyName
205
+ } : {},
206
+ ...locatePrompt ? {
207
+ locate: buildDetailedLocateParam(locatePrompt, keyboardPressTask)
208
+ } : {}
209
+ });
210
+ } else if ('aiScroll' in flowItem) {
211
+ const { aiScroll, ...scrollTask } = flowItem;
212
+ let locatePrompt;
213
+ locatePrompt = scrollTask.locate ? scrollTask.locate : aiScroll;
214
+ await agent.callActionInActionSpace('Scroll', {
215
+ ...scrollTask,
216
+ ...locatePrompt ? {
217
+ locate: buildDetailedLocateParam(locatePrompt, scrollTask)
218
+ } : {}
219
+ });
220
+ } else {
221
+ const actionSpace = this.actionSpace;
222
+ let locatePromptShortcut;
223
+ const matchedAction = actionSpace.find((action)=>{
224
+ const actionInterfaceAlias = action.interfaceAlias;
225
+ if (actionInterfaceAlias && Object.prototype.hasOwnProperty.call(flowItem, actionInterfaceAlias)) {
226
+ locatePromptShortcut = flowItem[actionInterfaceAlias];
227
+ return true;
228
+ }
229
+ const keyOfActionInActionSpace = action.name;
230
+ if (Object.prototype.hasOwnProperty.call(flowItem, keyOfActionInActionSpace)) {
231
+ locatePromptShortcut = flowItem[keyOfActionInActionSpace];
232
+ return true;
233
+ }
234
+ return false;
235
+ });
236
+ assert(matchedAction, `unknown flowItem in yaml: ${JSON.stringify(flowItem)}`);
237
+ assert(!(flowItem.prompt && locatePromptShortcut), `conflict locate prompt for item: ${JSON.stringify(flowItem)}`);
238
+ if (locatePromptShortcut) flowItem.prompt = locatePromptShortcut;
239
+ const { locateParam, restParams } = buildDetailedLocateParamAndRestParams(locatePromptShortcut || '', flowItem, [
240
+ matchedAction.name,
241
+ matchedAction.interfaceAlias || '_never_mind_'
242
+ ]);
243
+ const flowParams = {
244
+ ...restParams,
245
+ locate: locateParam
246
+ };
247
+ debug(`matchedAction: ${matchedAction.name}`, `flowParams: ${JSON.stringify(flowParams, null, 2)}`);
248
+ await agent.callActionInActionSpace(matchedAction.name, flowParams);
249
+ }
250
+ }
251
+ this.reportFile = agent.reportFile;
252
+ await this.flushUnstableLogContent();
253
+ }
254
+ async run() {
255
+ const { target, web, android, ios, tasks } = this.script;
256
+ const webEnv = web || target;
257
+ const androidEnv = android;
258
+ const iosEnv = ios;
259
+ const platform = webEnv || androidEnv || iosEnv;
260
+ this.setPlayerStatus('running');
261
+ let agent = null;
262
+ let freeFn = [];
263
+ try {
264
+ const { agent: newAgent, freeFn: newFreeFn } = await this.setupAgent(platform);
265
+ this.actionSpace = await newAgent.getActionSpace();
266
+ agent = newAgent;
267
+ const originalOnTaskStartTip = agent.onTaskStartTip;
268
+ agent.onTaskStartTip = (tip)=>{
269
+ if ('running' === this.status) this.agentStatusTip = tip;
270
+ null == originalOnTaskStartTip || originalOnTaskStartTip(tip);
271
+ };
272
+ freeFn = [
273
+ ...newFreeFn || [],
274
+ {
275
+ name: 'restore-agent-onTaskStartTip',
276
+ fn: ()=>{
277
+ if (agent) agent.onTaskStartTip = originalOnTaskStartTip;
278
+ }
279
+ }
280
+ ];
281
+ } catch (e) {
282
+ this.setPlayerStatus('error', e);
283
+ return;
284
+ }
285
+ this.interfaceAgent = agent;
286
+ let taskIndex = 0;
287
+ this.setPlayerStatus('running');
288
+ let errorFlag = false;
289
+ while(taskIndex < tasks.length){
290
+ const taskStatus = this.taskStatusList[taskIndex];
291
+ this.setTaskStatus(taskIndex, 'running');
292
+ this.setTaskIndex(taskIndex);
293
+ try {
294
+ await this.playTask(taskStatus, this.interfaceAgent);
295
+ this.setTaskStatus(taskIndex, 'done');
296
+ } catch (e) {
297
+ this.setTaskStatus(taskIndex, 'error', e);
298
+ if (taskStatus.continueOnError) ;
299
+ else {
300
+ this.reportFile = agent.reportFile;
301
+ errorFlag = true;
302
+ break;
303
+ }
304
+ }
305
+ this.reportFile = null == agent ? void 0 : agent.reportFile;
306
+ taskIndex++;
307
+ }
308
+ if (errorFlag) this.setPlayerStatus('error');
309
+ else this.setPlayerStatus('done');
310
+ this.agentStatusTip = '';
311
+ for (const fn of freeFn)try {
312
+ await fn.fn();
313
+ } catch (e) {}
314
+ }
315
+ constructor(script, setupAgent, onTaskStatusChange, scriptPath){
316
+ var _this_target, _this_target1, _this_target2;
317
+ _define_property(this, "script", void 0);
318
+ _define_property(this, "setupAgent", void 0);
319
+ _define_property(this, "onTaskStatusChange", void 0);
320
+ _define_property(this, "currentTaskIndex", void 0);
321
+ _define_property(this, "taskStatusList", void 0);
322
+ _define_property(this, "status", void 0);
323
+ _define_property(this, "reportFile", void 0);
324
+ _define_property(this, "result", void 0);
325
+ _define_property(this, "unnamedResultIndex", void 0);
326
+ _define_property(this, "output", void 0);
327
+ _define_property(this, "unstableLogContent", void 0);
328
+ _define_property(this, "errorInSetup", void 0);
329
+ _define_property(this, "interfaceAgent", void 0);
330
+ _define_property(this, "agentStatusTip", void 0);
331
+ _define_property(this, "target", void 0);
332
+ _define_property(this, "actionSpace", void 0);
333
+ _define_property(this, "scriptPath", void 0);
334
+ this.script = script;
335
+ this.setupAgent = setupAgent;
336
+ this.onTaskStatusChange = onTaskStatusChange;
337
+ this.taskStatusList = [];
338
+ this.status = 'init';
339
+ this.unnamedResultIndex = 0;
340
+ this.interfaceAgent = null;
341
+ this.actionSpace = [];
342
+ this.scriptPath = scriptPath;
343
+ this.result = {};
344
+ this.target = script.target || script.web || script.android || script.ios || script.config;
345
+ if (ifInBrowser || ifInWorker) {
346
+ this.output = void 0;
347
+ debug('output is undefined in browser or worker');
348
+ } else if (null == (_this_target = this.target) ? void 0 : _this_target.output) {
349
+ this.output = external_node_path_resolve(process.cwd(), this.target.output);
350
+ debug('setting output by config.output', this.output);
351
+ } else {
352
+ const scriptName = this.scriptPath ? basename(this.scriptPath, '.yaml').replace(/\.(ya?ml)$/i, '') : "script";
353
+ this.output = join(getRpasceneRunSubDir('output'), `${scriptName}-${Date.now()}.json`);
354
+ debug("setting output by script path", this.output);
355
+ }
356
+ if (ifInBrowser || ifInWorker) this.unstableLogContent = void 0;
357
+ else if ('string' == typeof (null == (_this_target1 = this.target) ? void 0 : _this_target1.unstableLogContent)) this.unstableLogContent = external_node_path_resolve(process.cwd(), this.target.unstableLogContent);
358
+ else if ((null == (_this_target2 = this.target) ? void 0 : _this_target2.unstableLogContent) === true) this.unstableLogContent = join(getRpasceneRunSubDir('output'), 'unstableLogContent.json');
359
+ this.taskStatusList = (script.tasks || []).map((task, taskIndex)=>{
360
+ var _task_flow;
361
+ return {
362
+ ...task,
363
+ index: taskIndex,
364
+ status: 'init',
365
+ totalSteps: (null == (_task_flow = task.flow) ? void 0 : _task_flow.length) || 0
366
+ };
367
+ });
368
+ }
369
+ }
370
+ export { ScriptPlayer };
371
+
372
+ //# sourceMappingURL=player.mjs.map