mcp-google-extras 1.0.4 → 1.0.5
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.
|
@@ -10,6 +10,8 @@ import { register as renameFile } from './renameFile.js';
|
|
|
10
10
|
import { register as deleteFile } from './deleteFile.js';
|
|
11
11
|
import { register as createDocument } from './createDocument.js';
|
|
12
12
|
import { register as createFromTemplate } from './createFromTemplate.js';
|
|
13
|
+
import { register as listSharedDrives } from './listSharedDrives.js';
|
|
14
|
+
import { register as listSharedWithMe } from './listSharedWithMe.js';
|
|
13
15
|
export function registerDriveTools(server) {
|
|
14
16
|
listGoogleDocs(server);
|
|
15
17
|
searchGoogleDocs(server);
|
|
@@ -23,4 +25,6 @@ export function registerDriveTools(server) {
|
|
|
23
25
|
deleteFile(server);
|
|
24
26
|
createDocument(server);
|
|
25
27
|
createFromTemplate(server);
|
|
28
|
+
listSharedDrives(server);
|
|
29
|
+
listSharedWithMe(server);
|
|
26
30
|
}
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
import { UserError } from 'fastmcp';
|
|
2
|
+
import { z } from 'zod';
|
|
3
|
+
import { getDriveClient } from '../../clients.js';
|
|
4
|
+
export function register(server) {
|
|
5
|
+
server.addTool({
|
|
6
|
+
name: 'listSharedDrives',
|
|
7
|
+
description: 'Lists all Shared Drives (Team Drives) the user has access to. Use the returned drive ID with listFolderContents to browse its contents.',
|
|
8
|
+
parameters: z.object({
|
|
9
|
+
maxResults: z
|
|
10
|
+
.number()
|
|
11
|
+
.int()
|
|
12
|
+
.min(1)
|
|
13
|
+
.max(100)
|
|
14
|
+
.optional()
|
|
15
|
+
.default(50)
|
|
16
|
+
.describe('Maximum number of shared drives to return.'),
|
|
17
|
+
pageToken: z
|
|
18
|
+
.string()
|
|
19
|
+
.optional()
|
|
20
|
+
.describe('Token for fetching the next page of results.'),
|
|
21
|
+
}),
|
|
22
|
+
execute: async (args, { log }) => {
|
|
23
|
+
const drive = await getDriveClient();
|
|
24
|
+
log.info('Listing shared drives');
|
|
25
|
+
try {
|
|
26
|
+
const response = await drive.drives.list({
|
|
27
|
+
pageSize: args.maxResults,
|
|
28
|
+
pageToken: args.pageToken || undefined,
|
|
29
|
+
fields: 'nextPageToken,drives(id,name,createdTime,hidden)',
|
|
30
|
+
});
|
|
31
|
+
const drives = (response.data.drives || []).map((d) => ({
|
|
32
|
+
id: d.id,
|
|
33
|
+
name: d.name,
|
|
34
|
+
createdTime: d.createdTime,
|
|
35
|
+
hidden: d.hidden || false,
|
|
36
|
+
}));
|
|
37
|
+
const result = {
|
|
38
|
+
drives,
|
|
39
|
+
nextPageToken: response.data.nextPageToken || null,
|
|
40
|
+
totalCount: drives.length,
|
|
41
|
+
};
|
|
42
|
+
return JSON.stringify(result, null, 2);
|
|
43
|
+
}
|
|
44
|
+
catch (error) {
|
|
45
|
+
log.error(`Error listing shared drives: ${error.message || error}`);
|
|
46
|
+
if (error.code === 403)
|
|
47
|
+
throw new UserError('Permission denied. Make sure the Drive API scope includes shared drives.');
|
|
48
|
+
throw new UserError(`Failed to list shared drives: ${error.message || 'Unknown error'}`);
|
|
49
|
+
}
|
|
50
|
+
},
|
|
51
|
+
});
|
|
52
|
+
}
|
|
@@ -0,0 +1,97 @@
|
|
|
1
|
+
import { UserError } from 'fastmcp';
|
|
2
|
+
import { z } from 'zod';
|
|
3
|
+
import { getDriveClient } from '../../clients.js';
|
|
4
|
+
export function register(server) {
|
|
5
|
+
server.addTool({
|
|
6
|
+
name: 'listSharedWithMe',
|
|
7
|
+
description: 'Lists files and folders that have been shared with the user ("Shared with me"). Returns items shared by others that are not in the user\'s own Drive. Use folder IDs with listFolderContents to navigate into shared folders.',
|
|
8
|
+
parameters: z.object({
|
|
9
|
+
mimeType: z
|
|
10
|
+
.string()
|
|
11
|
+
.optional()
|
|
12
|
+
.describe('Filter by MIME type. Use "application/vnd.google-apps.folder" for folders only, "application/vnd.google-apps.document" for Docs, "application/vnd.google-apps.spreadsheet" for Sheets, etc.'),
|
|
13
|
+
query: z
|
|
14
|
+
.string()
|
|
15
|
+
.optional()
|
|
16
|
+
.describe('Search query to filter shared items by name (uses "contains" matching).'),
|
|
17
|
+
maxResults: z
|
|
18
|
+
.number()
|
|
19
|
+
.int()
|
|
20
|
+
.min(1)
|
|
21
|
+
.max(100)
|
|
22
|
+
.optional()
|
|
23
|
+
.default(50)
|
|
24
|
+
.describe('Maximum number of items to return.'),
|
|
25
|
+
pageToken: z
|
|
26
|
+
.string()
|
|
27
|
+
.optional()
|
|
28
|
+
.describe('Token for fetching the next page of results.'),
|
|
29
|
+
orderBy: z
|
|
30
|
+
.enum(['name', 'modifiedTime', 'sharedWithMeTime'])
|
|
31
|
+
.optional()
|
|
32
|
+
.default('sharedWithMeTime')
|
|
33
|
+
.describe('How to sort results. Defaults to most recently shared first.'),
|
|
34
|
+
}),
|
|
35
|
+
execute: async (args, { log }) => {
|
|
36
|
+
const drive = await getDriveClient();
|
|
37
|
+
log.info('Listing shared with me');
|
|
38
|
+
try {
|
|
39
|
+
let queryParts = ['sharedWithMe=true', 'trashed=false'];
|
|
40
|
+
if (args.mimeType) {
|
|
41
|
+
queryParts.push(`mimeType='${args.mimeType}'`);
|
|
42
|
+
}
|
|
43
|
+
if (args.query) {
|
|
44
|
+
queryParts.push(`name contains '${args.query.replace(/'/g, "\\'")}'`);
|
|
45
|
+
}
|
|
46
|
+
const orderBy = args.orderBy === 'sharedWithMeTime'
|
|
47
|
+
? 'sharedWithMeTime desc'
|
|
48
|
+
: args.orderBy === 'modifiedTime'
|
|
49
|
+
? 'modifiedTime desc'
|
|
50
|
+
: 'name';
|
|
51
|
+
const response = await drive.files.list({
|
|
52
|
+
q: queryParts.join(' and '),
|
|
53
|
+
pageSize: args.maxResults,
|
|
54
|
+
pageToken: args.pageToken || undefined,
|
|
55
|
+
orderBy,
|
|
56
|
+
fields: 'nextPageToken,files(id,name,mimeType,modifiedTime,sharedWithMeTime,sharingUser(displayName,emailAddress),webViewLink,owners(displayName))',
|
|
57
|
+
supportsAllDrives: true,
|
|
58
|
+
includeItemsFromAllDrives: true,
|
|
59
|
+
});
|
|
60
|
+
const items = response.data.files || [];
|
|
61
|
+
const folders = items
|
|
62
|
+
.filter((f) => f.mimeType === 'application/vnd.google-apps.folder')
|
|
63
|
+
.map((f) => ({
|
|
64
|
+
id: f.id,
|
|
65
|
+
name: f.name,
|
|
66
|
+
sharedWithMeTime: f.sharedWithMeTime,
|
|
67
|
+
sharedBy: f.sharingUser?.displayName || f.owners?.[0]?.displayName || null,
|
|
68
|
+
sharedByEmail: f.sharingUser?.emailAddress || null,
|
|
69
|
+
}));
|
|
70
|
+
const files = items
|
|
71
|
+
.filter((f) => f.mimeType !== 'application/vnd.google-apps.folder')
|
|
72
|
+
.map((f) => ({
|
|
73
|
+
id: f.id,
|
|
74
|
+
name: f.name,
|
|
75
|
+
mimeType: f.mimeType,
|
|
76
|
+
modifiedTime: f.modifiedTime,
|
|
77
|
+
sharedWithMeTime: f.sharedWithMeTime,
|
|
78
|
+
sharedBy: f.sharingUser?.displayName || f.owners?.[0]?.displayName || null,
|
|
79
|
+
sharedByEmail: f.sharingUser?.emailAddress || null,
|
|
80
|
+
}));
|
|
81
|
+
const result = {
|
|
82
|
+
folders,
|
|
83
|
+
files,
|
|
84
|
+
nextPageToken: response.data.nextPageToken || null,
|
|
85
|
+
totalCount: items.length,
|
|
86
|
+
};
|
|
87
|
+
return JSON.stringify(result, null, 2);
|
|
88
|
+
}
|
|
89
|
+
catch (error) {
|
|
90
|
+
log.error(`Error listing shared items: ${error.message || error}`);
|
|
91
|
+
if (error.code === 403)
|
|
92
|
+
throw new UserError('Permission denied. Make sure you have the correct Drive API scopes.');
|
|
93
|
+
throw new UserError(`Failed to list shared items: ${error.message || 'Unknown error'}`);
|
|
94
|
+
}
|
|
95
|
+
},
|
|
96
|
+
});
|
|
97
|
+
}
|