nolo-cli 0.1.7 → 0.1.9

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 (247) hide show
  1. package/README.md +107 -5
  2. package/agentRuntimeCommands.ts +464 -0
  3. package/ai/agent/_executeModel.ts +118 -0
  4. package/ai/agent/agentSlice.ts +525 -0
  5. package/ai/agent/appWorkingMemory.ts +126 -0
  6. package/ai/agent/avatarUtils.ts +24 -0
  7. package/ai/agent/buildEditingContext.ts +373 -0
  8. package/ai/agent/buildSystemPrompt.ts +532 -0
  9. package/ai/agent/cleanAgentMessages.ts +140 -0
  10. package/ai/agent/cliChatClient.ts +119 -0
  11. package/ai/agent/cliExecutor.ts +733 -0
  12. package/ai/agent/cliPrompt.ts +10 -0
  13. package/ai/agent/contextCompiler.ts +107 -0
  14. package/ai/agent/contextLayerContract.ts +44 -0
  15. package/ai/agent/createAgentSchema.ts +234 -0
  16. package/ai/agent/executeToolCall.ts +58 -0
  17. package/ai/agent/fetchAgentContexts.ts +42 -0
  18. package/ai/agent/generatePrompt.ts +3 -0
  19. package/ai/agent/getFullChatContextKeys.ts +168 -0
  20. package/ai/agent/hooks/fetchPublicAgents.ts +133 -0
  21. package/ai/agent/hooks/useAgentConfig.ts +61 -0
  22. package/ai/agent/hooks/useAgentDialog.ts +35 -0
  23. package/ai/agent/hooks/useAgentFormValidation.ts +202 -0
  24. package/ai/agent/hooks/usePublicAgents.ts +473 -0
  25. package/ai/agent/machineRunPermissions.ts +95 -0
  26. package/ai/agent/persistMessageWithFixedId.ts +37 -0
  27. package/ai/agent/planSlice.ts +259 -0
  28. package/ai/agent/referenceUtils.ts +229 -0
  29. package/ai/agent/runAgentBackground.ts +238 -0
  30. package/ai/agent/runAgentClientLoop.ts +138 -0
  31. package/ai/agent/runtimeGuidance.ts +97 -0
  32. package/ai/agent/runtimeServerBase.ts +37 -0
  33. package/ai/agent/server/fetchPublicAgents.ts +128 -0
  34. package/ai/agent/startParallelAgentStreams.ts +424 -0
  35. package/ai/agent/startupProtocol.ts +53 -0
  36. package/ai/agent/streamAgentChatTurn.ts +1278 -0
  37. package/ai/agent/streamAgentChatTurnUtils.ts +738 -0
  38. package/ai/agent/types.ts +71 -0
  39. package/ai/agent/utils/imageOutput.ts +33 -0
  40. package/ai/agent/utils/sortUtils.ts +250 -0
  41. package/ai/agent/web/referencePickerUtils.ts +146 -0
  42. package/ai/ai.locale.ts +1075 -0
  43. package/ai/chat/accumulateToolCallChunks.ts +95 -0
  44. package/ai/chat/fetchUtils.native.ts +276 -0
  45. package/ai/chat/fetchUtils.ts +153 -0
  46. package/ai/chat/parseApiError.ts +64 -0
  47. package/ai/chat/parseMultilineSSE.ts +95 -0
  48. package/ai/chat/sendOpenAICompletionsRequest.native.ts +682 -0
  49. package/ai/chat/sendOpenAICompletionsRequest.ts +703 -0
  50. package/ai/chat/sendOpenAIResponseRequest.ts +491 -0
  51. package/ai/chat/shouldUseServerProxy.ts +18 -0
  52. package/ai/chat/sseClient.native.ts +91 -0
  53. package/ai/chat/sseClient.ts +67 -0
  54. package/ai/chat/streamReader.native.ts +31 -0
  55. package/ai/chat/streamReader.ts +62 -0
  56. package/ai/chat/updateTotalUsage.ts +72 -0
  57. package/ai/context/buildReferenceContext.ts +437 -0
  58. package/ai/context/calculateContextUsage.ts +133 -0
  59. package/ai/context/retention.ts +165 -0
  60. package/ai/context/tokenUtils.ts +78 -0
  61. package/ai/index.ts +1 -0
  62. package/ai/llm/calculateGeminiImageTokens.ts +57 -0
  63. package/ai/llm/deepinfra.ts +28 -0
  64. package/ai/llm/fireworks.ts +50 -0
  65. package/ai/llm/generateRequestBody.ts +165 -0
  66. package/ai/llm/getModelContextWindow.ts +84 -0
  67. package/ai/llm/getNoloKey.ts +31 -0
  68. package/ai/llm/getPricing.ts +199 -0
  69. package/ai/llm/hooks/useModelPricing.ts +75 -0
  70. package/ai/llm/imagePricing.ts +40 -0
  71. package/ai/llm/isResponseAPIModel.ts +13 -0
  72. package/ai/llm/mimo.ts +71 -0
  73. package/ai/llm/mistral.ts +22 -0
  74. package/ai/llm/modelAvatar.ts +427 -0
  75. package/ai/llm/models.ts +45 -0
  76. package/ai/llm/openrouterModels.ts +269 -0
  77. package/ai/llm/providers.ts +306 -0
  78. package/ai/llm/reasoningModels.ts +28 -0
  79. package/ai/llm/types.ts +59 -0
  80. package/ai/llm/usageRequestOptions.ts +59 -0
  81. package/ai/memory/capture.ts +148 -0
  82. package/ai/memory/consolidate.ts +104 -0
  83. package/ai/memory/delete.ts +147 -0
  84. package/ai/memory/overlay.ts +84 -0
  85. package/ai/memory/query.ts +38 -0
  86. package/ai/memory/queryShared.ts +160 -0
  87. package/ai/memory/rank.ts +105 -0
  88. package/ai/memory/recentRelationshipRecap.ts +249 -0
  89. package/ai/memory/remember.ts +167 -0
  90. package/ai/memory/runtime.ts +76 -0
  91. package/ai/memory/store.ts +20 -0
  92. package/ai/memory/storeShared.ts +76 -0
  93. package/ai/memory/types.ts +46 -0
  94. package/ai/memory/understanding.ts +349 -0
  95. package/ai/memory/understandingGreeting.ts +264 -0
  96. package/ai/messages/type.ts +20 -0
  97. package/ai/policy/personalizationDialog.ts +333 -0
  98. package/ai/policy/runtimePolicy.ts +440 -0
  99. package/ai/policy/selfUpdateFields.ts +48 -0
  100. package/ai/policy/types.ts +64 -0
  101. package/ai/skills/referenceRuntime.ts +274 -0
  102. package/ai/skills/skillDiagnostics.ts +251 -0
  103. package/ai/skills/skillDocBuilder.ts +139 -0
  104. package/ai/skills/skillDocProtocol.ts +434 -0
  105. package/ai/skills/skillReferenceSummary.ts +63 -0
  106. package/ai/skills/skillSummaryMarker.ts +26 -0
  107. package/ai/token/calculatePrice.ts +544 -0
  108. package/ai/token/db.ts +98 -0
  109. package/ai/token/externalToolCost.ts +330 -0
  110. package/ai/token/hooks/useRecords.ts +65 -0
  111. package/ai/token/missingUsageEstimate.ts +42 -0
  112. package/ai/token/modelUsageQuery.ts +252 -0
  113. package/ai/token/normalizeUsage.ts +84 -0
  114. package/ai/token/openaiImageGenerationUsage.ts +56 -0
  115. package/ai/token/prepareTokenUsageData.ts +88 -0
  116. package/ai/token/query.ts +88 -0
  117. package/ai/token/queryUserTokens.ts +59 -0
  118. package/ai/token/resolveBillingTarget.ts +52 -0
  119. package/ai/token/saveTokenRecord.ts +53 -0
  120. package/ai/token/serverDialogProjection.ts +78 -0
  121. package/ai/token/serverTokenWriter.ts +143 -0
  122. package/ai/token/stats.ts +21 -0
  123. package/ai/token/tokenThunks.ts +24 -0
  124. package/ai/token/types.ts +93 -0
  125. package/ai/tools/agent/agentTools.ts +176 -0
  126. package/ai/tools/agent/agentUpdateShared.ts +311 -0
  127. package/ai/tools/agent/callAgentTool.ts +139 -0
  128. package/ai/tools/agent/createAgentTool.ts +512 -0
  129. package/ai/tools/agent/createDialogTool.ts +69 -0
  130. package/ai/tools/agent/createSkillAgentTool.ts +62 -0
  131. package/ai/tools/agent/parallelBudget.ts +221 -0
  132. package/ai/tools/agent/presets/appBuilderPreset.ts +145 -0
  133. package/ai/tools/agent/runLlmTool.ts +96 -0
  134. package/ai/tools/agent/runStreamingAgentTool.ts +73 -0
  135. package/ai/tools/agent/skillAgentArgs.ts +106 -0
  136. package/ai/tools/agent/skillAgentPreset.ts +89 -0
  137. package/ai/tools/agent/streamParallelAgentsTool.ts +122 -0
  138. package/ai/tools/agent/updateAgentTool.ts +96 -0
  139. package/ai/tools/agent/updateSelfTool.ts +113 -0
  140. package/ai/tools/amazonProductScraperTool.ts +86 -0
  141. package/ai/tools/apifyActorClient.ts +45 -0
  142. package/ai/tools/appEditGuard.ts +372 -0
  143. package/ai/tools/appReadSnapshot.ts +153 -0
  144. package/ai/tools/appTools.ts +1549 -0
  145. package/ai/tools/applyEditTool.ts +256 -0
  146. package/ai/tools/applyLineEditsTool.ts +312 -0
  147. package/ai/tools/browserTools/click.ts +33 -0
  148. package/ai/tools/browserTools/closeSession.ts +29 -0
  149. package/ai/tools/browserTools/common.ts +27 -0
  150. package/ai/tools/browserTools/openSession.ts +48 -0
  151. package/ai/tools/browserTools/readContent.ts +38 -0
  152. package/ai/tools/browserTools/selectOption.ts +46 -0
  153. package/ai/tools/browserTools/typeText.ts +42 -0
  154. package/ai/tools/category/createCategoryTool.ts +66 -0
  155. package/ai/tools/category/queryContentsByCategoryTool.ts +69 -0
  156. package/ai/tools/category/updateContentCategoryTool.ts +75 -0
  157. package/ai/tools/cfBrowserTools.ts +319 -0
  158. package/ai/tools/cfSpeechToTextTool.ts +49 -0
  159. package/ai/tools/checkEnvTool.ts +65 -0
  160. package/ai/tools/cloudflareCrawlTool.ts +289 -0
  161. package/ai/tools/codeSearchTool.ts +111 -0
  162. package/ai/tools/codeTools.ts +101 -0
  163. package/ai/tools/createDocTool.ts +132 -0
  164. package/ai/tools/createPlanTool.ts +999 -0
  165. package/ai/tools/createSkillDocTool.ts +155 -0
  166. package/ai/tools/createWorkflowTool.ts +154 -0
  167. package/ai/tools/deepseekOcrTool.ts +34 -0
  168. package/ai/tools/delayTool.ts +31 -0
  169. package/ai/tools/deleteSpacesTool.ts +325 -0
  170. package/ai/tools/deleteSpacesToolModel.ts +159 -0
  171. package/ai/tools/devReloadUtils.ts +29 -0
  172. package/ai/tools/dialogMessageSearch.ts +137 -0
  173. package/ai/tools/doctorSkillTool.ts +72 -0
  174. package/ai/tools/ecommerceScraperTool.ts +86 -0
  175. package/ai/tools/emailTools.ts +549 -0
  176. package/ai/tools/evalSkillTool.ts +92 -0
  177. package/ai/tools/exaSearchTool.ts +64 -0
  178. package/ai/tools/execBashTool.ts +379 -0
  179. package/ai/tools/executeSqlTool.ts +192 -0
  180. package/ai/tools/fetchWebpageSupport.ts +309 -0
  181. package/ai/tools/fetchWebpageTool.ts +84 -0
  182. package/ai/tools/geminiImagePreviewTool.ts +361 -0
  183. package/ai/tools/generateDocxTool.ts +215 -0
  184. package/ai/tools/googleSearchScraperTool.ts +106 -0
  185. package/ai/tools/importDataTool.ts +133 -0
  186. package/ai/tools/importSkillTool.ts +162 -0
  187. package/ai/tools/index.ts +1858 -0
  188. package/ai/tools/listFilesTool.ts +82 -0
  189. package/ai/tools/listUserSpacesTool.ts +113 -0
  190. package/ai/tools/modelUsageTools.ts +142 -0
  191. package/ai/tools/olmOcrTool.ts +34 -0
  192. package/ai/tools/openaiImageTool.ts +218 -0
  193. package/ai/tools/paddleOcrTool.ts +34 -0
  194. package/ai/tools/prepareTools.ts +23 -0
  195. package/ai/tools/readDocTool.ts +84 -0
  196. package/ai/tools/readFileTool.ts +211 -0
  197. package/ai/tools/readTool.ts +163 -0
  198. package/ai/tools/readXPostTool.ts +233 -0
  199. package/ai/tools/rememberMemoryTool.ts +84 -0
  200. package/ai/tools/remotionVideoTool.ts +151 -0
  201. package/ai/tools/searchDialogMessagesTool.ts +222 -0
  202. package/ai/tools/searchRepoTool.ts +115 -0
  203. package/ai/tools/searchWorkspaceTool.ts +259 -0
  204. package/ai/tools/skillFollowup.ts +86 -0
  205. package/ai/tools/surfWeatherTool.ts +169 -0
  206. package/ai/tools/table/addTableRowTool.ts +217 -0
  207. package/ai/tools/table/createTableTool.ts +315 -0
  208. package/ai/tools/table/rowTools.ts +366 -0
  209. package/ai/tools/table/schemaTools.ts +244 -0
  210. package/ai/tools/table/shareTableTool.ts +148 -0
  211. package/ai/tools/table/toolShared.ts +129 -0
  212. package/ai/tools/toolApiClient.ts +198 -0
  213. package/ai/tools/toolNameAliases.ts +57 -0
  214. package/ai/tools/toolResultError.ts +42 -0
  215. package/ai/tools/toolRunSlice.ts +303 -0
  216. package/ai/tools/toolSchemaCompatibility.ts +53 -0
  217. package/ai/tools/toolVisibility.ts +4 -0
  218. package/ai/tools/types.ts +20 -0
  219. package/ai/tools/uiAskChoiceTool.ts +104 -0
  220. package/ai/tools/updateContentTitleTool.ts +84 -0
  221. package/ai/tools/updateDocTool.ts +105 -0
  222. package/ai/tools/updateUserPreferenceProfileTool.ts +145 -0
  223. package/ai/tools/whisperTool.ts +77 -0
  224. package/ai/tools/writeFileTool.ts +210 -0
  225. package/ai/tools/youtubeScraperTool.ts +116 -0
  226. package/ai/tools/ziweiChartTool.ts +678 -0
  227. package/ai/types.ts +55 -0
  228. package/ai/workflow/workflowExecutor.ts +323 -0
  229. package/ai/workflow/workflowSlice.ts +73 -0
  230. package/ai/workflow/workflowTypes.ts +106 -0
  231. package/client/agentRun.ts +198 -167
  232. package/client/compactDialog.ts +222 -0
  233. package/commandRegistry.ts +14 -0
  234. package/connector-experimental/capabilities.ts +73 -0
  235. package/connector-experimental/codexBinary.ts +41 -0
  236. package/connector-experimental/heartbeatLoop.ts +22 -0
  237. package/connector-experimental/index.ts +5 -0
  238. package/connector-experimental/machineInfo.ts +46 -0
  239. package/connector-experimental/protocol.ts +54 -0
  240. package/connectorWebSocketTarget.ts +29 -0
  241. package/defaultServer.ts +1 -0
  242. package/index.ts +158 -104
  243. package/machineCommands.ts +382 -0
  244. package/package.json +12 -2
  245. package/tui/readlineWorkspace.ts +50 -0
  246. package/tui/session.ts +40 -2
  247. package/updateCommands.ts +70 -5
