@node2flow/gmail-mcp 1.0.1 → 1.0.2
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.ts +34 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/server.d.ts +2 -2
- package/dist/server.d.ts.map +1 -1
- package/dist/server.js +142 -122
- package/dist/server.js.map +1 -1
- package/package.json +1 -1
- package/src/server.ts +174 -154
package/dist/index.d.ts
CHANGED
|
@@ -16,5 +16,38 @@ export default function createSmitheryServer(opts?: {
|
|
|
16
16
|
GOOGLE_CLIENT_SECRET?: string;
|
|
17
17
|
GOOGLE_REFRESH_TOKEN?: string;
|
|
18
18
|
};
|
|
19
|
-
}): import("@modelcontextprotocol/sdk/server
|
|
19
|
+
}): import("@modelcontextprotocol/sdk/server").Server<{
|
|
20
|
+
method: string;
|
|
21
|
+
params?: {
|
|
22
|
+
[x: string]: unknown;
|
|
23
|
+
_meta?: {
|
|
24
|
+
[x: string]: unknown;
|
|
25
|
+
progressToken?: string | number | undefined;
|
|
26
|
+
"io.modelcontextprotocol/related-task"?: {
|
|
27
|
+
taskId: string;
|
|
28
|
+
} | undefined;
|
|
29
|
+
} | undefined;
|
|
30
|
+
} | undefined;
|
|
31
|
+
}, {
|
|
32
|
+
method: string;
|
|
33
|
+
params?: {
|
|
34
|
+
[x: string]: unknown;
|
|
35
|
+
_meta?: {
|
|
36
|
+
[x: string]: unknown;
|
|
37
|
+
progressToken?: string | number | undefined;
|
|
38
|
+
"io.modelcontextprotocol/related-task"?: {
|
|
39
|
+
taskId: string;
|
|
40
|
+
} | undefined;
|
|
41
|
+
} | undefined;
|
|
42
|
+
} | undefined;
|
|
43
|
+
}, {
|
|
44
|
+
[x: string]: unknown;
|
|
45
|
+
_meta?: {
|
|
46
|
+
[x: string]: unknown;
|
|
47
|
+
progressToken?: string | number | undefined;
|
|
48
|
+
"io.modelcontextprotocol/related-task"?: {
|
|
49
|
+
taskId: string;
|
|
50
|
+
} | undefined;
|
|
51
|
+
} | undefined;
|
|
52
|
+
}>;
|
|
20
53
|
//# sourceMappingURL=index.d.ts.map
|
package/dist/index.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA;;;;;;;;;;GAUG;AA2JH,MAAM,CAAC,OAAO,UAAU,oBAAoB,CAAC,IAAI,CAAC,EAAE;IAClD,MAAM,CAAC,EAAE;QACP,gBAAgB,CAAC,EAAE,MAAM,CAAC;QAC1B,oBAAoB,CAAC,EAAE,MAAM,CAAC;QAC9B,oBAAoB,CAAC,EAAE,MAAM,CAAC;KAC/B,CAAC;CACH
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AACA;;;;;;;;;;GAUG;AA2JH,MAAM,CAAC,OAAO,UAAU,oBAAoB,CAAC,IAAI,CAAC,EAAE;IAClD,MAAM,CAAC,EAAE;QACP,gBAAgB,CAAC,EAAE,MAAM,CAAC;QAC1B,oBAAoB,CAAC,EAAE,MAAM,CAAC;QAC9B,oBAAoB,CAAC,EAAE,MAAM,CAAC;KAC/B,CAAC;CACH;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAMA"}
|
package/dist/server.d.ts
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Shared MCP Server — used by both Node.js (index.ts) and CF Worker (worker.ts)
|
|
3
3
|
*/
|
|
4
|
-
import {
|
|
4
|
+
import { Server } from '@modelcontextprotocol/sdk/server/index.js';
|
|
5
5
|
import { GmailClient } from './gmail-client.js';
|
|
6
6
|
export interface GmailMcpConfig {
|
|
7
7
|
clientId: string;
|
|
@@ -9,5 +9,5 @@ export interface GmailMcpConfig {
|
|
|
9
9
|
refreshToken: string;
|
|
10
10
|
}
|
|
11
11
|
export declare function handleToolCall(toolName: string, args: Record<string, unknown>, client: GmailClient): Promise<import("./types.js").MessageList> | Promise<import("./types.js").Message> | Promise<void> | Promise<import("./types.js").Attachment> | Promise<import("./types.js").DraftList> | Promise<import("./types.js").Draft> | Promise<import("./types.js").LabelList> | Promise<import("./types.js").Label> | Promise<import("./types.js").ThreadList> | Promise<import("./types.js").Thread> | Promise<import("./types.js").Profile> | Promise<import("./types.js").VacationSettings>;
|
|
12
|
-
export declare function createServer(config?: GmailMcpConfig):
|
|
12
|
+
export declare function createServer(config?: GmailMcpConfig): Server;
|
|
13
13
|
//# sourceMappingURL=server.d.ts.map
|
package/dist/server.d.ts.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"server.d.ts","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,MAAM,EAAE,MAAM,2CAA2C,CAAC;AASnE,OAAO,EAAE,WAAW,EAAE,MAAM,mBAAmB,CAAC;AAGhD,MAAM,WAAW,cAAc;IAC7B,QAAQ,EAAE,MAAM,CAAC;IACjB,YAAY,EAAE,MAAM,CAAC;IACrB,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,wBAAgB,cAAc,CAC5B,QAAQ,EAAE,MAAM,EAChB,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,EAC7B,MAAM,EAAE,WAAW,2dAoKpB;AAED,wBAAgB,YAAY,CAAC,MAAM,CAAC,EAAE,cAAc,GAAG,MAAM,CAwL5D"}
|
package/dist/server.js
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
/**
|
|
2
2
|
* Shared MCP Server — used by both Node.js (index.ts) and CF Worker (worker.ts)
|
|
3
3
|
*/
|
|
4
|
-
import {
|
|
5
|
-
import { ListToolsRequestSchema } from '@modelcontextprotocol/sdk/types.js';
|
|
4
|
+
import { Server } from '@modelcontextprotocol/sdk/server/index.js';
|
|
5
|
+
import { CallToolRequestSchema, ListToolsRequestSchema, ListPromptsRequestSchema, GetPromptRequestSchema, ListResourcesRequestSchema, ReadResourceRequestSchema, } from '@modelcontextprotocol/sdk/types.js';
|
|
6
6
|
import { GmailClient } from './gmail-client.js';
|
|
7
7
|
import { TOOLS } from './tools.js';
|
|
8
8
|
export function handleToolCall(toolName, args, client) {
|
|
@@ -165,23 +165,30 @@ export function handleToolCall(toolName, args, client) {
|
|
|
165
165
|
}
|
|
166
166
|
}
|
|
167
167
|
export function createServer(config) {
|
|
168
|
-
const server = new
|
|
169
|
-
name: 'gmail-mcp',
|
|
170
|
-
version: '1.0.0',
|
|
171
|
-
});
|
|
168
|
+
const server = new Server({ name: 'gmail-mcp', version: '1.0.0' }, { capabilities: { tools: {}, prompts: {}, resources: {} } });
|
|
172
169
|
let client = null;
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
|
|
170
|
+
// List available tools
|
|
171
|
+
server.setRequestHandler(ListToolsRequestSchema, async () => ({
|
|
172
|
+
tools: TOOLS.map(tool => {
|
|
173
|
+
const hasProperties = Object.keys(tool.inputSchema.properties).length > 0;
|
|
174
|
+
return {
|
|
175
|
+
name: tool.name,
|
|
176
|
+
description: tool.description,
|
|
177
|
+
...(hasProperties ? { inputSchema: tool.inputSchema } : {}),
|
|
178
|
+
annotations: tool.annotations,
|
|
179
|
+
};
|
|
180
|
+
}),
|
|
181
|
+
}));
|
|
182
|
+
// Handle tool calls
|
|
183
|
+
server.setRequestHandler(CallToolRequestSchema, async (request) => {
|
|
184
|
+
const { name, arguments: args } = request.params;
|
|
185
|
+
try {
|
|
179
186
|
const clientId = config?.clientId ||
|
|
180
|
-
args
|
|
187
|
+
args?.GOOGLE_CLIENT_ID;
|
|
181
188
|
const clientSecret = config?.clientSecret ||
|
|
182
|
-
args
|
|
189
|
+
args?.GOOGLE_CLIENT_SECRET;
|
|
183
190
|
const refreshToken = config?.refreshToken ||
|
|
184
|
-
args
|
|
191
|
+
args?.GOOGLE_REFRESH_TOKEN;
|
|
185
192
|
if (!clientId || !clientSecret || !refreshToken) {
|
|
186
193
|
return {
|
|
187
194
|
content: [{ type: 'text', text: 'Error: GOOGLE_CLIENT_ID, GOOGLE_CLIENT_SECRET, and GOOGLE_REFRESH_TOKEN are all required.' }],
|
|
@@ -191,120 +198,133 @@ export function createServer(config) {
|
|
|
191
198
|
if (!client || config?.clientId !== clientId) {
|
|
192
199
|
client = new GmailClient({ clientId, clientSecret, refreshToken });
|
|
193
200
|
}
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
198
|
-
|
|
199
|
-
|
|
200
|
-
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
|
|
208
|
-
|
|
209
|
-
|
|
210
|
-
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
content: {
|
|
215
|
-
type: 'text',
|
|
216
|
-
text: [
|
|
217
|
-
'You are a Gmail email assistant.',
|
|
218
|
-
'',
|
|
219
|
-
'Sending emails:',
|
|
220
|
-
'1. **Send directly** — gmail_send_message with to, subject, body',
|
|
221
|
-
'2. **HTML emails** — Include html parameter for rich formatting',
|
|
222
|
-
'3. **Reply to thread** — Set thread_id, in_reply_to (Message-ID header), and references',
|
|
223
|
-
'4. **CC/BCC** — Comma-separate multiple addresses',
|
|
224
|
-
'',
|
|
225
|
-
'Working with drafts:',
|
|
226
|
-
'1. **Create draft** — gmail_create_draft (same params as send)',
|
|
227
|
-
'2. **Update draft** — gmail_update_draft replaces the entire draft',
|
|
228
|
-
'3. **Send draft** — gmail_send_draft with the draft ID',
|
|
229
|
-
'4. **List drafts** — gmail_list_drafts to see all drafts',
|
|
230
|
-
'',
|
|
231
|
-
'Tips:',
|
|
232
|
-
'- For replies, always get the original message first to extract Message-ID and thread_id',
|
|
233
|
-
'- Use gmail_get_thread to see the full conversation before replying',
|
|
234
|
-
'- HTML body is sent alongside plain text as multipart/alternative',
|
|
235
|
-
].join('\n'),
|
|
236
|
-
},
|
|
237
|
-
}],
|
|
238
|
-
}));
|
|
239
|
-
server.prompt('search-and-organize', 'Guide for searching emails, managing labels, and organizing the mailbox', async () => ({
|
|
240
|
-
messages: [{
|
|
241
|
-
role: 'user',
|
|
242
|
-
content: {
|
|
243
|
-
type: 'text',
|
|
244
|
-
text: [
|
|
245
|
-
'You are a Gmail organization assistant.',
|
|
246
|
-
'',
|
|
247
|
-
'Search syntax (q parameter):',
|
|
248
|
-
'- from:user@example.com — Messages from a sender',
|
|
249
|
-
'- to:user@example.com — Messages to a recipient',
|
|
250
|
-
'- subject:"meeting notes" — Subject contains text',
|
|
251
|
-
'- has:attachment — Messages with attachments',
|
|
252
|
-
'- is:unread / is:starred / is:important',
|
|
253
|
-
'- label:custom-label — Messages with a specific label',
|
|
254
|
-
'- after:2026/01/01 / before:2026/12/31 — Date range',
|
|
255
|
-
'- newer_than:2d / older_than:1y — Relative dates',
|
|
256
|
-
'- filename:pdf — Attachments by type',
|
|
257
|
-
'- Combine: "from:boss@company.com has:attachment is:unread"',
|
|
258
|
-
'',
|
|
259
|
-
'Organizing with labels:',
|
|
260
|
-
'1. **List labels** — gmail_list_labels to see all labels',
|
|
261
|
-
'2. **Create label** — gmail_create_label with nested support ("Projects/Active")',
|
|
262
|
-
'3. **Apply labels** — gmail_modify_message to add/remove labels',
|
|
263
|
-
'4. **Mark as read** — Remove "UNREAD" label',
|
|
264
|
-
'5. **Star message** — Add "STARRED" label',
|
|
265
|
-
'6. **Batch operations** — gmail_batch_modify for bulk label changes',
|
|
266
|
-
'',
|
|
267
|
-
'System labels: INBOX, SENT, DRAFT, SPAM, TRASH, UNREAD, STARRED, IMPORTANT, CATEGORY_PERSONAL, CATEGORY_SOCIAL, CATEGORY_PROMOTIONS, CATEGORY_UPDATES, CATEGORY_FORUMS',
|
|
268
|
-
].join('\n'),
|
|
269
|
-
},
|
|
270
|
-
}],
|
|
201
|
+
const result = await handleToolCall(name, args || {}, client);
|
|
202
|
+
const text = result === undefined ? '{"success": true}' : JSON.stringify(result, null, 2);
|
|
203
|
+
return {
|
|
204
|
+
content: [{ type: 'text', text }],
|
|
205
|
+
isError: false,
|
|
206
|
+
};
|
|
207
|
+
}
|
|
208
|
+
catch (error) {
|
|
209
|
+
return {
|
|
210
|
+
content: [{ type: 'text', text: `Error: ${error instanceof Error ? error.message : String(error)}` }],
|
|
211
|
+
isError: true,
|
|
212
|
+
};
|
|
213
|
+
}
|
|
214
|
+
});
|
|
215
|
+
// List prompts
|
|
216
|
+
server.setRequestHandler(ListPromptsRequestSchema, async () => ({
|
|
217
|
+
prompts: [
|
|
218
|
+
{ name: 'compose-and-send', description: 'Guide for composing and sending emails, managing drafts' },
|
|
219
|
+
{ name: 'search-and-organize', description: 'Guide for searching emails, managing labels, and organizing the mailbox' },
|
|
220
|
+
],
|
|
271
221
|
}));
|
|
272
|
-
//
|
|
273
|
-
server.
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
222
|
+
// Get prompt
|
|
223
|
+
server.setRequestHandler(GetPromptRequestSchema, async (request) => {
|
|
224
|
+
const { name } = request.params;
|
|
225
|
+
if (name === 'compose-and-send') {
|
|
226
|
+
return {
|
|
227
|
+
messages: [{
|
|
228
|
+
role: 'user',
|
|
229
|
+
content: {
|
|
230
|
+
type: 'text',
|
|
231
|
+
text: [
|
|
232
|
+
'You are a Gmail email assistant.',
|
|
233
|
+
'',
|
|
234
|
+
'Sending emails:',
|
|
235
|
+
'1. **Send directly** — gmail_send_message with to, subject, body',
|
|
236
|
+
'2. **HTML emails** — Include html parameter for rich formatting',
|
|
237
|
+
'3. **Reply to thread** — Set thread_id, in_reply_to (Message-ID header), and references',
|
|
238
|
+
'4. **CC/BCC** — Comma-separate multiple addresses',
|
|
239
|
+
'',
|
|
240
|
+
'Working with drafts:',
|
|
241
|
+
'1. **Create draft** — gmail_create_draft (same params as send)',
|
|
242
|
+
'2. **Update draft** — gmail_update_draft replaces the entire draft',
|
|
243
|
+
'3. **Send draft** — gmail_send_draft with the draft ID',
|
|
244
|
+
'4. **List drafts** — gmail_list_drafts to see all drafts',
|
|
245
|
+
'',
|
|
246
|
+
'Tips:',
|
|
247
|
+
'- For replies, always get the original message first to extract Message-ID and thread_id',
|
|
248
|
+
'- Use gmail_get_thread to see the full conversation before replying',
|
|
249
|
+
'- HTML body is sent alongside plain text as multipart/alternative',
|
|
250
|
+
].join('\n'),
|
|
251
|
+
},
|
|
252
|
+
}],
|
|
253
|
+
};
|
|
254
|
+
}
|
|
255
|
+
if (name === 'search-and-organize') {
|
|
256
|
+
return {
|
|
257
|
+
messages: [{
|
|
258
|
+
role: 'user',
|
|
259
|
+
content: {
|
|
260
|
+
type: 'text',
|
|
261
|
+
text: [
|
|
262
|
+
'You are a Gmail organization assistant.',
|
|
263
|
+
'',
|
|
264
|
+
'Search syntax (q parameter):',
|
|
265
|
+
'- from:user@example.com — Messages from a sender',
|
|
266
|
+
'- to:user@example.com — Messages to a recipient',
|
|
267
|
+
'- subject:"meeting notes" — Subject contains text',
|
|
268
|
+
'- has:attachment — Messages with attachments',
|
|
269
|
+
'- is:unread / is:starred / is:important',
|
|
270
|
+
'- label:custom-label — Messages with a specific label',
|
|
271
|
+
'- after:2026/01/01 / before:2026/12/31 — Date range',
|
|
272
|
+
'- newer_than:2d / older_than:1y — Relative dates',
|
|
273
|
+
'- filename:pdf — Attachments by type',
|
|
274
|
+
'- Combine: "from:boss@company.com has:attachment is:unread"',
|
|
275
|
+
'',
|
|
276
|
+
'Organizing with labels:',
|
|
277
|
+
'1. **List labels** — gmail_list_labels to see all labels',
|
|
278
|
+
'2. **Create label** — gmail_create_label with nested support ("Projects/Active")',
|
|
279
|
+
'3. **Apply labels** — gmail_modify_message to add/remove labels',
|
|
280
|
+
'4. **Mark as read** — Remove "UNREAD" label',
|
|
281
|
+
'5. **Star message** — Add "STARRED" label',
|
|
282
|
+
'6. **Batch operations** — gmail_batch_modify for bulk label changes',
|
|
283
|
+
'',
|
|
284
|
+
'System labels: INBOX, SENT, DRAFT, SPAM, TRASH, UNREAD, STARRED, IMPORTANT, CATEGORY_PERSONAL, CATEGORY_SOCIAL, CATEGORY_PROMOTIONS, CATEGORY_UPDATES, CATEGORY_FORUMS',
|
|
285
|
+
].join('\n'),
|
|
286
|
+
},
|
|
287
|
+
}],
|
|
288
|
+
};
|
|
289
|
+
}
|
|
290
|
+
throw new Error(`Unknown prompt: ${name}`);
|
|
291
|
+
});
|
|
292
|
+
// List resources
|
|
293
|
+
server.setRequestHandler(ListResourcesRequestSchema, async () => ({
|
|
294
|
+
resources: [{
|
|
278
295
|
uri: 'gmail://server-info',
|
|
296
|
+
name: 'server-info',
|
|
297
|
+
description: 'Connection status and available tools for this Gmail MCP server',
|
|
279
298
|
mimeType: 'application/json',
|
|
280
|
-
text: JSON.stringify({
|
|
281
|
-
name: 'gmail-mcp',
|
|
282
|
-
version: '1.0.0',
|
|
283
|
-
connected: !!config,
|
|
284
|
-
has_oauth: !!(config?.clientId),
|
|
285
|
-
tools_available: TOOLS.length,
|
|
286
|
-
tool_categories: {
|
|
287
|
-
messages: 10,
|
|
288
|
-
drafts: 6,
|
|
289
|
-
labels: 5,
|
|
290
|
-
threads: 5,
|
|
291
|
-
settings: 2,
|
|
292
|
-
},
|
|
293
|
-
}, null, 2),
|
|
294
299
|
}],
|
|
295
300
|
}));
|
|
296
|
-
//
|
|
297
|
-
server.
|
|
298
|
-
|
|
299
|
-
|
|
301
|
+
// Read resource
|
|
302
|
+
server.setRequestHandler(ReadResourceRequestSchema, async (request) => {
|
|
303
|
+
const { uri } = request.params;
|
|
304
|
+
if (uri === 'gmail://server-info') {
|
|
300
305
|
return {
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
306
|
+
contents: [{
|
|
307
|
+
uri: 'gmail://server-info',
|
|
308
|
+
mimeType: 'application/json',
|
|
309
|
+
text: JSON.stringify({
|
|
310
|
+
name: 'gmail-mcp',
|
|
311
|
+
version: '1.0.0',
|
|
312
|
+
connected: !!config,
|
|
313
|
+
has_oauth: !!(config?.clientId),
|
|
314
|
+
tools_available: TOOLS.length,
|
|
315
|
+
tool_categories: {
|
|
316
|
+
messages: 10,
|
|
317
|
+
drafts: 6,
|
|
318
|
+
labels: 5,
|
|
319
|
+
threads: 5,
|
|
320
|
+
settings: 2,
|
|
321
|
+
},
|
|
322
|
+
}, null, 2),
|
|
323
|
+
}],
|
|
305
324
|
};
|
|
306
|
-
}
|
|
307
|
-
|
|
325
|
+
}
|
|
326
|
+
throw new Error(`Unknown resource: ${uri}`);
|
|
327
|
+
});
|
|
308
328
|
return server;
|
|
309
329
|
}
|
|
310
330
|
//# sourceMappingURL=server.js.map
|
package/dist/server.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,
|
|
1
|
+
{"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
|
package/package.json
CHANGED
package/src/server.ts
CHANGED
|
@@ -2,8 +2,15 @@
|
|
|
2
2
|
* Shared MCP Server — used by both Node.js (index.ts) and CF Worker (worker.ts)
|
|
3
3
|
*/
|
|
4
4
|
|
|
5
|
-
import {
|
|
6
|
-
import {
|
|
5
|
+
import { Server } from '@modelcontextprotocol/sdk/server/index.js';
|
|
6
|
+
import {
|
|
7
|
+
CallToolRequestSchema,
|
|
8
|
+
ListToolsRequestSchema,
|
|
9
|
+
ListPromptsRequestSchema,
|
|
10
|
+
GetPromptRequestSchema,
|
|
11
|
+
ListResourcesRequestSchema,
|
|
12
|
+
ReadResourceRequestSchema,
|
|
13
|
+
} from '@modelcontextprotocol/sdk/types.js';
|
|
7
14
|
import { GmailClient } from './gmail-client.js';
|
|
8
15
|
import { TOOLS } from './tools.js';
|
|
9
16
|
|
|
@@ -182,175 +189,188 @@ export function handleToolCall(
|
|
|
182
189
|
}
|
|
183
190
|
}
|
|
184
191
|
|
|
185
|
-
export function createServer(config?: GmailMcpConfig) {
|
|
186
|
-
const server = new
|
|
187
|
-
name: 'gmail-mcp',
|
|
188
|
-
|
|
189
|
-
|
|
192
|
+
export function createServer(config?: GmailMcpConfig): Server {
|
|
193
|
+
const server = new Server(
|
|
194
|
+
{ name: 'gmail-mcp', version: '1.0.0' },
|
|
195
|
+
{ capabilities: { tools: {}, prompts: {}, resources: {} } }
|
|
196
|
+
);
|
|
190
197
|
|
|
191
198
|
let client: GmailClient | null = null;
|
|
192
199
|
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
200
|
+
// List available tools
|
|
201
|
+
server.setRequestHandler(ListToolsRequestSchema, async () => ({
|
|
202
|
+
tools: TOOLS.map(tool => {
|
|
203
|
+
const hasProperties = Object.keys(tool.inputSchema.properties).length > 0;
|
|
204
|
+
return {
|
|
205
|
+
name: tool.name,
|
|
197
206
|
description: tool.description,
|
|
198
|
-
inputSchema: tool.inputSchema
|
|
207
|
+
...(hasProperties ? { inputSchema: tool.inputSchema } : {}),
|
|
199
208
|
annotations: tool.annotations,
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
|
|
203
|
-
config?.clientId ||
|
|
204
|
-
(args as Record<string, unknown>).GOOGLE_CLIENT_ID as string;
|
|
205
|
-
const clientSecret =
|
|
206
|
-
config?.clientSecret ||
|
|
207
|
-
(args as Record<string, unknown>).GOOGLE_CLIENT_SECRET as string;
|
|
208
|
-
const refreshToken =
|
|
209
|
-
config?.refreshToken ||
|
|
210
|
-
(args as Record<string, unknown>).GOOGLE_REFRESH_TOKEN as string;
|
|
209
|
+
};
|
|
210
|
+
}),
|
|
211
|
+
}));
|
|
211
212
|
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
isError: true,
|
|
216
|
-
};
|
|
217
|
-
}
|
|
213
|
+
// Handle tool calls
|
|
214
|
+
server.setRequestHandler(CallToolRequestSchema, async (request) => {
|
|
215
|
+
const { name, arguments: args } = request.params;
|
|
218
216
|
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
217
|
+
try {
|
|
218
|
+
const clientId =
|
|
219
|
+
config?.clientId ||
|
|
220
|
+
(args as Record<string, unknown>)?.GOOGLE_CLIENT_ID as string;
|
|
221
|
+
const clientSecret =
|
|
222
|
+
config?.clientSecret ||
|
|
223
|
+
(args as Record<string, unknown>)?.GOOGLE_CLIENT_SECRET as string;
|
|
224
|
+
const refreshToken =
|
|
225
|
+
config?.refreshToken ||
|
|
226
|
+
(args as Record<string, unknown>)?.GOOGLE_REFRESH_TOKEN as string;
|
|
222
227
|
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
isError: false,
|
|
229
|
-
};
|
|
230
|
-
} catch (error) {
|
|
231
|
-
return {
|
|
232
|
-
content: [{ type: 'text' as const, text: `Error: ${error instanceof Error ? error.message : String(error)}` }],
|
|
233
|
-
isError: true,
|
|
234
|
-
};
|
|
235
|
-
}
|
|
228
|
+
if (!clientId || !clientSecret || !refreshToken) {
|
|
229
|
+
return {
|
|
230
|
+
content: [{ type: 'text' as const, text: 'Error: GOOGLE_CLIENT_ID, GOOGLE_CLIENT_SECRET, and GOOGLE_REFRESH_TOKEN are all required.' }],
|
|
231
|
+
isError: true,
|
|
232
|
+
};
|
|
236
233
|
}
|
|
237
|
-
);
|
|
238
|
-
}
|
|
239
234
|
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
'Guide for composing and sending emails, managing drafts',
|
|
244
|
-
async () => ({
|
|
245
|
-
messages: [{
|
|
246
|
-
role: 'user' as const,
|
|
247
|
-
content: {
|
|
248
|
-
type: 'text' as const,
|
|
249
|
-
text: [
|
|
250
|
-
'You are a Gmail email assistant.',
|
|
251
|
-
'',
|
|
252
|
-
'Sending emails:',
|
|
253
|
-
'1. **Send directly** — gmail_send_message with to, subject, body',
|
|
254
|
-
'2. **HTML emails** — Include html parameter for rich formatting',
|
|
255
|
-
'3. **Reply to thread** — Set thread_id, in_reply_to (Message-ID header), and references',
|
|
256
|
-
'4. **CC/BCC** — Comma-separate multiple addresses',
|
|
257
|
-
'',
|
|
258
|
-
'Working with drafts:',
|
|
259
|
-
'1. **Create draft** — gmail_create_draft (same params as send)',
|
|
260
|
-
'2. **Update draft** — gmail_update_draft replaces the entire draft',
|
|
261
|
-
'3. **Send draft** — gmail_send_draft with the draft ID',
|
|
262
|
-
'4. **List drafts** — gmail_list_drafts to see all drafts',
|
|
263
|
-
'',
|
|
264
|
-
'Tips:',
|
|
265
|
-
'- For replies, always get the original message first to extract Message-ID and thread_id',
|
|
266
|
-
'- Use gmail_get_thread to see the full conversation before replying',
|
|
267
|
-
'- HTML body is sent alongside plain text as multipart/alternative',
|
|
268
|
-
].join('\n'),
|
|
269
|
-
},
|
|
270
|
-
}],
|
|
271
|
-
}),
|
|
272
|
-
);
|
|
235
|
+
if (!client || config?.clientId !== clientId) {
|
|
236
|
+
client = new GmailClient({ clientId, clientSecret, refreshToken });
|
|
237
|
+
}
|
|
273
238
|
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
|
|
278
|
-
|
|
279
|
-
|
|
280
|
-
|
|
281
|
-
|
|
282
|
-
|
|
283
|
-
|
|
284
|
-
|
|
285
|
-
|
|
286
|
-
|
|
287
|
-
'- to:user@example.com — Messages to a recipient',
|
|
288
|
-
'- subject:"meeting notes" — Subject contains text',
|
|
289
|
-
'- has:attachment — Messages with attachments',
|
|
290
|
-
'- is:unread / is:starred / is:important',
|
|
291
|
-
'- label:custom-label — Messages with a specific label',
|
|
292
|
-
'- after:2026/01/01 / before:2026/12/31 — Date range',
|
|
293
|
-
'- newer_than:2d / older_than:1y — Relative dates',
|
|
294
|
-
'- filename:pdf — Attachments by type',
|
|
295
|
-
'- Combine: "from:boss@company.com has:attachment is:unread"',
|
|
296
|
-
'',
|
|
297
|
-
'Organizing with labels:',
|
|
298
|
-
'1. **List labels** — gmail_list_labels to see all labels',
|
|
299
|
-
'2. **Create label** — gmail_create_label with nested support ("Projects/Active")',
|
|
300
|
-
'3. **Apply labels** — gmail_modify_message to add/remove labels',
|
|
301
|
-
'4. **Mark as read** — Remove "UNREAD" label',
|
|
302
|
-
'5. **Star message** — Add "STARRED" label',
|
|
303
|
-
'6. **Batch operations** — gmail_batch_modify for bulk label changes',
|
|
304
|
-
'',
|
|
305
|
-
'System labels: INBOX, SENT, DRAFT, SPAM, TRASH, UNREAD, STARRED, IMPORTANT, CATEGORY_PERSONAL, CATEGORY_SOCIAL, CATEGORY_PROMOTIONS, CATEGORY_UPDATES, CATEGORY_FORUMS',
|
|
306
|
-
].join('\n'),
|
|
307
|
-
},
|
|
308
|
-
}],
|
|
309
|
-
}),
|
|
310
|
-
);
|
|
239
|
+
const result = await handleToolCall(name, args || {}, client);
|
|
240
|
+
const text = result === undefined ? '{"success": true}' : JSON.stringify(result, null, 2);
|
|
241
|
+
return {
|
|
242
|
+
content: [{ type: 'text' as const, text }],
|
|
243
|
+
isError: false,
|
|
244
|
+
};
|
|
245
|
+
} catch (error) {
|
|
246
|
+
return {
|
|
247
|
+
content: [{ type: 'text' as const, text: `Error: ${error instanceof Error ? error.message : String(error)}` }],
|
|
248
|
+
isError: true,
|
|
249
|
+
};
|
|
250
|
+
}
|
|
251
|
+
});
|
|
311
252
|
|
|
312
|
-
//
|
|
313
|
-
server.
|
|
314
|
-
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
|
|
321
|
-
|
|
322
|
-
|
|
323
|
-
|
|
324
|
-
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
253
|
+
// List prompts
|
|
254
|
+
server.setRequestHandler(ListPromptsRequestSchema, async () => ({
|
|
255
|
+
prompts: [
|
|
256
|
+
{ name: 'compose-and-send', description: 'Guide for composing and sending emails, managing drafts' },
|
|
257
|
+
{ name: 'search-and-organize', description: 'Guide for searching emails, managing labels, and organizing the mailbox' },
|
|
258
|
+
],
|
|
259
|
+
}));
|
|
260
|
+
|
|
261
|
+
// Get prompt
|
|
262
|
+
server.setRequestHandler(GetPromptRequestSchema, async (request) => {
|
|
263
|
+
const { name } = request.params;
|
|
264
|
+
|
|
265
|
+
if (name === 'compose-and-send') {
|
|
266
|
+
return {
|
|
267
|
+
messages: [{
|
|
268
|
+
role: 'user' as const,
|
|
269
|
+
content: {
|
|
270
|
+
type: 'text' as const,
|
|
271
|
+
text: [
|
|
272
|
+
'You are a Gmail email assistant.',
|
|
273
|
+
'',
|
|
274
|
+
'Sending emails:',
|
|
275
|
+
'1. **Send directly** — gmail_send_message with to, subject, body',
|
|
276
|
+
'2. **HTML emails** — Include html parameter for rich formatting',
|
|
277
|
+
'3. **Reply to thread** — Set thread_id, in_reply_to (Message-ID header), and references',
|
|
278
|
+
'4. **CC/BCC** — Comma-separate multiple addresses',
|
|
279
|
+
'',
|
|
280
|
+
'Working with drafts:',
|
|
281
|
+
'1. **Create draft** — gmail_create_draft (same params as send)',
|
|
282
|
+
'2. **Update draft** — gmail_update_draft replaces the entire draft',
|
|
283
|
+
'3. **Send draft** — gmail_send_draft with the draft ID',
|
|
284
|
+
'4. **List drafts** — gmail_list_drafts to see all drafts',
|
|
285
|
+
'',
|
|
286
|
+
'Tips:',
|
|
287
|
+
'- For replies, always get the original message first to extract Message-ID and thread_id',
|
|
288
|
+
'- Use gmail_get_thread to see the full conversation before replying',
|
|
289
|
+
'- HTML body is sent alongside plain text as multipart/alternative',
|
|
290
|
+
].join('\n'),
|
|
336
291
|
},
|
|
337
|
-
},
|
|
338
|
-
}
|
|
339
|
-
}
|
|
340
|
-
);
|
|
292
|
+
}],
|
|
293
|
+
};
|
|
294
|
+
}
|
|
341
295
|
|
|
342
|
-
|
|
343
|
-
(server as any).server.setRequestHandler(ListToolsRequestSchema, () => ({
|
|
344
|
-
tools: TOOLS.map(tool => {
|
|
345
|
-
const hasProperties = Object.keys(tool.inputSchema.properties).length > 0;
|
|
296
|
+
if (name === 'search-and-organize') {
|
|
346
297
|
return {
|
|
347
|
-
|
|
348
|
-
|
|
349
|
-
|
|
350
|
-
|
|
298
|
+
messages: [{
|
|
299
|
+
role: 'user' as const,
|
|
300
|
+
content: {
|
|
301
|
+
type: 'text' as const,
|
|
302
|
+
text: [
|
|
303
|
+
'You are a Gmail organization assistant.',
|
|
304
|
+
'',
|
|
305
|
+
'Search syntax (q parameter):',
|
|
306
|
+
'- from:user@example.com — Messages from a sender',
|
|
307
|
+
'- to:user@example.com — Messages to a recipient',
|
|
308
|
+
'- subject:"meeting notes" — Subject contains text',
|
|
309
|
+
'- has:attachment — Messages with attachments',
|
|
310
|
+
'- is:unread / is:starred / is:important',
|
|
311
|
+
'- label:custom-label — Messages with a specific label',
|
|
312
|
+
'- after:2026/01/01 / before:2026/12/31 — Date range',
|
|
313
|
+
'- newer_than:2d / older_than:1y — Relative dates',
|
|
314
|
+
'- filename:pdf — Attachments by type',
|
|
315
|
+
'- Combine: "from:boss@company.com has:attachment is:unread"',
|
|
316
|
+
'',
|
|
317
|
+
'Organizing with labels:',
|
|
318
|
+
'1. **List labels** — gmail_list_labels to see all labels',
|
|
319
|
+
'2. **Create label** — gmail_create_label with nested support ("Projects/Active")',
|
|
320
|
+
'3. **Apply labels** — gmail_modify_message to add/remove labels',
|
|
321
|
+
'4. **Mark as read** — Remove "UNREAD" label',
|
|
322
|
+
'5. **Star message** — Add "STARRED" label',
|
|
323
|
+
'6. **Batch operations** — gmail_batch_modify for bulk label changes',
|
|
324
|
+
'',
|
|
325
|
+
'System labels: INBOX, SENT, DRAFT, SPAM, TRASH, UNREAD, STARRED, IMPORTANT, CATEGORY_PERSONAL, CATEGORY_SOCIAL, CATEGORY_PROMOTIONS, CATEGORY_UPDATES, CATEGORY_FORUMS',
|
|
326
|
+
].join('\n'),
|
|
327
|
+
},
|
|
328
|
+
}],
|
|
351
329
|
};
|
|
352
|
-
}
|
|
330
|
+
}
|
|
331
|
+
|
|
332
|
+
throw new Error(`Unknown prompt: ${name}`);
|
|
333
|
+
});
|
|
334
|
+
|
|
335
|
+
// List resources
|
|
336
|
+
server.setRequestHandler(ListResourcesRequestSchema, async () => ({
|
|
337
|
+
resources: [{
|
|
338
|
+
uri: 'gmail://server-info',
|
|
339
|
+
name: 'server-info',
|
|
340
|
+
description: 'Connection status and available tools for this Gmail MCP server',
|
|
341
|
+
mimeType: 'application/json',
|
|
342
|
+
}],
|
|
353
343
|
}));
|
|
354
344
|
|
|
345
|
+
// Read resource
|
|
346
|
+
server.setRequestHandler(ReadResourceRequestSchema, async (request) => {
|
|
347
|
+
const { uri } = request.params;
|
|
348
|
+
|
|
349
|
+
if (uri === 'gmail://server-info') {
|
|
350
|
+
return {
|
|
351
|
+
contents: [{
|
|
352
|
+
uri: 'gmail://server-info',
|
|
353
|
+
mimeType: 'application/json',
|
|
354
|
+
text: JSON.stringify({
|
|
355
|
+
name: 'gmail-mcp',
|
|
356
|
+
version: '1.0.0',
|
|
357
|
+
connected: !!config,
|
|
358
|
+
has_oauth: !!(config?.clientId),
|
|
359
|
+
tools_available: TOOLS.length,
|
|
360
|
+
tool_categories: {
|
|
361
|
+
messages: 10,
|
|
362
|
+
drafts: 6,
|
|
363
|
+
labels: 5,
|
|
364
|
+
threads: 5,
|
|
365
|
+
settings: 2,
|
|
366
|
+
},
|
|
367
|
+
}, null, 2),
|
|
368
|
+
}],
|
|
369
|
+
};
|
|
370
|
+
}
|
|
371
|
+
|
|
372
|
+
throw new Error(`Unknown resource: ${uri}`);
|
|
373
|
+
});
|
|
374
|
+
|
|
355
375
|
return server;
|
|
356
376
|
}
|