mcp-inspect 1.0.0

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 (72) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +107 -0
  3. package/dist/src/App.d.ts +7 -0
  4. package/dist/src/App.d.ts.map +1 -0
  5. package/dist/src/App.js +591 -0
  6. package/dist/src/App.js.map +1 -0
  7. package/dist/src/components/DetailsModal.d.ts +11 -0
  8. package/dist/src/components/DetailsModal.d.ts.map +1 -0
  9. package/dist/src/components/DetailsModal.js +50 -0
  10. package/dist/src/components/DetailsModal.js.map +1 -0
  11. package/dist/src/components/HistoryTab.d.ts +13 -0
  12. package/dist/src/components/HistoryTab.d.ts.map +1 -0
  13. package/dist/src/components/HistoryTab.js +122 -0
  14. package/dist/src/components/HistoryTab.js.map +1 -0
  15. package/dist/src/components/InfoTab.d.ts +13 -0
  16. package/dist/src/components/InfoTab.d.ts.map +1 -0
  17. package/dist/src/components/InfoTab.js +28 -0
  18. package/dist/src/components/InfoTab.js.map +1 -0
  19. package/dist/src/components/NotificationsTab.d.ts +13 -0
  20. package/dist/src/components/NotificationsTab.d.ts.map +1 -0
  21. package/dist/src/components/NotificationsTab.js +37 -0
  22. package/dist/src/components/NotificationsTab.js.map +1 -0
  23. package/dist/src/components/PromptsTab.d.ts +13 -0
  24. package/dist/src/components/PromptsTab.d.ts.map +1 -0
  25. package/dist/src/components/PromptsTab.js +60 -0
  26. package/dist/src/components/PromptsTab.js.map +1 -0
  27. package/dist/src/components/ResourcesTab.d.ts +13 -0
  28. package/dist/src/components/ResourcesTab.d.ts.map +1 -0
  29. package/dist/src/components/ResourcesTab.js +60 -0
  30. package/dist/src/components/ResourcesTab.js.map +1 -0
  31. package/dist/src/components/Tabs.d.ts +24 -0
  32. package/dist/src/components/Tabs.d.ts.map +1 -0
  33. package/dist/src/components/Tabs.js +22 -0
  34. package/dist/src/components/Tabs.js.map +1 -0
  35. package/dist/src/components/ToolTestModal.d.ts +11 -0
  36. package/dist/src/components/ToolTestModal.d.ts.map +1 -0
  37. package/dist/src/components/ToolTestModal.js +112 -0
  38. package/dist/src/components/ToolTestModal.js.map +1 -0
  39. package/dist/src/components/ToolsTab.d.ts +14 -0
  40. package/dist/src/components/ToolsTab.d.ts.map +1 -0
  41. package/dist/src/components/ToolsTab.js +76 -0
  42. package/dist/src/components/ToolsTab.js.map +1 -0
  43. package/dist/src/hooks/useMCPClient.d.ts +41 -0
  44. package/dist/src/hooks/useMCPClient.d.ts.map +1 -0
  45. package/dist/src/hooks/useMCPClient.js +179 -0
  46. package/dist/src/hooks/useMCPClient.js.map +1 -0
  47. package/dist/src/hooks/useMessageTracking.d.ts +9 -0
  48. package/dist/src/hooks/useMessageTracking.d.ts.map +1 -0
  49. package/dist/src/hooks/useMessageTracking.js +124 -0
  50. package/dist/src/hooks/useMessageTracking.js.map +1 -0
  51. package/dist/src/types/focus.d.ts +2 -0
  52. package/dist/src/types/focus.d.ts.map +1 -0
  53. package/dist/src/types/focus.js +2 -0
  54. package/dist/src/types/focus.js.map +1 -0
  55. package/dist/src/types/messages.d.ts +14 -0
  56. package/dist/src/types/messages.d.ts.map +1 -0
  57. package/dist/src/types/messages.js +2 -0
  58. package/dist/src/types/messages.js.map +1 -0
  59. package/dist/src/types.d.ts +48 -0
  60. package/dist/src/types.d.ts.map +1 -0
  61. package/dist/src/types.js +2 -0
  62. package/dist/src/types.js.map +1 -0
  63. package/dist/src/utils/schemaToForm.d.ts +9 -0
  64. package/dist/src/utils/schemaToForm.d.ts.map +1 -0
  65. package/dist/src/utils/schemaToForm.js +107 -0
  66. package/dist/src/utils/schemaToForm.js.map +1 -0
  67. package/dist/tui.d.ts +3 -0
  68. package/dist/tui.d.ts.map +1 -0
  69. package/dist/tui.js +43 -0
  70. package/dist/tui.js.map +1 -0
  71. package/package.json +58 -0
  72. package/screenshots/mcp-inspector.png +0 -0
