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.
- package/LICENSE +21 -0
- package/README.md +107 -0
- package/dist/src/App.d.ts +7 -0
- package/dist/src/App.d.ts.map +1 -0
- package/dist/src/App.js +591 -0
- package/dist/src/App.js.map +1 -0
- package/dist/src/components/DetailsModal.d.ts +11 -0
- package/dist/src/components/DetailsModal.d.ts.map +1 -0
- package/dist/src/components/DetailsModal.js +50 -0
- package/dist/src/components/DetailsModal.js.map +1 -0
- package/dist/src/components/HistoryTab.d.ts +13 -0
- package/dist/src/components/HistoryTab.d.ts.map +1 -0
- package/dist/src/components/HistoryTab.js +122 -0
- package/dist/src/components/HistoryTab.js.map +1 -0
- package/dist/src/components/InfoTab.d.ts +13 -0
- package/dist/src/components/InfoTab.d.ts.map +1 -0
- package/dist/src/components/InfoTab.js +28 -0
- package/dist/src/components/InfoTab.js.map +1 -0
- package/dist/src/components/NotificationsTab.d.ts +13 -0
- package/dist/src/components/NotificationsTab.d.ts.map +1 -0
- package/dist/src/components/NotificationsTab.js +37 -0
- package/dist/src/components/NotificationsTab.js.map +1 -0
- package/dist/src/components/PromptsTab.d.ts +13 -0
- package/dist/src/components/PromptsTab.d.ts.map +1 -0
- package/dist/src/components/PromptsTab.js +60 -0
- package/dist/src/components/PromptsTab.js.map +1 -0
- package/dist/src/components/ResourcesTab.d.ts +13 -0
- package/dist/src/components/ResourcesTab.d.ts.map +1 -0
- package/dist/src/components/ResourcesTab.js +60 -0
- package/dist/src/components/ResourcesTab.js.map +1 -0
- package/dist/src/components/Tabs.d.ts +24 -0
- package/dist/src/components/Tabs.d.ts.map +1 -0
- package/dist/src/components/Tabs.js +22 -0
- package/dist/src/components/Tabs.js.map +1 -0
- package/dist/src/components/ToolTestModal.d.ts +11 -0
- package/dist/src/components/ToolTestModal.d.ts.map +1 -0
- package/dist/src/components/ToolTestModal.js +112 -0
- package/dist/src/components/ToolTestModal.js.map +1 -0
- package/dist/src/components/ToolsTab.d.ts +14 -0
- package/dist/src/components/ToolsTab.d.ts.map +1 -0
- package/dist/src/components/ToolsTab.js +76 -0
- package/dist/src/components/ToolsTab.js.map +1 -0
- package/dist/src/hooks/useMCPClient.d.ts +41 -0
- package/dist/src/hooks/useMCPClient.d.ts.map +1 -0
- package/dist/src/hooks/useMCPClient.js +179 -0
- package/dist/src/hooks/useMCPClient.js.map +1 -0
- package/dist/src/hooks/useMessageTracking.d.ts +9 -0
- package/dist/src/hooks/useMessageTracking.d.ts.map +1 -0
- package/dist/src/hooks/useMessageTracking.js +124 -0
- package/dist/src/hooks/useMessageTracking.js.map +1 -0
- package/dist/src/types/focus.d.ts +2 -0
- package/dist/src/types/focus.d.ts.map +1 -0
- package/dist/src/types/focus.js +2 -0
- package/dist/src/types/focus.js.map +1 -0
- package/dist/src/types/messages.d.ts +14 -0
- package/dist/src/types/messages.d.ts.map +1 -0
- package/dist/src/types/messages.js +2 -0
- package/dist/src/types/messages.js.map +1 -0
- package/dist/src/types.d.ts +48 -0
- package/dist/src/types.d.ts.map +1 -0
- package/dist/src/types.js +2 -0
- package/dist/src/types.js.map +1 -0
- package/dist/src/utils/schemaToForm.d.ts +9 -0
- package/dist/src/utils/schemaToForm.d.ts.map +1 -0
- package/dist/src/utils/schemaToForm.js +107 -0
- package/dist/src/utils/schemaToForm.js.map +1 -0
- package/dist/tui.d.ts +3 -0
- package/dist/tui.d.ts.map +1 -0
- package/dist/tui.js +43 -0
- package/dist/tui.js.map +1 -0
- package/package.json +58 -0
- 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":";AAAA,OAAc,EAAE,QAAQ,EAAE,SAAS,EAAE,MAAM,EAAE,MAAM,OAAO,CAAC;AAC3D,OAAO,EAAE,GAAG,EAAE,IAAI,EAAE,QAAQ,EAAY,MAAM,KAAK,CAAC;AACpD,OAAO,EAAE,UAAU,EAAsB,MAAM,iBAAiB,CAAC;AAajE,MAAM,UAAU,YAAY,CAAC,EAAE,SAAS,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,aAAa,EAAE,WAAW,GAAG,IAAI,EAAE,aAAa,EAAqB;IACpI,MAAM,CAAC,aAAa,EAAE,gBAAgB,CAAC,GAAG,QAAQ,CAAS,CAAC,CAAC,CAAC;IAC9D,MAAM,CAAC,KAAK,EAAE,QAAQ,CAAC,GAAG,QAAQ,CAAgB,IAAI,CAAC,CAAC;IACxD,MAAM,aAAa,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC;IAElD,2CAA2C;IAC3C,QAAQ,CAAC,CAAC,KAAa,EAAE,GAAQ,EAAE,EAAE;QACnC,IAAI,WAAW,KAAK,MAAM,EAAE,CAAC;YAC3B,oBAAoB;YACpB,IAAI,GAAG,CAAC,OAAO,IAAI,aAAa,GAAG,CAAC,EAAE,CAAC;gBACrC,gBAAgB,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;YACtC,CAAC;iBAAM,IAAI,GAAG,CAAC,SAAS,IAAI,aAAa,GAAG,SAAS,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBACjE,gBAAgB,CAAC,aAAa,GAAG,CAAC,CAAC,CAAC;YACtC,CAAC;YACD,OAAO;QACT,CAAC;QAED,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC;YAC9B,8CAA8C;YAC9C,IAAI,KAAK,KAAK,GAAG,IAAI,gBAAgB,IAAI,aAAa,EAAE,CAAC;gBACvD,aAAa,CAAC,gBAAgB,CAAC,CAAC;gBAChC,OAAO;YACT,CAAC;YAED,gDAAgD;YAChD,IAAI,GAAG,CAAC,OAAO,EAAE,CAAC;gBAChB,aAAa,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;YACtC,CAAC;iBAAM,IAAI,GAAG,CAAC,SAAS,EAAE,CAAC;gBACzB,aAAa,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC;YACrC,CAAC;iBAAM,IAAI,GAAG,CAAC,MAAM,EAAE,CAAC;gBACtB,MAAM,cAAc,GAAG,aAAa,CAAC,OAAO,EAAE,iBAAiB,EAAE,IAAI,CAAC,CAAC;gBACvE,aAAa,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,cAAc,CAAC,CAAC;YACnD,CAAC;iBAAM,IAAI,GAAG,CAAC,QAAQ,EAAE,CAAC;gBACxB,MAAM,cAAc,GAAG,aAAa,CAAC,OAAO,EAAE,iBAAiB,EAAE,IAAI,CAAC,CAAC;gBACvE,aAAa,CAAC,OAAO,EAAE,QAAQ,CAAC,cAAc,CAAC,CAAC;YAClD,CAAC;QACH,CAAC;IACH,CAAC,EAAE,EAAE,QAAQ,EAAE,WAAW,KAAK,MAAM,IAAI,WAAW,KAAK,SAAS,EAAE,CAAC,CAAC;IAEtE,sCAAsC;IACtC,SAAS,CAAC,GAAG,EAAE;QACb,aAAa,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC,CAAC,CAAC;IACrC,CAAC,EAAE,CAAC,aAAa,CAAC,CAAC,CAAC;IAEpB,uEAAuE;IACvE,SAAS,CAAC,GAAG,EAAE;QACb,gBAAgB,CAAC,CAAC,CAAC,CAAC;IACtB,CAAC,EAAE,CAAC,SAAS,CAAC,CAAC,CAAC;IAEhB,MAAM,gBAAgB,GAAG,SAAS,CAAC,aAAa,CAAC,IAAI,IAAI,CAAC;IAE1D,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,CAAC,CAAC;IAC1C,MAAM,WAAW,GAAG,KAAK,GAAG,SAAS,CAAC;IAEtC,OAAO,CACL,MAAC,GAAG,IAAC,aAAa,EAAC,KAAK,EAAC,KAAK,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,aAEnD,MAAC,GAAG,IACF,KAAK,EAAE,SAAS,EAChB,MAAM,EAAE,MAAM,EACd,WAAW,EAAC,QAAQ,EACpB,SAAS,EAAE,KAAK,EAChB,YAAY,EAAE,KAAK,EACnB,UAAU,EAAE,KAAK,EACjB,WAAW,EAAE,IAAI,EACjB,aAAa,EAAC,QAAQ,EACtB,QAAQ,EAAE,CAAC,aAEX,KAAC,GAAG,IAAC,QAAQ,EAAE,CAAC,YACd,MAAC,IAAI,IAAC,IAAI,QAAC,eAAe,EAAE,WAAW,KAAK,MAAM,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,4BAAc,SAAS,CAAC,MAAM,SAAS,GAC5G,EACL,KAAK,CAAC,CAAC,CAAC,CACP,KAAC,GAAG,IAAC,QAAQ,EAAE,CAAC,YACd,KAAC,IAAI,IAAC,KAAK,EAAC,KAAK,YAAE,KAAK,GAAQ,GAC5B,CACP,CAAC,CAAC,CAAC,SAAS,CAAC,MAAM,KAAK,CAAC,CAAC,CAAC,CAAC,CAC3B,KAAC,GAAG,IAAC,QAAQ,EAAE,CAAC,YACd,KAAC,IAAI,IAAC,QAAQ,6CAA8B,GACxC,CACP,CAAC,CAAC,CAAC,CACF,KAAC,GAAG,IAAC,aAAa,EAAC,QAAQ,EAAC,QAAQ,EAAE,CAAC,YACpC,SAAS,CAAC,GAAG,CAAC,CAAC,QAAQ,EAAE,KAAK,EAAE,EAAE;4BACjC,MAAM,UAAU,GAAG,KAAK,KAAK,aAAa,CAAC;4BAC3C,OAAO,CACL,KAAC,GAAG,IAA6B,QAAQ,EAAE,CAAC,YAC1C,MAAC,IAAI,eACF,UAAU,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,EACxB,QAAQ,CAAC,IAAI,IAAI,QAAQ,CAAC,GAAG,IAAI,YAAY,KAAK,GAAG,CAAC,EAAE,IACpD,IAJC,QAAQ,CAAC,GAAG,IAAI,KAAK,CAKzB,CACP,CAAC;wBACJ,CAAC,CAAC,GACE,CACP,IACG,EAGN,KAAC,GAAG,IAAC,KAAK,EAAE,WAAW,EAAE,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,EAAE,aAAa,EAAC,QAAQ,EAAC,QAAQ,EAAC,QAAQ,YAC3F,gBAAgB,CAAC,CAAC,CAAC,CAClB,8BAEE,KAAC,GAAG,IAAC,UAAU,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,YAC/B,KAAC,IAAI,IAAC,IAAI,QAAC,eAAe,EAAE,WAAW,KAAK,SAAS,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,EAAE,KAAK,EAAC,MAAM,YACvF,gBAAgB,CAAC,IAAI,IAAI,gBAAgB,CAAC,GAAG,GACzC,GACH,EAGN,MAAC,UAAU,IAAC,GAAG,EAAE,aAAa,EAAE,MAAM,EAAE,MAAM,GAAG,CAAC,aAE/C,gBAAgB,CAAC,WAAW,IAAI,CAC/B,4BACG,gBAAgB,CAAC,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,IAAY,EAAE,GAAW,EAAE,EAAE,CAAC,CAC3E,KAAC,GAAG,IAAqB,SAAS,EAAE,GAAG,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,UAAU,EAAE,CAAC,YAClE,KAAC,IAAI,IAAC,QAAQ,kBAAE,IAAI,GAAQ,IADpB,QAAQ,GAAG,EAAE,CAEjB,CACP,CAAC,GACD,CACJ,EAGA,gBAAgB,CAAC,GAAG,IAAI,CACvB,KAAC,GAAG,IAAC,SAAS,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,YAC9B,MAAC,IAAI,IAAC,QAAQ,4BAAO,gBAAgB,CAAC,GAAG,IAAQ,GAC7C,CACP,EAGA,gBAAgB,CAAC,QAAQ,IAAI,CAC5B,KAAC,GAAG,IAAC,SAAS,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,YAC9B,MAAC,IAAI,IAAC,QAAQ,kCAAa,gBAAgB,CAAC,QAAQ,IAAQ,GACxD,CACP,IACU,EAGZ,WAAW,KAAK,SAAS,IAAI,CAC5B,KAAC,GAAG,IAAC,UAAU,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,cAAc,EAAC,QAAQ,EAAC,eAAe,EAAC,MAAM,YAC3E,KAAC,IAAI,IAAC,IAAI,QAAC,KAAK,EAAC,OAAO,mDAAgC,GACpD,CACP,IACA,CACJ,CAAC,CAAC,CAAC,CACF,KAAC,GAAG,IAAC,QAAQ,EAAE,CAAC,EAAE,UAAU,EAAE,CAAC,YAC7B,KAAC,IAAI,IAAC,QAAQ,wDAAyC,GACnD,CACP,GACG,IACF,CACP,CAAC;AACJ,CAAC"}
|
|
@@ -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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"}
|