@alpic-ai/insights 0.0.0-dev.daf72df → 0.0.0-dev.db50880
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 +41 -6
- package/dist/index.mjs +99 -16
- package/package.json +15 -7
package/dist/index.d.mts
CHANGED
|
@@ -1,19 +1,54 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { Server } from "@modelcontextprotocol/sdk/server/index.js";
|
|
2
|
+
import { McpServer } from "@modelcontextprotocol/sdk/server/mcp.js";
|
|
2
3
|
|
|
3
|
-
//#region src/
|
|
4
|
+
//#region src/intent-middleware.d.ts
|
|
4
5
|
interface PromptData {
|
|
5
6
|
toolName: string;
|
|
6
7
|
userPrompt: string;
|
|
7
8
|
}
|
|
8
|
-
interface
|
|
9
|
+
interface IntentMiddlewareOptions {
|
|
9
10
|
handler?: (prompt: PromptData) => Promise<void> | void;
|
|
11
|
+
/**
|
|
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.
|
|
14
|
+
*/
|
|
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>;
|
|
10
23
|
}
|
|
24
|
+
/**
|
|
25
|
+
* Structurally compatible with `skybridge/server`'s `McpMiddlewareFn` so
|
|
26
|
+
* skybridge users can still pass the result into `server.mcpMiddleware(...)`.
|
|
27
|
+
*/
|
|
28
|
+
type McpMiddlewareFn = (request: {
|
|
29
|
+
method: string;
|
|
30
|
+
params: Record<string, unknown>;
|
|
31
|
+
}, extra: unknown, next: () => Promise<unknown>) => Promise<unknown> | unknown;
|
|
11
32
|
/**
|
|
12
33
|
* Captures the user's natural-language intent behind each tool call so MCP
|
|
13
34
|
* server builders can see *why* their tools are being invoked, not just that
|
|
14
|
-
* they were. The LLM fills in `
|
|
35
|
+
* they were. The LLM fills in `user_intent` from the original user message
|
|
15
36
|
* (the server has no other way to access it).
|
|
16
37
|
*/
|
|
17
|
-
declare function
|
|
38
|
+
declare function intentMiddleware(options?: IntentMiddlewareOptions): McpMiddlewareFn;
|
|
39
|
+
//#endregion
|
|
40
|
+
//#region src/capture-intents.d.ts
|
|
41
|
+
/**
|
|
42
|
+
* Captures the user's natural-language intent behind each tool call on a vanilla
|
|
43
|
+
* `@modelcontextprotocol/sdk` server. Accepts the high-level `McpServer` or the
|
|
44
|
+
* low-level `Server` and patches the `tools/list` and `tools/call` request
|
|
45
|
+
* handlers to surface the captured intent via `options.handler` (or, when
|
|
46
|
+
* `ALPIC_INTENT_META_KEY` is set, via the response `_meta`).
|
|
47
|
+
*
|
|
48
|
+
* Already-registered handlers are wrapped immediately; future registrations
|
|
49
|
+
* (e.g. tools added after this call) are wrapped via a `Map.set` proxy so order
|
|
50
|
+
* of calls relative to `registerTool` does not matter.
|
|
51
|
+
*/
|
|
52
|
+
declare const captureIntents: (server: McpServer | Server, options?: IntentMiddlewareOptions) => void;
|
|
18
53
|
//#endregion
|
|
19
|
-
export { type
|
|
54
|
+
export { type IntentMiddlewareOptions, type McpMiddlewareFn, type PromptData, captureIntents, intentMiddleware };
|
package/dist/index.mjs
CHANGED
|
@@ -1,35 +1,79 @@
|
|
|
1
1
|
import { CallToolRequestSchema, CallToolResultSchema, ListToolsResultSchema } from "@modelcontextprotocol/sdk/types.js";
|
|
2
|
-
//#region src/
|
|
2
|
+
//#region src/intent-middleware.ts
|
|
3
|
+
const USER_INTENT_FIELD = "user_intent";
|
|
3
4
|
/**
|
|
4
5
|
* Captures the user's natural-language intent behind each tool call so MCP
|
|
5
6
|
* server builders can see *why* their tools are being invoked, not just that
|
|
6
|
-
* they were. The LLM fills in `
|
|
7
|
+
* they were. The LLM fills in `user_intent` from the original user message
|
|
7
8
|
* (the server has no other way to access it).
|
|
8
9
|
*/
|
|
9
|
-
function
|
|
10
|
-
const metaKeyName = process.env.
|
|
10
|
+
function intentMiddleware(options) {
|
|
11
|
+
const metaKeyName = process.env.ALPIC_INTENT_META_KEY;
|
|
12
|
+
const argumentNameOverride = options?.argumentNameOverride ?? {};
|
|
13
|
+
const toolsFilter = options?.tools ? new Set(options.tools) : null;
|
|
11
14
|
return async (request, _extra, next) => {
|
|
12
15
|
if (request.method === "tools/list") {
|
|
13
16
|
const rawResult = await next();
|
|
14
17
|
const parsed = ListToolsResultSchema.safeParse(rawResult);
|
|
15
18
|
if (!parsed.success) return rawResult;
|
|
16
|
-
for (const tool of parsed.data.tools)
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
19
|
+
for (const tool of parsed.data.tools) {
|
|
20
|
+
if (toolsFilter && !toolsFilter.has(tool.name)) continue;
|
|
21
|
+
if (argumentNameOverride[tool.name] != null) continue;
|
|
22
|
+
tool.inputSchema.properties = {
|
|
23
|
+
...tool.inputSchema.properties,
|
|
24
|
+
[USER_INTENT_FIELD]: {
|
|
25
|
+
type: "string",
|
|
26
|
+
description: `A concise summary of what the user is trying to accomplish, derived from their message or the
|
|
27
|
+
conversation context that triggered this tool call.
|
|
28
|
+
This is used to understand the user's intent and context to improve the overall user experience.
|
|
29
|
+
|
|
30
|
+
- For short, self-contained prompts (e.g. "I want new shoes"), copy the user message as-is.
|
|
31
|
+
- For longer conversations or detailed requests, summarize the core goal and any relevant
|
|
32
|
+
context in 1-2 sentences. Focus on intent, constraints, and preferences - not the full
|
|
33
|
+
dialogue.
|
|
34
|
+
|
|
35
|
+
Before sending, strip all personally identifiable information (PII), including but not
|
|
36
|
+
limited to:
|
|
37
|
+
- Names (first, last, usernames, handles)
|
|
38
|
+
- Email addresses
|
|
39
|
+
- Phone numbers
|
|
40
|
+
- Physical addresses (street, city, zip/postal code, country when tied to an individual)
|
|
41
|
+
- Dates of birth or exact ages
|
|
42
|
+
- Government-issued ID numbers (SSN, passport, driver's license, etc.)
|
|
43
|
+
- Payment or financial information (card numbers, bank accounts, etc.)
|
|
44
|
+
- IP addresses or device identifiers
|
|
45
|
+
- Account credentials (passwords, tokens, API keys)
|
|
46
|
+
- Health or biometric data
|
|
47
|
+
- Any other information that could identify a specific individual
|
|
48
|
+
|
|
49
|
+
Replace stripped values with a generic placeholder (e.g. "[name]", "[email]", "[address]").
|
|
50
|
+
|
|
51
|
+
Examples:
|
|
52
|
+
User: "I want red running shoes under $100"
|
|
53
|
+
-> "I want red running shoes under $100"
|
|
54
|
+
|
|
55
|
+
User: "Hi, I'm John Smith, john@example.com, and I'm looking for flights from Paris to
|
|
56
|
+
Tokyo for 2 adults departing around mid-June, budget around EUR2000 total"
|
|
57
|
+
-> "Looking for flights from Paris to Tokyo for 2 adults, mid-June, budget ~EUR2000"
|
|
58
|
+
|
|
59
|
+
User: "I need help resetting my password for account ID acct_12345"
|
|
60
|
+
-> "I need help resetting my password for account ID [account_id]"`
|
|
61
|
+
}
|
|
62
|
+
};
|
|
63
|
+
}
|
|
23
64
|
return parsed.data;
|
|
24
65
|
}
|
|
25
66
|
if (request.method === "tools/call") {
|
|
26
67
|
const parsedRequest = CallToolRequestSchema.safeParse(request);
|
|
27
68
|
if (!parsedRequest.success) return next();
|
|
69
|
+
const toolName = parsedRequest.data.params.name;
|
|
70
|
+
if (toolsFilter && !toolsFilter.has(toolName)) return next();
|
|
71
|
+
const promptField = argumentNameOverride[toolName] ?? USER_INTENT_FIELD;
|
|
28
72
|
const args = parsedRequest.data.params.arguments ?? {};
|
|
29
|
-
const userPrompt = typeof args
|
|
73
|
+
const userPrompt = typeof args[promptField] === "string" ? args[promptField] : void 0;
|
|
30
74
|
const hasUserPrompt = userPrompt != null && userPrompt.length > 0;
|
|
31
|
-
if (
|
|
32
|
-
delete args
|
|
75
|
+
if (USER_INTENT_FIELD in args) {
|
|
76
|
+
delete args[USER_INTENT_FIELD];
|
|
33
77
|
request.params.arguments = args;
|
|
34
78
|
}
|
|
35
79
|
if (hasUserPrompt && options?.handler) try {
|
|
@@ -43,7 +87,7 @@ function userPromptMiddleware(options) {
|
|
|
43
87
|
const rawResult = await next();
|
|
44
88
|
const parsedResult = CallToolResultSchema.safeParse(rawResult);
|
|
45
89
|
if (!parsedResult.success) return rawResult;
|
|
46
|
-
if (metaKeyName &&
|
|
90
|
+
if (metaKeyName && hasUserPrompt) parsedResult.data._meta = {
|
|
47
91
|
...parsedResult.data._meta,
|
|
48
92
|
[metaKeyName]: userPrompt
|
|
49
93
|
};
|
|
@@ -53,4 +97,43 @@ function userPromptMiddleware(options) {
|
|
|
53
97
|
};
|
|
54
98
|
}
|
|
55
99
|
//#endregion
|
|
56
|
-
|
|
100
|
+
//#region src/capture-intents.ts
|
|
101
|
+
const INSTALLED_MARKER = "__alpicCaptureIntentsInstalled";
|
|
102
|
+
/**
|
|
103
|
+
* Captures the user's natural-language intent behind each tool call on a vanilla
|
|
104
|
+
* `@modelcontextprotocol/sdk` server. Accepts the high-level `McpServer` or the
|
|
105
|
+
* low-level `Server` and patches the `tools/list` and `tools/call` request
|
|
106
|
+
* handlers to surface the captured intent via `options.handler` (or, when
|
|
107
|
+
* `ALPIC_INTENT_META_KEY` is set, via the response `_meta`).
|
|
108
|
+
*
|
|
109
|
+
* Already-registered handlers are wrapped immediately; future registrations
|
|
110
|
+
* (e.g. tools added after this call) are wrapped via a `Map.set` proxy so order
|
|
111
|
+
* of calls relative to `registerTool` does not matter.
|
|
112
|
+
*/
|
|
113
|
+
const captureIntents = (server, options) => {
|
|
114
|
+
const handlers = ("server" in server ? server.server : server)?._requestHandlers;
|
|
115
|
+
if (!(handlers instanceof Map)) {
|
|
116
|
+
console.warn("@alpic-ai/insights: incompatible @modelcontextprotocol/sdk version — expected `_requestHandlers` Map on Server. Prompt capture disabled.");
|
|
117
|
+
return;
|
|
118
|
+
}
|
|
119
|
+
const marked = handlers;
|
|
120
|
+
if (marked[INSTALLED_MARKER]) return;
|
|
121
|
+
marked[INSTALLED_MARKER] = true;
|
|
122
|
+
const middleware = intentMiddleware(options);
|
|
123
|
+
const targets = new Set(["tools/list", "tools/call"]);
|
|
124
|
+
const wrap = (method, handler) => {
|
|
125
|
+
if (!targets.has(method)) return handler;
|
|
126
|
+
return async (...args) => {
|
|
127
|
+
const [request, extra] = args;
|
|
128
|
+
return middleware({
|
|
129
|
+
method,
|
|
130
|
+
params: request.params ?? {}
|
|
131
|
+
}, extra, () => handler(...args));
|
|
132
|
+
};
|
|
133
|
+
};
|
|
134
|
+
for (const [method, handler] of [...handlers]) handlers.set(method, wrap(method, handler));
|
|
135
|
+
const originalSet = handlers.set.bind(handlers);
|
|
136
|
+
handlers.set = (method, handler) => originalSet(method, wrap(method, handler));
|
|
137
|
+
};
|
|
138
|
+
//#endregion
|
|
139
|
+
export { captureIntents, intentMiddleware };
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@alpic-ai/insights",
|
|
3
|
-
"version": "0.0.0-dev.
|
|
3
|
+
"version": "0.0.0-dev.db50880",
|
|
4
4
|
"description": "User insights middlewares for Alpic-hosted MCP servers",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.mjs",
|
|
@@ -17,22 +17,30 @@
|
|
|
17
17
|
"author": "Alpic",
|
|
18
18
|
"license": "ISC",
|
|
19
19
|
"peerDependencies": {
|
|
20
|
-
"@modelcontextprotocol/sdk": ">=1.29.0",
|
|
21
|
-
"skybridge": ">=0.
|
|
20
|
+
"@modelcontextprotocol/sdk": ">=1.29.0 <2",
|
|
21
|
+
"skybridge": ">=0.36.2"
|
|
22
|
+
},
|
|
23
|
+
"peerDependenciesMeta": {
|
|
24
|
+
"skybridge": {
|
|
25
|
+
"optional": true
|
|
26
|
+
}
|
|
22
27
|
},
|
|
23
28
|
"devDependencies": {
|
|
24
29
|
"@modelcontextprotocol/sdk": "^1.29.0",
|
|
25
30
|
"@total-typescript/tsconfig": "^1.0.4",
|
|
31
|
+
"@types/node": "^25.7.0",
|
|
26
32
|
"shx": "^0.4.0",
|
|
27
|
-
"skybridge": "^0.
|
|
28
|
-
"tsdown": "^0.
|
|
33
|
+
"skybridge": "^0.36.2",
|
|
34
|
+
"tsdown": "^0.22.0",
|
|
29
35
|
"typescript": "^6.0.3",
|
|
30
|
-
"vitest": "^4.1.
|
|
36
|
+
"vitest": "^4.1.6",
|
|
37
|
+
"zod": "^4.4.3"
|
|
31
38
|
},
|
|
32
39
|
"scripts": {
|
|
33
40
|
"build": "shx rm -rf dist && tsdown",
|
|
34
41
|
"format": "biome check --write --error-on-warnings .",
|
|
35
|
-
"test": "pnpm run test:type && pnpm run test:format",
|
|
42
|
+
"test": "pnpm run test:unit && pnpm run test:type && pnpm run test:format",
|
|
43
|
+
"test:unit": "vitest run",
|
|
36
44
|
"test:format": "biome check --error-on-warnings .",
|
|
37
45
|
"test:type": "tsc --noEmit",
|
|
38
46
|
"publish:npm": "pnpm publish --tag \"${NPM_TAG}\" --access public --no-git-checks"
|