@@ -0,0 +1,60 @@
1
+ import { jsxs as _jsxs, jsx as _jsx, Fragment as _Fragment } from "react/jsx-runtime";
2
+ import { useState, useEffect, useRef } from 'react';
3
+ import { Box, Text, useInput } from 'ink';
4
+ import { ScrollView } from 'ink-scroll-view';
5
+ export function ResourcesTab({ resources, client, width, height, onCountChange, focusedPane = null, onViewDetails }) {
6
+ const [selectedIndex, setSelectedIndex] = useState(0);
7
+ const [error, setError] = useState(null);
8
+ const scrollViewRef = useRef(null);
9
+ // Handle arrow key navigation when focused
10
+ useInput((input, key) => {
11
+ if (focusedPane === 'list') {
12
+ // Navigate the list
13
+ if (key.upArrow && selectedIndex > 0) {
14
+ setSelectedIndex(selectedIndex - 1);
15
+ }
16
+ else if (key.downArrow && selectedIndex < resources.length - 1) {
17
+ setSelectedIndex(selectedIndex + 1);
18
+ }
19
+ return;
20
+ }
21
+ if (focusedPane === 'details') {
22
+ // Handle '+' key to view in full screen modal
23
+ if (input === '+' && selectedResource && onViewDetails) {
24
+ onViewDetails(selectedResource);
25
+ return;
26
+ }
27
+ // Scroll the details pane using ink-scroll-view
28
+ if (key.upArrow) {
29
+ scrollViewRef.current?.scrollBy(-1);
30
+ }
31
+ else if (key.downArrow) {
32
+ scrollViewRef.current?.scrollBy(1);
33
+ }
34
+ else if (key.pageUp) {
35
+ const viewportHeight = scrollViewRef.current?.getViewportHeight() || 1;
36
+ scrollViewRef.current?.scrollBy(-viewportHeight);
37
+ }
38
+ else if (key.pageDown) {
39
+ const viewportHeight = scrollViewRef.current?.getViewportHeight() || 1;
40
+ scrollViewRef.current?.scrollBy(viewportHeight);
41
+ }
42
+ }
43
+ }, { isActive: focusedPane === 'list' || focusedPane === 'details' });
44
+ // Reset scroll when selection changes
45
+ useEffect(() => {
46
+ scrollViewRef.current?.scrollTo(0);
47
+ }, [selectedIndex]);
48
+ // Reset selected index when resources array changes (different server)
49
+ useEffect(() => {
50
+ setSelectedIndex(0);
51
+ }, [resources]);
52
+ const selectedResource = resources[selectedIndex] || null;
53
+ const listWidth = Math.floor(width * 0.4);
54
+ const detailWidth = width - listWidth;
55
+ return (_jsxs(Box, { flexDirection: "row", width: width, height: height, children: [_jsxs(Box, { width: listWidth, height: height, borderStyle: "single", borderTop: false, borderBottom: false, borderLeft: false, borderRight: true, flexDirection: "column", paddingX: 1, children: [_jsx(Box, { paddingY: 1, children: _jsxs(Text, { bold: true, backgroundColor: focusedPane === 'list' ? 'yellow' : undefined, children: ["Resources (", resources.length, ")"] }) }), error ? (_jsx(Box, { paddingY: 1, children: _jsx(Text, { color: "red", children: error }) })) : resources.length === 0 ? (_jsx(Box, { paddingY: 1, children: _jsx(Text, { dimColor: true, children: "No resources available" }) })) : (_jsx(Box, { flexDirection: "column", flexGrow: 1, children: resources.map((resource, index) => {
56
+ const isSelected = index === selectedIndex;
57
+ return (_jsx(Box, { paddingY: 0, children: _jsxs(Text, { children: [isSelected ? '▶ ' : ' ', resource.name || resource.uri || `Resource ${index + 1}`] }) }, resource.uri || index));
58
+ }) }))] }), _jsx(Box, { width: detailWidth, height: height, paddingX: 1, flexDirection: "column", overflow: "hidden", children: selectedResource ? (_jsxs(_Fragment, { children: [_jsx(Box, { flexShrink: 0, paddingTop: 1, children: _jsx(Text, { bold: true, backgroundColor: focusedPane === 'details' ? 'yellow' : undefined, color: "cyan", children: selectedResource.name || selectedResource.uri }) }), _jsxs(ScrollView, { ref: scrollViewRef, height: height - 5, children: [selectedResource.description && (_jsx(_Fragment, { children: selectedResource.description.split('\n').map((line, idx) => (_jsx(Box, { marginTop: idx === 0 ? 1 : 0, flexShrink: 0, children: _jsx(Text, { dimColor: true, children: line }) }, `desc-${idx}`))) })), selectedResource.uri && (_jsx(Box, { marginTop: 1, flexShrink: 0, children: _jsxs(Text, { dimColor: true, children: ["URI: ", selectedResource.uri] }) })), selectedResource.mimeType && (_jsx(Box, { marginTop: 1, flexShrink: 0, children: _jsxs(Text, { dimColor: true, children: ["MIME Type: ", selectedResource.mimeType] }) }))] }), focusedPane === 'details' && (_jsx(Box, { flexShrink: 0, height: 1, justifyContent: "center", backgroundColor: "gray", children: _jsx(Text, { bold: true, color: "white", children: "\u2191/\u2193 to scroll, + to zoom" }) }))] })) : (_jsx(Box, { paddingY: 1, flexShrink: 0, children: _jsx(Text, { dimColor: true, children: "Select a resource to view details" }) })) })] }));
59
+ }
60
+ //# sourceMappingURL=ResourcesTab.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ResourcesTab.js","sourceRoot":"","sources":["../../../src/components/ResourcesTab.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,24 @@
1
+ export type TabType = 'info' | 'resources' | 'prompts' | 'tools' | 'messages' | 'logging';
2
+ interface TabsProps {
3
+ activeTab: TabType;
4
+ onTabChange: (tab: TabType) => void;
5
+ width: number;
6
+ counts?: {
7
+ info?: number;
8
+ resources?: number;
9
+ prompts?: number;
10
+ tools?: number;
11
+ messages?: number;
12
+ logging?: number;
13
+ };
14
+ focused?: boolean;
15
+ showLogging?: boolean;
16
+ }
17
+ export declare const tabs: {
18
+ id: TabType;
19
+ label: string;
20
+ accelerator: string;
21
+ }[];
22
+ export declare function Tabs({ activeTab, onTabChange, width, counts, focused, showLogging }: TabsProps): import("react/jsx-runtime").JSX.Element;
23
+ export {};
24
+ //# sourceMappingURL=Tabs.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Tabs.d.ts","sourceRoot":"","sources":["../../../src/components/Tabs.tsx"],"names":[],"mappings":"AAGA,MAAM,MAAM,OAAO,GAAG,MAAM,GAAG,WAAW,GAAG,SAAS,GAAG,OAAO,GAAG,UAAU,GAAG,SAAS,CAAC;AAE1F,UAAU,SAAS;IACjB,SAAS,EAAE,OAAO,CAAC;IACnB,WAAW,EAAE,CAAC,GAAG,EAAE,OAAO,KAAK,IAAI,CAAC;IACpC,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,CAAC,EAAE;QACP,IAAI,CAAC,EAAE,MAAM,CAAC;QACd,SAAS,CAAC,EAAE,MAAM,CAAC;QACnB,OAAO,CAAC,EAAE,MAAM,CAAC;QACjB,KAAK,CAAC,EAAE,MAAM,CAAC;QACf,QAAQ,CAAC,EAAE,MAAM,CAAC;QAClB,OAAO,CAAC,EAAE,MAAM,CAAC;KAClB,CAAC;IACF,OAAO,CAAC,EAAE,OAAO,CAAC;IAClB,WAAW,CAAC,EAAE,OAAO,CAAC;CACvB;AAED,eAAO,MAAM,IAAI,EAAE;IAAE,EAAE,EAAE,OAAO,CAAC;IAAC,KAAK,EAAE,MAAM,CAAC;IAAC,WAAW,EAAE,MAAM,CAAA;CAAE,EAOrE,CAAC;AAEF,wBAAgB,IAAI,CAAC,EAAE,SAAS,EAAE,WAAW,EAAE,KAAK,EAAE,MAAW,EAAE,OAAe,EAAE,WAAkB,EAAE,EAAE,SAAS,2CA0ClH"}
@@ -0,0 +1,22 @@
1
+ import { jsx as _jsx, jsxs as _jsxs } from "react/jsx-runtime";
2
+ import { Box, Text } from 'ink';
3
+ export const tabs = [
4
+ { id: 'info', label: 'Info', accelerator: 'i' },
5
+ { id: 'resources', label: 'Resources', accelerator: 'r' },
6
+ { id: 'prompts', label: 'Prompts', accelerator: 'p' },
7
+ { id: 'tools', label: 'Tools', accelerator: 't' },
8
+ { id: 'messages', label: 'Messages', accelerator: 'm' },
9
+ { id: 'logging', label: 'Logging', accelerator: 'l' },
10
+ ];
11
+ export function Tabs({ activeTab, onTabChange, width, counts = {}, focused = false, showLogging = true }) {
12
+ const visibleTabs = showLogging ? tabs : tabs.filter(tab => tab.id !== 'logging');
13
+ return (_jsx(Box, { width: width, borderStyle: "single", borderTop: false, borderLeft: false, borderRight: false, borderBottom: true, flexDirection: "row", justifyContent: "space-between", flexWrap: "wrap", paddingX: 1, children: visibleTabs.map((tab) => {
14
+ const isActive = activeTab === tab.id;
15
+ const count = counts[tab.id];
16
+ const countText = count !== undefined ? ` (${count})` : '';
17
+ const firstChar = tab.label[0];
18
+ const restOfLabel = tab.label.slice(1);
19
+ return (_jsx(Box, { flexShrink: 0, children: _jsxs(Text, { bold: isActive, color: isActive ? 'cyan' : 'gray', backgroundColor: isActive && focused ? 'yellow' : undefined, children: [isActive ? '▶ ' : ' ', _jsx(Text, { underline: true, children: firstChar }), restOfLabel, countText] }) }, tab.id));
20
+ }) }));
21
+ }
22
+ //# sourceMappingURL=Tabs.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"Tabs.js","sourceRoot":"","sources":["../../../src/components/Tabs.tsx"],"names":[],"mappings":";AACA,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,MAAM,KAAK,CAAC;AAoBhC,MAAM,CAAC,MAAM,IAAI,GAA0D;IACzE,EAAE,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,WAAW,EAAE,GAAG,EAAE;IAC/C,EAAE,EAAE,EAAE,WAAW,EAAE,KAAK,EAAE,WAAW,EAAE,WAAW,EAAE,GAAG,EAAE;IACzD,EAAE,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,EAAE,WAAW,EAAE,GAAG,EAAE;IACrD,EAAE,EAAE,EAAE,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,WAAW,EAAE,GAAG,EAAE;IACjD,EAAE,EAAE,EAAE,UAAU,EAAE,KAAK,EAAE,UAAU,EAAE,WAAW,EAAE,GAAG,EAAE;IACvD,EAAE,EAAE,EAAE,SAAS,EAAE,KAAK,EAAE,SAAS,EAAE,WAAW,EAAE,GAAG,EAAE;CACtD,CAAC;AAEF,MAAM,UAAU,IAAI,CAAC,EAAE,SAAS,EAAE,WAAW,EAAE,KAAK,EAAE,MAAM,GAAG,EAAE,EAAE,OAAO,GAAG,KAAK,EAAE,WAAW,GAAG,IAAI,EAAa;IACjH,MAAM,WAAW,GAAG,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,KAAK,SAAS,CAAC,CAAC;IAElF,OAAO,CACL,KAAC,GAAG,IACF,KAAK,EAAE,KAAK,EACZ,WAAW,EAAC,QAAQ,EACpB,SAAS,EAAE,KAAK,EAChB,UAAU,EAAE,KAAK,EACjB,WAAW,EAAE,KAAK,EAClB,YAAY,EAAE,IAAI,EAClB,aAAa,EAAC,KAAK,EACnB,cAAc,EAAC,eAAe,EAC9B,QAAQ,EAAC,MAAM,EACf,QAAQ,EAAE,CAAC,YAEV,WAAW,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE;YACvB,MAAM,QAAQ,GAAG,SAAS,KAAK,GAAG,CAAC,EAAE,CAAC;YACtC,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC7B,MAAM,SAAS,GAAG,KAAK,KAAK,SAAS,CAAC,CAAC,CAAC,KAAK,KAAK,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;YAC3D,MAAM,SAAS,GAAG,GAAG,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAC/B,MAAM,WAAW,GAAG,GAAG,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAEvC,OAAO,CACL,KAAC,GAAG,IAEF,UAAU,EAAE,CAAC,YAEb,MAAC,IAAI,IACH,IAAI,EAAE,QAAQ,EACd,KAAK,EAAE,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,EACjC,eAAe,EAAE,QAAQ,IAAI,OAAO,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,aAE1D,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,EACvB,KAAC,IAAI,IAAC,SAAS,kBAAE,SAAS,GAAQ,EACjC,WAAW,EAAE,SAAS,IAClB,IAXF,GAAG,CAAC,EAAE,CAYP,CACP,CAAC;QACJ,CAAC,CAAC,GACE,CACP,CAAC;AACJ,CAAC"}
@@ -0,0 +1,11 @@
1
+ import type { Client } from '@modelcontextprotocol/sdk/client/index.js';
2
+ interface ToolTestModalProps {
3
+ tool: any;
4
+ client: Client | null;
5
+ width: number;
6
+ height: number;
7
+ onClose: () => void;
8
+ }
9
+ export declare function ToolTestModal({ tool, client, width, height, onClose }: ToolTestModalProps): import("react/jsx-runtime").JSX.Element;
10
+ export {};
11
+ //# sourceMappingURL=ToolTestModal.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ToolTestModal.d.ts","sourceRoot":"","sources":["../../../src/components/ToolTestModal.tsx"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,2CAA2C,CAAC;AAIxE,UAAU,kBAAkB;IAC1B,IAAI,EAAE,GAAG,CAAC;IACV,MAAM,EAAE,MAAM,GAAG,IAAI,CAAC;IACtB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC;IACf,OAAO,EAAE,MAAM,IAAI,CAAC;CACrB;AAYD,wBAAgB,aAAa,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,OAAO,EAAE,EAAE,kBAAkB,2CA+NzF"}
@@ -0,0 +1,112 @@
1
+ import { jsx as _jsx, jsxs as _jsxs, Fragment as _Fragment } from "react/jsx-runtime";
2
+ import React, { useState } from 'react';
3
+ import { Box, Text, useInput } from 'ink';
4
+ import { Form } from 'ink-form';
5
+ import { schemaToForm } from '../utils/schemaToForm.js';
6
+ import { ScrollView } from 'ink-scroll-view';
7
+ export function ToolTestModal({ tool, client, width, height, onClose }) {
8
+ const [state, setState] = useState('form');
9
+ const [result, setResult] = useState(null);
10
+ const scrollViewRef = React.useRef(null);
11
+ // Use full terminal dimensions instead of passed dimensions
12
+ const [terminalDimensions, setTerminalDimensions] = React.useState({
13
+ width: process.stdout.columns || width,
14
+ height: process.stdout.rows || height,
15
+ });
16
+ React.useEffect(() => {
17
+ const updateDimensions = () => {
18
+ setTerminalDimensions({
19
+ width: process.stdout.columns || width,
20
+ height: process.stdout.rows || height,
21
+ });
22
+ };
23
+ process.stdout.on('resize', updateDimensions);
24
+ updateDimensions();
25
+ return () => {
26
+ process.stdout.off('resize', updateDimensions);
27
+ };
28
+ }, [width, height]);
29
+ const formStructure = tool?.inputSchema
30
+ ? schemaToForm(tool.inputSchema, tool.name || 'Unknown Tool')
31
+ : {
32
+ title: `Test Tool: ${tool?.name || 'Unknown'}`,
33
+ sections: [{ title: 'Parameters', fields: [] }],
34
+ };
35
+ // Reset state when modal closes
36
+ React.useEffect(() => {
37
+ return () => {
38
+ // Cleanup: reset state when component unmounts
39
+ setState('form');
40
+ setResult(null);
41
+ };
42
+ }, []);
43
+ // Handle escape to close
44
+ useInput((input, key) => {
45
+ if (key.escape) {
46
+ setState('form');
47
+ setResult(null);
48
+ onClose();
49
+ }
50
+ else if (state === 'results') {
51
+ // Allow scrolling in results view
52
+ if (key.downArrow) {
53
+ scrollViewRef.current?.scrollBy(1);
54
+ }
55
+ else if (key.upArrow) {
56
+ scrollViewRef.current?.scrollBy(-1);
57
+ }
58
+ else if (key.pageDown) {
59
+ const viewportHeight = scrollViewRef.current?.getViewportHeight() || 1;
60
+ scrollViewRef.current?.scrollBy(viewportHeight);
61
+ }
62
+ else if (key.pageUp) {
63
+ const viewportHeight = scrollViewRef.current?.getViewportHeight() || 1;
64
+ scrollViewRef.current?.scrollBy(-viewportHeight);
65
+ }
66
+ }
67
+ }, { isActive: true });
68
+ const handleFormSubmit = async (values) => {
69
+ if (!client || !tool)
70
+ return;
71
+ setState('loading');
72
+ const startTime = Date.now();
73
+ try {
74
+ const response = await client.callTool({
75
+ name: tool.name,
76
+ arguments: values,
77
+ });
78
+ const duration = Date.now() - startTime;
79
+ // Handle MCP SDK response format
80
+ const output = response.isError
81
+ ? { error: true, content: response.content }
82
+ : response.structuredContent || response.content || response;
83
+ setResult({
84
+ input: values,
85
+ output: response.isError ? null : output,
86
+ error: response.isError ? 'Tool returned an error' : undefined,
87
+ errorDetails: response.isError ? output : undefined,
88
+ duration,
89
+ });
90
+ setState('results');
91
+ }
92
+ catch (error) {
93
+ const duration = Date.now() - startTime;
94
+ const errorObj = error instanceof Error
95
+ ? { message: error.message, name: error.name, stack: error.stack }
96
+ : { error: String(error) };
97
+ setResult({
98
+ input: values,
99
+ output: null,
100
+ error: error instanceof Error ? error.message : 'Unknown error',
101
+ errorDetails: errorObj,
102
+ duration,
103
+ });
104
+ setState('results');
105
+ }
106
+ };
107
+ // Calculate modal dimensions - use almost full screen
108
+ const modalWidth = terminalDimensions.width - 2;
109
+ const modalHeight = terminalDimensions.height - 2;
110
+ return (_jsx(Box, { position: "absolute", width: terminalDimensions.width, height: terminalDimensions.height, flexDirection: "column", justifyContent: "center", alignItems: "center", children: _jsxs(Box, { width: modalWidth, height: modalHeight, borderStyle: "single", borderColor: "cyan", flexDirection: "column", paddingX: 1, paddingY: 1, backgroundColor: "black", children: [_jsxs(Box, { flexShrink: 0, marginBottom: 1, children: [_jsx(Text, { bold: true, color: "cyan", children: formStructure.title }), _jsx(Text, { children: " " }), _jsx(Text, { dimColor: true, children: "(Press ESC to close)" })] }), _jsxs(Box, { flexGrow: 1, flexDirection: "column", overflow: "hidden", children: [state === 'form' && (_jsx(Box, { flexGrow: 1, width: "100%", children: _jsx(Form, { form: formStructure, onSubmit: handleFormSubmit }) })), state === 'loading' && (_jsx(Box, { flexGrow: 1, justifyContent: "center", alignItems: "center", children: _jsx(Text, { color: "yellow", children: "Calling tool..." }) })), state === 'results' && result && (_jsx(Box, { flexGrow: 1, flexDirection: "column", overflow: "hidden", children: _jsxs(ScrollView, { ref: scrollViewRef, children: [_jsx(Box, { marginBottom: 1, flexShrink: 0, children: _jsxs(Text, { bold: true, color: "green", children: ["Duration: ", result.duration, "ms"] }) }), _jsxs(Box, { marginBottom: 1, flexShrink: 0, flexDirection: "column", children: [_jsx(Text, { bold: true, color: "cyan", children: "Input:" }), _jsx(Box, { paddingLeft: 2, children: _jsx(Text, { dimColor: true, children: JSON.stringify(result.input, null, 2) }) })] }), result.error ? (_jsxs(Box, { flexShrink: 0, flexDirection: "column", children: [_jsx(Text, { bold: true, color: "red", children: "Error:" }), _jsx(Box, { paddingLeft: 2, children: _jsx(Text, { color: "red", children: result.error }) }), result.errorDetails && (_jsxs(_Fragment, { children: [_jsx(Box, { marginTop: 1, children: _jsx(Text, { bold: true, color: "red", dimColor: true, children: "Error Details:" }) }), _jsx(Box, { paddingLeft: 2, children: _jsx(Text, { dimColor: true, children: JSON.stringify(result.errorDetails, null, 2) }) })] }))] })) : (_jsxs(Box, { flexShrink: 0, flexDirection: "column", children: [_jsx(Text, { bold: true, color: "green", children: "Output:" }), _jsx(Box, { paddingLeft: 2, children: _jsx(Text, { dimColor: true, children: JSON.stringify(result.output, null, 2) }) })] }))] }) }))] })] }) }));
111
+ }
112
+ //# sourceMappingURL=ToolTestModal.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ToolTestModal.js","sourceRoot":"","sources":["../../../src/components/ToolTestModal.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,14 @@
1
+ import type { Client } from '@modelcontextprotocol/sdk/client/index.js';
2
+ interface ToolsTabProps {
3
+ tools: any[];
4
+ client: Client | null;
5
+ width: number;
6
+ height: number;
7
+ onCountChange?: (count: number) => void;
8
+ focusedPane?: 'list' | 'details' | null;
9
+ onTestTool?: (tool: any) => void;
10
+ onViewDetails?: (tool: any) => void;
11
+ }
12
+ export declare function ToolsTab({ tools, client, width, height, onCountChange, focusedPane, onTestTool, onViewDetails }: ToolsTabProps): import("react/jsx-runtime").JSX.Element;
13
+ export {};
14
+ //# sourceMappingURL=ToolsTab.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ToolsTab.d.ts","sourceRoot":"","sources":["../../../src/components/ToolsTab.tsx"],"names":[],"mappings":"AAGA,OAAO,KAAK,EAAE,MAAM,EAAE,MAAM,2CAA2C,CAAC;AAExE,UAAU,aAAa;IACrB,KAAK,EAAE,GAAG,EAAE,CAAC;IACb,MAAM,EAAE,MAAM,GAAG,IAAI,CAAC;IACtB,KAAK,EAAE,MAAM,CAAC;IACd,MAAM,EAAE,MAAM,CAAC;IACf,aAAa,CAAC,EAAE,CAAC,KAAK,EAAE,MAAM,KAAK,IAAI,CAAC;IACxC,WAAW,CAAC,EAAE,MAAM,GAAG,SAAS,GAAG,IAAI,CAAC;IACxC,UAAU,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,KAAK,IAAI,CAAC;IACjC,aAAa,CAAC,EAAE,CAAC,IAAI,EAAE,GAAG,KAAK,IAAI,CAAC;CACrC;AAED,wBAAgB,QAAQ,CAAC,EAAE,KAAK,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,aAAa,EAAE,WAAkB,EAAE,UAAU,EAAE,aAAa,EAAE,EAAE,aAAa,2CA8KrI"}
@@ -0,0 +1,76 @@
1
+ import { jsxs as _jsxs, jsx as _jsx, Fragment as _Fragment } from "react/jsx-runtime";
2
+ import { useState, useEffect, useRef } from 'react';
3
+ import { Box, Text, useInput } from 'ink';
4
+ import { ScrollView } from 'ink-scroll-view';
5
+ export function ToolsTab({ tools, client, width, height, onCountChange, focusedPane = null, onTestTool, onViewDetails }) {
6
+ const [selectedIndex, setSelectedIndex] = useState(0);
7
+ const [error, setError] = useState(null);
8
+ const scrollViewRef = useRef(null);
9
+ const listWidth = Math.floor(width * 0.4);
10
+ const detailWidth = width - listWidth;
11
+ // Handle arrow key navigation when focused
12
+ useInput((input, key) => {
13
+ // Handle Enter key to test tool (works from both list and details)
14
+ if (key.return && selectedTool && client && onTestTool) {
15
+ onTestTool(selectedTool);
16
+ return;
17
+ }
18
+ if (focusedPane === 'list') {
19
+ // Navigate the list
20
+ if (key.upArrow && selectedIndex > 0) {
21
+ setSelectedIndex(selectedIndex - 1);
22
+ }
23
+ else if (key.downArrow && selectedIndex < tools.length - 1) {
24
+ setSelectedIndex(selectedIndex + 1);
25
+ }
26
+ return;
27
+ }
28
+ if (focusedPane === 'details') {
29
+ // Handle '+' key to view in full screen modal
30
+ if (input === '+' && selectedTool && onViewDetails) {
31
+ onViewDetails(selectedTool);
32
+ return;
33
+ }
34
+ // Scroll the details pane using ink-scroll-view
35
+ if (key.upArrow) {
36
+ scrollViewRef.current?.scrollBy(-1);
37
+ }
38
+ else if (key.downArrow) {
39
+ scrollViewRef.current?.scrollBy(1);
40
+ }
41
+ else if (key.pageUp) {
42
+ const viewportHeight = scrollViewRef.current?.getViewportHeight() || 1;
43
+ scrollViewRef.current?.scrollBy(-viewportHeight);
44
+ }
45
+ else if (key.pageDown) {
46
+ const viewportHeight = scrollViewRef.current?.getViewportHeight() || 1;
47
+ scrollViewRef.current?.scrollBy(viewportHeight);
48
+ }
49
+ }
50
+ }, { isActive: focusedPane === 'list' || focusedPane === 'details' });
51
+ // Helper to calculate content lines for a tool
52
+ const calculateToolContentLines = (tool) => {
53
+ let lines = 1; // Name
54
+ if (tool.description)
55
+ lines += tool.description.split('\n').length + 1;
56
+ if (tool.inputSchema) {
57
+ const schemaStr = JSON.stringify(tool.inputSchema, null, 2);
58
+ lines += schemaStr.split('\n').length + 2; // +2 for "Input Schema:" label
59
+ }
60
+ return lines;
61
+ };
62
+ // Reset scroll when selection changes
63
+ useEffect(() => {
64
+ scrollViewRef.current?.scrollTo(0);
65
+ }, [selectedIndex]);
66
+ // Reset selected index when tools array changes (different server)
67
+ useEffect(() => {
68
+ setSelectedIndex(0);
69
+ }, [tools]);
70
+ const selectedTool = tools[selectedIndex] || null;
71
+ return (_jsxs(Box, { flexDirection: "row", width: width, height: height, children: [_jsxs(Box, { width: listWidth, height: height, borderStyle: "single", borderTop: false, borderBottom: false, borderLeft: false, borderRight: true, flexDirection: "column", paddingX: 1, children: [_jsx(Box, { paddingY: 1, children: _jsxs(Text, { bold: true, backgroundColor: focusedPane === 'list' ? 'yellow' : undefined, children: ["Tools (", tools.length, ")"] }) }), error ? (_jsx(Box, { paddingY: 1, children: _jsx(Text, { color: "red", children: error }) })) : tools.length === 0 ? (_jsx(Box, { paddingY: 1, children: _jsx(Text, { dimColor: true, children: "No tools available" }) })) : (_jsx(Box, { flexDirection: "column", flexGrow: 1, children: tools.map((tool, index) => {
72
+ const isSelected = index === selectedIndex;
73
+ return (_jsx(Box, { paddingY: 0, children: _jsxs(Text, { children: [isSelected ? '▶ ' : ' ', tool.name || `Tool ${index + 1}`] }) }, tool.name || index));
74
+ }) }))] }), _jsx(Box, { width: detailWidth, height: height, paddingX: 1, flexDirection: "column", overflow: "hidden", children: selectedTool ? (_jsxs(_Fragment, { children: [_jsxs(Box, { flexShrink: 0, flexDirection: "row", justifyContent: "space-between", paddingTop: 1, children: [_jsx(Text, { bold: true, backgroundColor: focusedPane === 'details' ? 'yellow' : undefined, color: "cyan", children: selectedTool.name }), client && (_jsx(Text, { children: _jsx(Text, { color: "cyan", bold: true, children: "[Enter to Test]" }) }))] }), _jsxs(ScrollView, { ref: scrollViewRef, height: height - 5, children: [selectedTool.description && (_jsx(_Fragment, { children: selectedTool.description.split('\n').map((line, idx) => (_jsx(Box, { marginTop: idx === 0 ? 1 : 0, flexShrink: 0, children: _jsx(Text, { dimColor: true, children: line }) }, `desc-${idx}`))) })), selectedTool.inputSchema && (_jsxs(_Fragment, { children: [_jsx(Box, { marginTop: 1, flexShrink: 0, children: _jsx(Text, { bold: true, children: "Input Schema:" }) }), JSON.stringify(selectedTool.inputSchema, null, 2).split('\n').map((line, idx) => (_jsx(Box, { marginTop: idx === 0 ? 1 : 0, paddingLeft: 2, flexShrink: 0, children: _jsx(Text, { dimColor: true, children: line }) }, `schema-${idx}`)))] }))] }), focusedPane === 'details' && (_jsx(Box, { flexShrink: 0, height: 1, justifyContent: "center", backgroundColor: "gray", children: _jsx(Text, { bold: true, color: "white", children: "\u2191/\u2193 to scroll, + to zoom" }) }))] })) : (_jsx(Box, { paddingY: 1, flexShrink: 0, children: _jsx(Text, { dimColor: true, children: "Select a tool to view details" }) })) })] }));
75
+ }
76
+ //# sourceMappingURL=ToolsTab.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ToolsTab.js","sourceRoot":"","sources":["../../../src/components/ToolsTab.tsx"],"names":[],"mappings":"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"}
@@ -0,0 +1,41 @@
1
+ import { Client } from '@modelcontextprotocol/sdk/client/index.js';
2
+ import type { MCPServerConfig } from '../types.js';
3
+ import type { Transport, TransportSendOptions } from '@modelcontextprotocol/sdk/shared/transport.js';
4
+ import type { JSONRPCMessage, MessageExtraInfo } from '@modelcontextprotocol/sdk/types.js';
5
+ import type { JSONRPCRequest, JSONRPCNotification, JSONRPCResultResponse, JSONRPCErrorResponse } from '@modelcontextprotocol/sdk/types.js';
6
+ export type ConnectionStatus = 'disconnected' | 'connecting' | 'connected' | 'error';
7
+ export interface ServerConnection {
8
+ name: string;
9
+ config: MCPServerConfig;
10
+ client: Client | null;
11
+ status: ConnectionStatus;
12
+ error: string | null;
13
+ }
14
+ export interface MessageTrackingCallbacks {
15
+ trackRequest?: (message: JSONRPCRequest) => void;
16
+ trackResponse?: (message: JSONRPCResultResponse | JSONRPCErrorResponse) => void;
17
+ trackNotification?: (message: JSONRPCNotification) => void;
18
+ }
19
+ declare class LoggingProxyTransport implements Transport {
20
+ private baseTransport;
21
+ private callbacks;
22
+ constructor(baseTransport: Transport, callbacks: MessageTrackingCallbacks);
23
+ start(): Promise<void>;
24
+ send(message: JSONRPCMessage, options?: TransportSendOptions): Promise<void>;
25
+ close(): Promise<void>;
26
+ get onclose(): (() => void) | undefined;
27
+ set onclose(handler: (() => void) | undefined);
28
+ get onerror(): ((error: Error) => void) | undefined;
29
+ set onerror(handler: ((error: Error) => void) | undefined);
30
+ get onmessage(): (<T extends JSONRPCMessage>(message: T, extra?: MessageExtraInfo) => void) | undefined;
31
+ set onmessage(handler: (<T extends JSONRPCMessage>(message: T, extra?: MessageExtraInfo) => void) | undefined);
32
+ get sessionId(): string | undefined;
33
+ get setProtocolVersion(): ((version: string) => void) | undefined;
34
+ }
35
+ export { LoggingProxyTransport };
36
+ export declare function useMCPClient(serverName: string | null, config: MCPServerConfig | null, messageTracking?: MessageTrackingCallbacks): {
37
+ connection: ServerConnection | null;
38
+ connect: () => Promise<Client | null>;
39
+ disconnect: () => Promise<void>;
40
+ };
41
+ //# sourceMappingURL=useMCPClient.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useMCPClient.d.ts","sourceRoot":"","sources":["../../../src/hooks/useMCPClient.ts"],"names":[],"mappings":"AACA,OAAO,EAAE,MAAM,EAAE,MAAM,2CAA2C,CAAC;AAEnE,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;AACnD,OAAO,KAAK,EAAE,SAAS,EAAE,oBAAoB,EAAE,MAAM,+CAA+C,CAAC;AACrG,OAAO,KAAK,EAAE,cAAc,EAAE,gBAAgB,EAAE,MAAM,oCAAoC,CAAC;AAC3F,OAAO,KAAK,EACV,cAAc,EACd,mBAAmB,EACnB,qBAAqB,EACrB,oBAAoB,EACrB,MAAM,oCAAoC,CAAC;AAE5C,MAAM,MAAM,gBAAgB,GAAG,cAAc,GAAG,YAAY,GAAG,WAAW,GAAG,OAAO,CAAC;AAErF,MAAM,WAAW,gBAAgB;IAC/B,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,eAAe,CAAC;IACxB,MAAM,EAAE,MAAM,GAAG,IAAI,CAAC;IACtB,MAAM,EAAE,gBAAgB,CAAC;IACzB,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;CACtB;AAED,MAAM,WAAW,wBAAwB;IACvC,YAAY,CAAC,EAAE,CAAC,OAAO,EAAE,cAAc,KAAK,IAAI,CAAC;IACjD,aAAa,CAAC,EAAE,CAAC,OAAO,EAAE,qBAAqB,GAAG,oBAAoB,KAAK,IAAI,CAAC;IAChF,iBAAiB,CAAC,EAAE,CAAC,OAAO,EAAE,mBAAmB,KAAK,IAAI,CAAC;CAC5D;AAGD,cAAM,qBAAsB,YAAW,SAAS;IAE5C,OAAO,CAAC,aAAa;IACrB,OAAO,CAAC,SAAS;gBADT,aAAa,EAAE,SAAS,EACxB,SAAS,EAAE,wBAAwB;IAGvC,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAItB,IAAI,CAAC,OAAO,EAAE,cAAc,EAAE,OAAO,CAAC,EAAE,oBAAoB,GAAG,OAAO,CAAC,IAAI,CAAC;IAQ5E,KAAK,IAAI,OAAO,CAAC,IAAI,CAAC;IAI5B,IAAI,OAAO,IAAI,CAAC,MAAM,IAAI,CAAC,GAAG,SAAS,CAEtC;IAED,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC,MAAM,IAAI,CAAC,GAAG,SAAS,EAE5C;IAED,IAAI,OAAO,IAAI,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,SAAS,CAElD;IAED,IAAI,OAAO,CAAC,OAAO,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,KAAK,IAAI,CAAC,GAAG,SAAS,EAExD;IAED,IAAI,SAAS,IAAI,CAAC,CAAC,CAAC,SAAS,cAAc,EAAE,OAAO,EAAE,CAAC,EAAE,KAAK,CAAC,EAAE,gBAAgB,KAAK,IAAI,CAAC,GAAG,SAAS,CAEtG;IAED,IAAI,SAAS,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC,SAAS,cAAc,EAAE,OAAO,EAAE,CAAC,EAAE,KAAK,CAAC,EAAE,gBAAgB,KAAK,IAAI,CAAC,GAAG,SAAS,EAuB5G;IAED,IAAI,SAAS,IAAI,MAAM,GAAG,SAAS,CAElC;IAED,IAAI,kBAAkB,IAAI,CAAC,CAAC,OAAO,EAAE,MAAM,KAAK,IAAI,CAAC,GAAG,SAAS,CAEhE;CACF;AAGD,OAAO,EAAE,qBAAqB,EAAE,CAAC;AAEjC,wBAAgB,YAAY,CAC1B,UAAU,EAAE,MAAM,GAAG,IAAI,EACzB,MAAM,EAAE,eAAe,GAAG,IAAI,EAC9B,eAAe,CAAC,EAAE,wBAAwB;;mBAYJ,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;;EAyG7D"}
@@ -0,0 +1,179 @@
1
+ import { useState, useRef, useCallback } from 'react';
2
+ import { Client } from '@modelcontextprotocol/sdk/client/index.js';
3
+ import { StdioClientTransport } from '@modelcontextprotocol/sdk/client/stdio.js';
4
+ // Proxy Transport that intercepts all messages for logging/tracking
5
+ class LoggingProxyTransport {
6
+ baseTransport;
7
+ callbacks;
8
+ constructor(baseTransport, callbacks) {
9
+ this.baseTransport = baseTransport;
10
+ this.callbacks = callbacks;
11
+ }
12
+ async start() {
13
+ return this.baseTransport.start();
14
+ }
15
+ async send(message, options) {
16
+ // Track outgoing requests (only requests have a method and are sent by the client)
17
+ if ('method' in message && 'id' in message) {
18
+ this.callbacks.trackRequest?.(message);
19
+ }
20
+ return this.baseTransport.send(message, options);
21
+ }
22
+ async close() {
23
+ return this.baseTransport.close();
24
+ }
25
+ get onclose() {
26
+ return this.baseTransport.onclose;
27
+ }
28
+ set onclose(handler) {
29
+ this.baseTransport.onclose = handler;
30
+ }
31
+ get onerror() {
32
+ return this.baseTransport.onerror;
33
+ }
34
+ set onerror(handler) {
35
+ this.baseTransport.onerror = handler;
36
+ }
37
+ get onmessage() {
38
+ return this.baseTransport.onmessage;
39
+ }
40
+ set onmessage(handler) {
41
+ if (handler) {
42
+ // Wrap the handler to track incoming messages
43
+ this.baseTransport.onmessage = (message, extra) => {
44
+ // Track incoming messages
45
+ if ('id' in message && message.id !== null && message.id !== undefined) {
46
+ // Check if it's a response (has 'result' or 'error' property)
47
+ if ('result' in message || 'error' in message) {
48
+ this.callbacks.trackResponse?.(message);
49
+ }
50
+ else if ('method' in message) {
51
+ // This is a request coming from the server
52
+ this.callbacks.trackRequest?.(message);
53
+ }
54
+ }
55
+ else if ('method' in message) {
56
+ // Notification (no ID, has method)
57
+ this.callbacks.trackNotification?.(message);
58
+ }
59
+ // Call the original handler
60
+ handler(message, extra);
61
+ };
62
+ }
63
+ else {
64
+ this.baseTransport.onmessage = undefined;
65
+ }
66
+ }
67
+ get sessionId() {
68
+ return this.baseTransport.sessionId;
69
+ }
70
+ get setProtocolVersion() {
71
+ return this.baseTransport.setProtocolVersion;
72
+ }
73
+ }
74
+ // Export LoggingProxyTransport for use in other hooks
75
+ export { LoggingProxyTransport };
76
+ export function useMCPClient(serverName, config, messageTracking) {
77
+ const [connection, setConnection] = useState(null);
78
+ const clientRef = useRef(null);
79
+ const messageTrackingRef = useRef(messageTracking);
80
+ const isMountedRef = useRef(true);
81
+ // Update ref when messageTracking changes
82
+ if (messageTracking) {
83
+ messageTrackingRef.current = messageTracking;
84
+ }
85
+ const connect = useCallback(async () => {
86
+ if (!serverName || !config) {
87
+ return null;
88
+ }
89
+ // If already connected, return existing client
90
+ if (clientRef.current && connection?.status === 'connected') {
91
+ return clientRef.current;
92
+ }
93
+ setConnection({
94
+ name: serverName,
95
+ config,
96
+ client: null,
97
+ status: 'connecting',
98
+ error: null,
99
+ });
100
+ try {
101
+ // Only support stdio in useMCPClient hook (legacy support)
102
+ // For full transport support, use the transport creation in App.tsx
103
+ if ('type' in config && config.type !== 'stdio' && config.type !== undefined) {
104
+ throw new Error(`Transport type ${config.type} not supported in useMCPClient hook`);
105
+ }
106
+ const stdioConfig = config;
107
+ const baseTransport = new StdioClientTransport({
108
+ command: stdioConfig.command,
109
+ args: stdioConfig.args || [],
110
+ env: stdioConfig.env,
111
+ });
112
+ // Wrap with proxy transport if message tracking is enabled
113
+ const transport = messageTrackingRef.current
114
+ ? new LoggingProxyTransport(baseTransport, messageTrackingRef.current)
115
+ : baseTransport;
116
+ const client = new Client({
117
+ name: 'mcp-inspect',
118
+ version: '1.0.0',
119
+ }, {
120
+ capabilities: {},
121
+ });
122
+ await client.connect(transport);
123
+ if (!isMountedRef.current) {
124
+ await client.close();
125
+ return null;
126
+ }
127
+ clientRef.current = client;
128
+ setConnection({
129
+ name: serverName,
130
+ config,
131
+ client,
132
+ status: 'connected',
133
+ error: null,
134
+ });
135
+ return client;
136
+ }
137
+ catch (error) {
138
+ if (!isMountedRef.current)
139
+ return null;
140
+ setConnection({
141
+ name: serverName,
142
+ config,
143
+ client: null,
144
+ status: 'error',
145
+ error: error instanceof Error ? error.message : 'Unknown error',
146
+ });
147
+ return null;
148
+ }
149
+ }, [serverName, config, connection?.status]);
150
+ const disconnect = useCallback(async () => {
151
+ if (clientRef.current) {
152
+ try {
153
+ await clientRef.current.close();
154
+ }
155
+ catch (error) {
156
+ // Ignore errors on close
157
+ }
158
+ clientRef.current = null;
159
+ }
160
+ if (serverName && config) {
161
+ setConnection({
162
+ name: serverName,
163
+ config,
164
+ client: null,
165
+ status: 'disconnected',
166
+ error: null,
167
+ });
168
+ }
169
+ else {
170
+ setConnection(null);
171
+ }
172
+ }, [serverName, config]);
173
+ return {
174
+ connection,
175
+ connect,
176
+ disconnect,
177
+ };
178
+ }
179
+ //# sourceMappingURL=useMCPClient.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useMCPClient.js","sourceRoot":"","sources":["../../../src/hooks/useMCPClient.ts"],"names":[],"mappings":"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"}