@within-7/minto 0.3.6 → 0.3.10

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (238) hide show
  1. package/{cli.js → cli.cjs} +25 -23
  2. package/dist/commands/agents/AgentsCommand.js +459 -655
  3. package/dist/commands/agents/AgentsCommand.js.map +2 -2
  4. package/dist/commands/agents/types.js +1 -0
  5. package/dist/commands/agents/types.js.map +2 -2
  6. package/dist/commands/agents/utils/fileOperations.js +96 -36
  7. package/dist/commands/agents/utils/fileOperations.js.map +3 -3
  8. package/dist/commands/agents/utils/index.js +3 -1
  9. package/dist/commands/agents/utils/index.js.map +2 -2
  10. package/dist/commands/context.js +54 -23
  11. package/dist/commands/context.js.map +2 -2
  12. package/dist/commands/export.js +673 -93
  13. package/dist/commands/export.js.map +2 -2
  14. package/dist/commands/language.js +110 -0
  15. package/dist/commands/language.js.map +7 -0
  16. package/dist/commands/mcp-interactive.js +419 -217
  17. package/dist/commands/mcp-interactive.js.map +2 -2
  18. package/dist/commands/model.js +415 -66
  19. package/dist/commands/model.js.map +2 -2
  20. package/dist/commands/new.js +56 -0
  21. package/dist/commands/new.js.map +7 -0
  22. package/dist/commands/permissions.js +75 -49
  23. package/dist/commands/permissions.js.map +2 -2
  24. package/dist/commands/plugin.js +882 -185
  25. package/dist/commands/plugin.js.map +3 -3
  26. package/dist/commands/resume.js +251 -16
  27. package/dist/commands/resume.js.map +2 -2
  28. package/dist/commands/sandbox.js +168 -70
  29. package/dist/commands/sandbox.js.map +2 -2
  30. package/dist/commands/sessions.js +224 -0
  31. package/dist/commands/sessions.js.map +7 -0
  32. package/dist/commands/setup.js +596 -109
  33. package/dist/commands/setup.js.map +2 -2
  34. package/dist/commands/stats.js +292 -0
  35. package/dist/commands/stats.js.map +7 -0
  36. package/dist/commands/status.js +75 -7
  37. package/dist/commands/status.js.map +2 -2
  38. package/dist/commands/undo.js +154 -180
  39. package/dist/commands/undo.js.map +2 -2
  40. package/dist/commands.js +6 -0
  41. package/dist/commands.js.map +2 -2
  42. package/dist/components/AskUserQuestionDialog/AskUserQuestionDialog.js +3 -2
  43. package/dist/components/AskUserQuestionDialog/AskUserQuestionDialog.js.map +2 -2
  44. package/dist/components/Config.js +9 -8
  45. package/dist/components/Config.js.map +2 -2
  46. package/dist/components/HeaderBar.js +2 -1
  47. package/dist/components/HeaderBar.js.map +2 -2
  48. package/dist/components/Help.js +166 -32
  49. package/dist/components/Help.js.map +2 -2
  50. package/dist/components/HotkeyHelpPanel.js +46 -44
  51. package/dist/components/HotkeyHelpPanel.js.map +2 -2
  52. package/dist/components/InfoPanel/InfoPanel.js +123 -0
  53. package/dist/components/InfoPanel/InfoPanel.js.map +7 -0
  54. package/dist/components/InfoPanel/index.js +5 -0
  55. package/dist/components/InfoPanel/index.js.map +7 -0
  56. package/dist/components/InfoPanel/types.js +1 -0
  57. package/dist/components/InfoPanel/types.js.map +7 -0
  58. package/dist/components/Logo.js +5 -2
  59. package/dist/components/Logo.js.map +2 -2
  60. package/dist/components/MCPServerApprovalDialog.js +6 -5
  61. package/dist/components/MCPServerApprovalDialog.js.map +2 -2
  62. package/dist/components/MCPServerMultiselectDialog.js +5 -4
  63. package/dist/components/MCPServerMultiselectDialog.js.map +2 -2
  64. package/dist/components/MessageSelector.js +4 -3
  65. package/dist/components/MessageSelector.js.map +2 -2
  66. package/dist/components/ModelConfig.js +13 -12
  67. package/dist/components/ModelConfig.js.map +2 -2
  68. package/dist/components/ModelListManager.js +4 -3
  69. package/dist/components/ModelListManager.js.map +2 -2
  70. package/dist/components/ModelSelector/BrandTextInput.js +43 -0
  71. package/dist/components/ModelSelector/BrandTextInput.js.map +7 -0
  72. package/dist/components/ModelSelector/ModelSelector.js +419 -501
  73. package/dist/components/ModelSelector/ModelSelector.js.map +2 -2
  74. package/dist/components/ModelSelector/WizardContainer.js +45 -0
  75. package/dist/components/ModelSelector/WizardContainer.js.map +7 -0
  76. package/dist/components/ModelSelector/index.js +1 -3
  77. package/dist/components/ModelSelector/index.js.map +2 -2
  78. package/dist/components/PromptInput.js +77 -44
  79. package/dist/components/PromptInput.js.map +2 -2
  80. package/dist/components/SensitiveFileWarning.js +12 -8
  81. package/dist/components/SensitiveFileWarning.js.map +2 -2
  82. package/dist/components/SimpleSelector/SimpleSelector.js +154 -0
  83. package/dist/components/SimpleSelector/SimpleSelector.js.map +7 -0
  84. package/dist/components/SimpleSelector/index.js +5 -0
  85. package/dist/components/SimpleSelector/index.js.map +7 -0
  86. package/dist/components/SimpleSelector/types.js +1 -0
  87. package/dist/components/SimpleSelector/types.js.map +7 -0
  88. package/dist/components/StatusOverlayContent.js +21 -0
  89. package/dist/components/StatusOverlayContent.js.map +7 -0
  90. package/dist/components/TabbedListView/ScrollableList.js +117 -0
  91. package/dist/components/TabbedListView/ScrollableList.js.map +7 -0
  92. package/dist/components/TabbedListView/SearchInput.js +23 -0
  93. package/dist/components/TabbedListView/SearchInput.js.map +7 -0
  94. package/dist/components/TabbedListView/TabBar.js +20 -0
  95. package/dist/components/TabbedListView/TabBar.js.map +7 -0
  96. package/dist/components/TabbedListView/TabbedListView.js +246 -0
  97. package/dist/components/TabbedListView/TabbedListView.js.map +7 -0
  98. package/dist/components/TabbedListView/index.js +11 -0
  99. package/dist/components/TabbedListView/index.js.map +7 -0
  100. package/dist/components/TabbedListView/types.js +1 -0
  101. package/dist/components/TabbedListView/types.js.map +7 -0
  102. package/dist/components/TodoChangeBlock.js +6 -5
  103. package/dist/components/TodoChangeBlock.js.map +3 -3
  104. package/dist/components/TodoPanel.js +6 -3
  105. package/dist/components/TodoPanel.js.map +3 -3
  106. package/dist/components/TrustDialog.js +6 -5
  107. package/dist/components/TrustDialog.js.map +2 -2
  108. package/dist/components/messages/UserToolResultMessage/UserToolCanceledMessage.js +2 -1
  109. package/dist/components/messages/UserToolResultMessage/UserToolCanceledMessage.js.map +2 -2
  110. package/dist/constants/macros.js +1 -1
  111. package/dist/constants/macros.js.map +1 -1
  112. package/dist/constants/product.js +2 -2
  113. package/dist/constants/product.js.map +1 -1
  114. package/dist/constants/prompts.js +17 -0
  115. package/dist/constants/prompts.js.map +2 -2
  116. package/dist/constants/toolInputExamples.js +5 -1
  117. package/dist/constants/toolInputExamples.js.map +2 -2
  118. package/dist/core/backupHook.js +29 -0
  119. package/dist/core/backupHook.js.map +7 -0
  120. package/dist/core/config/defaults.js +8 -2
  121. package/dist/core/config/defaults.js.map +2 -2
  122. package/dist/core/config/schema.js +14 -2
  123. package/dist/core/config/schema.js.map +2 -2
  124. package/dist/core/costTracker.js +0 -16
  125. package/dist/core/costTracker.js.map +2 -2
  126. package/dist/core/tokenStatsManager.js +5 -0
  127. package/dist/core/tokenStatsManager.js.map +2 -2
  128. package/dist/cost-tracker.js +0 -16
  129. package/dist/cost-tracker.js.map +2 -2
  130. package/dist/entrypoints/bootstrap.js +56 -0
  131. package/dist/entrypoints/bootstrap.js.map +7 -0
  132. package/dist/entrypoints/cli.js +164 -23
  133. package/dist/entrypoints/cli.js.map +3 -3
  134. package/dist/history.js +75 -15
  135. package/dist/history.js.map +2 -2
  136. package/dist/i18n/index.js +2 -2
  137. package/dist/i18n/index.js.map +2 -2
  138. package/dist/i18n/locales/en.js +582 -1
  139. package/dist/i18n/locales/en.js.map +2 -2
  140. package/dist/i18n/locales/zh-CN.js +582 -1
  141. package/dist/i18n/locales/zh-CN.js.map +2 -2
  142. package/dist/i18n/types.js.map +1 -1
  143. package/dist/index.js +1 -1
  144. package/dist/index.js.map +2 -2
  145. package/dist/messages.js +11 -0
  146. package/dist/messages.js.map +2 -2
  147. package/dist/permissions.js.map +2 -2
  148. package/dist/query.js +9 -0
  149. package/dist/query.js.map +2 -2
  150. package/dist/screens/REPL.js +45 -7
  151. package/dist/screens/REPL.js.map +2 -2
  152. package/dist/services/customCommands.js +44 -16
  153. package/dist/services/customCommands.js.map +2 -2
  154. package/dist/services/plugins/lspServers.js +1 -1
  155. package/dist/services/plugins/lspServers.js.map +2 -2
  156. package/dist/services/plugins/pluginRuntime.js +2 -1
  157. package/dist/services/plugins/pluginRuntime.js.map +2 -2
  158. package/dist/services/plugins/pluginValidation.js +10 -3
  159. package/dist/services/plugins/pluginValidation.js.map +2 -2
  160. package/dist/services/plugins/skillMarketplace.js +16 -8
  161. package/dist/services/plugins/skillMarketplace.js.map +2 -2
  162. package/dist/services/systemReminder.js +17 -6
  163. package/dist/services/systemReminder.js.map +2 -2
  164. package/dist/tools/FileEditTool/FileEditTool.js +7 -0
  165. package/dist/tools/FileEditTool/FileEditTool.js.map +2 -2
  166. package/dist/tools/FileWriteTool/FileWriteTool.js +7 -0
  167. package/dist/tools/FileWriteTool/FileWriteTool.js.map +2 -2
  168. package/dist/tools/MultiEditTool/MultiEditTool.js +7 -0
  169. package/dist/tools/MultiEditTool/MultiEditTool.js.map +2 -2
  170. package/dist/tools/NotebookEditTool/NotebookEditTool.js +2 -0
  171. package/dist/tools/NotebookEditTool/NotebookEditTool.js.map +2 -2
  172. package/dist/tools/TaskTool/TaskTool.js +179 -1
  173. package/dist/tools/TaskTool/TaskTool.js.map +2 -2
  174. package/dist/tools/TodoWriteTool/prompt.js +21 -0
  175. package/dist/tools/TodoWriteTool/prompt.js.map +2 -2
  176. package/dist/tools/URLFetcherTool/prompt.js +14 -9
  177. package/dist/tools/URLFetcherTool/prompt.js.map +2 -2
  178. package/dist/tools/WebSearchTool/prompt.js +12 -6
  179. package/dist/tools/WebSearchTool/prompt.js.map +2 -2
  180. package/dist/types/PermissionMode.js +30 -1
  181. package/dist/types/PermissionMode.js.map +2 -2
  182. package/dist/types/plugin.js +2 -4
  183. package/dist/types/plugin.js.map +2 -2
  184. package/dist/utils/agentHookExecutor.js +103 -0
  185. package/dist/utils/agentHookExecutor.js.map +7 -0
  186. package/dist/utils/agentLoader.js +272 -32
  187. package/dist/utils/agentLoader.js.map +2 -2
  188. package/dist/utils/agentMemory.js +134 -0
  189. package/dist/utils/agentMemory.js.map +7 -0
  190. package/dist/utils/claudeCodeSync.js +439 -0
  191. package/dist/utils/claudeCodeSync.js.map +7 -0
  192. package/dist/utils/config.js +52 -24
  193. package/dist/utils/config.js.map +2 -2
  194. package/dist/utils/configPaths.js +199 -0
  195. package/dist/utils/configPaths.js.map +7 -0
  196. package/dist/utils/execFileNoThrow.js +2 -1
  197. package/dist/utils/execFileNoThrow.js.map +2 -2
  198. package/dist/utils/historyManager.js +234 -0
  199. package/dist/utils/historyManager.js.map +7 -0
  200. package/dist/utils/marketplaceManager.js +80 -43
  201. package/dist/utils/marketplaceManager.js.map +2 -2
  202. package/dist/utils/messages.js +13 -8
  203. package/dist/utils/messages.js.map +2 -2
  204. package/dist/utils/migration/index.js +37 -0
  205. package/dist/utils/migration/index.js.map +7 -0
  206. package/dist/utils/migration/migrateHistory.js +273 -0
  207. package/dist/utils/migration/migrateHistory.js.map +7 -0
  208. package/dist/utils/migration/migrateTodos.js +323 -0
  209. package/dist/utils/migration/migrateTodos.js.map +7 -0
  210. package/dist/utils/pasteCache.js +309 -0
  211. package/dist/utils/pasteCache.js.map +7 -0
  212. package/dist/utils/pluginInstaller.js +34 -24
  213. package/dist/utils/pluginInstaller.js.map +2 -2
  214. package/dist/utils/pluginLoader.js +54 -28
  215. package/dist/utils/pluginLoader.js.map +2 -2
  216. package/dist/utils/repoFetcher.js +110 -0
  217. package/dist/utils/repoFetcher.js.map +7 -0
  218. package/dist/utils/sessionIndex.js +192 -0
  219. package/dist/utils/sessionIndex.js.map +7 -0
  220. package/dist/utils/sessionTracker.js +170 -0
  221. package/dist/utils/sessionTracker.js.map +7 -0
  222. package/dist/utils/skillLoader.js +103 -5
  223. package/dist/utils/skillLoader.js.map +2 -2
  224. package/dist/utils/stats.js +417 -0
  225. package/dist/utils/stats.js.map +7 -0
  226. package/dist/utils/stringSubstitution.js +106 -0
  227. package/dist/utils/stringSubstitution.js.map +7 -0
  228. package/dist/utils/teamConfig.js +156 -14
  229. package/dist/utils/teamConfig.js.map +2 -2
  230. package/dist/utils/terminal.js +1 -1
  231. package/dist/utils/terminal.js.map +2 -2
  232. package/dist/utils/todoStorage.js +51 -19
  233. package/dist/utils/todoStorage.js.map +2 -2
  234. package/dist/utils/tooling/safeRender.js.map +2 -2
  235. package/dist/version.js +2 -2
  236. package/dist/version.js.map +1 -1
  237. package/package.json +71 -28
  238. package/scripts/{postinstall.js → postinstall.cjs} +1 -1
