nolo-cli 0.1.8 โ†’ 0.1.10

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 (239) hide show
  1. package/README.md +32 -0
  2. package/agentRuntimeCommands.ts +3 -3
  3. package/ai/agent/_executeModel.ts +118 -0
  4. package/ai/agent/agentSlice.ts +525 -0
  5. package/ai/agent/appWorkingMemory.ts +126 -0
  6. package/ai/agent/avatarUtils.ts +24 -0
  7. package/ai/agent/buildEditingContext.ts +373 -0
  8. package/ai/agent/buildSystemPrompt.ts +532 -0
  9. package/ai/agent/cleanAgentMessages.ts +140 -0
  10. package/ai/agent/cliChatClient.ts +119 -0
  11. package/ai/agent/cliExecutor.ts +733 -0
  12. package/ai/agent/cliPrompt.ts +10 -0
  13. package/ai/agent/contextCompiler.ts +107 -0
  14. package/ai/agent/contextLayerContract.ts +44 -0
  15. package/ai/agent/createAgentSchema.ts +234 -0
  16. package/ai/agent/executeToolCall.ts +58 -0
  17. package/ai/agent/fetchAgentContexts.ts +42 -0
  18. package/ai/agent/generatePrompt.ts +3 -0
  19. package/ai/agent/getFullChatContextKeys.ts +168 -0
  20. package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
  21. package/ai/agent/hooks/useAgentConfig.ts +61 -0
  22. package/ai/agent/hooks/useAgentDialog.ts +35 -0
  23. package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
  24. package/ai/agent/hooks/usePublicAgents.ts +473 -0
  25. package/ai/agent/machineRunPermissions.ts +95 -0
  26. package/ai/agent/persistMessageWithFixedId.ts +37 -0
  27. package/ai/agent/planSlice.ts +259 -0
  28. package/ai/agent/referenceUtils.ts +229 -0
  29. package/ai/agent/runAgentBackground.ts +238 -0
  30. package/ai/agent/runAgentClientLoop.ts +138 -0
  31. package/ai/agent/runtimeGuidance.ts +97 -0
  32. package/ai/agent/runtimeServerBase.ts +37 -0
  33. package/ai/agent/server/fetchPublicAgents.ts +128 -0
  34. package/ai/agent/startParallelAgentStreams.ts +424 -0
  35. package/ai/agent/startupProtocol.ts +53 -0
  36. package/ai/agent/streamAgentChatTurn.ts +1278 -0
  37. package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
  38. package/ai/agent/types.ts +71 -0
  39. package/ai/agent/utils/imageOutput.ts +33 -0
  40. package/ai/agent/utils/sortUtils.ts +250 -0
  41. package/ai/agent/web/referencePickerUtils.ts +146 -0
  42. package/ai/ai.locale.ts +1079 -0
  43. package/ai/chat/accumulateToolCallChunks.ts +95 -0
  44. package/ai/chat/fetchUtils.native.ts +276 -0
  45. package/ai/chat/fetchUtils.ts +153 -0
  46. package/ai/chat/parseApiError.ts +64 -0
  47. package/ai/chat/parseMultilineSSE.ts +95 -0
  48. package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
  49. package/ai/chat/sendOpenAICompletionsRequest.ts +703 -0
  50. package/ai/chat/sendOpenAIResponseRequest.ts +491 -0
  51. package/ai/chat/shouldUseServerProxy.ts +18 -0
  52. package/ai/chat/sseClient.native.ts +91 -0
  53. package/ai/chat/sseClient.ts +67 -0
  54. package/ai/chat/streamReader.native.ts +31 -0
  55. package/ai/chat/streamReader.ts +62 -0
  56. package/ai/chat/updateTotalUsage.ts +72 -0
  57. package/ai/context/buildReferenceContext.ts +437 -0
  58. package/ai/context/calculateContextUsage.ts +133 -0
  59. package/ai/context/retention.ts +165 -0
  60. package/ai/context/tokenUtils.ts +78 -0
  61. package/ai/index.ts +1 -0
  62. package/ai/llm/calculateGeminiImageTokens.ts +57 -0
  63. package/ai/llm/deepinfra.ts +28 -0
  64. package/ai/llm/fireworks.ts +50 -0
  65. package/ai/llm/generateRequestBody.ts +165 -0
  66. package/ai/llm/getModelContextWindow.ts +84 -0
  67. package/ai/llm/getNoloKey.ts +31 -0
  68. package/ai/llm/getPricing.ts +199 -0
  69. package/ai/llm/hooks/useModelPricing.ts +75 -0
  70. package/ai/llm/imagePricing.ts +40 -0
  71. package/ai/llm/isResponseAPIModel.ts +13 -0
  72. package/ai/llm/mimo.ts +71 -0
  73. package/ai/llm/mistral.ts +22 -0
  74. package/ai/llm/modelAvatar.ts +427 -0
  75. package/ai/llm/models.ts +45 -0
  76. package/ai/llm/openrouterModels.ts +269 -0
  77. package/ai/llm/providers.ts +306 -0
  78. package/ai/llm/reasoningModels.ts +28 -0
  79. package/ai/llm/types.ts +59 -0
  80. package/ai/llm/usageRequestOptions.ts +59 -0
  81. package/ai/memory/capture.ts +148 -0
  82. package/ai/memory/consolidate.ts +104 -0
  83. package/ai/memory/delete.ts +147 -0
  84. package/ai/memory/overlay.ts +84 -0
  85. package/ai/memory/query.ts +38 -0
  86. package/ai/memory/queryShared.ts +160 -0
  87. package/ai/memory/rank.ts +105 -0
  88. package/ai/memory/recentRelationshipRecap.ts +249 -0
  89. package/ai/memory/remember.ts +167 -0
  90. package/ai/memory/runtime.ts +76 -0
  91. package/ai/memory/store.ts +20 -0
  92. package/ai/memory/storeShared.ts +76 -0
  93. package/ai/memory/types.ts +46 -0
  94. package/ai/memory/understanding.ts +349 -0
  95. package/ai/memory/understandingGreeting.ts +264 -0
  96. package/ai/messages/type.ts +20 -0
  97. package/ai/policy/personalizationDialog.ts +333 -0
  98. package/ai/policy/runtimePolicy.ts +440 -0
  99. package/ai/policy/selfUpdateFields.ts +48 -0
  100. package/ai/policy/types.ts +64 -0
  101. package/ai/skills/referenceRuntime.ts +274 -0
  102. package/ai/skills/skillDiagnostics.ts +251 -0
  103. package/ai/skills/skillDocBuilder.ts +139 -0
  104. package/ai/skills/skillDocProtocol.ts +434 -0
  105. package/ai/skills/skillReferenceSummary.ts +63 -0
  106. package/ai/skills/skillSummaryMarker.ts +26 -0
  107. package/ai/token/calculatePrice.ts +544 -0
  108. package/ai/token/db.ts +98 -0
  109. package/ai/token/externalToolCost.ts +330 -0
  110. package/ai/token/hooks/useRecords.ts +65 -0
  111. package/ai/token/missingUsageEstimate.ts +42 -0
  112. package/ai/token/modelUsageQuery.ts +252 -0
  113. package/ai/token/normalizeUsage.ts +84 -0
  114. package/ai/token/openaiImageGenerationUsage.ts +56 -0
  115. package/ai/token/prepareTokenUsageData.ts +88 -0
  116. package/ai/token/query.ts +88 -0
  117. package/ai/token/queryUserTokens.ts +59 -0
  118. package/ai/token/resolveBillingTarget.ts +52 -0
  119. package/ai/token/saveTokenRecord.ts +53 -0
  120. package/ai/token/serverDialogProjection.ts +78 -0
  121. package/ai/token/serverTokenWriter.ts +143 -0
  122. package/ai/token/stats.ts +21 -0
  123. package/ai/token/tokenThunks.ts +24 -0
  124. package/ai/token/types.ts +93 -0
  125. package/ai/tools/agent/agentTools.ts +176 -0
  126. package/ai/tools/agent/agentUpdateShared.ts +311 -0
  127. package/ai/tools/agent/callAgentTool.ts +139 -0
  128. package/ai/tools/agent/createAgentTool.ts +512 -0
  129. package/ai/tools/agent/createDialogTool.ts +69 -0
  130. package/ai/tools/agent/createSkillAgentTool.ts +62 -0
  131. package/ai/tools/agent/parallelBudget.ts +221 -0
  132. package/ai/tools/agent/presets/appBuilderPreset.ts +145 -0
  133. package/ai/tools/agent/runLlmTool.ts +96 -0
  134. package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
  135. package/ai/tools/agent/skillAgentArgs.ts +106 -0
  136. package/ai/tools/agent/skillAgentPreset.ts +89 -0
  137. package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
  138. package/ai/tools/agent/updateAgentTool.ts +96 -0
  139. package/ai/tools/agent/updateSelfTool.ts +113 -0
  140. package/ai/tools/amazonProductScraperTool.ts +86 -0
  141. package/ai/tools/apifyActorClient.ts +45 -0
  142. package/ai/tools/appEditGuard.ts +372 -0
  143. package/ai/tools/appReadSnapshot.ts +153 -0
  144. package/ai/tools/appTools.ts +1549 -0
  145. package/ai/tools/applyEditTool.ts +256 -0
  146. package/ai/tools/applyLineEditsTool.ts +312 -0
  147. package/ai/tools/browserTools/click.ts +33 -0
  148. package/ai/tools/browserTools/closeSession.ts +29 -0
  149. package/ai/tools/browserTools/common.ts +27 -0
  150. package/ai/tools/browserTools/openSession.ts +48 -0
  151. package/ai/tools/browserTools/readContent.ts +38 -0
  152. package/ai/tools/browserTools/selectOption.ts +46 -0
  153. package/ai/tools/browserTools/typeText.ts +42 -0
  154. package/ai/tools/category/createCategoryTool.ts +66 -0
  155. package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
  156. package/ai/tools/category/updateContentCategoryTool.ts +75 -0
  157. package/ai/tools/cfBrowserTools.ts +319 -0
  158. package/ai/tools/cfSpeechToTextTool.ts +49 -0
  159. package/ai/tools/checkEnvTool.ts +65 -0
  160. package/ai/tools/cloudflareCrawlTool.ts +289 -0
  161. package/ai/tools/codeSearchTool.ts +111 -0
  162. package/ai/tools/codeTools.ts +101 -0
  163. package/ai/tools/createDocTool.ts +132 -0
  164. package/ai/tools/createPlanTool.ts +999 -0
  165. package/ai/tools/createSkillDocTool.ts +155 -0
  166. package/ai/tools/createWorkflowTool.ts +154 -0
  167. package/ai/tools/deepseekOcrTool.ts +34 -0
  168. package/ai/tools/delayTool.ts +31 -0
  169. package/ai/tools/deleteSpacesTool.ts +325 -0
  170. package/ai/tools/deleteSpacesToolModel.ts +159 -0
  171. package/ai/tools/devReloadUtils.ts +29 -0
  172. package/ai/tools/dialogMessageSearch.ts +137 -0
  173. package/ai/tools/doctorSkillTool.ts +72 -0
  174. package/ai/tools/ecommerceScraperTool.ts +86 -0
  175. package/ai/tools/emailTools.ts +549 -0
  176. package/ai/tools/evalSkillTool.ts +92 -0
  177. package/ai/tools/exaSearchTool.ts +64 -0
  178. package/ai/tools/execBashTool.ts +379 -0
  179. package/ai/tools/executeSqlTool.ts +192 -0
  180. package/ai/tools/fetchWebpageSupport.ts +309 -0
  181. package/ai/tools/fetchWebpageTool.ts +84 -0
  182. package/ai/tools/geminiImagePreviewTool.ts +361 -0
  183. package/ai/tools/generateDocxTool.ts +215 -0
  184. package/ai/tools/googleSearchScraperTool.ts +106 -0
  185. package/ai/tools/importDataTool.ts +133 -0
  186. package/ai/tools/importSkillTool.ts +162 -0
  187. package/ai/tools/index.ts +1858 -0
  188. package/ai/tools/listFilesTool.ts +82 -0
  189. package/ai/tools/listUserSpacesTool.ts +113 -0
  190. package/ai/tools/modelUsageTools.ts +142 -0
  191. package/ai/tools/olmOcrTool.ts +34 -0
  192. package/ai/tools/openaiImageTool.ts +218 -0
  193. package/ai/tools/paddleOcrTool.ts +34 -0
  194. package/ai/tools/prepareTools.ts +23 -0
  195. package/ai/tools/readDocTool.ts +84 -0
  196. package/ai/tools/readFileTool.ts +211 -0
  197. package/ai/tools/readTool.ts +163 -0
  198. package/ai/tools/readXPostTool.ts +233 -0
  199. package/ai/tools/rememberMemoryTool.ts +84 -0
  200. package/ai/tools/remotionVideoTool.ts +151 -0
  201. package/ai/tools/searchDialogMessagesTool.ts +222 -0
  202. package/ai/tools/searchRepoTool.ts +115 -0
  203. package/ai/tools/searchWorkspaceTool.ts +259 -0
  204. package/ai/tools/skillFollowup.ts +86 -0
  205. package/ai/tools/surfWeatherTool.ts +169 -0
  206. package/ai/tools/table/addTableRowTool.ts +217 -0
  207. package/ai/tools/table/createTableTool.ts +315 -0
  208. package/ai/tools/table/rowTools.ts +366 -0
  209. package/ai/tools/table/schemaTools.ts +244 -0
  210. package/ai/tools/table/shareTableTool.ts +148 -0
  211. package/ai/tools/table/toolShared.ts +129 -0
  212. package/ai/tools/toolApiClient.ts +198 -0
  213. package/ai/tools/toolNameAliases.ts +57 -0
  214. package/ai/tools/toolResultError.ts +42 -0
  215. package/ai/tools/toolRunSlice.ts +303 -0
  216. package/ai/tools/toolSchemaCompatibility.ts +53 -0
  217. package/ai/tools/toolVisibility.ts +4 -0
  218. package/ai/tools/types.ts +20 -0
  219. package/ai/tools/uiAskChoiceTool.ts +104 -0
  220. package/ai/tools/updateContentTitleTool.ts +84 -0
  221. package/ai/tools/updateDocTool.ts +105 -0
  222. package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
  223. package/ai/tools/whisperTool.ts +77 -0
  224. package/ai/tools/writeFileTool.ts +210 -0
  225. package/ai/tools/youtubeScraperTool.ts +116 -0
  226. package/ai/tools/ziweiChartTool.ts +678 -0
  227. package/ai/types.ts +55 -0
  228. package/ai/workflow/workflowExecutor.ts +323 -0
  229. package/ai/workflow/workflowSlice.ts +73 -0
  230. package/ai/workflow/workflowTypes.ts +106 -0
  231. package/client/compactDialog.ts +222 -0
  232. package/connector-experimental/capabilities.ts +73 -0
  233. package/connector-experimental/codexBinary.ts +41 -0
  234. package/connector-experimental/heartbeatLoop.ts +22 -0
  235. package/connector-experimental/index.ts +5 -0
  236. package/connector-experimental/machineInfo.ts +46 -0
  237. package/connector-experimental/protocol.ts +54 -0
  238. package/machineCommands.ts +4 -4
  239. package/package.json +22 -6
