nolo-cli 0.1.10 โ†’ 0.1.12

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