@@ -1,62 +1,34 @@
1
- import React, { useState, useEffect, useCallback } from "react";
2
- import { Box, Text, useInput } from "ink";
3
- import { Select } from "../components/CustomSelect/select.js";
4
- import { getTheme } from "../utils/theme.js";
5
- import { SEMANTIC_COLORS } from "../constants/colors.js";
1
+ import React, { useState, useEffect, useCallback, useMemo } from "react";
6
2
  import {
7
3
  listMCPServers,
8
4
  getClients,
9
5
  refreshMCPConnections,
10
- getMcpServer
6
+ getMcpServer,
7
+ addMcpServer,
8
+ removeMcpServer
11
9
  } from "../services/mcpClient.js";
12
- import { SimpleSpinner } from "../components/Spinner.js";
13
10
  import {
14
11
  getCurrentProjectConfig,
15
12
  saveCurrentProjectConfig
16
13
  } from "../utils/config.js";
17
- const MainMenu = ({ onNavigate, onExit, isActive = true }) => {
18
- const theme = getTheme();
19
- const options = [
20
- { label: "\u{1F4CB} View all MCP servers", value: "list" },
21
- { label: "\u{1F504} Refresh connections", value: "refresh" }
22
- ];
23
- useInput(
24
- (input, key) => {
25
- if (key.escape) onExit();
26
- },
27
- { isActive }
28
- );
29
- return /* @__PURE__ */ React.createElement(
30
- Box,
31
- {
32
- flexDirection: "column",
33
- borderStyle: "round",
34
- borderColor: theme.primary,
35
- padding: 1
36
- },
37
- /* @__PURE__ */ React.createElement(Box, { marginBottom: 1 }, /* @__PURE__ */ React.createElement(Text, { bold: true, color: theme.primary }, "MCP Server Manager")),
38
- /* @__PURE__ */ React.createElement(
39
- Select,
40
- {
41
- options,
42
- onChange: (value) => {
43
- if (value === "list") onNavigate({ screen: "server-list" });
44
- else if (value === "refresh") {
45
- refreshMCPConnections();
46
- onNavigate({ screen: "server-list" });
47
- }
48
- }
49
- }
50
- ),
51
- /* @__PURE__ */ React.createElement(Box, { marginTop: 1 }, /* @__PURE__ */ React.createElement(Text, { color: SEMANTIC_COLORS.dim }, "Press ", /* @__PURE__ */ React.createElement(Text, { bold: true }, "ESC"), " to exit"))
52
- );
53
- };
54
- const ServerList = ({ onNavigate, onBack, isActive = true }) => {
55
- const theme = getTheme();
56
- const [loading, setLoading] = useState(true);
14
+ import { TabbedListView } from "../components/TabbedListView/TabbedListView.js";
15
+ import { t } from "../i18n/index.js";
16
+ const ID_ADD_SERVER = "__add__";
17
+ const ID_REFRESH_ALL = "__refresh__";
18
+ const ID_TOGGLE = "__toggle__";
19
+ const ID_DELETE = "__delete__";
20
+ const ID_BACK = "__back__";
21
+ const ID_CONFIRM_YES = "__yes__";
22
+ const ID_CONFIRM_NO = "__no__";
23
+ const MCPInteractive = ({ onDone }) => {
24
+ const [phase, setPhase] = useState({ type: "main" });
25
+ const [activeTab, setActiveTab] = useState("all");
26
+ const [searchQuery, setSearchQuery] = useState("");
27
+ const [statusOverlay, setStatusOverlay] = useState(null);
57
28
  const [servers, setServers] = useState([]);
58
- useEffect(() => {
59
- const loadServers = async () => {
29
+ const [loading, setLoading] = useState(true);
30
+ const loadServers = useCallback(async () => {
31
+ try {
60
32
  const serverConfigs = listMCPServers();
61
33
  const clients = await getClients();
62
34
  const serverStatuses = Object.entries(serverConfigs).map(
@@ -71,198 +43,428 @@ const ServerList = ({ onNavigate, onBack, isActive = true }) => {
71
43
  }
72
44
  );
73
45
  setServers(serverStatuses);
74
- setLoading(false);
75
- };
76
- loadServers();
46
+ } catch {
47
+ setServers([]);
48
+ }
77
49
  }, []);
78
- const handleServerSelect = useCallback(
79
- (serverName) => {
80
- onNavigate({ screen: "server-details", serverName });
81
- },
82
- [onNavigate]
83
- );
84
- useInput(
85
- (input, key) => {
86
- if (key.escape) onBack();
50
+ useEffect(() => {
51
+ setLoading(true);
52
+ loadServers().finally(() => setLoading(false));
53
+ }, [loadServers]);
54
+ useEffect(() => {
55
+ if (phase.type === "main") {
56
+ loadServers();
57
+ setSearchQuery("");
58
+ }
59
+ }, [phase, loadServers]);
60
+ const toggleServer = useCallback(
61
+ async (serverName) => {
62
+ setStatusOverlay({
63
+ type: "loading",
64
+ message: t("commands.mcp.toggling")
65
+ });
66
+ try {
67
+ const projectConfig = getCurrentProjectConfig();
68
+ const currentServer = projectConfig.mcpServers?.[serverName];
69
+ if (currentServer) {
70
+ const newEnabled = !(currentServer.enabled ?? true);
71
+ await saveCurrentProjectConfig({
72
+ ...projectConfig,
73
+ mcpServers: {
74
+ ...projectConfig.mcpServers,
75
+ [serverName]: { ...currentServer, enabled: newEnabled }
76
+ }
77
+ });
78
+ } else {
79
+ const serverConfig = getMcpServer(serverName);
80
+ if (serverConfig) {
81
+ await saveCurrentProjectConfig({
82
+ ...projectConfig,
83
+ mcpServers: {
84
+ ...projectConfig.mcpServers,
85
+ [serverName]: { ...serverConfig, enabled: false }
86
+ }
87
+ });
88
+ }
89
+ }
90
+ refreshMCPConnections();
91
+ await loadServers();
92
+ setStatusOverlay({
93
+ type: "success",
94
+ message: t("commands.mcp.toggleSuccess")
95
+ });
96
+ } catch (err) {
97
+ const msg = err instanceof Error ? err.message : String(err);
98
+ setStatusOverlay({ type: "error", message: msg });
99
+ }
87
100
  },
88
- { isActive }
101
+ [loadServers]
89
102
  );
90
- if (loading) {
91
- return /* @__PURE__ */ React.createElement(Box, { flexDirection: "column", padding: 1 }, /* @__PURE__ */ React.createElement(SimpleSpinner, { label: "Loading MCP servers..." }));
92
- }
93
- if (servers.length === 0) {
94
- return /* @__PURE__ */ React.createElement(Box, { flexDirection: "column", padding: 1 }, /* @__PURE__ */ React.createElement(Text, null, "No MCP servers configured."), /* @__PURE__ */ React.createElement(Box, { marginTop: 1 }, /* @__PURE__ */ React.createElement(Text, { color: SEMANTIC_COLORS.dim }, "Press ", /* @__PURE__ */ React.createElement(Text, { bold: true }, "ESC"), " to go back")));
95
- }
96
- const options = servers.map((server) => {
97
- const statusIcon = server.status === "connected" ? "\u2713" : "\u2717";
98
- const statusColor = server.status === "connected" ? theme.success : theme.error;
99
- return {
100
- label: /* @__PURE__ */ React.createElement(Text, null, /* @__PURE__ */ React.createElement(Text, { color: statusColor }, statusIcon), " ", server.name, " ", /* @__PURE__ */ React.createElement(Text, { color: SEMANTIC_COLORS.dim }, "(", server.config.scope, ")")),
101
- value: server.name
102
- };
103
- });
104
- return /* @__PURE__ */ React.createElement(
105
- Box,
106
- {
107
- flexDirection: "column",
108
- borderStyle: "round",
109
- borderColor: theme.primary,
110
- padding: 1
103
+ const deleteServer = useCallback(
104
+ async (serverName, scope) => {
105
+ setStatusOverlay({
106
+ type: "loading",
107
+ message: t("commands.mcp.deleting")
108
+ });
109
+ try {
110
+ removeMcpServer(serverName, scope);
111
+ refreshMCPConnections();
112
+ await loadServers();
113
+ setStatusOverlay({
114
+ type: "success",
115
+ message: t("commands.mcp.deleteSuccess")
116
+ });
117
+ } catch (err) {
118
+ const msg = err instanceof Error ? err.message : String(err);
119
+ setStatusOverlay({ type: "error", message: msg });
120
+ }
111
121
  },
112
- /* @__PURE__ */ React.createElement(Box, { marginBottom: 1 }, /* @__PURE__ */ React.createElement(Text, { bold: true, color: theme.primary }, "MCP Servers")),
113
- /* @__PURE__ */ React.createElement(Select, { options, onChange: handleServerSelect }),
114
- /* @__PURE__ */ React.createElement(Box, { marginTop: 1 }, /* @__PURE__ */ React.createElement(Text, { color: SEMANTIC_COLORS.dim }, "Press ", /* @__PURE__ */ React.createElement(Text, { bold: true }, "ESC"), " to go back"))
122
+ [loadServers]
115
123
  );
116
- };
117
- const ServerDetails = ({ serverName, onBack, onRefresh, isActive = true }) => {
118
- const theme = getTheme();
119
- const [loading, setLoading] = useState(true);
120
- const [server, setServer] = useState(null);
121
- const [toggling, setToggling] = useState(false);
122
- useEffect(() => {
123
- const loadServer = async () => {
124
- const config = getMcpServer(serverName);
125
- if (!config) {
126
- onBack();
127
- return;
128
- }
129
- const clients = await getClients();
130
- const client = clients.find((c) => c.name === serverName);
131
- setServer({
132
- name: serverName,
133
- status: client?.type === "connected" ? "connected" : "failed",
134
- config
124
+ const addNewServer = useCallback(
125
+ async (name, commandOrUrl) => {
126
+ setStatusOverlay({
127
+ type: "loading",
128
+ message: t("commands.mcp.adding")
135
129
  });
136
- setLoading(false);
137
- };
138
- loadServer();
139
- }, [serverName]);
140
- const toggleEnabled = async () => {
141
- if (!server) return;
142
- setToggling(true);
143
- const projectConfig = getCurrentProjectConfig();
144
- const currentServer = projectConfig.mcpServers?.[serverName];
145
- if (currentServer) {
146
- const newEnabled = !(currentServer.enabled ?? true);
147
- await saveCurrentProjectConfig({
148
- ...projectConfig,
149
- mcpServers: {
150
- ...projectConfig.mcpServers,
151
- [serverName]: {
152
- ...currentServer,
153
- enabled: newEnabled
154
- }
130
+ try {
131
+ if (commandOrUrl.startsWith("http://") || commandOrUrl.startsWith("https://")) {
132
+ addMcpServer(name, { type: "sse", url: commandOrUrl }, "project");
133
+ } else {
134
+ const parts = commandOrUrl.split(/\s+/);
135
+ const command = parts[0];
136
+ const args = parts.slice(1);
137
+ addMcpServer(
138
+ name,
139
+ { type: "stdio", command, args: args.length > 0 ? args : [] },
140
+ "project"
141
+ );
155
142
  }
143
+ refreshMCPConnections();
144
+ await loadServers();
145
+ setStatusOverlay({
146
+ type: "success",
147
+ message: t("commands.mcp.addSuccess")
148
+ });
149
+ } catch (err) {
150
+ const msg = err instanceof Error ? err.message : String(err);
151
+ setStatusOverlay({
152
+ type: "error",
153
+ message: `${t("commands.mcp.addFailed")}: ${msg}`
154
+ });
155
+ }
156
+ },
157
+ [loadServers]
158
+ );
159
+ const refreshAll = useCallback(async () => {
160
+ setStatusOverlay({
161
+ type: "loading",
162
+ message: t("commands.mcp.refreshing")
163
+ });
164
+ try {
165
+ refreshMCPConnections();
166
+ await loadServers();
167
+ setStatusOverlay({
168
+ type: "success",
169
+ message: t("commands.mcp.refreshSuccess")
156
170
  });
157
- } else {
158
- const serverConfig = getMcpServer(serverName);
159
- if (serverConfig) {
160
- await saveCurrentProjectConfig({
161
- ...projectConfig,
162
- mcpServers: {
163
- ...projectConfig.mcpServers,
164
- [serverName]: {
165
- ...serverConfig,
166
- enabled: false
167
- }
171
+ } catch (err) {
172
+ const msg = err instanceof Error ? err.message : String(err);
173
+ setStatusOverlay({ type: "error", message: msg });
174
+ }
175
+ }, [loadServers]);
176
+ const { tabs, items, title, footerHint, searchEnabled, searchPlaceholder } = useMemo(() => {
177
+ switch (phase.type) {
178
+ case "main": {
179
+ const projectCount = servers.filter(
180
+ (s) => s.config.scope === "project"
181
+ ).length;
182
+ const globalCount = servers.filter(
183
+ (s) => s.config.scope === "global"
184
+ ).length;
185
+ const mainTabs = [
186
+ {
187
+ id: "all",
188
+ label: t("commands.mcp.tabAll"),
189
+ badge: servers.length
190
+ },
191
+ {
192
+ id: "project",
193
+ label: t("commands.mcp.tabProject"),
194
+ badge: projectCount
195
+ },
196
+ {
197
+ id: "global",
198
+ label: t("commands.mcp.tabGlobal"),
199
+ badge: globalCount
168
200
  }
201
+ ];
202
+ const filtered = activeTab === "all" ? servers : servers.filter((s) => s.config.scope === activeTab);
203
+ const mainItems = filtered.map((server) => {
204
+ const isEnabled = "enabled" in server.config ? server.config.enabled : true;
205
+ return {
206
+ id: server.name,
207
+ label: server.name,
208
+ description: `${server.config.scope} \xB7 ${server.config.type}`,
209
+ status: !isEnabled ? "disabled" : server.status === "connected" ? "enabled" : "error",
210
+ data: { type: "server", server }
211
+ };
169
212
  });
213
+ mainItems.push({
214
+ id: ID_ADD_SERVER,
215
+ label: `+ ${t("commands.mcp.addServer")}`,
216
+ data: { type: "action" }
217
+ });
218
+ mainItems.push({
219
+ id: ID_REFRESH_ALL,
220
+ label: `\u21BB ${t("commands.mcp.refreshAll")}`,
221
+ data: { type: "action" }
222
+ });
223
+ return {
224
+ tabs: mainTabs,
225
+ items: mainItems,
226
+ title: t("commands.mcp.title"),
227
+ footerHint: "\u2191\u2193 Navigate \xB7 Enter Select \xB7 Tab Switch \xB7 / Search \xB7 Esc Exit",
228
+ searchEnabled: true,
229
+ searchPlaceholder: void 0
230
+ };
231
+ }
232
+ case "server-actions": {
233
+ const server = servers.find((s) => s.name === phase.serverName);
234
+ const isEnabled = server ? "enabled" in server.config ? server.config.enabled : true : true;
235
+ const actionItems = [
236
+ {
237
+ id: ID_TOGGLE,
238
+ label: isEnabled ? t("commands.mcp.disable") : t("commands.mcp.enable")
239
+ },
240
+ { id: ID_DELETE, label: t("commands.mcp.deleteServer") },
241
+ { id: ID_BACK, label: `\u2190 ${t("commands.mcp.back")}` }
242
+ ];
243
+ return {
244
+ tabs: [
245
+ { id: "actions", label: t("commands.mcp.actions") }
246
+ ],
247
+ items: actionItems,
248
+ title: phase.serverName,
249
+ footerHint: "\u2191\u2193 Navigate \xB7 Enter Select \xB7 Esc Back",
250
+ searchEnabled: false,
251
+ searchPlaceholder: void 0
252
+ };
170
253
  }
254
+ case "confirm-delete": {
255
+ const confirmItems = [
256
+ { id: ID_CONFIRM_YES, label: t("commands.mcp.confirmYes") },
257
+ { id: ID_CONFIRM_NO, label: t("commands.mcp.confirmNo") }
258
+ ];
259
+ return {
260
+ tabs: [
261
+ { id: "actions", label: t("commands.mcp.actions") }
262
+ ],
263
+ items: confirmItems,
264
+ title: t("commands.mcp.confirmDeleteTitle").replace(
265
+ "{name}",
266
+ phase.serverName
267
+ ),
268
+ footerHint: "\u2191\u2193 Navigate \xB7 Enter Select \xB7 Esc Back",
269
+ searchEnabled: false,
270
+ searchPlaceholder: void 0
271
+ };
272
+ }
273
+ case "add-name": {
274
+ return {
275
+ tabs: [
276
+ { id: "add", label: t("commands.mcp.addServer") }
277
+ ],
278
+ items: [],
279
+ title: t("commands.mcp.addServer"),
280
+ footerHint: t("commands.mcp.addFooterHint"),
281
+ searchEnabled: true,
282
+ searchPlaceholder: t("commands.mcp.addNamePlaceholder")
283
+ };
284
+ }
285
+ case "add-command": {
286
+ const exampleItems = [
287
+ {
288
+ id: "example-stdio",
289
+ label: "npx -y @modelcontextprotocol/server-xxx",
290
+ description: "stdio command"
291
+ },
292
+ {
293
+ id: "example-sse",
294
+ label: "https://example.com/mcp/sse",
295
+ description: "SSE endpoint"
296
+ }
297
+ ];
298
+ return {
299
+ tabs: [
300
+ { id: "add", label: t("commands.mcp.addServer") }
301
+ ],
302
+ items: exampleItems,
303
+ title: `${t("commands.mcp.addServer")}: ${phase.name}`,
304
+ footerHint: t("commands.mcp.addCommandFooterHint"),
305
+ searchEnabled: true,
306
+ searchPlaceholder: t("commands.mcp.addCommandPlaceholder")
307
+ };
308
+ }
309
+ }
310
+ }, [phase, activeTab, servers]);
311
+ const handleTabChange = useCallback((tabId) => {
312
+ setActiveTab(tabId);
313
+ setSearchQuery("");
314
+ }, []);
315
+ const handleStatusDismiss = useCallback(() => {
316
+ setStatusOverlay(null);
317
+ setPhase({ type: "main" });
318
+ }, []);
319
+ const handleBack = useCallback(() => {
320
+ if (statusOverlay && statusOverlay.type !== "loading") {
321
+ handleStatusDismiss();
322
+ return;
323
+ }
324
+ switch (phase.type) {
325
+ case "main":
326
+ onDone();
327
+ break;
328
+ case "server-actions":
329
+ setPhase({ type: "main" });
330
+ setSearchQuery("");
331
+ break;
332
+ case "confirm-delete":
333
+ setPhase({
334
+ type: "server-actions",
335
+ serverName: phase.serverName,
336
+ scope: phase.scope
337
+ });
338
+ break;
339
+ case "add-name":
340
+ case "add-command":
341
+ setPhase({ type: "main" });
342
+ setSearchQuery("");
343
+ break;
344
+ }
345
+ }, [phase, statusOverlay, onDone, handleStatusDismiss]);
346
+ const handleClose = useCallback(() => {
347
+ if (statusOverlay && statusOverlay.type !== "loading") {
348
+ handleStatusDismiss();
349
+ return;
350
+ }
351
+ if (phase.type === "main") {
352
+ onDone();
353
+ } else {
354
+ setPhase({ type: "main" });
355
+ setSearchQuery("");
171
356
  }
172
- setToggling(false);
173
- refreshMCPConnections();
174
- onRefresh();
175
- };
176
- useInput(
177
- (input, key) => {
178
- if (key.escape) onBack();
357
+ }, [phase, statusOverlay, onDone, handleStatusDismiss]);
358
+ const handleSearchSubmit = useCallback(
359
+ async (query) => {
360
+ if (!query.trim()) return;
361
+ if (phase.type === "add-name") {
362
+ setPhase({ type: "add-command", name: query.trim() });
363
+ setSearchQuery("");
364
+ return;
365
+ }
366
+ if (phase.type === "add-command") {
367
+ await addNewServer(phase.name, query.trim());
368
+ setSearchQuery("");
369
+ }
179
370
  },
180
- { isActive }
371
+ [phase, addNewServer]
181
372
  );
182
- if (loading) {
183
- return /* @__PURE__ */ React.createElement(Box, { flexDirection: "column", padding: 1 }, /* @__PURE__ */ React.createElement(SimpleSpinner, { label: "Loading server details..." }));
184
- }
185
- if (!server) {
186
- return /* @__PURE__ */ React.createElement(Box, { flexDirection: "column", padding: 1 }, /* @__PURE__ */ React.createElement(Text, { color: theme.error }, "Server not found"));
187
- }
188
- const isEnabled = "enabled" in server.config ? server.config.enabled : true;
189
- const statusColor = server.status === "connected" ? theme.success : theme.error;
190
- const enabledColor = isEnabled ? theme.success : theme.error;
191
- const options = [
192
- {
193
- label: isEnabled ? "\u{1F534} Disable server" : "\u{1F7E2} Enable server",
194
- value: "toggle"
195
- },
196
- { label: "\u{1F504} Refresh connection", value: "refresh" },
197
- { label: "\u2190 Back to list", value: "back" }
198
- ];
199
- return /* @__PURE__ */ React.createElement(
200
- Box,
201
- {
202
- flexDirection: "column",
203
- borderStyle: "round",
204
- borderColor: theme.primary,
205
- padding: 1
206
- },
207
- /* @__PURE__ */ React.createElement(Box, { marginBottom: 1 }, /* @__PURE__ */ React.createElement(Text, { bold: true, color: theme.primary }, server.name)),
208
- /* @__PURE__ */ React.createElement(Box, { flexDirection: "column", marginBottom: 1 }, /* @__PURE__ */ React.createElement(Text, null, "Status: ", /* @__PURE__ */ React.createElement(Text, { color: statusColor }, server.status)), /* @__PURE__ */ React.createElement(Text, null, "Enabled: ", /* @__PURE__ */ React.createElement(Text, { color: enabledColor }, isEnabled ? "Yes" : "No")), /* @__PURE__ */ React.createElement(Text, null, "Scope: ", /* @__PURE__ */ React.createElement(Text, { color: SEMANTIC_COLORS.dim }, server.config.scope)), /* @__PURE__ */ React.createElement(Text, null, "Type: ", /* @__PURE__ */ React.createElement(Text, { color: SEMANTIC_COLORS.dim }, server.config.type))),
209
- toggling ? /* @__PURE__ */ React.createElement(SimpleSpinner, { label: "Toggling server..." }) : /* @__PURE__ */ React.createElement(
210
- Select,
211
- {
212
- options,
213
- onChange: (value) => {
214
- if (value === "toggle") toggleEnabled();
215
- else if (value === "refresh") {
216
- refreshMCPConnections();
217
- onRefresh();
218
- } else if (value === "back") onBack();
373
+ const handleSelect = useCallback(
374
+ async (item) => {
375
+ switch (phase.type) {
376
+ case "main": {
377
+ if (item.id === ID_ADD_SERVER) {
378
+ setPhase({ type: "add-name" });
379
+ setSearchQuery("");
380
+ return;
381
+ }
382
+ if (item.id === ID_REFRESH_ALL) {
383
+ await refreshAll();
384
+ return;
385
+ }
386
+ const data = item.data;
387
+ if (data.type === "server" && data.server) {
388
+ setPhase({
389
+ type: "server-actions",
390
+ serverName: data.server.name,
391
+ scope: data.server.config.scope
392
+ });
393
+ }
394
+ break;
395
+ }
396
+ case "server-actions": {
397
+ if (item.id === ID_BACK) {
398
+ setPhase({ type: "main" });
399
+ return;
400
+ }
401
+ if (item.id === ID_TOGGLE) {
402
+ await toggleServer(phase.serverName);
403
+ return;
404
+ }
405
+ if (item.id === ID_DELETE) {
406
+ setPhase({
407
+ type: "confirm-delete",
408
+ serverName: phase.serverName,
409
+ scope: phase.scope
410
+ });
411
+ }
412
+ break;
413
+ }
414
+ case "confirm-delete": {
415
+ if (item.id === ID_CONFIRM_YES) {
416
+ await deleteServer(phase.serverName, phase.scope);
417
+ } else {
418
+ setPhase({
419
+ type: "server-actions",
420
+ serverName: phase.serverName,
421
+ scope: phase.scope
422
+ });
423
+ }
424
+ break;
425
+ }
426
+ case "add-command": {
427
+ setSearchQuery(item.label);
428
+ break;
219
429
  }
220
430
  }
221
- ),
222
- /* @__PURE__ */ React.createElement(Box, { marginTop: 1 }, /* @__PURE__ */ React.createElement(Text, { color: SEMANTIC_COLORS.dim }, "Press ", /* @__PURE__ */ React.createElement(Text, { bold: true }, "ESC"), " to go back"))
431
+ },
432
+ [phase, refreshAll, toggleServer, deleteServer]
223
433
  );
224
- };
225
- const MCPInteractive = ({ onDone }) => {
226
- const [navState, setNavState] = useState({
227
- screen: "main-menu"
228
- });
229
- const handleNavigate = (state) => {
230
- setNavState(state);
231
- };
232
- const handleBack = () => {
233
- if (navState.screen === "server-details") {
234
- setNavState({ screen: "server-list" });
235
- } else {
236
- setNavState({ screen: "main-menu" });
434
+ const currentTab = phase.type === "main" ? activeTab : phase.type === "server-actions" || phase.type === "confirm-delete" ? "actions" : "add";
435
+ const needsSearchSubmit = phase.type === "add-name" || phase.type === "add-command";
436
+ return /* @__PURE__ */ React.createElement(
437
+ TabbedListView,
438
+ {
439
+ title,
440
+ tabs,
441
+ activeTab: currentTab,
442
+ onTabChange: handleTabChange,
443
+ items,
444
+ searchEnabled,
445
+ searchPlaceholder,
446
+ searchQuery,
447
+ onSearchChange: setSearchQuery,
448
+ onSearchSubmit: needsSearchSubmit ? handleSearchSubmit : void 0,
449
+ onSelect: handleSelect,
450
+ onClose: handleClose,
451
+ onBack: handleBack,
452
+ statusOverlay,
453
+ footerHint,
454
+ groupByCategory: false,
455
+ isLoading: loading,
456
+ emptyText: t("commands.mcp.noServers"),
457
+ statusDismissHint: "Enter Continue \xB7 Esc Back"
237
458
  }
238
- };
239
- const handleRefresh = () => {
240
- setNavState({ ...navState });
241
- };
242
- switch (navState.screen) {
243
- case "main-menu":
244
- return /* @__PURE__ */ React.createElement(MainMenu, { onNavigate: handleNavigate, onExit: () => onDone() });
245
- case "server-list":
246
- return /* @__PURE__ */ React.createElement(ServerList, { onNavigate: handleNavigate, onBack: handleBack });
247
- case "server-details":
248
- return /* @__PURE__ */ React.createElement(
249
- ServerDetails,
250
- {
251
- serverName: navState.serverName,
252
- onBack: handleBack,
253
- onRefresh: handleRefresh
254
- }
255
- );
256
- default:
257
- return /* @__PURE__ */ React.createElement(Text, null, "Unknown screen");
258
- }
459
+ );
259
460
  };
260
461
  const mcpInteractive = {
261
462
  type: "local-jsx",
262
463
  name: "mcp",
263
- description: "Interactive MCP server management",
464
+ description: t("commands.mcp.description"),
264
465
  isEnabled: true,
265
466
  isHidden: false,
467
+ hidePromptInput: true,
266
468
  async call(onDone) {
267
469
  return /* @__PURE__ */ React.createElement(MCPInteractive, { onDone });
268
470
  },