@@ -0,0 +1,678 @@
1
+ /**
2
+ * ziweiChartTool — 紫微斗数排盘工具
3
+ *
4
+ * 改进点:
5
+ * 1. displayData 从4行扩展为完整的十二宫全信息文本(所有星曜、大限、小限)
6
+ * 2. gridText:ASCII 四行四列宫位排布,一眼看清空间关系
7
+ * 3. analysisContext:AI 解盘所需的关键结构化数据
8
+ * - 四化落宫、空宫列表、重要星曜位置
9
+ * 4. 星曜亮度符号(旺▲ 庙○ 陷▼ 平— 利+ 闲·)标注
10
+ * 5. 四化标记(禄权科忌)直接嵌入星名
11
+ */
12
+
13
+ import { astro } from "iztro";
14
+
15
+ // ─── Types ──────────────────────────────────────────────────────────────────
16
+
17
+ export type ZiweiCalendarType = "solar" | "lunar";
18
+ export type ZiweiGender = "男" | "女" | "male" | "female";
19
+ export type ZiweiLanguage =
20
+ | "zh-CN"
21
+ | "zh-TW"
22
+ | "en-US"
23
+ | "ja-JP"
24
+ | "ko-KR"
25
+ | "vi-VN";
26
+
27
+ export interface ZiweiChartToolArgs {
28
+ dateStr: string;
29
+ timeIndex: number;
30
+ gender: ZiweiGender;
31
+ calendarType?: ZiweiCalendarType;
32
+ isLeapMonth?: boolean;
33
+ fixLeap?: boolean;
34
+ language?: ZiweiLanguage;
35
+ }
36
+
37
+ interface StarSummary {
38
+ name: string;
39
+ brightness?: string;
40
+ mutagen?: string;
41
+ type: string;
42
+ }
43
+
44
+ interface NamedStarSummary {
45
+ name: string;
46
+ mutagen?: string;
47
+ }
48
+
49
+ interface PalaceSummary {
50
+ index: number;
51
+ name: string;
52
+ heavenlyStem: string;
53
+ earthlyBranch: string;
54
+ isBodyPalace: boolean;
55
+ isOriginalPalace: boolean;
56
+ majorStars: StarSummary[];
57
+ minorStars: string[];
58
+ adjectiveStars: string[];
59
+ changsheng12: string;
60
+ boshi12: string;
61
+ jiangqian12: string;
62
+ suiqian12: string;
63
+ decadal: {
64
+ range: [number, number];
65
+ heavenlyStem: string;
66
+ earthlyBranch: string;
67
+ };
68
+ ages: number[];
69
+ }
70
+
71
+ interface FourTransformations {
72
+ huaLu?: { star: string; palace: string };
73
+ huaQuan?: { star: string; palace: string };
74
+ huaKe?: { star: string; palace: string };
75
+ huaJi?: { star: string; palace: string };
76
+ }
77
+
78
+ interface AnalysisContext {
79
+ /** 命宫详情 */
80
+ mingPalace: {
81
+ name: string;
82
+ earthlyBranch: string;
83
+ heavenlyStem: string;
84
+ majorStars: StarSummary[];
85
+ minorStars: string[];
86
+ decadalRange: [number, number];
87
+ };
88
+ /** 身宫详情 */
89
+ bodyPalace: {
90
+ name: string;
91
+ earthlyBranch: string;
92
+ majorStars: StarSummary[];
93
+ };
94
+ /** 生年四化落宫 */
95
+ fourTransformations: FourTransformations;
96
+ /** 空宫(无主星)宫位列表 */
97
+ emptyPalaces: string[];
98
+ /** 主星所在宫位索引 starName → palaceName */
99
+ majorStarLocations: Record<string, string>;
100
+ /** 大限当前建议:按年龄找当前大限宫 */
101
+ decadalPalaces: Array<{
102
+ palaceName: string;
103
+ earthlyBranch: string;
104
+ range: [number, number];
105
+ majorStars: string[];
106
+ }>;
107
+ }
108
+
109
+ export interface ZiweiChartToolResult {
110
+ success: true;
111
+ input: {
112
+ calendarType: ZiweiCalendarType;
113
+ dateStr: string;
114
+ timeIndex: number;
115
+ gender: string;
116
+ isLeapMonth: boolean;
117
+ fixLeap: boolean;
118
+ language: ZiweiLanguage;
119
+ };
120
+ chart: {
121
+ solarDate: string;
122
+ lunarDate: string;
123
+ chineseDate: string;
124
+ heavenlyStemOfYear: string;
125
+ time: string;
126
+ timeRange: string;
127
+ sign: string;
128
+ zodiac: string;
129
+ earthlyBranchOfSoulPalace: string;
130
+ earthlyBranchOfBodyPalace: string;
131
+ soul: string;
132
+ body: string;
133
+ fiveElementsClass: string;
134
+ mutagenByYear: NamedStarSummary[];
135
+ palaces: PalaceSummary[];
136
+ };
137
+ summary: {
138
+ palaceCount: number;
139
+ mingGong: string;
140
+ shenGong: string;
141
+ mingZhu: string;
142
+ shenZhu: string;
143
+ fiveElementsClass: string;
144
+ };
145
+ summaryText: string;
146
+ /** 完整宫位信息文本(所有12宫全量展开,适合AI逐宫分析) */
147
+ displayData: string;
148
+ /** 四行四列 ASCII 宫位排布图 */
149
+ gridText: string;
150
+ /** AI 解盘关键结构化数据 */
151
+ analysisContext: AnalysisContext;
152
+ }
153
+
154
+ // ─── Constants ───────────────────────────────────────────────────────────────
155
+
156
+ const TIME_LABELS = [
157
+ "早子时(00:00~01:00)",
158
+ "丑时(01:00~03:00)",
159
+ "寅时(03:00~05:00)",
160
+ "卯时(05:00~07:00)",
161
+ "辰时(07:00~09:00)",
162
+ "巳时(09:00~11:00)",
163
+ "午时(11:00~13:00)",
164
+ "未时(13:00~15:00)",
165
+ "申时(15:00~17:00)",
166
+ "酉时(17:00~19:00)",
167
+ "戌时(19:00~21:00)",
168
+ "亥时(21:00~23:00)",
169
+ "晚子时(23:00~24:00)",
170
+ ] as const;
171
+
172
+ /**
173
+ * 地支顺序 → 宫位网格位置映射
174
+ * 传统紫微斗数盘面:
175
+ * 巳 午 未 申 (top row, left→right)
176
+ * 辰 酉 (second row, outer only)
177
+ * 卯 戌 (third row, outer only)
178
+ * 寅 丑 子 亥 (bottom row, left→right)
179
+ */
180
+ const BRANCH_TO_GRID: Record<string, [number, number]> = {
181
+ 巳: [0, 0],
182
+ 午: [0, 1],
183
+ 未: [0, 2],
184
+ 申: [0, 3],
185
+ 酉: [1, 3],
186
+ 戌: [2, 3],
187
+ 亥: [3, 3],
188
+ 子: [3, 2],
189
+ 丑: [3, 1],
190
+ 寅: [3, 0],
191
+ 卯: [2, 0],
192
+ 辰: [1, 0],
193
+ };
194
+
195
+ const BRIGHTNESS_SYMBOL: Record<string, string> = {
196
+ 旺: "旺▲",
197
+ 庙: "庙○",
198
+ 陷: "陷▼",
199
+ 平: "平—",
200
+ 利: "利+",
201
+ 闲: "闲·",
202
+ 得: "得✦",
203
+ 不: "不✧",
204
+ };
205
+
206
+ const MUTAGEN_SYMBOL: Record<string, string> = {
207
+ 禄: "化禄",
208
+ 权: "化权",
209
+ 科: "化科",
210
+ 忌: "化忌",
211
+ };
212
+
213
+ const PALACE_ALIAS: Record<string, string> = {
214
+ 仆役: "交友",
215
+ 官禄: "事业",
216
+ };
217
+
218
+ // ─── Normalization helpers ───────────────────────────────────────────────────
219
+
220
+ function normalizeGender(input: string): ZiweiGender {
221
+ const value = input.trim().toLowerCase();
222
+ if (value === "男" || value === "male") return value as ZiweiGender;
223
+ if (value === "女" || value === "female") return value as ZiweiGender;
224
+ throw new Error('gender 只支持 "男"、"女"、"male"、"female"。');
225
+ }
226
+
227
+ function normalizeCalendarType(input: unknown): ZiweiCalendarType {
228
+ return input === "lunar" ? "lunar" : "solar";
229
+ }
230
+
231
+ function normalizeLanguage(input: unknown): ZiweiLanguage {
232
+ const value = typeof input === "string" ? input : "zh-CN";
233
+ const supported = new Set<ZiweiLanguage>([
234
+ "zh-CN",
235
+ "zh-TW",
236
+ "en-US",
237
+ "ja-JP",
238
+ "ko-KR",
239
+ "vi-VN",
240
+ ]);
241
+ if (!supported.has(value as ZiweiLanguage)) {
242
+ throw new Error(
243
+ 'language 只支持 "zh-CN"、"zh-TW"、"en-US"、"ja-JP"、"ko-KR"、"vi-VN"。'
244
+ );
245
+ }
246
+ return value as ZiweiLanguage;
247
+ }
248
+
249
+ function normalizeTimeIndex(input: unknown): number {
250
+ const value = typeof input === "number" ? Math.trunc(input) : Number(input);
251
+ if (!Number.isInteger(value) || value < 0 || value > 12) {
252
+ throw new Error("timeIndex 必须是 0 到 12 的整数。");
253
+ }
254
+ return value;
255
+ }
256
+
257
+ function normalizeDateStr(input: unknown): string {
258
+ const value = typeof input === "string" ? input.trim() : "";
259
+ if (!/^\d{4}-\d{1,2}-\d{1,2}$/.test(value)) {
260
+ throw new Error('dateStr 必须是 "YYYY-M-D" 或 "YYYY-MM-DD" 格式。');
261
+ }
262
+ return value;
263
+ }
264
+
265
+ // ─── Formatting helpers ───────────────────────────────────────────────────────
266
+
267
+ function formatStarWithSymbols(star: StarSummary): string {
268
+ const brightStr = star.brightness
269
+ ? (BRIGHTNESS_SYMBOL[star.brightness] ?? star.brightness)
270
+ : "";
271
+ const mutagenStr = star.mutagen
272
+ ? (MUTAGEN_SYMBOL[star.mutagen] ?? `化${star.mutagen}`)
273
+ : "";
274
+ const tags = [brightStr, mutagenStr].filter(Boolean);
275
+ return tags.length > 0 ? `${star.name}[${tags.join(" ")}]` : star.name;
276
+ }
277
+
278
+ function dedupeMutagens(stars: StarSummary[]): NamedStarSummary[] {
279
+ const seen = new Set<string>();
280
+ return stars
281
+ .filter((s) => !!s.mutagen)
282
+ .map((s) => ({ name: s.name, mutagen: s.mutagen }))
283
+ .filter((s) => {
284
+ const key = `${s.mutagen}:${s.name}`;
285
+ if (seen.has(key)) return false;
286
+ seen.add(key);
287
+ return true;
288
+ });
289
+ }
290
+
291
+ // ─── Display data builder (full 12-palace breakdown) ─────────────────────────
292
+
293
+ function buildFullDisplayData(
294
+ palaces: PalaceSummary[],
295
+ chart: ZiweiChartToolResult["chart"],
296
+ input: ZiweiChartToolResult["input"]
297
+ ): string {
298
+ const divider = "─".repeat(60);
299
+ const header = [
300
+ `◆ 紫微斗数命盘 ◆`,
301
+ `阳历:${chart.solarDate} 农历:${chart.lunarDate}`,
302
+ `干支:${chart.chineseDate}`,
303
+ `五行局:${chart.fiveElementsClass} 命主:${chart.soul} 身主:${chart.body}`,
304
+ `生年四化:${
305
+ chart.mutagenByYear.length > 0
306
+ ? chart.mutagenByYear
307
+ .map((m) => `${m.name}${MUTAGEN_SYMBOL[m.mutagen ?? ""] ?? `化${m.mutagen}`}`)
308
+ .join("、")
309
+ : "无"
310
+ }`,
311
+ `性别:${input.gender} 时辰:${TIME_LABELS[input.timeIndex] ?? input.timeIndex}`,
312
+ ].join("\n");
313
+
314
+ const palaceLines = palaces.map((p) => {
315
+ const flags: string[] = [];
316
+ if (p.isBodyPalace) flags.push("身宫");
317
+ if (p.isOriginalPalace) flags.push("来因宫");
318
+ const flagStr = flags.length > 0 ? ` [${flags.join(" ")}]` : "";
319
+
320
+ const majorStr =
321
+ p.majorStars.length > 0
322
+ ? p.majorStars.map(formatStarWithSymbols).join(" ")
323
+ : "(空宫)";
324
+ const minorStr = p.minorStars.length > 0 ? p.minorStars.join("、") : "无";
325
+ const adjStr =
326
+ p.adjectiveStars.length > 0 ? p.adjectiveStars.join("、") : "无";
327
+
328
+ const displayPalaceName = PALACE_ALIAS[p.name] ?? p.name;
329
+ const palateName = displayPalaceName.endsWith("宫")
330
+ ? displayPalaceName
331
+ : `${displayPalaceName}宫`;
332
+ return [
333
+ `【${p.heavenlyStem}${p.earthlyBranch} ${palateName}${flagStr}】`,
334
+ ` 主星:${majorStr}`,
335
+ ` 辅星:${minorStr}`,
336
+ ` 杂耀:${adjStr}`,
337
+ ` 长生十二神:${p.changsheng12} 博士十二神:${p.boshi12}`,
338
+ ` 将前十二神:${p.jiangqian12} 岁前十二神:${p.suiqian12}`,
339
+ ` 大限:${p.decadal.range[0]}-${p.decadal.range[1]}岁(${p.decadal.heavenlyStem}${p.decadal.earthlyBranch})`,
340
+ ` 小限:${p.ages.join(" ")}`,
341
+ ].join("\n");
342
+ });
343
+
344
+ return [header, divider, ...palaceLines].join("\n" + divider + "\n");
345
+ }
346
+
347
+ // ─── Grid text builder ───────────────────────────────────────────────────────
348
+
349
+ /**
350
+ * 生成 ASCII 四行四列宫位图,中央2×2格显示基本信息。
351
+ * 每格宽约 20 字符,展示:地支+宫名、主星、大限范围。
352
+ */
353
+ function buildGridText(
354
+ palaces: PalaceSummary[],
355
+ chart: ZiweiChartToolResult["chart"]
356
+ ): string {
357
+ const branchToPalace: Record<string, PalaceSummary> = {};
358
+ for (const p of palaces) {
359
+ branchToPalace[p.earthlyBranch] = p;
360
+ }
361
+
362
+ const CELL_W = 22;
363
+ const pad = (s: string) => s.slice(0, CELL_W).padEnd(CELL_W);
364
+
365
+ function cellLines(branch: string): string[] {
366
+ const p = branchToPalace[branch];
367
+ if (!p) return ["", "", "", ""];
368
+ const displayPalaceName = PALACE_ALIAS[p.name] ?? p.name;
369
+ const flags = [p.isBodyPalace ? "身" : "", p.isOriginalPalace ? "来" : ""]
370
+ .filter(Boolean)
371
+ .join("");
372
+ const flagStr = flags ? `[${flags}]` : "";
373
+ const line1 = `${p.heavenlyStem}${branch} ${displayPalaceName}宫${flagStr}`;
374
+ const majorNames =
375
+ p.majorStars.length > 0
376
+ ? p.majorStars.map((s) => {
377
+ const b = s.brightness ? (BRIGHTNESS_SYMBOL[s.brightness]?.slice(0, 1) ?? "") : "";
378
+ const m = s.mutagen ? `化${s.mutagen}` : "";
379
+ return `${s.name}${b}${m}`;
380
+ })
381
+ : ["空宫"];
382
+ const line2 = majorNames.join(" ");
383
+ const line3 = `大限${p.decadal.range[0]}-${p.decadal.range[1]}`;
384
+ const line4 = `小限 ${p.ages.slice(0, 4).join(" ")}`;
385
+ return [line1, line2, line3, line4];
386
+ }
387
+
388
+ const centerInfo = [
389
+ `阳历 ${chart.solarDate}`,
390
+ `农历 ${chart.lunarDate}`,
391
+ `${chart.fiveElementsClass}`,
392
+ `命主:${chart.soul} 身主:${chart.body}`,
393
+ `四化:${chart.mutagenByYear
394
+ .map((m) => `${m.name}化${m.mutagen}`)
395
+ .join(" ")}`,
396
+ ``,
397
+ ].map((s) => s.slice(0, CELL_W * 2).padEnd(CELL_W * 2));
398
+
399
+ const branchRows: [string, string, string, string][] = [
400
+ ["巳", "午", "未", "申"],
401
+ ["辰", "", "", "酉"],
402
+ ["卯", "", "", "戌"],
403
+ ["寅", "丑", "子", "亥"],
404
+ ];
405
+
406
+ const sep = "+" + (["-".repeat(CELL_W), "-".repeat(CELL_W), "-".repeat(CELL_W), "-".repeat(CELL_W)].join("+")) + "+";
407
+
408
+ const lines: string[] = [sep];
409
+
410
+ for (let row = 0; row < 4; row++) {
411
+ const rowBranches = branchRows[row];
412
+ // build 4 cell content (4 lines each)
413
+ const cellData: string[][] = rowBranches.map((b) =>
414
+ b ? cellLines(b) : ["", "", "", ""]
415
+ );
416
+
417
+ for (let lineIdx = 0; lineIdx < 4; lineIdx++) {
418
+ // rows 1-2, cols 1-2 are center
419
+ const parts: string[] = [];
420
+ for (let col = 0; col < 4; col++) {
421
+ const isCenter = (row === 1 || row === 2) && (col === 1 || col === 2);
422
+ if (isCenter) {
423
+ // center cells: span handled separately, skip col 2 merging
424
+ if (col === 1) {
425
+ // render merged center 2-col wide cell
426
+ const centerLineIdx = (row - 1) * 4 + lineIdx;
427
+ parts.push(centerInfo[centerLineIdx] ?? " ".repeat(CELL_W * 2));
428
+ }
429
+ // col 2 is absorbed into the merged center; skip rendering a separator
430
+ } else {
431
+ parts.push(pad(cellData[col][lineIdx] ?? ""));
432
+ }
433
+ }
434
+ // build line: |col0|center(merged)|col3|
435
+ if ((row === 1 || row === 2)) {
436
+ lines.push(`|${pad(cellData[0][lineIdx])}|${centerInfo[(row - 1) * 4 + lineIdx] ?? " ".repeat(CELL_W * 2)}|${pad(cellData[3][lineIdx])}|`);
437
+ } else {
438
+ lines.push(`|${cellData.map((c) => pad(c[lineIdx] ?? "")).join("|")}|`);
439
+ }
440
+ }
441
+ lines.push(sep);
442
+ }
443
+
444
+ return lines.join("\n");
445
+ }
446
+
447
+ // ─── Analysis context builder ─────────────────────────────────────────────────
448
+
449
+ function buildAnalysisContext(
450
+ palaces: PalaceSummary[],
451
+ chart: ZiweiChartToolResult["chart"]
452
+ ): AnalysisContext {
453
+ const byBranch = new Map<string, PalaceSummary>();
454
+ for (const p of palaces) byBranch.set(p.earthlyBranch, p);
455
+
456
+ const mingPalaceData =
457
+ byBranch.get(chart.earthlyBranchOfSoulPalace) ??
458
+ palaces.find((p) => p.name === "命宫")!;
459
+ const bodyPalaceData =
460
+ byBranch.get(chart.earthlyBranchOfBodyPalace) ??
461
+ palaces.find((p) => p.isBodyPalace)!;
462
+
463
+ // 四化落宫
464
+ const fourTransformations: FourTransformations = {};
465
+ for (const p of palaces) {
466
+ for (const s of p.majorStars) {
467
+ if (!s.mutagen) continue;
468
+ const entry = { star: s.name, palace: `${p.name}宫` };
469
+ if (s.mutagen === "禄") fourTransformations.huaLu = entry;
470
+ else if (s.mutagen === "权") fourTransformations.huaQuan = entry;
471
+ else if (s.mutagen === "科") fourTransformations.huaKe = entry;
472
+ else if (s.mutagen === "忌") fourTransformations.huaJi = entry;
473
+ }
474
+ }
475
+
476
+ // 空宫
477
+ const emptyPalaces = palaces
478
+ .filter((p) => p.majorStars.length === 0)
479
+ .map((p) => `${p.name}宫(${p.heavenlyStem}${p.earthlyBranch})`);
480
+
481
+ // 主星位置
482
+ const majorStarLocations: Record<string, string> = {};
483
+ for (const p of palaces) {
484
+ for (const s of p.majorStars) {
485
+ majorStarLocations[s.name] = `${p.name}宫`;
486
+ }
487
+ }
488
+
489
+ // 大限宫列表(按年龄排序)
490
+ const decadalPalaces = [...palaces]
491
+ .sort((a, b) => a.decadal.range[0] - b.decadal.range[0])
492
+ .map((p) => ({
493
+ palaceName: p.name,
494
+ earthlyBranch: p.earthlyBranch,
495
+ range: p.decadal.range,
496
+ majorStars: p.majorStars.map((s) => s.name),
497
+ }));
498
+
499
+ return {
500
+ mingPalace: {
501
+ name: mingPalaceData?.name ?? "",
502
+ earthlyBranch: chart.earthlyBranchOfSoulPalace,
503
+ heavenlyStem: mingPalaceData?.heavenlyStem ?? "",
504
+ majorStars: mingPalaceData?.majorStars ?? [],
505
+ minorStars: mingPalaceData?.minorStars ?? [],
506
+ decadalRange: mingPalaceData?.decadal.range ?? [0, 0],
507
+ },
508
+ bodyPalace: {
509
+ name: bodyPalaceData?.name ?? "",
510
+ earthlyBranch: chart.earthlyBranchOfBodyPalace,
511
+ majorStars: bodyPalaceData?.majorStars ?? [],
512
+ },
513
+ fourTransformations,
514
+ emptyPalaces,
515
+ majorStarLocations,
516
+ decadalPalaces,
517
+ };
518
+ }
519
+
520
+ // ─── Main function ────────────────────────────────────────────────────────────
521
+
522
+ export function runZiweiChart(input: ZiweiChartToolArgs): ZiweiChartToolResult {
523
+ const calendarType = normalizeCalendarType(input.calendarType);
524
+ const language = normalizeLanguage(input.language);
525
+ const dateStr = normalizeDateStr(input.dateStr);
526
+ const timeIndex = normalizeTimeIndex(input.timeIndex);
527
+ const gender = normalizeGender(input.gender);
528
+ const fixLeap = input.fixLeap !== false;
529
+ const isLeapMonth = input.isLeapMonth === true;
530
+
531
+ const rawChart =
532
+ calendarType === "lunar"
533
+ ? astro.byLunar(dateStr, timeIndex, gender, isLeapMonth, fixLeap, language)
534
+ : astro.bySolar(dateStr, timeIndex, gender, fixLeap, language);
535
+
536
+ const palaces: PalaceSummary[] = rawChart.palaces.map((palace) => ({
537
+ index: palace.index,
538
+ name: palace.name,
539
+ heavenlyStem: palace.heavenlyStem,
540
+ earthlyBranch: palace.earthlyBranch,
541
+ isBodyPalace: palace.isBodyPalace,
542
+ isOriginalPalace: palace.isOriginalPalace,
543
+ majorStars: palace.majorStars.map((star) => ({
544
+ name: star.name,
545
+ brightness: star.brightness || undefined,
546
+ mutagen: star.mutagen || undefined,
547
+ type: star.type,
548
+ })),
549
+ minorStars: palace.minorStars.map((star) => star.name),
550
+ adjectiveStars: palace.adjectiveStars.map((star) => star.name),
551
+ changsheng12: palace.changsheng12,
552
+ boshi12: palace.boshi12,
553
+ jiangqian12: palace.jiangqian12,
554
+ suiqian12: palace.suiqian12,
555
+ decadal: palace.decadal,
556
+ ages: palace.ages,
557
+ }));
558
+
559
+ const mutagenByYear = dedupeMutagens(palaces.flatMap((p) => p.majorStars));
560
+ const mingGong = `${rawChart.earthlyBranchOfSoulPalace}宫`;
561
+ const shenGong = `${rawChart.earthlyBranchOfBodyPalace}宫`;
562
+ const heavenlyStemOfYear =
563
+ rawChart.chineseDate.split(" ")[0]?.slice(0, 1) ?? "";
564
+
565
+ const chartData: ZiweiChartToolResult["chart"] = {
566
+ solarDate: rawChart.solarDate,
567
+ lunarDate: rawChart.lunarDate,
568
+ chineseDate: rawChart.chineseDate,
569
+ heavenlyStemOfYear,
570
+ time: rawChart.time,
571
+ timeRange: rawChart.timeRange,
572
+ sign: rawChart.sign,
573
+ zodiac: rawChart.zodiac,
574
+ earthlyBranchOfSoulPalace: rawChart.earthlyBranchOfSoulPalace,
575
+ earthlyBranchOfBodyPalace: rawChart.earthlyBranchOfBodyPalace,
576
+ soul: rawChart.soul,
577
+ body: rawChart.body,
578
+ fiveElementsClass: rawChart.fiveElementsClass,
579
+ mutagenByYear,
580
+ palaces,
581
+ };
582
+
583
+ const inputNorm: ZiweiChartToolResult["input"] = {
584
+ calendarType,
585
+ dateStr,
586
+ timeIndex,
587
+ gender,
588
+ isLeapMonth,
589
+ fixLeap,
590
+ language,
591
+ };
592
+
593
+ const displayData = buildFullDisplayData(palaces, chartData, inputNorm);
594
+ const gridText = buildGridText(palaces, chartData);
595
+ const analysisContext = buildAnalysisContext(palaces, chartData);
596
+ const summaryText = `命宫 ${mingGong}|身宫 ${shenGong}|命主 ${rawChart.soul}|身主 ${rawChart.body}|五行局 ${rawChart.fiveElementsClass}`;
597
+
598
+ return {
599
+ success: true,
600
+ input: inputNorm,
601
+ chart: chartData,
602
+ summary: {
603
+ palaceCount: palaces.length,
604
+ mingGong,
605
+ shenGong,
606
+ mingZhu: rawChart.soul,
607
+ shenZhu: rawChart.body,
608
+ fiveElementsClass: rawChart.fiveElementsClass,
609
+ },
610
+ summaryText,
611
+ displayData,
612
+ gridText,
613
+ analysisContext,
614
+ };
615
+ }
616
+
617
+ // ─── Function schema ──────────────────────────────────────────────────────────
618
+
619
+ export const ziweiChartFunctionSchema = {
620
+ name: "ziweiChart",
621
+ description: [
622
+ "根据出生日期、时辰和性别生成紫微斗数完整排盘。",
623
+ "输出包含:",
624
+ " 1. chart:结构化十二宫完整数据(主星亮度/四化/大限/小限)",
625
+ " 2. displayData:十二宫全展开文本,每宫含主星/辅星/杂耀/长生/博士/将前/岁前",
626
+ " 3. gridText:ASCII 四行四列宫位排布图,直观显示空间关系",
627
+ " 4. analysisContext:AI 解盘关键数据(四化落宫、空宫、主星位置、大限序列)",
628
+ "timeIndex 0~12:0=早子,1=丑,2=寅,3=卯,4=辰,5=巳,6=午,7=未,8=申,9=酉,10=戌,11=亥,12=晚子。",
629
+ ].join("\n"),
630
+ parameters: {
631
+ type: "object",
632
+ properties: {
633
+ dateStr: {
634
+ type: "string",
635
+ description: '出生日期,"YYYY-M-D" 或 "YYYY-MM-DD"。例如 "1983-11-7"。',
636
+ },
637
+ timeIndex: {
638
+ type: "number",
639
+ description:
640
+ "出生时辰序号 0~12。0=早子,1=丑,2=寅,3=卯,4=辰,5=巳,6=午,7=未,8=申,9=酉,10=戌,11=亥,12=晚子。",
641
+ },
642
+ gender: {
643
+ type: "string",
644
+ description: '性别:"男"、"女"、"male"、"female"。',
645
+ },
646
+ calendarType: {
647
+ type: "string",
648
+ enum: ["solar", "lunar"],
649
+ description: '日期类型,默认 "solar"(阳历)。"lunar" 为农历。',
650
+ },
651
+ isLeapMonth: {
652
+ type: "boolean",
653
+ description: '农历闰月时设为 true,默认 false。',
654
+ },
655
+ fixLeap: {
656
+ type: "boolean",
657
+ description: "闰月修正,默认 true。",
658
+ },
659
+ language: {
660
+ type: "string",
661
+ enum: ["zh-CN", "zh-TW", "en-US", "ja-JP", "ko-KR", "vi-VN"],
662
+ description: '输出语言,默认 "zh-CN"。',
663
+ },
664
+ },
665
+ required: ["dateStr", "timeIndex", "gender"],
666
+ },
667
+ } as const;
668
+
669
+ export async function ziweiChartFunc(
670
+ args: ZiweiChartToolArgs
671
+ ): Promise<{ rawData: ZiweiChartToolResult; displayData: string; gridText: string }> {
672
+ const result = runZiweiChart(args);
673
+ return {
674
+ rawData: result,
675
+ displayData: result.displayData,
676
+ gridText: result.gridText,
677
+ };
678
+ }