@alpic-ai/insights 0.0.0-dev.a4748b9 → 0.0.0-dev.a4d6925
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/dist/index.d.mts +40 -15
- package/dist/index.mjs +104 -16
- package/package.json +1 -1
package/dist/index.d.mts
CHANGED
|
@@ -1,21 +1,25 @@
|
|
|
1
1
|
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
|
|
2
2
|
import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
|
|
3
3
|
|
|
4
|
-
//#region src/
|
|
4
|
+
//#region src/intent-middleware.d.ts
|
|
5
5
|
interface PromptData {
|
|
6
6
|
toolName: string;
|
|
7
7
|
userPrompt: string;
|
|
8
8
|
}
|
|
9
|
-
interface
|
|
9
|
+
interface IntentMiddlewareOptions {
|
|
10
10
|
handler?: (prompt: PromptData) => Promise<void> | void;
|
|
11
11
|
/**
|
|
12
|
-
*
|
|
13
|
-
*
|
|
14
|
-
* Use this when the tool already has a parameter (e.g. `query`, `question`) that conveys user intent.
|
|
15
|
-
* For tools in this mapping, the synthetic `user_intent` field is not injected into the schema and
|
|
16
|
-
* the field's value is read straight from the tool call arguments without being stripped.
|
|
12
|
+
* If provided, only these tool names will have the `user_intent` field injected and their
|
|
13
|
+
* prompts captured. All other tools are left untouched.
|
|
17
14
|
*/
|
|
18
|
-
|
|
15
|
+
tools?: string[];
|
|
16
|
+
/**
|
|
17
|
+
* Mapping of tool names to argument names whose values should be captured as the intent.
|
|
18
|
+
* Use this when the tool already has an argument (e.g. `query`, `question`) that conveys user
|
|
19
|
+
* intent. For tools in this mapping, the synthetic `user_intent` argument is not injected into the
|
|
20
|
+
* schema and the argument's value is read straight from the tool call arguments without being stripped.
|
|
21
|
+
*/
|
|
22
|
+
argumentNameOverride?: Record<string, string>;
|
|
19
23
|
}
|
|
20
24
|
/**
|
|
21
25
|
* Structurally compatible with `skybridge/server`'s `McpMiddlewareFn` so
|
|
@@ -31,20 +35,41 @@ type McpMiddlewareFn = (request: {
|
|
|
31
35
|
* they were. The LLM fills in `user_intent` from the original user message
|
|
32
36
|
* (the server has no other way to access it).
|
|
33
37
|
*/
|
|
34
|
-
declare function
|
|
38
|
+
declare function intentMiddleware(options?: IntentMiddlewareOptions): McpMiddlewareFn;
|
|
35
39
|
//#endregion
|
|
36
|
-
//#region src/capture-
|
|
40
|
+
//#region src/capture-intents.d.ts
|
|
37
41
|
/**
|
|
38
|
-
* Captures the user's natural-language
|
|
42
|
+
* Captures the user's natural-language intent behind each tool call on a vanilla
|
|
39
43
|
* `@modelcontextprotocol/sdk` server. Accepts the high-level `McpServer` or the
|
|
40
44
|
* low-level `Server` and patches the `tools/list` and `tools/call` request
|
|
41
|
-
* handlers to surface the captured
|
|
42
|
-
* `
|
|
45
|
+
* handlers to surface the captured intent via `options.handler` (or, when
|
|
46
|
+
* `ALPIC_INTENT_META_KEY` is set, via the response `_meta`).
|
|
43
47
|
*
|
|
44
48
|
* Already-registered handlers are wrapped immediately; future registrations
|
|
45
49
|
* (e.g. tools added after this call) are wrapped via a `Map.set` proxy so order
|
|
46
50
|
* of calls relative to `registerTool` does not matter.
|
|
47
51
|
*/
|
|
48
|
-
declare const
|
|
52
|
+
declare const captureIntents: (server: McpServer | Server, options?: IntentMiddlewareOptions) => void;
|
|
53
|
+
//#endregion
|
|
54
|
+
//#region src/feedback-middleware.d.ts
|
|
55
|
+
interface FeedbackData {
|
|
56
|
+
content: string;
|
|
57
|
+
source: "model" | "user";
|
|
58
|
+
}
|
|
59
|
+
interface FeedbackMiddlewareOptions {
|
|
60
|
+
/**
|
|
61
|
+
* Custom handler invoked with the user's feedback. When provided, the middleware still attaches the feedback to the response `_meta`.
|
|
62
|
+
* The handler runs **in addition to** Alpic's dashboard delivery, feedback are still captured when deployed on Alpic.
|
|
63
|
+
*/
|
|
64
|
+
handler?: (feedback: FeedbackData) => Promise<void> | void;
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Lets MCP server builders collect qualitative feedback from end users about their
|
|
68
|
+
* tool/server. Injects a `send_feedback` tool at `tools/list` time and intercepts calls
|
|
69
|
+
* to it at `tools/call` time. The tool has no handler on the server. The middleware
|
|
70
|
+
* short-circuits the call and either invokes the provided `handler` or attaches the
|
|
71
|
+
* feedback to the response `_meta`.
|
|
72
|
+
*/
|
|
73
|
+
declare function feedbackMiddleware(options?: FeedbackMiddlewareOptions): McpMiddlewareFn;
|
|
49
74
|
//#endregion
|
|
50
|
-
export { type
|
|
75
|
+
export { type FeedbackData, type FeedbackMiddlewareOptions, type IntentMiddlewareOptions, type McpMiddlewareFn, type PromptData, captureIntents, feedbackMiddleware, intentMiddleware };
|
package/dist/index.mjs
CHANGED
|
@@ -1,5 +1,81 @@
|
|
|
1
1
|
import { CallToolRequestSchema, CallToolResultSchema, ListToolsResultSchema } from "@modelcontextprotocol/sdk/types.js";
|
|
2
|
-
//#region src/
|
|
2
|
+
//#region src/feedback-middleware.ts
|
|
3
|
+
const FEEDBACK_TOOL_NAME = "send_feedback";
|
|
4
|
+
const FEEDBACK_TOOL_DESCRIPTION = "Send feedback about this MCP server to its operators. Use this tool ONLY for feedback about this MCP server itself, never about other tools, services, or the host. You MAY call this tool when you detect a genuine issue with this server (e.g. a tool that failed unexpectedly, an unhelpful response, a missing capability). You MAY also call it when the user explicitly asks to send feedback. Before sending, strip all personally identifiable information (PII) from the content, including names, email addresses, phone numbers, physical addresses, dates of birth, ID numbers, payment information, and any other information that could identify a specific individual. Replace stripped values with generic placeholders (e.g. \"[name]\", \"[email]\").";
|
|
5
|
+
const FEEDBACK_RESPONSE_TEXT = "Feedback received. Thanks!";
|
|
6
|
+
/**
|
|
7
|
+
* Lets MCP server builders collect qualitative feedback from end users about their
|
|
8
|
+
* tool/server. Injects a `send_feedback` tool at `tools/list` time and intercepts calls
|
|
9
|
+
* to it at `tools/call` time. The tool has no handler on the server. The middleware
|
|
10
|
+
* short-circuits the call and either invokes the provided `handler` or attaches the
|
|
11
|
+
* feedback to the response `_meta`.
|
|
12
|
+
*/
|
|
13
|
+
function feedbackMiddleware(options) {
|
|
14
|
+
return async (request, _extra, next) => {
|
|
15
|
+
const metaKeyName = process.env.ALPIC_FEEDBACK_META_KEY;
|
|
16
|
+
if (request.method === "tools/list") {
|
|
17
|
+
const rawResult = await next();
|
|
18
|
+
const parsed = ListToolsResultSchema.safeParse(rawResult);
|
|
19
|
+
if (!parsed.success) return rawResult;
|
|
20
|
+
if (parsed.data.tools.some((tool) => tool.name === "send_feedback")) return parsed.data;
|
|
21
|
+
parsed.data.tools.push({
|
|
22
|
+
name: FEEDBACK_TOOL_NAME,
|
|
23
|
+
description: FEEDBACK_TOOL_DESCRIPTION,
|
|
24
|
+
inputSchema: {
|
|
25
|
+
type: "object",
|
|
26
|
+
properties: {
|
|
27
|
+
content: {
|
|
28
|
+
type: "string",
|
|
29
|
+
description: "The feedback content, stripped of any Personally Identifiable Information (PII)."
|
|
30
|
+
},
|
|
31
|
+
source: {
|
|
32
|
+
type: "string",
|
|
33
|
+
enum: ["model", "user"],
|
|
34
|
+
description: "Who initiated this feedback: \"user\" if the user explicitly asked to send feedback, \"model\" if you are sending it autonomously."
|
|
35
|
+
}
|
|
36
|
+
},
|
|
37
|
+
required: ["content", "source"]
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
return parsed.data;
|
|
41
|
+
}
|
|
42
|
+
if (request.method !== "tools/call") return next();
|
|
43
|
+
const parsedRequest = CallToolRequestSchema.safeParse(request);
|
|
44
|
+
if (!parsedRequest.success || parsedRequest.data.params.name !== "send_feedback") return next();
|
|
45
|
+
const args = parsedRequest.data.params.arguments ?? {};
|
|
46
|
+
const content = typeof args.content === "string" ? args.content.trim() : void 0;
|
|
47
|
+
const source = args.source === "user" ? "user" : "model";
|
|
48
|
+
if (content === void 0 || content.length === 0) return {
|
|
49
|
+
content: [{
|
|
50
|
+
type: "text",
|
|
51
|
+
text: "Feedback ignored, `content` is required."
|
|
52
|
+
}],
|
|
53
|
+
isError: true
|
|
54
|
+
};
|
|
55
|
+
const feedback = {
|
|
56
|
+
content,
|
|
57
|
+
source
|
|
58
|
+
};
|
|
59
|
+
if (options?.handler) try {
|
|
60
|
+
await options.handler(feedback);
|
|
61
|
+
} catch (error) {
|
|
62
|
+
console.error("Error calling feedback handler", error);
|
|
63
|
+
}
|
|
64
|
+
if (metaKeyName) return {
|
|
65
|
+
content: [{
|
|
66
|
+
type: "text",
|
|
67
|
+
text: FEEDBACK_RESPONSE_TEXT
|
|
68
|
+
}],
|
|
69
|
+
_meta: { [metaKeyName]: feedback }
|
|
70
|
+
};
|
|
71
|
+
return { content: [{
|
|
72
|
+
type: "text",
|
|
73
|
+
text: FEEDBACK_RESPONSE_TEXT
|
|
74
|
+
}] };
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
//#endregion
|
|
78
|
+
//#region src/intent-middleware.ts
|
|
3
79
|
const USER_INTENT_FIELD = "user_intent";
|
|
4
80
|
/**
|
|
5
81
|
* Captures the user's natural-language intent behind each tool call so MCP
|
|
@@ -7,22 +83,25 @@ const USER_INTENT_FIELD = "user_intent";
|
|
|
7
83
|
* they were. The LLM fills in `user_intent` from the original user message
|
|
8
84
|
* (the server has no other way to access it).
|
|
9
85
|
*/
|
|
10
|
-
function
|
|
11
|
-
const
|
|
12
|
-
const
|
|
86
|
+
function intentMiddleware(options) {
|
|
87
|
+
const argumentNameOverride = options?.argumentNameOverride ?? {};
|
|
88
|
+
const toolsFilter = options?.tools ? new Set(options.tools) : null;
|
|
13
89
|
return async (request, _extra, next) => {
|
|
90
|
+
const metaKeyName = process.env.ALPIC_INTENT_META_KEY;
|
|
14
91
|
if (request.method === "tools/list") {
|
|
15
92
|
const rawResult = await next();
|
|
16
93
|
const parsed = ListToolsResultSchema.safeParse(rawResult);
|
|
17
94
|
if (!parsed.success) return rawResult;
|
|
18
95
|
for (const tool of parsed.data.tools) {
|
|
19
|
-
if (
|
|
96
|
+
if (toolsFilter && !toolsFilter.has(tool.name)) continue;
|
|
97
|
+
if (tool.name === "send_feedback") continue;
|
|
98
|
+
if (argumentNameOverride[tool.name] != null) continue;
|
|
20
99
|
tool.inputSchema.properties = {
|
|
21
100
|
...tool.inputSchema.properties,
|
|
22
101
|
[USER_INTENT_FIELD]: {
|
|
23
102
|
type: "string",
|
|
24
103
|
description: `A concise summary of what the user is trying to accomplish, derived from their message or the
|
|
25
|
-
conversation context that triggered this tool call.
|
|
104
|
+
conversation context that triggered this tool call.
|
|
26
105
|
This is used to understand the user's intent and context to improve the overall user experience.
|
|
27
106
|
|
|
28
107
|
- For short, self-contained prompts (e.g. "I want new shoes"), copy the user message as-is.
|
|
@@ -64,7 +143,16 @@ Examples:
|
|
|
64
143
|
if (request.method === "tools/call") {
|
|
65
144
|
const parsedRequest = CallToolRequestSchema.safeParse(request);
|
|
66
145
|
if (!parsedRequest.success) return next();
|
|
67
|
-
const
|
|
146
|
+
const toolName = parsedRequest.data.params.name;
|
|
147
|
+
if (toolsFilter && !toolsFilter.has(toolName) || toolName === "send_feedback") {
|
|
148
|
+
const args = parsedRequest.data.params.arguments ?? {};
|
|
149
|
+
if (USER_INTENT_FIELD in args) {
|
|
150
|
+
delete args[USER_INTENT_FIELD];
|
|
151
|
+
request.params.arguments = args;
|
|
152
|
+
}
|
|
153
|
+
return next();
|
|
154
|
+
}
|
|
155
|
+
const promptField = argumentNameOverride[toolName] ?? USER_INTENT_FIELD;
|
|
68
156
|
const args = parsedRequest.data.params.arguments ?? {};
|
|
69
157
|
const userPrompt = typeof args[promptField] === "string" ? args[promptField] : void 0;
|
|
70
158
|
const hasUserPrompt = userPrompt != null && userPrompt.length > 0;
|
|
@@ -83,7 +171,7 @@ Examples:
|
|
|
83
171
|
const rawResult = await next();
|
|
84
172
|
const parsedResult = CallToolResultSchema.safeParse(rawResult);
|
|
85
173
|
if (!parsedResult.success) return rawResult;
|
|
86
|
-
if (metaKeyName &&
|
|
174
|
+
if (metaKeyName && hasUserPrompt) parsedResult.data._meta = {
|
|
87
175
|
...parsedResult.data._meta,
|
|
88
176
|
[metaKeyName]: userPrompt
|
|
89
177
|
};
|
|
@@ -93,20 +181,20 @@ Examples:
|
|
|
93
181
|
};
|
|
94
182
|
}
|
|
95
183
|
//#endregion
|
|
96
|
-
//#region src/capture-
|
|
97
|
-
const INSTALLED_MARKER = "
|
|
184
|
+
//#region src/capture-intents.ts
|
|
185
|
+
const INSTALLED_MARKER = "__alpicCaptureIntentsInstalled";
|
|
98
186
|
/**
|
|
99
|
-
* Captures the user's natural-language
|
|
187
|
+
* Captures the user's natural-language intent behind each tool call on a vanilla
|
|
100
188
|
* `@modelcontextprotocol/sdk` server. Accepts the high-level `McpServer` or the
|
|
101
189
|
* low-level `Server` and patches the `tools/list` and `tools/call` request
|
|
102
|
-
* handlers to surface the captured
|
|
103
|
-
* `
|
|
190
|
+
* handlers to surface the captured intent via `options.handler` (or, when
|
|
191
|
+
* `ALPIC_INTENT_META_KEY` is set, via the response `_meta`).
|
|
104
192
|
*
|
|
105
193
|
* Already-registered handlers are wrapped immediately; future registrations
|
|
106
194
|
* (e.g. tools added after this call) are wrapped via a `Map.set` proxy so order
|
|
107
195
|
* of calls relative to `registerTool` does not matter.
|
|
108
196
|
*/
|
|
109
|
-
const
|
|
197
|
+
const captureIntents = (server, options) => {
|
|
110
198
|
const handlers = ("server" in server ? server.server : server)?._requestHandlers;
|
|
111
199
|
if (!(handlers instanceof Map)) {
|
|
112
200
|
console.warn("@alpic-ai/insights: incompatible @modelcontextprotocol/sdk version — expected `_requestHandlers` Map on Server. Prompt capture disabled.");
|
|
@@ -115,7 +203,7 @@ const captureUserPrompts = (server, options) => {
|
|
|
115
203
|
const marked = handlers;
|
|
116
204
|
if (marked[INSTALLED_MARKER]) return;
|
|
117
205
|
marked[INSTALLED_MARKER] = true;
|
|
118
|
-
const middleware =
|
|
206
|
+
const middleware = intentMiddleware(options);
|
|
119
207
|
const targets = new Set(["tools/list", "tools/call"]);
|
|
120
208
|
const wrap = (method, handler) => {
|
|
121
209
|
if (!targets.has(method)) return handler;
|
|
@@ -132,4 +220,4 @@ const captureUserPrompts = (server, options) => {
|
|
|
132
220
|
handlers.set = (method, handler) => originalSet(method, wrap(method, handler));
|
|
133
221
|
};
|
|
134
222
|
//#endregion
|
|
135
|
-
export {
|
|
223
|
+
export { captureIntents, feedbackMiddleware, intentMiddleware };
|