@@ -0,0 +1,1549 @@
1
+ // packages/ai/tools/appTools.ts
2
+ // Web ๅบ”็”จ้ƒจ็ฝฒ/็ฎก็†ๅทฅๅ…ท
3
+ // ๅฝ“ๅ‰็ปŸไธ€ๅ‘ๅธƒๅˆฐๅนณๅฐๆ‰˜็ฎก่ฟ่กŒๆ—ถ
4
+
5
+ import { toolRunUpdated, type ToolRunStep } from "./toolRunSlice";
6
+ import { callToolApi, getToolRequestContext } from "./toolApiClient";
7
+ import { syncAppRecord } from "app/actions/syncAppRecord";
8
+ import { deleteDbKey } from "app/hooks/deleteDbKey";
9
+ import { selectAllMsgs, selectCurrentDialogId } from "chat/messages/messageSlice";
10
+ import { ToolResultError } from "./toolResultError";
11
+ import {
12
+ analyzeAppStyleSystem,
13
+ buildAppReadSnapshotWarning,
14
+ buildAppStyleSystemHint,
15
+ classifyAppReadSnapshot,
16
+ } from "./appReadSnapshot";
17
+ import { evaluateSmallVisualEditGuard } from "./appEditGuard";
18
+
19
+ type AppSourceFile = { name: string; code: string };
20
+ type AppDeployFramework = "worker" | "react-spa";
21
+
22
+ interface AppDeployArgs {
23
+ name?: string;
24
+ code?: string;
25
+ files?: AppSourceFile[];
26
+ pages?: AppSourceFile[];
27
+ appId?: string;
28
+ framework?: AppDeployFramework;
29
+ spaceId?: string;
30
+ }
31
+
32
+ interface AppDeployApiResult {
33
+ success: boolean;
34
+ url: string;
35
+ customUrl?: string;
36
+ routeRegistered?: boolean;
37
+ previewReady?: boolean;
38
+ modifiedOn?: string;
39
+ userFriendlyName: string;
40
+ appId?: string;
41
+ appKey?: string;
42
+ appRecord?: Record<string, any> | null;
43
+ bundleWarnings?: string[];
44
+ deployMode?: "platform";
45
+ framework?: AppDeployFramework;
46
+ previewCheck?: {
47
+ attempted: boolean;
48
+ ready: boolean;
49
+ status?: number;
50
+ attempts: number;
51
+ };
52
+ }
53
+
54
+ interface AppDeployStartResult {
55
+ success: boolean;
56
+ jobId: string;
57
+ eventChannel?: string;
58
+ status: "pending" | "running";
59
+ summary?: string;
60
+ steps?: ToolRunStep[];
61
+ }
62
+
63
+ interface AppDeployStatusResult {
64
+ success: boolean;
65
+ jobId: string;
66
+ status: "pending" | "running" | "succeeded" | "failed";
67
+ summary?: string;
68
+ steps?: ToolRunStep[];
69
+ result?: AppDeployApiResult;
70
+ error?: {
71
+ message?: string;
72
+ code?: string;
73
+ details?: unknown;
74
+ };
75
+ }
76
+
77
+ interface AppPreflightIssue {
78
+ code: string;
79
+ message: string;
80
+ file?: string;
81
+ importSpecifier?: string;
82
+ symbol?: string;
83
+ suggestion?: string;
84
+ }
85
+
86
+ interface AppPreflightResult {
87
+ success: boolean;
88
+ ok: boolean;
89
+ framework: AppDeployFramework;
90
+ summary: string;
91
+ issues: AppPreflightIssue[];
92
+ warnings: string[];
93
+ entryFile?: string;
94
+ externalImports?: string[];
95
+ }
96
+
97
+ interface AppRepairPlanStep {
98
+ action: string;
99
+ reason: string;
100
+ }
101
+
102
+ interface AppRepairPlan {
103
+ strategy: "targeted-repair";
104
+ scope: "existing-files";
105
+ mode: "preflight-first";
106
+ summary: string;
107
+ steps: AppRepairPlanStep[];
108
+ issueCodes: string[];
109
+ suggestedFiles?: string[];
110
+ keepFiles?: string[];
111
+ revertFiles?: string[];
112
+ preferTokenFiles?: string[];
113
+ targetStyleFields?: string[];
114
+ targetElements?: string[];
115
+ rerun: ["appPreflight", "appDeploy"];
116
+ }
117
+
118
+ interface AppStoplossPayload {
119
+ success: false;
120
+ ok: false;
121
+ error: true;
122
+ code:
123
+ | "DEPLOY_TRANSPORT_FAILURE"
124
+ | "PREFLIGHT_TRANSPORT_FAILURE";
125
+ summary: string;
126
+ framework: AppDeployFramework;
127
+ stopReason: "invalid-json-response" | "html-response";
128
+ retryable: false;
129
+ responsePreview?: string;
130
+ nextAction: string;
131
+ }
132
+
133
+ const normalizeOptionalString = (value: unknown): string | undefined => {
134
+ if (typeof value !== "string") return undefined;
135
+ const trimmed = value.trim();
136
+ return trimmed ? trimmed : undefined;
137
+ };
138
+
139
+ export function decideAppDeploySpaceId(params: {
140
+ explicitSpaceId?: string | null;
141
+ currentSpaceId?: string | null;
142
+ existingAppSpaceId?: string | null;
143
+ }): string | undefined {
144
+ const existingAppSpaceId = normalizeOptionalString(params.existingAppSpaceId);
145
+ if (existingAppSpaceId) {
146
+ return undefined;
147
+ }
148
+ return (
149
+ normalizeOptionalString(params.explicitSpaceId) ??
150
+ normalizeOptionalString(params.currentSpaceId)
151
+ );
152
+ }
153
+
154
+ export async function resolveAppDeploySpaceId(
155
+ args: AppDeployArgs,
156
+ thunkApi: any
157
+ ): Promise<string | undefined> {
158
+ const currentSpaceId = normalizeOptionalString(
159
+ thunkApi?.getState?.()?.space?.currentSpaceId
160
+ );
161
+
162
+ if (!args.appId) {
163
+ return decideAppDeploySpaceId({
164
+ explicitSpaceId: args.spaceId,
165
+ currentSpaceId,
166
+ });
167
+ }
168
+
169
+ try {
170
+ const existing = await callToolApi<{
171
+ success: boolean;
172
+ spaceId?: string | null;
173
+ }>(thunkApi, "/api/app/get", { appId: args.appId }, { withAuth: true });
174
+
175
+ return decideAppDeploySpaceId({
176
+ explicitSpaceId: args.spaceId,
177
+ currentSpaceId,
178
+ existingAppSpaceId: existing.spaceId,
179
+ });
180
+ } catch {
181
+ return decideAppDeploySpaceId({
182
+ explicitSpaceId: args.spaceId,
183
+ currentSpaceId,
184
+ });
185
+ }
186
+ }
187
+
188
+ const TOOL_STEP_STATUS_RANK: Record<ToolRunStep["status"], number> = {
189
+ pending: 0,
190
+ running: 1,
191
+ succeeded: 2,
192
+ failed: 3,
193
+ };
194
+
195
+ function parseSseChunk(chunk: string): Array<Record<string, unknown>> {
196
+ const results: Array<Record<string, unknown>> = [];
197
+ for (const line of chunk.split("\n")) {
198
+ const trimmed = line.trim();
199
+ if (!trimmed.startsWith("data:")) continue;
200
+ const json = trimmed.slice(5).trim();
201
+ if (!json) continue;
202
+ try {
203
+ results.push(JSON.parse(json));
204
+ } catch {
205
+ // ignore malformed lines / heartbeat
206
+ }
207
+ }
208
+ return results;
209
+ }
210
+
211
+ const APP_DEPLOY_STEP_LABELS: Record<string, string> = {
212
+ prepare: "ๆ•ด็†้ƒจ็ฝฒๅ‚ๆ•ฐ",
213
+ preflight: "้ข„ๆฃ€ไปฃ็ ",
214
+ build: "ๆ‰“ๅŒ…ๅบ”็”จ",
215
+ deploy: "ๅ‘ๅธƒ็ซ™็‚น",
216
+ verify: "้ชŒ่ฏ่ฎฟ้—ฎ",
217
+ };
218
+
219
+ function buildDeploySteps(
220
+ currentStepId: keyof typeof APP_DEPLOY_STEP_LABELS,
221
+ currentStatus: ToolRunStep["status"],
222
+ detail?: string
223
+ ): ToolRunStep[] {
224
+ const ids = Object.keys(APP_DEPLOY_STEP_LABELS) as Array<
225
+ keyof typeof APP_DEPLOY_STEP_LABELS
226
+ >;
227
+ const currentIndex = ids.indexOf(currentStepId);
228
+ return ids.map((id, index) => ({
229
+ id,
230
+ label: APP_DEPLOY_STEP_LABELS[id],
231
+ status:
232
+ index < currentIndex
233
+ ? "succeeded"
234
+ : index === currentIndex
235
+ ? currentStatus
236
+ : "pending",
237
+ ...(index === currentIndex && detail ? { detail } : {}),
238
+ }));
239
+ }
240
+
241
+ function updateDeployProgress(
242
+ thunkApi: any,
243
+ toolRunId: string | undefined,
244
+ stepIdOrSteps: keyof typeof APP_DEPLOY_STEP_LABELS | ToolRunStep[],
245
+ summary: string,
246
+ currentStatus: ToolRunStep["status"] = "running",
247
+ detail?: string
248
+ ) {
249
+ if (!toolRunId) return;
250
+ thunkApi.dispatch(
251
+ toolRunUpdated({
252
+ id: toolRunId,
253
+ outputSummary: summary,
254
+ steps: Array.isArray(stepIdOrSteps)
255
+ ? stepIdOrSteps
256
+ : buildDeploySteps(stepIdOrSteps, currentStatus, detail),
257
+ })
258
+ );
259
+ }
260
+
261
+ function normalizeAppDeployArgs(args: AppDeployArgs) {
262
+ const normalizedFiles = Array.isArray(args.files) && args.files.length > 0
263
+ ? args.files
264
+ : Array.isArray(args.pages) && args.pages.length > 0
265
+ ? args.pages
266
+ : undefined;
267
+ return {
268
+ ...args,
269
+ files: normalizedFiles,
270
+ };
271
+ }
272
+
273
+ function isLikelyReactWorkerMisuse(code: string | undefined): boolean {
274
+ if (!code) return false;
275
+ return /from\s+["']react["']|from\s+["']react-dom|react-icons\/lu|createRoot\s*\(|<\w+[^>]*>/.test(
276
+ code
277
+ );
278
+ }
279
+
280
+ function rewriteAppDeployError(errorMessage: string, args: AppDeployArgs): string {
281
+ if (errorMessage.includes("React SPA ๆจกๅผๅฟ…้กปๆไพ› files ๅ‚ๆ•ฐ")) {
282
+ return 'React SPA ้œ€่ฆไผ  `files`๏ผˆ่‡ณๅฐ‘ `main.tsx` + `App.tsx`๏ผ‰๏ผŒไธ่ƒฝๅชไผ  `code`ใ€‚ๅฆ‚ๆžœไฝ ็Žฐๅœจๆ‹ฟๅˆฐ็š„ๆ˜ฏๅคšๆ–‡ไปถๆบ็ ๏ผŒไนŸๅฏไปฅ็›ดๆŽฅไผ  `pages`๏ผŒ็ณป็ปŸไผš่‡ชๅŠจๅ…ผๅฎนไธบ `files`ใ€‚';
283
+ }
284
+ if (errorMessage.includes("้œ€่ฆๆไพ› code ๆˆ– files ๅ‚ๆ•ฐ")) {
285
+ return "็ผบๅฐ‘ๅฏ้ƒจ็ฝฒๆบ็ ใ€‚่ฏทไผ  `code`๏ผˆๅ•ๆ–‡ไปถ Worker๏ผ‰ๆˆ– `files`๏ผˆๅคšๆ–‡ไปถ้กน็›ฎ๏ผ‰๏ผ›ๅฆ‚ๆžœไฝ ๆ‰‹ไธŠๅญ—ๆฎตๅๆ˜ฏ `pages`๏ผŒ็ŽฐๅœจไนŸๅฏไปฅ็›ดๆŽฅไผ ใ€‚";
286
+ }
287
+ if (
288
+ errorMessage.includes("Bundle failed") &&
289
+ args.framework !== "react-spa" &&
290
+ isLikelyReactWorkerMisuse(args.code)
291
+ ) {
292
+ return 'ๆฃ€ๆต‹ๅˆฐไฝ ๆŠŠ React ็ป„ไปถไปฃ็ ๅฝ“ๆˆๅ•ๆ–‡ไปถ Worker ๅŽป้ƒจ็ฝฒไบ†ใ€‚่ฆๅšไบคไบ’็ฝ‘้กต๏ผŒ่ฏทๆ”น็”จ `framework: "react-spa"` ๅนถไผ  `files`๏ผˆ้€šๅธธๆ˜ฏ `main.tsx` + `App.tsx`๏ผ‰๏ผ›ๅฆ‚ๆžœไฝ ๅชๆƒณ่ฟ”ๅ›ž้™ๆ€ HTML๏ผŒ่ฏทๅŽปๆމ `react` / `react-dom` / `react-icons` importใ€‚';
293
+ }
294
+ return errorMessage;
295
+ }
296
+
297
+ function messageContentToText(content: unknown): string {
298
+ if (typeof content === "string") return content;
299
+ if (Array.isArray(content)) {
300
+ return content
301
+ .map((part) =>
302
+ part && typeof part === "object" && "text" in part
303
+ ? String((part as { text?: unknown }).text ?? "")
304
+ : ""
305
+ )
306
+ .join("\n");
307
+ }
308
+ return "";
309
+ }
310
+
311
+ function getLatestUserInputFromThunk(thunkApi: any): string | undefined {
312
+ try {
313
+ const state = thunkApi?.getState?.();
314
+ if (!state) return undefined;
315
+ const dialogId = selectCurrentDialogId(state);
316
+ const messages = selectAllMsgs(state, dialogId);
317
+ for (let i = messages.length - 1; i >= 0; i -= 1) {
318
+ const message = messages[i];
319
+ if (message?.role !== "user") continue;
320
+ const text = messageContentToText(message.content).trim();
321
+ if (text) return text;
322
+ }
323
+ } catch {
324
+ return undefined;
325
+ }
326
+ return undefined;
327
+ }
328
+
329
+ function formatPreflightIssues(issues: AppPreflightIssue[] | undefined): string {
330
+ if (!Array.isArray(issues) || issues.length === 0) return "";
331
+ return issues
332
+ .slice(0, 6)
333
+ .map((issue) => {
334
+ const parts = [`- ${issue.message}`];
335
+ if (issue.suggestion) parts.push(`ๅปบ่ฎฎๆ”นไธบ๏ผš${issue.suggestion}`);
336
+ return parts.join("๏ผ›");
337
+ })
338
+ .join("\n");
339
+ }
340
+
341
+ function inferRepairSuggestedFiles(issues: AppPreflightIssue[] | undefined): string[] {
342
+ const files = new Set<string>();
343
+ for (const issue of issues ?? []) {
344
+ if (issue.file) files.add(issue.file);
345
+ if (issue.code === "missing-entry-file") {
346
+ files.add("main.tsx");
347
+ files.add("App.tsx");
348
+ }
349
+ }
350
+ return [...files];
351
+ }
352
+
353
+ function buildRepairSteps(
354
+ issues: AppPreflightIssue[] | undefined,
355
+ framework: AppDeployFramework
356
+ ): AppRepairPlanStep[] {
357
+ const steps: AppRepairPlanStep[] = [];
358
+ const seen = new Set<string>();
359
+
360
+ const push = (action: string, reason: string) => {
361
+ const key = `${action}::${reason}`;
362
+ if (seen.has(key)) return;
363
+ seen.add(key);
364
+ steps.push({ action, reason });
365
+ };
366
+
367
+ for (const issue of issues ?? []) {
368
+ switch (issue.code) {
369
+ case "missing-files":
370
+ push(
371
+ framework === "react-spa"
372
+ ? "่กฅ้ฝ React SPA ็š„ files ๆ•ฐ็ป„๏ผŒไธ่ฆๅชไผ  code"
373
+ : "่กฅ้ฝๅฏ้ƒจ็ฝฒๆบ็ ๏ผŒ็กฎไฟ่‡ณๅฐ‘ไผ  code ๆˆ– files",
374
+ issue.message
375
+ );
376
+ break;
377
+ case "missing-entry-file":
378
+ push("ๆ–ฐๅขžๅนถไฟ็•™็จณๅฎšๅ…ฅๅฃๆ–‡ไปถ main.tsx ไธŽ App.tsx", issue.message);
379
+ break;
380
+ case "invalid-file":
381
+ case "invalid-file-path":
382
+ push("ไฟฎๆญฃ้žๆณ•ๆ–‡ไปถๅๆˆ–็ฉบๆ–‡ไปถๅ†…ๅฎน๏ผŒๅชๆ”นๅ‡บ้—ฎ้ข˜็š„ๆ–‡ไปถ", issue.message);
383
+ break;
384
+ case "css-import-disallowed":
385
+ push("็งป้™ค CSS import๏ผŒๆ”นๆˆๅ†…่”ๆ ทๅผใ€style ๅฏน่ฑกๆˆ–็ป„ไปถๅ†… style ๆ ‡็ญพ", issue.message);
386
+ break;
387
+ case "unsupported-import":
388
+ push(
389
+ issue.importSpecifier
390
+ ? `็งป้™คๆˆ–ๆ›ฟๆขๆœชๆ”ฏๆŒไพ่ต– ${issue.importSpecifier}๏ผŒๅชไฟ็•™ๅนณๅฐ็™ฝๅๅ•ไพ่ต–`
391
+ : "็งป้™คๆˆ–ๆ›ฟๆขๆœชๆ”ฏๆŒไพ่ต–๏ผŒๅชไฟ็•™ๅนณๅฐ็™ฝๅๅ•ไพ่ต–",
392
+ issue.message
393
+ );
394
+ break;
395
+ case "invalid-icon-import":
396
+ push(
397
+ issue.suggestion
398
+ ? `ๆŠŠๆ— ๆ•ˆๅ›พๆ ‡ๆ›ฟๆขๆˆ ${issue.suggestion}`
399
+ : "ๆŠŠๆ— ๆ•ˆๅ›พๆ ‡ๆ›ฟๆขๆˆ react-icons/lu ไธญ็œŸๅฎžๅญ˜ๅœจ็š„ๅ›พๆ ‡ๅ",
400
+ issue.message
401
+ );
402
+ break;
403
+ default:
404
+ push("ๆ นๆฎ้ข„ๆฃ€้—ฎ้ข˜ๅšๅฑ€้ƒจไฟฎๅค๏ผŒไธ่ฆๆ•ด้กต้‡ๅ†™", issue.message);
405
+ }
406
+ }
407
+
408
+ push("ไฟฎๅฎŒๅŽๅ…ˆ้‡ๆ–ฐ่ฐƒ็”จ appPreflight", "็กฎ่ฎคๅฝ“ๅ‰ issues ๅทฒๆถˆ้™ค");
409
+ push("ๅชๆœ‰ preflight ้€š่ฟ‡ๅŽๅ†่ฐƒ็”จ appDeploy", "้ฟๅ…้‡ๅค่ฟ›ๅ…ฅๅคฑ่ดฅ้ƒจ็ฝฒ");
410
+ return steps;
411
+ }
412
+
413
+ function buildAppRepairPlan(args: {
414
+ summary: string;
415
+ framework: AppDeployFramework;
416
+ issues?: AppPreflightIssue[];
417
+ }): AppRepairPlan {
418
+ return {
419
+ strategy: "targeted-repair",
420
+ scope: "existing-files",
421
+ mode: "preflight-first",
422
+ summary: args.summary,
423
+ steps: buildRepairSteps(args.issues, args.framework),
424
+ issueCodes: [...new Set((args.issues ?? []).map((issue) => issue.code))],
425
+ suggestedFiles: inferRepairSuggestedFiles(args.issues),
426
+ rerun: ["appPreflight", "appDeploy"],
427
+ };
428
+ }
429
+
430
+ function buildAppRepairPayload(args: {
431
+ summary: string;
432
+ framework: AppDeployFramework;
433
+ issues?: AppPreflightIssue[];
434
+ warnings?: string[];
435
+ entryFile?: string;
436
+ externalImports?: string[];
437
+ code?: string;
438
+ }): {
439
+ success: false;
440
+ ok: false;
441
+ error: true;
442
+ code: "PREFLIGHT_FAILED" | "DEPLOY_FAILED";
443
+ summary: string;
444
+ framework: AppDeployFramework;
445
+ issues: AppPreflightIssue[];
446
+ warnings?: string[];
447
+ entryFile?: string;
448
+ externalImports?: string[];
449
+ repairPlan: AppRepairPlan;
450
+ nextAction: string;
451
+ } {
452
+ const issues = args.issues ?? [];
453
+ return {
454
+ success: false,
455
+ ok: false,
456
+ error: true,
457
+ code: issues.length > 0 ? "PREFLIGHT_FAILED" : "DEPLOY_FAILED",
458
+ summary: args.summary,
459
+ framework: args.framework,
460
+ issues,
461
+ ...(args.warnings?.length ? { warnings: args.warnings } : {}),
462
+ ...(args.entryFile ? { entryFile: args.entryFile } : {}),
463
+ ...(args.externalImports?.length ? { externalImports: args.externalImports } : {}),
464
+ repairPlan: buildAppRepairPlan({
465
+ summary: args.summary,
466
+ framework: args.framework,
467
+ issues,
468
+ }),
469
+ nextAction:
470
+ "ๅชไฟฎๅคๅฝ“ๅ‰ issues ๅ‘ฝไธญ็š„ๆ–‡ไปถๅ’Œไพ่ต–๏ผŒ็„ถๅŽ้‡ๆ–ฐ่ฐƒ็”จ appPreflight๏ผ›ๅชๆœ‰้€š่ฟ‡ๅŽๅ† appDeployใ€‚",
471
+ };
472
+ }
473
+
474
+ function formatRepairPlan(plan: AppRepairPlan | undefined): string {
475
+ if (!plan) return "";
476
+ return [
477
+ "ไฟฎๅคๅปบ่ฎฎ๏ผš",
478
+ ...plan.steps.slice(0, 6).map((step, index) => `${index + 1}. ${step.action}๏ผˆ${step.reason}๏ผ‰`),
479
+ ...(plan.keepFiles?.length ? [`- ไฟ็•™ๆ–‡ไปถ๏ผš${plan.keepFiles.join(", ")}`] : []),
480
+ ...(plan.revertFiles?.length ? [`- ๅ›ž้€€ๆ–‡ไปถ๏ผš${plan.revertFiles.join(", ")}`] : []),
481
+ ...(plan.preferTokenFiles?.length
482
+ ? [`- ไผ˜ๅ…ˆๆŠŠ่ง†่ง‰ไฟฎๆ”นๆ”ถๆ•›ๅˆฐ่ฟ™ไบ› token ๆ–‡ไปถ๏ผš${plan.preferTokenFiles.join(", ")}`]
483
+ : []),
484
+ ...(plan.targetStyleFields?.length
485
+ ? [`- ไป…็ปง็ปญ่ฐƒๆ•ด่ฟ™ไบ›่ง†่ง‰ๅญ—ๆฎต๏ผš${plan.targetStyleFields.join(", ")}`]
486
+ : []),
487
+ ...(plan.targetElements?.length
488
+ ? [`- ไป…็ปง็ปญ่ฐƒๆ•ด่ฟ™ไบ›ๅ…ƒ็ด ๏ผš${plan.targetElements.join(", ")}`]
489
+ : []),
490
+ "ไฟฎๅฎŒๅŽ๏ผšๅ…ˆ appPreflight๏ผŒๅ† appDeployใ€‚",
491
+ ].join("\n");
492
+ }
493
+
494
+ function isTransportStoplossError(error: {
495
+ code?: string;
496
+ message?: string;
497
+ details?: unknown;
498
+ } | null | undefined): boolean {
499
+ const code = error?.code ?? "";
500
+ if (
501
+ code === "HTML_RESPONSE" ||
502
+ code === "INVALID_JSON_RESPONSE" ||
503
+ code === "HTML_ERROR_RESPONSE" ||
504
+ code === "NON_JSON_ERROR_RESPONSE"
505
+ ) {
506
+ return true;
507
+ }
508
+ const message = error?.message ?? "";
509
+ return message.includes("Unexpected token '<'") || message.includes("<!DOCTYPE");
510
+ }
511
+
512
+ function buildAppStoplossPayload(args: {
513
+ summary: string;
514
+ framework: AppDeployFramework;
515
+ stage: "deploy" | "preflight";
516
+ error?: {
517
+ code?: string;
518
+ details?: unknown;
519
+ message?: string;
520
+ } | null;
521
+ }): AppStoplossPayload {
522
+ const details =
523
+ args.error?.details && typeof args.error.details === "object"
524
+ ? (args.error.details as { responsePreview?: string })
525
+ : null;
526
+ const errorCode = args.error?.code ?? "";
527
+ const stopReason =
528
+ errorCode.includes("HTML") || (args.error?.message ?? "").includes("<!DOCTYPE")
529
+ ? "html-response"
530
+ : "invalid-json-response";
531
+ return {
532
+ success: false,
533
+ ok: false,
534
+ error: true,
535
+ code:
536
+ args.stage === "preflight"
537
+ ? "PREFLIGHT_TRANSPORT_FAILURE"
538
+ : "DEPLOY_TRANSPORT_FAILURE",
539
+ summary: args.summary,
540
+ framework: args.framework,
541
+ stopReason,
542
+ retryable: false,
543
+ ...(details?.responsePreview ? { responsePreview: details.responsePreview } : {}),
544
+ nextAction:
545
+ "่ฟ™ไธๆ˜ฏไปฃ็ ็บง issues๏ผŒ่€Œๆ˜ฏ้ƒจ็ฝฒ้€š้“่ฟ”ๅ›žไบ†ๅผ‚ๅธธๅ“ๅบ”ใ€‚ๅœๆญข่‡ชๅŠจ deploy / preflight ้‡่ฏ•๏ผŒๅ‘็”จๆˆท่ฏดๆ˜Žๅฝ“ๅ‰ๅนณๅฐๆŽฅๅฃๅผ‚ๅธธ๏ผŒ็ญ‰ๅพ…ๆœๅŠกๆขๅคๅŽๅ†็ปง็ปญใ€‚",
546
+ };
547
+ }
548
+
549
+ function formatStoplossPlan(payload: AppStoplossPayload): string {
550
+ return [
551
+ payload.summary,
552
+ payload.responsePreview ? `ๅ“ๅบ”้ข„่งˆ: ${payload.responsePreview}` : "",
553
+ "ๅˆคๆ–ญ: ๅฝ“ๅ‰ๆ˜ฏ้ƒจ็ฝฒ/้ข„ๆฃ€้€š้“ๅผ‚ๅธธ๏ผŒไธๆ˜ฏๅบ”็”จไปฃ็ ้—ฎ้ข˜ใ€‚",
554
+ "ไธ‹ไธ€ๆญฅ: ๅœๆญข่‡ชๅŠจ้‡่ฏ•๏ผŒๅ‘Š่ฏ‰็”จๆˆทๅฝ“ๅ‰ๅนณๅฐๆŽฅๅฃ่ฟ”ๅ›žๅผ‚ๅธธ๏ผŒ็จๅŽๅ†่ฏ•ใ€‚",
555
+ ]
556
+ .filter(Boolean)
557
+ .join("\n");
558
+ }
559
+
560
+ const sleep = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms));
561
+
562
+ async function pollAppDeployJob(
563
+ thunkApi: any,
564
+ args: AppDeployArgs,
565
+ toolRunId: string | undefined,
566
+ jobId: string,
567
+ options?: {
568
+ sharedState?: {
569
+ done: boolean;
570
+ result?: AppDeployApiResult;
571
+ failure?: Error;
572
+ error?: string;
573
+ lastSteps: ToolRunStep[];
574
+ };
575
+ }
576
+ ): Promise<AppDeployApiResult> {
577
+ const sharedState = options?.sharedState;
578
+ let lastSteps: ToolRunStep[] = sharedState?.lastSteps ?? [];
579
+ try {
580
+ for (let attempt = 0; attempt < 180; attempt += 1) {
581
+ if (sharedState?.done) {
582
+ if (sharedState.result) return sharedState.result;
583
+ if (sharedState.failure) throw sharedState.failure;
584
+ throw new Error(sharedState.error || "้ƒจ็ฝฒๅคฑ่ดฅ");
585
+ }
586
+ const statusData = await callToolApi<AppDeployStatusResult>(
587
+ thunkApi,
588
+ "/api/app/deploy/status",
589
+ { jobId },
590
+ { withAuth: true }
591
+ );
592
+
593
+ const nextSteps = Array.isArray(statusData.steps)
594
+ ? statusData.steps.map((step) => {
595
+ const previous = lastSteps.find((item) => item.id === step.id);
596
+ if (
597
+ previous &&
598
+ TOOL_STEP_STATUS_RANK[step.status] < TOOL_STEP_STATUS_RANK[previous.status]
599
+ ) {
600
+ return previous;
601
+ }
602
+ return step;
603
+ })
604
+ : lastSteps;
605
+ // Only advance lastSteps when server returns a non-empty steps array; this
606
+ // preserves the last known UI state if the server briefly returns empty steps.
607
+ if (nextSteps.length > 0) {
608
+ lastSteps = nextSteps;
609
+ if (sharedState) {
610
+ sharedState.lastSteps = nextSteps;
611
+ }
612
+ }
613
+
614
+ updateDeployProgress(
615
+ thunkApi,
616
+ toolRunId,
617
+ lastSteps.length > 0 ? lastSteps : "prepare",
618
+ statusData.summary ?? "ๆญฃๅœจๅŒๆญฅๆœๅŠก็ซฏ้ƒจ็ฝฒ็Šถๆ€โ€ฆ",
619
+ statusData.status === "failed" ? "failed" : "running"
620
+ );
621
+
622
+ if (statusData.status === "succeeded" && statusData.result) {
623
+ if (sharedState) {
624
+ sharedState.done = true;
625
+ sharedState.result = statusData.result;
626
+ }
627
+ return statusData.result;
628
+ }
629
+ if (statusData.status === "failed") {
630
+ if (
631
+ isTransportStoplossError({
632
+ code: statusData.error?.code,
633
+ details: statusData.error?.details,
634
+ message: statusData.error?.message || statusData.summary,
635
+ })
636
+ ) {
637
+ const payload = buildAppStoplossPayload({
638
+ summary: "้ƒจ็ฝฒ็Šถๆ€ๆŽฅๅฃ่ฟ”ๅ›žไบ†ๅผ‚ๅธธๅ“ๅบ”๏ผŒๅทฒๅœๆญข่‡ชๅŠจ้‡่ฏ•ใ€‚",
639
+ framework: args.framework ?? "worker",
640
+ stage: "deploy",
641
+ error: {
642
+ code: statusData.error?.code,
643
+ details: statusData.error?.details,
644
+ message: statusData.error?.message || statusData.summary,
645
+ },
646
+ });
647
+ const displayMessage = formatStoplossPlan(payload);
648
+ const stoplossError = new ToolResultError(payload.summary, {
649
+ code: payload.code,
650
+ rawData: payload,
651
+ displayData: displayMessage,
652
+ retryable: false,
653
+ });
654
+ if (sharedState) {
655
+ sharedState.done = true;
656
+ sharedState.failure = stoplossError;
657
+ sharedState.error = displayMessage;
658
+ }
659
+ updateDeployProgress(
660
+ thunkApi,
661
+ toolRunId,
662
+ statusData.steps ?? "prepare",
663
+ displayMessage,
664
+ "failed"
665
+ );
666
+ throw stoplossError;
667
+ }
668
+ const rewrittenBase = rewriteAppDeployError(
669
+ statusData.error?.message || statusData.summary || "้ƒจ็ฝฒๅคฑ่ดฅ",
670
+ args
671
+ );
672
+ const issues =
673
+ statusData.error?.code === "PREFLIGHT_FAILED"
674
+ ? ((statusData.error?.details as { issues?: AppPreflightIssue[] } | undefined)?.issues ?? [])
675
+ : [];
676
+ const repairPayload = buildAppRepairPayload({
677
+ summary: rewrittenBase,
678
+ framework: args.framework ?? "worker",
679
+ issues,
680
+ });
681
+ const rewritten = [
682
+ rewrittenBase,
683
+ formatPreflightIssues(issues),
684
+ formatRepairPlan(repairPayload.repairPlan),
685
+ ]
686
+ .filter(Boolean)
687
+ .join("\n");
688
+ const repairError = new ToolResultError(rewrittenBase, {
689
+ code: statusData.error?.code ?? repairPayload.code,
690
+ rawData: repairPayload,
691
+ displayData: rewritten,
692
+ retryable: true,
693
+ });
694
+ if (sharedState) {
695
+ sharedState.done = true;
696
+ sharedState.failure = repairError;
697
+ sharedState.error = rewritten;
698
+ }
699
+ updateDeployProgress(
700
+ thunkApi,
701
+ toolRunId,
702
+ statusData.steps ?? "prepare",
703
+ rewritten,
704
+ "failed"
705
+ );
706
+ throw repairError;
707
+ }
708
+
709
+ await sleep(Math.min(400 + attempt * 50, 2000));
710
+ }
711
+
712
+ throw new Error("้ƒจ็ฝฒไปปๅŠกไปๅœจๆœๅŠก็ซฏๆ‰ง่กŒ๏ผŒ่ฏท็จๅŽ้‡่ฏ•ๆŸฅ็œ‹็ป“ๆžœใ€‚");
713
+ } finally {
714
+ // Ensure SSE subscriber always sees done=true so it can cancel the reader,
715
+ // even when polling times out or throws an unexpected error.
716
+ if (sharedState && !sharedState.done) {
717
+ sharedState.done = true;
718
+ }
719
+ }
720
+ }
721
+
722
+ async function subscribeToDeployEvents(args: {
723
+ thunkApi: any;
724
+ deployArgs: AppDeployArgs;
725
+ toolRunId?: string;
726
+ jobId: string;
727
+ eventChannel: string;
728
+ sharedState: {
729
+ done: boolean;
730
+ result?: AppDeployApiResult;
731
+ failure?: Error;
732
+ error?: string;
733
+ lastSteps: ToolRunStep[];
734
+ };
735
+ }): Promise<void> {
736
+ const { thunkApi, deployArgs, toolRunId, jobId, eventChannel, sharedState } = args;
737
+ const { baseUrl, token } = getToolRequestContext(thunkApi);
738
+ if (!token) return;
739
+
740
+ const res = await fetch(`${baseUrl}/api/events/${encodeURIComponent(eventChannel)}`, {
741
+ method: "GET",
742
+ headers: {
743
+ Accept: "text/event-stream",
744
+ Authorization: `Bearer ${token}`,
745
+ },
746
+ });
747
+ if (!res.ok || !res.body) return;
748
+
749
+ const reader = res.body.getReader();
750
+ const decoder = new TextDecoder();
751
+
752
+ try {
753
+ while (true) {
754
+ if (sharedState.done) {
755
+ return;
756
+ }
757
+ const { done, value } = await reader.read();
758
+ if (done) return;
759
+ const chunk = decoder.decode(value, { stream: true });
760
+ for (const event of parseSseChunk(chunk)) {
761
+ if (event.type !== "app-deploy-progress" || event.jobId !== jobId) continue;
762
+ const eventSteps = Array.isArray(event.steps)
763
+ ? (event.steps as ToolRunStep[]).map((step) => {
764
+ const previous = sharedState.lastSteps.find((item) => item.id === step.id);
765
+ if (
766
+ previous &&
767
+ TOOL_STEP_STATUS_RANK[step.status] < TOOL_STEP_STATUS_RANK[previous.status]
768
+ ) {
769
+ return previous;
770
+ }
771
+ return step;
772
+ })
773
+ : sharedState.lastSteps;
774
+ sharedState.lastSteps = eventSteps;
775
+ const status = typeof event.status === "string" ? event.status : "running";
776
+ const summary =
777
+ typeof event.summary === "string" ? event.summary : "ๆญฃๅœจๆŽฅๆ”ถๆœๅŠก็ซฏ้ƒจ็ฝฒไบ‹ไปถโ€ฆ";
778
+
779
+ updateDeployProgress(
780
+ thunkApi,
781
+ toolRunId,
782
+ eventSteps.length > 0 ? eventSteps : "prepare",
783
+ summary,
784
+ status === "failed" ? "failed" : status === "succeeded" ? "succeeded" : "running"
785
+ );
786
+
787
+ if (status === "succeeded" && event.result) {
788
+ sharedState.done = true;
789
+ sharedState.result = event.result as AppDeployApiResult;
790
+ return;
791
+ }
792
+ if (status === "failed") {
793
+ if (
794
+ isTransportStoplossError({
795
+ code: (event.error as { code?: string } | undefined)?.code,
796
+ details: (event.error as { details?: unknown } | undefined)?.details,
797
+ message: (event.error as { message?: string } | undefined)?.message || summary,
798
+ })
799
+ ) {
800
+ const payload = buildAppStoplossPayload({
801
+ summary: "้ƒจ็ฝฒไบ‹ไปถๆต่ฟ”ๅ›žไบ†ๅผ‚ๅธธๅ“ๅบ”๏ผŒๅทฒๅœๆญข่‡ชๅŠจ้‡่ฏ•ใ€‚",
802
+ framework: deployArgs.framework ?? "worker",
803
+ stage: "deploy",
804
+ error: {
805
+ code: (event.error as { code?: string } | undefined)?.code,
806
+ details: (event.error as { details?: unknown } | undefined)?.details,
807
+ message: (event.error as { message?: string } | undefined)?.message || summary,
808
+ },
809
+ });
810
+ const displayMessage = formatStoplossPlan(payload);
811
+ const stoplossError = new ToolResultError(payload.summary, {
812
+ code: payload.code,
813
+ rawData: payload,
814
+ displayData: displayMessage,
815
+ retryable: false,
816
+ });
817
+ sharedState.done = true;
818
+ sharedState.failure = stoplossError;
819
+ sharedState.error = displayMessage;
820
+ updateDeployProgress(
821
+ thunkApi,
822
+ toolRunId,
823
+ eventSteps.length > 0 ? eventSteps : "prepare",
824
+ displayMessage,
825
+ "failed"
826
+ );
827
+ return;
828
+ }
829
+ const rewrittenBase = rewriteAppDeployError(
830
+ (event.error as { message?: string } | undefined)?.message || summary || "้ƒจ็ฝฒๅคฑ่ดฅ",
831
+ deployArgs
832
+ );
833
+ const issues =
834
+ (event.error as { code?: string; details?: { issues?: AppPreflightIssue[] } } | undefined)?.code === "PREFLIGHT_FAILED"
835
+ ? (((event.error as { details?: { issues?: AppPreflightIssue[] } } | undefined)?.details)
836
+ ?.issues ?? [])
837
+ : [];
838
+ const repairPayload = buildAppRepairPayload({
839
+ summary: rewrittenBase,
840
+ framework: deployArgs.framework ?? "worker",
841
+ issues,
842
+ });
843
+ const rewritten = [
844
+ rewrittenBase,
845
+ formatPreflightIssues(issues),
846
+ formatRepairPlan(repairPayload.repairPlan),
847
+ ]
848
+ .filter(Boolean)
849
+ .join("\n");
850
+ const repairError = new ToolResultError(rewrittenBase, {
851
+ code: (event.error as { code?: string } | undefined)?.code ?? repairPayload.code,
852
+ rawData: repairPayload,
853
+ displayData: rewritten,
854
+ retryable: true,
855
+ });
856
+ sharedState.done = true;
857
+ sharedState.failure = repairError;
858
+ sharedState.error = rewritten;
859
+ updateDeployProgress(
860
+ thunkApi,
861
+ toolRunId,
862
+ eventSteps.length > 0 ? eventSteps : "prepare",
863
+ rewritten,
864
+ "failed"
865
+ );
866
+ return;
867
+ }
868
+ }
869
+ }
870
+ } finally {
871
+ // Cancel the SSE reader on every exit path (success, failure, external done,
872
+ // natural stream end) so the underlying HTTP connection is always released.
873
+ try {
874
+ await reader.cancel();
875
+ } catch {}
876
+ try {
877
+ reader.releaseLock();
878
+ } catch {}
879
+ }
880
+ }
881
+
882
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
883
+ // appDeploy โ€” ้ƒจ็ฝฒๆˆ–ๆ›ดๆ–ฐไธ€ไธช Web ๅบ”็”จ
884
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
885
+ export const appDeployFunctionSchema = {
886
+ name: "appDeploy",
887
+ description:
888
+ "ๅฐ† JavaScript/TypeScript ไปฃ็ ้ƒจ็ฝฒไธบ Web ๅบ”็”จใ€‚" +
889
+ "้ป˜่ฎค้ƒจ็ฝฒๅˆฐๅนณๅฐๆœๅŠกๅ™จ๏ผˆnolo.chat/apps/{appId}/๏ผ‰๏ผŒๆ— ้œ€็”จๆˆท้…็ฝฎไปปไฝ•้ขๅค–่ดฆๅท๏ผŒ็ซ‹ๅณๅฏ่ฎฟ้—ฎใ€‚" +
890
+ "ไปฃ็ ๅฟ…้กปๆ˜ฏ ES Module ๆ ผๅผ๏ผˆexport default { fetch(req) {} }๏ผ‰ใ€‚" +
891
+ "ๆ”ฏๆŒๅคšๆ–‡ไปถ้กน็›ฎ๏ผš้€š่ฟ‡ files ๆ•ฐ็ป„ไผ ๅ…ฅ๏ผŒๆœๅŠก็ซฏ่‡ชๅŠจๆ‰“ๅŒ…ใ€‚" +
892
+ "ๆ–ฐๅปบๅบ”็”จๆ—ถไฝฟ็”จ name๏ผ›ๆ›ดๆ–ฐๅทฒๆœ‰ๅบ”็”จๆ—ถๅฟ…้กปไผ˜ๅ…ˆไผ  appId๏ผŒ้ฟๅ…ๅ› ไธบๅ็งฐ้‡ๅค่€Œ่ฏฏๅปบๆ–ฐๅบ”็”จใ€‚",
893
+ parameters: {
894
+ type: "object",
895
+ properties: {
896
+ name: {
897
+ type: "string",
898
+ description: "ๅบ”็”จๅ็งฐ๏ผŒๅชๅ…่ฎธๅญ—ๆฏใ€ๆ•ฐๅญ—ใ€่ฟžๅญ—็ฌฆ๏ผŒไพ‹ๅฆ‚ 'my-app' ๆˆ– 'bmi-calculator'ใ€‚ไป…็”จไบŽๆ–ฐๅปบๅบ”็”จ๏ผ›ๆ›ดๆ–ฐๅทฒๆœ‰ๅบ”็”จๆ—ถๅบ”ไผ˜ๅ…ˆไผ  appId๏ผŒๆœๅŠก็ซฏไผš่‡ชๅŠจๆฒฟ็”จๅކๅฒๅ็งฐใ€‚",
899
+ },
900
+ code: {
901
+ type: "string",
902
+ description:
903
+ "ๅ•ๆ–‡ไปถๅบ”็”จไปฃ็ ๏ผŒๅฟ…้กปๆ˜ฏ ES Module ๆ ผๅผใ€‚ไธŽ files ไบŒ้€‰ไธ€ใ€‚" +
904
+ "็คบไพ‹๏ผš\n" +
905
+ "export default {\n" +
906
+ " async fetch(request, env, ctx) {\n" +
907
+ " return new Response('Hello World!');\n" +
908
+ " }\n" +
909
+ "};",
910
+ },
911
+ files: {
912
+ type: "array",
913
+ description: "ๅคšๆ–‡ไปถ้กน็›ฎ๏ผŒไธŽ code ไบŒ้€‰ไธ€ใ€‚ๆœๅŠก็ซฏ่‡ชๅŠจๆ‰“ๅŒ…ใ€‚ๆ™ฎ้€š Worker ๅ…ฅๅฃ็”จ index.ts/main.ts/worker.ts๏ผ›React SPA ๆŽจ่ main.tsx + App.tsxใ€‚",
914
+ items: {
915
+ type: "object",
916
+ properties: {
917
+ name: { type: "string", description: "ๆ–‡ไปถๅ๏ผŒๅฆ‚ 'index.ts' ๆˆ– 'utils/helper.ts'" },
918
+ code: { type: "string", description: "ๆ–‡ไปถๅ†…ๅฎน" },
919
+ },
920
+ required: ["name", "code"],
921
+ },
922
+ },
923
+ pages: {
924
+ type: "array",
925
+ description: "ๅ…ผๅฎนๅˆซๅ๏ผŒ็ญ‰ๅŒไบŽ filesใ€‚่‹ฅไธŠๆธธ็”Ÿๆˆ็š„ๆ˜ฏ pages ๅญ—ๆฎต๏ผŒ็ณป็ปŸไผš่‡ชๅŠจๆŒ‰ files ๅค„็†ใ€‚",
926
+ items: {
927
+ type: "object",
928
+ properties: {
929
+ name: { type: "string", description: "ๆ–‡ไปถๅ๏ผŒๅฆ‚ 'App.tsx'" },
930
+ code: { type: "string", description: "ๆ–‡ไปถๅ†…ๅฎน" },
931
+ },
932
+ required: ["name", "code"],
933
+ },
934
+ },
935
+ appId: {
936
+ type: "string",
937
+ description:
938
+ "ๅบ”็”จ ID๏ผˆๆœๅŠกๅ™จ่ฟ”ๅ›ž็š„ ULID๏ผ‰ใ€‚" +
939
+ "ๅˆ›ๅปบๆ–ฐๅบ”็”จๆ—ถไธ่ฆไผ ๆญคๅญ—ๆฎต๏ผŒๆœๅŠกๅ™จไผš่‡ชๅŠจ็”Ÿๆˆๅนถๅœจๅ“ๅบ”ไธญ่ฟ”ๅ›žใ€‚" +
940
+ "ๆ›ดๆ–ฐๅทฒๆœ‰ๅบ”็”จๆ—ถ๏ผŒๅกซๅ…ฅไธŠๆฌก้ƒจ็ฝฒๅ“ๅบ”ไธญ่ฟ”ๅ›ž็š„ appId๏ผŒๅฎž็Žฐ่ฆ†็›–ๆ›ดๆ–ฐ่€Œ้žๆ–ฐๅปบใ€‚",
941
+ },
942
+ framework: {
943
+ type: "string",
944
+ enum: ["worker", "react-spa"],
945
+ description:
946
+ "ๅบ”็”จๆก†ๆžถใ€‚" +
947
+ "'worker'๏ผˆ้ป˜่ฎค๏ผ‰๏ผš็›ดๆŽฅ้ƒจ็ฝฒ export default { fetch } ไปฃ็ ใ€‚" +
948
+ "'react-spa'๏ผš้ƒจ็ฝฒๅคšๆ–‡ไปถ React ๅ•้กตๅบ”็”จ๏ผŒ้€‚ๅˆๅคๆ‚ไบคไบ’ใ€ๅ›พ่กจๅ’Œ็ป„ไปถๅŒ–้กต้ขใ€‚React SPA ๆจกๅผๅฟ…้กป้…ๅˆ files ไฝฟ็”จใ€‚",
949
+ },
950
+ spaceId: {
951
+ type: "string",
952
+ description:
953
+ "ๅฏ้€‰็š„ Space IDใ€‚้ป˜่ฎคไผšไผ˜ๅ…ˆไฝฟ็”จๅฝ“ๅ‰ๆญฃๅœจ็ผ–่พ‘/ๅฏน่ฏ็š„็ฉบ้—ดใ€‚" +
954
+ "ๅฆ‚ๆžœๆ˜ฏๅฐšๆœช็ป‘ๅฎš็ฉบ้—ด็š„ๆ—งๅบ”็”จ๏ผŒ้‡ๆ–ฐ้ƒจ็ฝฒๆ—ถไผšๆŠŠๅฎƒ็ป‘ๅฎšๅˆฐ่ฏฅ็ฉบ้—ด๏ผ›" +
955
+ "ๅทฒ็ป็ป‘ๅฎš็ฉบ้—ด็š„ๅบ”็”จไธไผšๅ› ไธบ่ฟ™ไธชๅญ—ๆฎต่ขซ่ฟ็งปใ€‚",
956
+ },
957
+ },
958
+ anyOf: [
959
+ { required: ["name"] },
960
+ { required: ["appId"] },
961
+ ],
962
+ // note: ๆ–ฐๅปบๅบ”็”จๆ—ถๅฟ…้กปๆ˜พๅผๆไพ› name๏ผ›ๆ›ดๆ–ฐๅทฒๆœ‰ๅบ”็”จๆ—ถๅ…่ฎธไป…ไผ  appId
963
+ },
964
+ };
965
+
966
+ export const appPreflightFunctionSchema = {
967
+ name: "appPreflight",
968
+ description:
969
+ "ๅœจ็œŸๆญฃ้ƒจ็ฝฒๅ‰ๅ…ˆๅšๅบ”็”จ้ข„ๆฃ€๏ผŒๆฃ€ๆŸฅ React SPA / Worker ็š„ๅ…ฅๅฃๆ–‡ไปถใ€็™ฝๅๅ•ไพ่ต–ใ€ๅ›พๆ ‡ๅใ€CSS import ๅ’Œๅธธ่ง้ƒจ็ฝฒ้”™่ฏฏใ€‚",
970
+ parameters: {
971
+ type: "object",
972
+ properties: {
973
+ name: {
974
+ type: "string",
975
+ description: "ๅบ”็”จๅ็งฐใ€‚ๆ–ฐๅปบๅบ”็”จๅปบ่ฎฎไผ ๏ผ›ๆ›ดๆ–ฐๅทฒๆœ‰ๅบ”็”จๆ—ถๅฏ้…ๅˆ appId ไธ€่ตทไผ ใ€‚",
976
+ },
977
+ code: {
978
+ type: "string",
979
+ description: "ๅ•ๆ–‡ไปถ Worker ไปฃ็ ๏ผŒไธŽ files ไบŒ้€‰ไธ€ใ€‚",
980
+ },
981
+ files: {
982
+ type: "array",
983
+ description: "ๅคšๆ–‡ไปถๆบ็ ๏ผŒไธŽ code ไบŒ้€‰ไธ€ใ€‚React SPA ๆŽจ่ main.tsx + App.tsxใ€‚",
984
+ items: {
985
+ type: "object",
986
+ properties: {
987
+ name: { type: "string" },
988
+ code: { type: "string" },
989
+ },
990
+ required: ["name", "code"],
991
+ },
992
+ },
993
+ pages: {
994
+ type: "array",
995
+ description: "ๅ…ผๅฎนๅˆซๅ๏ผŒ็ญ‰ๅŒไบŽ filesใ€‚",
996
+ items: {
997
+ type: "object",
998
+ properties: {
999
+ name: { type: "string" },
1000
+ code: { type: "string" },
1001
+ },
1002
+ required: ["name", "code"],
1003
+ },
1004
+ },
1005
+ appId: {
1006
+ type: "string",
1007
+ description: "ๅทฒๅญ˜ๅœจๅบ”็”จ็š„ appId๏ผ›็”จไบŽๆ›ดๆ–ฐๅ‰ๆ ก้ชŒใ€‚",
1008
+ },
1009
+ framework: {
1010
+ type: "string",
1011
+ enum: ["worker", "react-spa"],
1012
+ },
1013
+ spaceId: {
1014
+ type: "string",
1015
+ description: "ๅฏ้€‰ Space IDใ€‚",
1016
+ },
1017
+ },
1018
+ anyOf: [{ required: ["name"] }, { required: ["appId"] }],
1019
+ },
1020
+ };
1021
+
1022
+ export async function appPreflightFunc(
1023
+ rawArgs: AppDeployArgs,
1024
+ thunkApi: any
1025
+ ): Promise<{ rawData: any; displayData: string }> {
1026
+ const args = normalizeAppDeployArgs(rawArgs);
1027
+ const deploySpaceId = await resolveAppDeploySpaceId(args, thunkApi);
1028
+ const data = await callToolApi<AppPreflightResult>(
1029
+ thunkApi,
1030
+ "/api/app/preflight",
1031
+ {
1032
+ name: args.name,
1033
+ code: args.code,
1034
+ files: args.files,
1035
+ appId: args.appId,
1036
+ framework: args.framework,
1037
+ ...(deploySpaceId ? { spaceId: deploySpaceId } : {}),
1038
+ },
1039
+ { withAuth: true }
1040
+ );
1041
+
1042
+ const lines = [
1043
+ data.ok ? "โœ… ้ข„ๆฃ€้€š่ฟ‡" : "โŒ ้ข„ๆฃ€ๅคฑ่ดฅ",
1044
+ `- ๆก†ๆžถ: ${data.framework}`,
1045
+ `- ๆ‘˜่ฆ: ${data.summary}`,
1046
+ ];
1047
+ const repairPayload = data.ok
1048
+ ? null
1049
+ : buildAppRepairPayload({
1050
+ summary: data.summary,
1051
+ framework: data.framework,
1052
+ issues: data.issues,
1053
+ warnings: data.warnings,
1054
+ entryFile: data.entryFile,
1055
+ externalImports: data.externalImports,
1056
+ });
1057
+ if (data.entryFile) lines.push(`- ๅ…ฅๅฃๆ–‡ไปถ: ${data.entryFile}`);
1058
+ if (data.issues?.length) {
1059
+ lines.push(`\n้—ฎ้ข˜:\n${formatPreflightIssues(data.issues)}`);
1060
+ }
1061
+ if (repairPayload?.repairPlan) {
1062
+ lines.push(`\n${formatRepairPlan(repairPayload.repairPlan)}`);
1063
+ }
1064
+ if (data.warnings?.length) {
1065
+ lines.push(`\n่ญฆๅ‘Š:\n${data.warnings.map((warning) => `- ${warning}`).join("\n")}`);
1066
+ }
1067
+
1068
+ return {
1069
+ rawData: repairPayload ?? data,
1070
+ displayData: lines.join("\n"),
1071
+ };
1072
+ }
1073
+
1074
+ export async function appDeployFunc(
1075
+ rawArgs: AppDeployArgs,
1076
+ thunkApi: any,
1077
+ context?: { parentMessageId: string; toolRunId?: string; userInput?: string }
1078
+ ): Promise<{ rawData: any; displayData: string }> {
1079
+ const args = normalizeAppDeployArgs(rawArgs);
1080
+ const { name, code, files, appId, framework } = args;
1081
+ const deploySpaceId = await resolveAppDeploySpaceId(args, thunkApi);
1082
+ const toolRunId = context?.toolRunId;
1083
+ if (!name && !appId) throw new Error("ๅฟ…้กปๆไพ› name ๅ‚ๆ•ฐ๏ผˆๆ–ฐๅปบๅบ”็”จๆ—ถๅฟ…ๅกซ๏ผ›ๆ›ดๆ–ฐๅทฒๆœ‰ๅบ”็”จๆ—ถๅฏ็œ็•ฅ๏ผŒๆœๅŠก็ซฏไปŽ appId ่ฎฐๅฝ•ไธญ่‡ชๅŠจ่กฅๅ……๏ผ‰");
1084
+ if (framework === "react-spa" && code && (!files || files.length === 0)) {
1085
+ throw new Error('React SPA ้œ€่ฆไผ  `files`๏ผˆ่‡ณๅฐ‘ `main.tsx` + `App.tsx`๏ผ‰๏ผŒไธ่ƒฝๅชไผ  `code`ใ€‚');
1086
+ }
1087
+ if (!code && (!files || files.length === 0)) throw new Error("ๅฟ…้กปๆไพ› code ๆˆ– files ๅ‚ๆ•ฐ");
1088
+ if ((rawArgs as { deployTarget?: string }).deployTarget && (rawArgs as { deployTarget?: string }).deployTarget !== "platform") {
1089
+ throw new Error("ๅทฒไธๅ†ๆ”ฏๆŒ Cloudflare ้ƒจ็ฝฒ็›ฎๆ ‡๏ผŒ่ฏทไฝฟ็”จๅนณๅฐๆ‰˜็ฎกใ€‚");
1090
+ }
1091
+
1092
+ updateDeployProgress(
1093
+ thunkApi,
1094
+ toolRunId,
1095
+ "prepare",
1096
+ "ๆญฃๅœจๆ•ด็†้ƒจ็ฝฒๅ‚ๆ•ฐโ€ฆ",
1097
+ "running",
1098
+ framework === "react-spa" ? "ๅ‡†ๅค‡ React SPA ๅคšๆ–‡ไปถๆž„ๅปบ" : "ๅ‡†ๅค‡ๅ•ๆ–‡ไปถ Worker / ๅคšๆ–‡ไปถ Worker ้ƒจ็ฝฒ"
1099
+ );
1100
+
1101
+ const guardUserInput = context?.userInput ?? getLatestUserInputFromThunk(thunkApi);
1102
+ if (appId && (code || (files && files.length > 0)) && guardUserInput) {
1103
+ try {
1104
+ const previousSource = await callToolApi<{
1105
+ code?: string;
1106
+ files?: AppSourceFile[];
1107
+ }>(thunkApi, "/api/app/prepare-edit", { appId }, { withAuth: true });
1108
+ const guardResult = evaluateSmallVisualEditGuard({
1109
+ userInput: guardUserInput,
1110
+ previousSource,
1111
+ nextSource: { code, files },
1112
+ });
1113
+ if (!guardResult.ok) {
1114
+ updateDeployProgress(
1115
+ thunkApi,
1116
+ toolRunId,
1117
+ "prepare",
1118
+ guardResult.summary,
1119
+ "failed",
1120
+ "ๆฃ€ๆต‹ๅˆฐๅฐ่ง†่ง‰ไฟฎๆ”น่ถ…ๅ‡บ่Œƒๅ›ด๏ผŒ่ฆๆฑ‚ๅ…ˆๆ”ถๆ•›ๆ”นๅŠจ"
1121
+ );
1122
+ throw new ToolResultError(guardResult.summary, {
1123
+ code: guardResult.rawData.code,
1124
+ rawData: guardResult.rawData,
1125
+ displayData: guardResult.displayData,
1126
+ retryable: true,
1127
+ });
1128
+ }
1129
+ } catch (error: any) {
1130
+ if (
1131
+ error instanceof ToolResultError ||
1132
+ error?.name === "ToolResultError" ||
1133
+ error?.code === "SMALL_VISUAL_SCOPE_EXCEEDED"
1134
+ ) {
1135
+ throw error;
1136
+ }
1137
+ }
1138
+ }
1139
+
1140
+ let preflightData: AppPreflightResult;
1141
+ try {
1142
+ preflightData = await callToolApi<AppPreflightResult>(
1143
+ thunkApi,
1144
+ "/api/app/preflight",
1145
+ {
1146
+ name,
1147
+ code,
1148
+ files,
1149
+ appId,
1150
+ framework,
1151
+ ...(deploySpaceId ? { spaceId: deploySpaceId } : {}),
1152
+ },
1153
+ { withAuth: true }
1154
+ );
1155
+ } catch (error: any) {
1156
+ if (isTransportStoplossError(error)) {
1157
+ const payload = buildAppStoplossPayload({
1158
+ summary: "้ข„ๆฃ€ๆŽฅๅฃ่ฟ”ๅ›žไบ†ๅผ‚ๅธธๅ“ๅบ”๏ผŒๆš‚ๆ—ถๆ— ๆณ•ๅˆคๆ–ญไปฃ็ ้—ฎ้ข˜ใ€‚",
1159
+ framework: framework ?? "worker",
1160
+ stage: "preflight",
1161
+ error,
1162
+ });
1163
+ const displayMessage = formatStoplossPlan(payload);
1164
+ updateDeployProgress(
1165
+ thunkApi,
1166
+ toolRunId,
1167
+ "preflight",
1168
+ displayMessage,
1169
+ "failed",
1170
+ "ๆฃ€ๆต‹ๅˆฐ้ข„ๆฃ€้€š้“ๅผ‚ๅธธ๏ผŒๅทฒๅœๆญข่‡ชๅŠจ้‡่ฏ•"
1171
+ );
1172
+ throw new ToolResultError(payload.summary, {
1173
+ code: payload.code,
1174
+ rawData: payload,
1175
+ displayData: displayMessage,
1176
+ retryable: false,
1177
+ });
1178
+ }
1179
+ const rewritten = rewriteAppDeployError(error?.message || String(error), args);
1180
+ const issues = (error?.details as { issues?: AppPreflightIssue[] } | undefined)?.issues ?? [];
1181
+ const repairPayload = buildAppRepairPayload({
1182
+ summary: rewritten,
1183
+ framework: framework ?? "worker",
1184
+ issues,
1185
+ });
1186
+ const displayMessage = [
1187
+ rewritten,
1188
+ formatPreflightIssues(issues),
1189
+ formatRepairPlan(repairPayload.repairPlan),
1190
+ ]
1191
+ .filter(Boolean)
1192
+ .join("\n");
1193
+ updateDeployProgress(
1194
+ thunkApi,
1195
+ toolRunId,
1196
+ "preflight",
1197
+ displayMessage,
1198
+ "failed",
1199
+ "้ข„ๆฃ€ๆŽฅๅฃๆ‰ง่กŒๅคฑ่ดฅ"
1200
+ );
1201
+ throw new ToolResultError(rewritten, {
1202
+ code: error?.code ?? repairPayload.code,
1203
+ rawData: repairPayload,
1204
+ displayData: displayMessage,
1205
+ retryable: true,
1206
+ });
1207
+ }
1208
+
1209
+ if (!preflightData.ok) {
1210
+ const repairPayload = buildAppRepairPayload({
1211
+ summary: preflightData.summary,
1212
+ framework: preflightData.framework,
1213
+ issues: preflightData.issues,
1214
+ warnings: preflightData.warnings,
1215
+ entryFile: preflightData.entryFile,
1216
+ externalImports: preflightData.externalImports,
1217
+ });
1218
+ const displayMessage = [
1219
+ preflightData.summary,
1220
+ formatPreflightIssues(preflightData.issues),
1221
+ formatRepairPlan(repairPayload.repairPlan),
1222
+ ]
1223
+ .filter(Boolean)
1224
+ .join("\n");
1225
+ updateDeployProgress(
1226
+ thunkApi,
1227
+ toolRunId,
1228
+ "preflight",
1229
+ displayMessage,
1230
+ "failed",
1231
+ "่ฏทๅ…ˆไฟฎๅค้ข„ๆฃ€้—ฎ้ข˜๏ผŒๅ†้‡ๆ–ฐ้ƒจ็ฝฒ"
1232
+ );
1233
+ throw new ToolResultError(preflightData.summary, {
1234
+ code: "PREFLIGHT_FAILED",
1235
+ rawData: repairPayload,
1236
+ displayData: displayMessage,
1237
+ retryable: true,
1238
+ });
1239
+ }
1240
+
1241
+ updateDeployProgress(
1242
+ thunkApi,
1243
+ toolRunId,
1244
+ "preflight",
1245
+ preflightData.summary || "้ข„ๆฃ€้€š่ฟ‡",
1246
+ "succeeded",
1247
+ preflightData.entryFile
1248
+ ? `ๅ…ฅๅฃๆ–‡ไปถ๏ผš${preflightData.entryFile}`
1249
+ : framework === "react-spa"
1250
+ ? "React SPA ็บฆๆŸๆฃ€ๆŸฅ้€š่ฟ‡"
1251
+ : "Worker ็บฆๆŸๆฃ€ๆŸฅ้€š่ฟ‡"
1252
+ );
1253
+
1254
+ let startData: AppDeployStartResult;
1255
+ try {
1256
+ startData = await callToolApi<AppDeployStartResult>(
1257
+ thunkApi,
1258
+ "/api/app/deploy",
1259
+ {
1260
+ name,
1261
+ code,
1262
+ files,
1263
+ appId,
1264
+ framework,
1265
+ ...(deploySpaceId ? { spaceId: deploySpaceId } : {}),
1266
+ },
1267
+ { withAuth: true }
1268
+ );
1269
+ } catch (error: any) {
1270
+ if (isTransportStoplossError(error)) {
1271
+ const payload = buildAppStoplossPayload({
1272
+ summary: "้ƒจ็ฝฒๆŽฅๅฃ่ฟ”ๅ›žไบ†ๅผ‚ๅธธๅ“ๅบ”๏ผŒๅฝ“ๅ‰ไธ่ƒฝ็ปง็ปญ่‡ชๅŠจ้ƒจ็ฝฒใ€‚",
1273
+ framework: framework ?? "worker",
1274
+ stage: "deploy",
1275
+ error,
1276
+ });
1277
+ const displayMessage = formatStoplossPlan(payload);
1278
+ updateDeployProgress(
1279
+ thunkApi,
1280
+ toolRunId,
1281
+ framework === "react-spa" ? "build" : "prepare",
1282
+ displayMessage,
1283
+ "failed",
1284
+ "ๆฃ€ๆต‹ๅˆฐ้ƒจ็ฝฒ้€š้“ๅผ‚ๅธธ๏ผŒๅทฒๅœๆญข่‡ชๅŠจ้‡่ฏ•"
1285
+ );
1286
+ throw new ToolResultError(payload.summary, {
1287
+ code: payload.code,
1288
+ rawData: payload,
1289
+ displayData: displayMessage,
1290
+ retryable: false,
1291
+ });
1292
+ }
1293
+ const rewritten = rewriteAppDeployError(error?.message || String(error), args);
1294
+ const issues = (error?.details as { issues?: AppPreflightIssue[] } | undefined)?.issues ?? [];
1295
+ const repairPayload = buildAppRepairPayload({
1296
+ summary: rewritten,
1297
+ framework: framework ?? "worker",
1298
+ issues,
1299
+ });
1300
+ const displayMessage = [
1301
+ rewritten,
1302
+ formatPreflightIssues(issues),
1303
+ formatRepairPlan(repairPayload.repairPlan),
1304
+ ]
1305
+ .filter(Boolean)
1306
+ .join("\n");
1307
+ updateDeployProgress(
1308
+ thunkApi,
1309
+ toolRunId,
1310
+ framework === "react-spa" ? "build" : "prepare",
1311
+ displayMessage,
1312
+ "failed",
1313
+ "่ฏทๆ นๆฎๆ็คบ่ฐƒๆ•ดๅ‚ๆ•ฐๆˆ–ไปฃ็ ๅŽ้‡่ฏ•"
1314
+ );
1315
+ throw new ToolResultError(rewritten, {
1316
+ code: error?.code ?? repairPayload.code,
1317
+ rawData: repairPayload,
1318
+ displayData: displayMessage,
1319
+ retryable: true,
1320
+ });
1321
+ }
1322
+
1323
+ updateDeployProgress(
1324
+ thunkApi,
1325
+ toolRunId,
1326
+ startData.steps ?? "prepare",
1327
+ startData.summary ?? "้ƒจ็ฝฒ่ฏทๆฑ‚ๅทฒๅ‘้€ๅˆฐๆœๅŠก็ซฏโ€ฆ",
1328
+ "running"
1329
+ );
1330
+ const sharedState = {
1331
+ done: false,
1332
+ result: undefined as AppDeployApiResult | undefined,
1333
+ failure: undefined as Error | undefined,
1334
+ error: undefined as string | undefined,
1335
+ lastSteps: Array.isArray(startData.steps) ? startData.steps : [],
1336
+ };
1337
+ const ssePromise = startData.eventChannel
1338
+ ? subscribeToDeployEvents({
1339
+ thunkApi,
1340
+ deployArgs: args,
1341
+ toolRunId,
1342
+ jobId: startData.jobId,
1343
+ eventChannel: startData.eventChannel,
1344
+ sharedState,
1345
+ }).catch(() => {
1346
+ // SSE ไป…ไฝœไธบๅŠ ้€Ÿ้€š้“๏ผŒๅคฑ่ดฅๅŽ็”ฑ่ฝฎ่ฏขๅ…œๅบ•
1347
+ })
1348
+ : Promise.resolve();
1349
+
1350
+ const data = await pollAppDeployJob(thunkApi, args, toolRunId, startData.jobId, {
1351
+ sharedState,
1352
+ });
1353
+ void ssePromise;
1354
+ const primaryUrl = data.customUrl ?? data.url;
1355
+ const previewCheck = data.previewCheck;
1356
+ const previewReady = data.previewReady ?? previewCheck?.ready;
1357
+
1358
+ // ้ƒจ็ฝฒๆˆๅŠŸๅŽๅฐ† appRecord ๅŒๆญฅๅˆฐๆœฌๅœฐ DB ๅ’Œๆ‰€ๆœ‰ syncServers
1359
+ if (data.appKey && data.appRecord) {
1360
+ void thunkApi.dispatch(syncAppRecord(data.appKey, data.appRecord));
1361
+ }
1362
+
1363
+ const lines = [
1364
+ `๐Ÿš€ ๅบ”็”จ้ƒจ็ฝฒๆˆๅŠŸ๏ผ`,
1365
+ `- ๅ็งฐ: ${data.userFriendlyName}`,
1366
+ ...(data.appId ? [`- appId: ${data.appId}`] : []),
1367
+ `- ่ฎฟ้—ฎๅœฐๅ€: ${primaryUrl}`,
1368
+ `- ๆ›ดๆ–ฐๆ—ถ้—ด: ${data.modifiedOn ?? "ๅˆšๅˆš"}`,
1369
+ ];
1370
+
1371
+ if (data.bundleWarnings?.length) {
1372
+ lines.push(`\nโš ๏ธ ๆ‰“ๅŒ…่ญฆๅ‘Š:\n${data.bundleWarnings.join("\n")}`);
1373
+ }
1374
+ if (previewCheck?.attempted && previewReady === false) {
1375
+ lines.push("\nโณ ็ซ™็‚นๅทฒๅ‘ๅธƒ๏ผŒไฝ†้ฆ–ๆฌกๅŠ ่ฝฝๅฏ่ƒฝ็จๆ…ข๏ผ›่Šๅคฉๅก็‰‡้‡Œไผš็ปง็ปญๆ˜พ็คบ้ข„่งˆๅŠ ่ฝฝ็Šถๆ€ใ€‚");
1376
+ }
1377
+ lines.push(`\nๅฏไปฅ็›ดๆŽฅ่ฎฟ้—ฎ ${primaryUrl} ๆต‹่ฏ•ๆ•ˆๆžœใ€‚`);
1378
+
1379
+ return {
1380
+ rawData: { ...data, appUrl: primaryUrl, previewCheck },
1381
+ displayData: lines.join("\n"),
1382
+ };
1383
+ }
1384
+
1385
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
1386
+ // appList โ€” ๅˆ—ๅ‡บ็”จๆˆทๅทฒ้ƒจ็ฝฒ็š„ๆ‰€ๆœ‰ๅบ”็”จ
1387
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
1388
+ export const appListFunctionSchema = {
1389
+ name: "appList",
1390
+ description: "ๅˆ—ๅ‡บๅบ”็”จๅˆ—่กจใ€‚ไธไผ  spaceId ๆ—ถๅˆ—ๅ‡บๅฝ“ๅ‰็”จๆˆท่‡ชๅทฑ็š„ๅบ”็”จ๏ผ›ไผ ๅ…ฅ spaceId ๆ—ถๅˆ—ๅ‡บ่ฏฅ Space ไธ‹็š„ๅบ”็”จใ€‚",
1391
+ parameters: {
1392
+ type: "object",
1393
+ properties: {
1394
+ spaceId: {
1395
+ type: "string",
1396
+ description: "ๅฏ้€‰ใ€‚่‹ฅๆไพ›๏ผŒๅˆ™ๅชๅˆ—ๅ‡บ่ฏฅ Space ไธ‹็š„ๅบ”็”จใ€‚",
1397
+ },
1398
+ },
1399
+ },
1400
+ };
1401
+
1402
+ export async function appListFunc(
1403
+ args: { spaceId?: string },
1404
+ thunkApi: any
1405
+ ): Promise<{ rawData: any; displayData: string }> {
1406
+ const data = await callToolApi<{
1407
+ success: boolean;
1408
+ workers: Array<{
1409
+ userFriendlyName: string;
1410
+ url: string;
1411
+ customUrl?: string;
1412
+ appId?: string;
1413
+ modifiedOn: string;
1414
+ }>;
1415
+ }>(thunkApi, "/api/app/list", args?.spaceId ? { spaceId: args.spaceId } : {}, { withAuth: true });
1416
+
1417
+ if (!data.workers.length) {
1418
+ return {
1419
+ rawData: data,
1420
+ displayData: "๐Ÿ“ญ ไฝ ่ฟ˜ๆฒกๆœ‰้ƒจ็ฝฒไปปไฝ•ๅบ”็”จใ€‚",
1421
+ };
1422
+ }
1423
+
1424
+ const list = data.workers
1425
+ .map((w) => {
1426
+ const url = w.customUrl ?? w.url;
1427
+ const id = w.appId ? ` (appId: ${w.appId})` : "";
1428
+ return `- **${w.userFriendlyName}**${id}: ${url} (ๆ›ดๆ–ฐ: ${w.modifiedOn?.slice(0, 10) ?? "-"})`;
1429
+ })
1430
+ .join("\n");
1431
+
1432
+ return {
1433
+ rawData: data,
1434
+ displayData: `๐Ÿ“‹ ๅทฒ้ƒจ็ฝฒ็š„ๅบ”็”จ (${data.workers.length} ไธช):\n${list}`,
1435
+ };
1436
+ }
1437
+
1438
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
1439
+ // appDelete โ€” ๅˆ ้™คไธ€ไธชๅบ”็”จ
1440
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
1441
+ export const appDeleteFunctionSchema = {
1442
+ name: "appDelete",
1443
+ description: "ๅˆ ้™คไธ€ไธชๅทฒ้ƒจ็ฝฒ็š„ๅบ”็”จใ€‚ๅˆ ้™คๅŽ URL ็ซ‹ๅณๅคฑๆ•ˆ๏ผŒไธๅฏๆขๅคใ€‚",
1444
+ parameters: {
1445
+ type: "object",
1446
+ properties: {
1447
+ appId: {
1448
+ type: "string",
1449
+ description: "ๅบ”็”จ IDใ€‚ๅฟ…้กปๅ…ˆ็”จ appList / appDeploy / appRead ๆ‹ฟๅˆฐๅฎƒ๏ผŒๅ†ๆŒ‰ appId ๅˆ ้™คใ€‚",
1450
+ },
1451
+ },
1452
+ required: ["appId"],
1453
+ },
1454
+ };
1455
+
1456
+ export async function appDeleteFunc(
1457
+ args: { appId: string },
1458
+ thunkApi: any
1459
+ ): Promise<{ rawData: any; displayData: string }> {
1460
+ const { appId } = args;
1461
+ if (!appId) throw new Error("ๅฟ…้กปๆไพ› appId ๅ‚ๆ•ฐ๏ผ›่ฏทๅ…ˆ่ฐƒ็”จ appList ๆˆ– appRead ่Žทๅ–็›ฎๆ ‡ๅบ”็”จ ID");
1462
+
1463
+ // ๅ…ˆ่Žทๅ– appKey๏ผŒๅ†่ตฐๅ…จ server ๅˆ ้™ค่ทฏๅพ„
1464
+ const appInfo = await callToolApi<{
1465
+ appKey?: string;
1466
+ appId?: string;
1467
+ name?: string;
1468
+ }>(thunkApi, "/api/app/get", { appId }, { withAuth: true });
1469
+
1470
+ if (!appInfo.appKey) {
1471
+ throw new Error("ๆ— ๆณ•่งฃๆž appKey๏ผŒๆ— ๆณ•ๆ‰ง่กŒ็ปŸไธ€ๅˆ ้™ค");
1472
+ }
1473
+
1474
+ await thunkApi.dispatch(deleteDbKey(appInfo.appKey));
1475
+
1476
+ return {
1477
+ rawData: { deleted: true },
1478
+ displayData: `๐Ÿ—‘๏ธ ๅบ”็”จ "${appInfo.name ?? appId}" ๅทฒๆˆๅŠŸๅˆ ้™คใ€‚`,
1479
+ };
1480
+ }
1481
+
1482
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
1483
+ // appRead โ€” ่ฏปๅ–ๅบ”็”จๅฝ“ๅ‰ไปฃ็ ๏ผˆ็”จไบŽไฟฎๆ”นๅ‰่Žทๅ–็Žฐๆœ‰ๅ†…ๅฎน๏ผ‰
1484
+ // โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€
1485
+ export const appReadFunctionSchema = {
1486
+ name: "appRead",
1487
+ description:
1488
+ "่ฏปๅ–ๅทฒ้ƒจ็ฝฒๅบ”็”จ็š„ๅฝ“ๅ‰ไปฃ็ ใ€‚ๅœจไฟฎๆ”นๅบ”็”จ๏ผˆๅขžๅˆ ๅŠŸ่ƒฝใ€่ฐƒๆ•ดๆ ทๅผ็ญ‰๏ผ‰ไน‹ๅ‰๏ผŒๅฟ…้กปๅ…ˆ่ฐƒ็”จๆญคๅทฅๅ…ท่Žทๅ–็Žฐๆœ‰ไปฃ็ ๏ผŒๅ†ๅŸบไบŽ็Žฐๆœ‰ไปฃ็ ่ฟ›่กŒไฟฎๆ”นๅŽ้‡ๆ–ฐ้ƒจ็ฝฒใ€‚ๅฟ…้กปไผ  appId๏ผ›ๅฆ‚่ฟ˜ไธ็Ÿฅ้“ appId๏ผŒ่ฏทๅ…ˆ่ฐƒ็”จ appListใ€‚",
1489
+ parameters: {
1490
+ type: "object",
1491
+ properties: {
1492
+ appId: {
1493
+ type: "string",
1494
+ description: "ๅบ”็”จ ID๏ผˆไปŽ appList ๆˆ–ไน‹ๅ‰็š„ appRead/appDeploy ่ฟ”ๅ›ž็ป“ๆžœไธญ่Žทๅ–๏ผ‰ใ€‚ๆฏ” name ๆ›ด็ฒพ็กฎ๏ผŒๅŽ็ปญๆ›ดๆ–ฐ/ๅˆ ้™ค้ƒฝๅบ”ไผ˜ๅ…ˆไฝฟ็”จใ€‚",
1495
+ },
1496
+ },
1497
+ required: ["appId"],
1498
+ },
1499
+ };
1500
+
1501
+ export async function appReadFunc(
1502
+ args: { appId: string },
1503
+ thunkApi: any
1504
+ ): Promise<{ rawData: any; displayData: string }> {
1505
+ const { appId } = args;
1506
+ if (!appId) throw new Error("ๅฟ…้กปๆไพ› appId ๅ‚ๆ•ฐ๏ผ›่ฏทๅ…ˆ่ฐƒ็”จ appList ่Žทๅ–็›ฎๆ ‡ๅบ”็”จ ID");
1507
+
1508
+ const data = await callToolApi<{
1509
+ success: boolean;
1510
+ appId: string;
1511
+ userFriendlyName: string;
1512
+ url: string;
1513
+ customUrl?: string;
1514
+ code: string;
1515
+ files?: Array<{ name: string; code: string }>;
1516
+ framework?: "worker" | "react-spa";
1517
+ }>(thunkApi, "/api/app/prepare-edit", { appId }, { withAuth: true });
1518
+
1519
+ const primaryUrl = data.customUrl ?? data.url;
1520
+ const displayBody = Array.isArray(data.files) && data.files.length > 0
1521
+ ? data.files
1522
+ .map((file) => `### ${file.name}\n\`\`\`${file.name.endsWith(".tsx") || file.name.endsWith(".ts") ? "typescript" : "javascript"}\n${file.code}\n\`\`\``)
1523
+ .join("\n\n")
1524
+ : "```javascript\n" + data.code + "\n```";
1525
+ const snapshotWarning = buildAppReadSnapshotWarning(
1526
+ classifyAppReadSnapshot(data)
1527
+ );
1528
+ const styleSystemAnalysis = analyzeAppStyleSystem(data);
1529
+ const styleSystemHint = buildAppStyleSystemHint(styleSystemAnalysis);
1530
+
1531
+ return {
1532
+ rawData: {
1533
+ ...data,
1534
+ styleSystemStatus: styleSystemAnalysis.status,
1535
+ legacyMigrationRecommended:
1536
+ styleSystemAnalysis.legacyMigrationRecommended,
1537
+ styleSystemEvidence: styleSystemAnalysis.evidence,
1538
+ ...(styleSystemHint ? { styleSystemHint } : {}),
1539
+ },
1540
+ displayData:
1541
+ `๐Ÿ“„ ๅบ”็”จ "${data.userFriendlyName}" ๅฝ“ๅ‰ไปฃ็ ๏ผš\n` +
1542
+ `- appId: ${data.appId}\n` +
1543
+ `- ่ฎฟ้—ฎๅœฐๅ€: ${primaryUrl}\n` +
1544
+ (snapshotWarning ? `\n${snapshotWarning}\n` : "\n") +
1545
+ (styleSystemHint ? `\n${styleSystemHint}\n` : "") +
1546
+ "\n" +
1547
+ displayBody,
1548
+ };
1549
+ }