nolo-cli 0.1.10 โ†’ 0.1.11

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 (240) hide show
  1. package/README.md +0 -32
  2. package/agentRuntimeCommands.ts +3 -3
  3. package/commandRegistry.ts +2 -2
  4. package/machineCommands.ts +31 -6
  5. package/package.json +6 -22
  6. package/ai/agent/_executeModel.ts +0 -118
  7. package/ai/agent/agentSlice.ts +0 -525
  8. package/ai/agent/appWorkingMemory.ts +0 -126
  9. package/ai/agent/avatarUtils.ts +0 -24
  10. package/ai/agent/buildEditingContext.ts +0 -373
  11. package/ai/agent/buildSystemPrompt.ts +0 -532
  12. package/ai/agent/cleanAgentMessages.ts +0 -140
  13. package/ai/agent/cliChatClient.ts +0 -119
  14. package/ai/agent/cliExecutor.ts +0 -733
  15. package/ai/agent/cliPrompt.ts +0 -10
  16. package/ai/agent/contextCompiler.ts +0 -107
  17. package/ai/agent/contextLayerContract.ts +0 -44
  18. package/ai/agent/createAgentSchema.ts +0 -234
  19. package/ai/agent/executeToolCall.ts +0 -58
  20. package/ai/agent/fetchAgentContexts.ts +0 -42
  21. package/ai/agent/generatePrompt.ts +0 -3
  22. package/ai/agent/getFullChatContextKeys.ts +0 -168
  23. package/ai/agent/hooks/fetchPublicAgents.ts +0 -133
  24. package/ai/agent/hooks/useAgentConfig.ts +0 -61
  25. package/ai/agent/hooks/useAgentDialog.ts +0 -35
  26. package/ai/agent/hooks/useAgentFormValidation.ts +0 -202
  27. package/ai/agent/hooks/usePublicAgents.ts +0 -473
  28. package/ai/agent/machineRunPermissions.ts +0 -95
  29. package/ai/agent/persistMessageWithFixedId.ts +0 -37
  30. package/ai/agent/planSlice.ts +0 -259
  31. package/ai/agent/referenceUtils.ts +0 -229
  32. package/ai/agent/runAgentBackground.ts +0 -238
  33. package/ai/agent/runAgentClientLoop.ts +0 -138
  34. package/ai/agent/runtimeGuidance.ts +0 -97
  35. package/ai/agent/runtimeServerBase.ts +0 -37
  36. package/ai/agent/server/fetchPublicAgents.ts +0 -128
  37. package/ai/agent/startParallelAgentStreams.ts +0 -424
  38. package/ai/agent/startupProtocol.ts +0 -53
  39. package/ai/agent/streamAgentChatTurn.ts +0 -1278
  40. package/ai/agent/streamAgentChatTurnUtils.ts +0 -738
  41. package/ai/agent/types.ts +0 -71
  42. package/ai/agent/utils/imageOutput.ts +0 -33
  43. package/ai/agent/utils/sortUtils.ts +0 -250
  44. package/ai/agent/web/referencePickerUtils.ts +0 -146
  45. package/ai/ai.locale.ts +0 -1079
  46. package/ai/chat/accumulateToolCallChunks.ts +0 -95
  47. package/ai/chat/fetchUtils.native.ts +0 -276
  48. package/ai/chat/fetchUtils.ts +0 -153
  49. package/ai/chat/parseApiError.ts +0 -64
  50. package/ai/chat/parseMultilineSSE.ts +0 -95
  51. package/ai/chat/sendOpenAICompletionsRequest.native.ts +0 -682
  52. package/ai/chat/sendOpenAICompletionsRequest.ts +0 -703
  53. package/ai/chat/sendOpenAIResponseRequest.ts +0 -491
  54. package/ai/chat/shouldUseServerProxy.ts +0 -18
  55. package/ai/chat/sseClient.native.ts +0 -91
  56. package/ai/chat/sseClient.ts +0 -67
  57. package/ai/chat/streamReader.native.ts +0 -31
  58. package/ai/chat/streamReader.ts +0 -62
  59. package/ai/chat/updateTotalUsage.ts +0 -72
  60. package/ai/context/buildReferenceContext.ts +0 -437
  61. package/ai/context/calculateContextUsage.ts +0 -133
  62. package/ai/context/retention.ts +0 -165
  63. package/ai/context/tokenUtils.ts +0 -78
  64. package/ai/index.ts +0 -1
  65. package/ai/llm/calculateGeminiImageTokens.ts +0 -57
  66. package/ai/llm/deepinfra.ts +0 -28
  67. package/ai/llm/fireworks.ts +0 -50
  68. package/ai/llm/generateRequestBody.ts +0 -165
  69. package/ai/llm/getModelContextWindow.ts +0 -84
  70. package/ai/llm/getNoloKey.ts +0 -31
  71. package/ai/llm/getPricing.ts +0 -199
  72. package/ai/llm/hooks/useModelPricing.ts +0 -75
  73. package/ai/llm/imagePricing.ts +0 -40
  74. package/ai/llm/isResponseAPIModel.ts +0 -13
  75. package/ai/llm/mimo.ts +0 -71
  76. package/ai/llm/mistral.ts +0 -22
  77. package/ai/llm/modelAvatar.ts +0 -427
  78. package/ai/llm/models.ts +0 -45
  79. package/ai/llm/openrouterModels.ts +0 -269
  80. package/ai/llm/providers.ts +0 -306
  81. package/ai/llm/reasoningModels.ts +0 -28
  82. package/ai/llm/types.ts +0 -59
  83. package/ai/llm/usageRequestOptions.ts +0 -59
  84. package/ai/memory/capture.ts +0 -148
  85. package/ai/memory/consolidate.ts +0 -104
  86. package/ai/memory/delete.ts +0 -147
  87. package/ai/memory/overlay.ts +0 -84
  88. package/ai/memory/query.ts +0 -38
  89. package/ai/memory/queryShared.ts +0 -160
  90. package/ai/memory/rank.ts +0 -105
  91. package/ai/memory/recentRelationshipRecap.ts +0 -249
  92. package/ai/memory/remember.ts +0 -167
  93. package/ai/memory/runtime.ts +0 -76
  94. package/ai/memory/store.ts +0 -20
  95. package/ai/memory/storeShared.ts +0 -76
  96. package/ai/memory/types.ts +0 -46
  97. package/ai/memory/understanding.ts +0 -349
  98. package/ai/memory/understandingGreeting.ts +0 -264
  99. package/ai/messages/type.ts +0 -20
  100. package/ai/policy/personalizationDialog.ts +0 -333
  101. package/ai/policy/runtimePolicy.ts +0 -440
  102. package/ai/policy/selfUpdateFields.ts +0 -48
  103. package/ai/policy/types.ts +0 -64
  104. package/ai/skills/referenceRuntime.ts +0 -274
  105. package/ai/skills/skillDiagnostics.ts +0 -251
  106. package/ai/skills/skillDocBuilder.ts +0 -139
  107. package/ai/skills/skillDocProtocol.ts +0 -434
  108. package/ai/skills/skillReferenceSummary.ts +0 -63
  109. package/ai/skills/skillSummaryMarker.ts +0 -26
  110. package/ai/token/calculatePrice.ts +0 -544
  111. package/ai/token/db.ts +0 -98
  112. package/ai/token/externalToolCost.ts +0 -330
  113. package/ai/token/hooks/useRecords.ts +0 -65
  114. package/ai/token/missingUsageEstimate.ts +0 -42
  115. package/ai/token/modelUsageQuery.ts +0 -252
  116. package/ai/token/normalizeUsage.ts +0 -84
  117. package/ai/token/openaiImageGenerationUsage.ts +0 -56
  118. package/ai/token/prepareTokenUsageData.ts +0 -88
  119. package/ai/token/query.ts +0 -88
  120. package/ai/token/queryUserTokens.ts +0 -59
  121. package/ai/token/resolveBillingTarget.ts +0 -52
  122. package/ai/token/saveTokenRecord.ts +0 -53
  123. package/ai/token/serverDialogProjection.ts +0 -78
  124. package/ai/token/serverTokenWriter.ts +0 -143
  125. package/ai/token/stats.ts +0 -21
  126. package/ai/token/tokenThunks.ts +0 -24
  127. package/ai/token/types.ts +0 -93
  128. package/ai/tools/agent/agentTools.ts +0 -176
  129. package/ai/tools/agent/agentUpdateShared.ts +0 -311
  130. package/ai/tools/agent/callAgentTool.ts +0 -139
  131. package/ai/tools/agent/createAgentTool.ts +0 -512
  132. package/ai/tools/agent/createDialogTool.ts +0 -69
  133. package/ai/tools/agent/createSkillAgentTool.ts +0 -62
  134. package/ai/tools/agent/parallelBudget.ts +0 -221
  135. package/ai/tools/agent/presets/appBuilderPreset.ts +0 -145
  136. package/ai/tools/agent/runLlmTool.ts +0 -96
  137. package/ai/tools/agent/runStreamingAgentTool.ts +0 -73
  138. package/ai/tools/agent/skillAgentArgs.ts +0 -106
  139. package/ai/tools/agent/skillAgentPreset.ts +0 -89
  140. package/ai/tools/agent/streamParallelAgentsTool.ts +0 -122
  141. package/ai/tools/agent/updateAgentTool.ts +0 -96
  142. package/ai/tools/agent/updateSelfTool.ts +0 -113
  143. package/ai/tools/amazonProductScraperTool.ts +0 -86
  144. package/ai/tools/apifyActorClient.ts +0 -45
  145. package/ai/tools/appEditGuard.ts +0 -372
  146. package/ai/tools/appReadSnapshot.ts +0 -153
  147. package/ai/tools/appTools.ts +0 -1549
  148. package/ai/tools/applyEditTool.ts +0 -256
  149. package/ai/tools/applyLineEditsTool.ts +0 -312
  150. package/ai/tools/browserTools/click.ts +0 -33
  151. package/ai/tools/browserTools/closeSession.ts +0 -29
  152. package/ai/tools/browserTools/common.ts +0 -27
  153. package/ai/tools/browserTools/openSession.ts +0 -48
  154. package/ai/tools/browserTools/readContent.ts +0 -38
  155. package/ai/tools/browserTools/selectOption.ts +0 -46
  156. package/ai/tools/browserTools/typeText.ts +0 -42
  157. package/ai/tools/category/createCategoryTool.ts +0 -66
  158. package/ai/tools/category/queryContentsByCategoryTool.ts +0 -69
  159. package/ai/tools/category/updateContentCategoryTool.ts +0 -75
  160. package/ai/tools/cfBrowserTools.ts +0 -319
  161. package/ai/tools/cfSpeechToTextTool.ts +0 -49
  162. package/ai/tools/checkEnvTool.ts +0 -65
  163. package/ai/tools/cloudflareCrawlTool.ts +0 -289
  164. package/ai/tools/codeSearchTool.ts +0 -111
  165. package/ai/tools/codeTools.ts +0 -101
  166. package/ai/tools/createDocTool.ts +0 -132
  167. package/ai/tools/createPlanTool.ts +0 -999
  168. package/ai/tools/createSkillDocTool.ts +0 -155
  169. package/ai/tools/createWorkflowTool.ts +0 -154
  170. package/ai/tools/deepseekOcrTool.ts +0 -34
  171. package/ai/tools/delayTool.ts +0 -31
  172. package/ai/tools/deleteSpacesTool.ts +0 -325
  173. package/ai/tools/deleteSpacesToolModel.ts +0 -159
  174. package/ai/tools/devReloadUtils.ts +0 -29
  175. package/ai/tools/dialogMessageSearch.ts +0 -137
  176. package/ai/tools/doctorSkillTool.ts +0 -72
  177. package/ai/tools/ecommerceScraperTool.ts +0 -86
  178. package/ai/tools/emailTools.ts +0 -549
  179. package/ai/tools/evalSkillTool.ts +0 -92
  180. package/ai/tools/exaSearchTool.ts +0 -64
  181. package/ai/tools/execBashTool.ts +0 -379
  182. package/ai/tools/executeSqlTool.ts +0 -192
  183. package/ai/tools/fetchWebpageSupport.ts +0 -309
  184. package/ai/tools/fetchWebpageTool.ts +0 -84
  185. package/ai/tools/geminiImagePreviewTool.ts +0 -361
  186. package/ai/tools/generateDocxTool.ts +0 -215
  187. package/ai/tools/googleSearchScraperTool.ts +0 -106
  188. package/ai/tools/importDataTool.ts +0 -133
  189. package/ai/tools/importSkillTool.ts +0 -162
  190. package/ai/tools/index.ts +0 -1858
  191. package/ai/tools/listFilesTool.ts +0 -82
  192. package/ai/tools/listUserSpacesTool.ts +0 -113
  193. package/ai/tools/modelUsageTools.ts +0 -142
  194. package/ai/tools/olmOcrTool.ts +0 -34
  195. package/ai/tools/openaiImageTool.ts +0 -218
  196. package/ai/tools/paddleOcrTool.ts +0 -34
  197. package/ai/tools/prepareTools.ts +0 -23
  198. package/ai/tools/readDocTool.ts +0 -84
  199. package/ai/tools/readFileTool.ts +0 -211
  200. package/ai/tools/readTool.ts +0 -163
  201. package/ai/tools/readXPostTool.ts +0 -233
  202. package/ai/tools/rememberMemoryTool.ts +0 -84
  203. package/ai/tools/remotionVideoTool.ts +0 -151
  204. package/ai/tools/searchDialogMessagesTool.ts +0 -222
  205. package/ai/tools/searchRepoTool.ts +0 -115
  206. package/ai/tools/searchWorkspaceTool.ts +0 -259
  207. package/ai/tools/skillFollowup.ts +0 -86
  208. package/ai/tools/surfWeatherTool.ts +0 -169
  209. package/ai/tools/table/addTableRowTool.ts +0 -217
  210. package/ai/tools/table/createTableTool.ts +0 -315
  211. package/ai/tools/table/rowTools.ts +0 -366
  212. package/ai/tools/table/schemaTools.ts +0 -244
  213. package/ai/tools/table/shareTableTool.ts +0 -148
  214. package/ai/tools/table/toolShared.ts +0 -129
  215. package/ai/tools/toolApiClient.ts +0 -198
  216. package/ai/tools/toolNameAliases.ts +0 -57
  217. package/ai/tools/toolResultError.ts +0 -42
  218. package/ai/tools/toolRunSlice.ts +0 -303
  219. package/ai/tools/toolSchemaCompatibility.ts +0 -53
  220. package/ai/tools/toolVisibility.ts +0 -4
  221. package/ai/tools/types.ts +0 -20
  222. package/ai/tools/uiAskChoiceTool.ts +0 -104
  223. package/ai/tools/updateContentTitleTool.ts +0 -84
  224. package/ai/tools/updateDocTool.ts +0 -105
  225. package/ai/tools/updateUserPreferenceProfileTool.ts +0 -145
  226. package/ai/tools/whisperTool.ts +0 -77
  227. package/ai/tools/writeFileTool.ts +0 -210
  228. package/ai/tools/youtubeScraperTool.ts +0 -116
  229. package/ai/tools/ziweiChartTool.ts +0 -678
  230. package/ai/types.ts +0 -55
  231. package/ai/workflow/workflowExecutor.ts +0 -323
  232. package/ai/workflow/workflowSlice.ts +0 -73
  233. package/ai/workflow/workflowTypes.ts +0 -106
  234. package/client/compactDialog.ts +0 -222
  235. package/connector-experimental/capabilities.ts +0 -73
  236. package/connector-experimental/codexBinary.ts +0 -41
  237. package/connector-experimental/heartbeatLoop.ts +0 -22
  238. package/connector-experimental/index.ts +0 -5
  239. package/connector-experimental/machineInfo.ts +0 -46
  240. package/connector-experimental/protocol.ts +0 -54
@@ -1,1549 +0,0 @@
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
- }