@mcp-z/mcp-drive 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 +163 -0
- package/bin/server.js +5 -0
- package/dist/cjs/constants.d.cts +7 -0
- package/dist/cjs/constants.d.ts +7 -0
- package/dist/cjs/constants.js +18 -0
- package/dist/cjs/constants.js.map +1 -0
- package/dist/cjs/index.d.cts +8 -0
- package/dist/cjs/index.d.ts +8 -0
- package/dist/cjs/index.js +314 -0
- package/dist/cjs/index.js.map +1 -0
- package/dist/cjs/lib/create-store.d.cts +2 -0
- package/dist/cjs/lib/create-store.d.ts +2 -0
- package/dist/cjs/lib/create-store.js +166 -0
- package/dist/cjs/lib/create-store.js.map +1 -0
- package/dist/cjs/lib/query-builder.d.cts +45 -0
- package/dist/cjs/lib/query-builder.d.ts +45 -0
- package/dist/cjs/lib/query-builder.js +219 -0
- package/dist/cjs/lib/query-builder.js.map +1 -0
- package/dist/cjs/mcp/index.d.cts +3 -0
- package/dist/cjs/mcp/index.d.ts +3 -0
- package/dist/cjs/mcp/index.js +66 -0
- package/dist/cjs/mcp/index.js.map +1 -0
- package/dist/cjs/mcp/prompts/index.d.cts +2 -0
- package/dist/cjs/mcp/prompts/index.d.ts +2 -0
- package/dist/cjs/mcp/prompts/index.js +26 -0
- package/dist/cjs/mcp/prompts/index.js.map +1 -0
- package/dist/cjs/mcp/prompts/organize-files.d.cts +16 -0
- package/dist/cjs/mcp/prompts/organize-files.d.ts +16 -0
- package/dist/cjs/mcp/prompts/organize-files.js +169 -0
- package/dist/cjs/mcp/prompts/organize-files.js.map +1 -0
- package/dist/cjs/mcp/prompts/query-syntax.d.cts +19 -0
- package/dist/cjs/mcp/prompts/query-syntax.d.ts +19 -0
- package/dist/cjs/mcp/prompts/query-syntax.js +169 -0
- package/dist/cjs/mcp/prompts/query-syntax.js.map +1 -0
- package/dist/cjs/mcp/resources/file.d.cts +9 -0
- package/dist/cjs/mcp/resources/file.d.ts +9 -0
- package/dist/cjs/mcp/resources/file.js +247 -0
- package/dist/cjs/mcp/resources/file.js.map +1 -0
- package/dist/cjs/mcp/resources/index.d.cts +1 -0
- package/dist/cjs/mcp/resources/index.d.ts +1 -0
- package/dist/cjs/mcp/resources/index.js +17 -0
- package/dist/cjs/mcp/resources/index.js.map +1 -0
- package/dist/cjs/mcp/tools/file-move-to-trash.d.cts +59 -0
- package/dist/cjs/mcp/tools/file-move-to-trash.d.ts +59 -0
- package/dist/cjs/mcp/tools/file-move-to-trash.js +334 -0
- package/dist/cjs/mcp/tools/file-move-to-trash.js.map +1 -0
- package/dist/cjs/mcp/tools/file-move.d.cts +73 -0
- package/dist/cjs/mcp/tools/file-move.d.ts +73 -0
- package/dist/cjs/mcp/tools/file-move.js +613 -0
- package/dist/cjs/mcp/tools/file-move.js.map +1 -0
- package/dist/cjs/mcp/tools/files-search.d.cts +135 -0
- package/dist/cjs/mcp/tools/files-search.d.ts +135 -0
- package/dist/cjs/mcp/tools/files-search.js +558 -0
- package/dist/cjs/mcp/tools/files-search.js.map +1 -0
- package/dist/cjs/mcp/tools/folder-contents.d.cts +139 -0
- package/dist/cjs/mcp/tools/folder-contents.d.ts +139 -0
- package/dist/cjs/mcp/tools/folder-contents.js +513 -0
- package/dist/cjs/mcp/tools/folder-contents.js.map +1 -0
- package/dist/cjs/mcp/tools/folder-create.d.cts +59 -0
- package/dist/cjs/mcp/tools/folder-create.d.ts +59 -0
- package/dist/cjs/mcp/tools/folder-create.js +368 -0
- package/dist/cjs/mcp/tools/folder-create.js.map +1 -0
- package/dist/cjs/mcp/tools/folder-path.d.cts +49 -0
- package/dist/cjs/mcp/tools/folder-path.d.ts +49 -0
- package/dist/cjs/mcp/tools/folder-path.js +367 -0
- package/dist/cjs/mcp/tools/folder-path.js.map +1 -0
- package/dist/cjs/mcp/tools/folder-search.d.cts +139 -0
- package/dist/cjs/mcp/tools/folder-search.d.ts +139 -0
- package/dist/cjs/mcp/tools/folder-search.js +760 -0
- package/dist/cjs/mcp/tools/folder-search.js.map +1 -0
- package/dist/cjs/mcp/tools/index.d.cts +7 -0
- package/dist/cjs/mcp/tools/index.d.ts +7 -0
- package/dist/cjs/mcp/tools/index.js +46 -0
- package/dist/cjs/mcp/tools/index.js.map +1 -0
- package/dist/cjs/package.json +1 -0
- package/dist/cjs/schemas/drive-query-schema.d.cts +40 -0
- package/dist/cjs/schemas/drive-query-schema.d.ts +40 -0
- package/dist/cjs/schemas/drive-query-schema.js +90 -0
- package/dist/cjs/schemas/drive-query-schema.js.map +1 -0
- package/dist/cjs/schemas/drive-validation.d.cts +48 -0
- package/dist/cjs/schemas/drive-validation.d.ts +48 -0
- package/dist/cjs/schemas/drive-validation.js +96 -0
- package/dist/cjs/schemas/drive-validation.js.map +1 -0
- package/dist/cjs/schemas/index.d.cts +2 -0
- package/dist/cjs/schemas/index.d.ts +2 -0
- package/dist/cjs/schemas/index.js +20 -0
- package/dist/cjs/schemas/index.js.map +1 -0
- package/dist/cjs/setup/config.d.cts +44 -0
- package/dist/cjs/setup/config.d.ts +44 -0
- package/dist/cjs/setup/config.js +201 -0
- package/dist/cjs/setup/config.js.map +1 -0
- package/dist/cjs/setup/http.d.cts +8 -0
- package/dist/cjs/setup/http.d.ts +8 -0
- package/dist/cjs/setup/http.js +260 -0
- package/dist/cjs/setup/http.js.map +1 -0
- package/dist/cjs/setup/index.d.cts +5 -0
- package/dist/cjs/setup/index.d.ts +5 -0
- package/dist/cjs/setup/index.js +46 -0
- package/dist/cjs/setup/index.js.map +1 -0
- package/dist/cjs/setup/oauth-google.d.cts +64 -0
- package/dist/cjs/setup/oauth-google.d.ts +64 -0
- package/dist/cjs/setup/oauth-google.js +347 -0
- package/dist/cjs/setup/oauth-google.js.map +1 -0
- package/dist/cjs/setup/runtime.d.cts +10 -0
- package/dist/cjs/setup/runtime.d.ts +10 -0
- package/dist/cjs/setup/runtime.js +353 -0
- package/dist/cjs/setup/runtime.js.map +1 -0
- package/dist/cjs/setup/stdio.d.cts +7 -0
- package/dist/cjs/setup/stdio.d.ts +7 -0
- package/dist/cjs/setup/stdio.js +239 -0
- package/dist/cjs/setup/stdio.js.map +1 -0
- package/dist/cjs/types.d.cts +45 -0
- package/dist/cjs/types.d.ts +45 -0
- package/dist/cjs/types.js +5 -0
- package/dist/cjs/types.js.map +1 -0
- package/dist/esm/constants.d.ts +7 -0
- package/dist/esm/constants.js +7 -0
- package/dist/esm/constants.js.map +1 -0
- package/dist/esm/index.d.ts +8 -0
- package/dist/esm/index.js +34 -0
- package/dist/esm/index.js.map +1 -0
- package/dist/esm/lib/create-store.d.ts +2 -0
- package/dist/esm/lib/create-store.js +6 -0
- package/dist/esm/lib/create-store.js.map +1 -0
- package/dist/esm/lib/query-builder.d.ts +45 -0
- package/dist/esm/lib/query-builder.js +184 -0
- package/dist/esm/lib/query-builder.js.map +1 -0
- package/dist/esm/mcp/index.d.ts +3 -0
- package/dist/esm/mcp/index.js +6 -0
- package/dist/esm/mcp/index.js.map +1 -0
- package/dist/esm/mcp/prompts/index.d.ts +2 -0
- package/dist/esm/mcp/prompts/index.js +2 -0
- package/dist/esm/mcp/prompts/index.js.map +1 -0
- package/dist/esm/mcp/prompts/organize-files.d.ts +16 -0
- package/dist/esm/mcp/prompts/organize-files.js +21 -0
- package/dist/esm/mcp/prompts/organize-files.js.map +1 -0
- package/dist/esm/mcp/prompts/query-syntax.d.ts +19 -0
- package/dist/esm/mcp/prompts/query-syntax.js +82 -0
- package/dist/esm/mcp/prompts/query-syntax.js.map +1 -0
- package/dist/esm/mcp/resources/file.d.ts +9 -0
- package/dist/esm/mcp/resources/file.js +77 -0
- package/dist/esm/mcp/resources/file.js.map +1 -0
- package/dist/esm/mcp/resources/index.d.ts +1 -0
- package/dist/esm/mcp/resources/index.js +1 -0
- package/dist/esm/mcp/resources/index.js.map +1 -0
- package/dist/esm/mcp/tools/file-move-to-trash.d.ts +59 -0
- package/dist/esm/mcp/tools/file-move-to-trash.js +118 -0
- package/dist/esm/mcp/tools/file-move-to-trash.js.map +1 -0
- package/dist/esm/mcp/tools/file-move.d.ts +73 -0
- package/dist/esm/mcp/tools/file-move.js +274 -0
- package/dist/esm/mcp/tools/file-move.js.map +1 -0
- package/dist/esm/mcp/tools/files-search.d.ts +135 -0
- package/dist/esm/mcp/tools/files-search.js +254 -0
- package/dist/esm/mcp/tools/files-search.js.map +1 -0
- package/dist/esm/mcp/tools/folder-contents.d.ts +139 -0
- package/dist/esm/mcp/tools/folder-contents.js +214 -0
- package/dist/esm/mcp/tools/folder-contents.js.map +1 -0
- package/dist/esm/mcp/tools/folder-create.d.ts +59 -0
- package/dist/esm/mcp/tools/folder-create.js +140 -0
- package/dist/esm/mcp/tools/folder-create.js.map +1 -0
- package/dist/esm/mcp/tools/folder-path.d.ts +49 -0
- package/dist/esm/mcp/tools/folder-path.js +147 -0
- package/dist/esm/mcp/tools/folder-path.js.map +1 -0
- package/dist/esm/mcp/tools/folder-search.d.ts +139 -0
- package/dist/esm/mcp/tools/folder-search.js +343 -0
- package/dist/esm/mcp/tools/folder-search.js.map +1 -0
- package/dist/esm/mcp/tools/index.d.ts +7 -0
- package/dist/esm/mcp/tools/index.js +7 -0
- package/dist/esm/mcp/tools/index.js.map +1 -0
- package/dist/esm/package.json +1 -0
- package/dist/esm/schemas/drive-query-schema.d.ts +40 -0
- package/dist/esm/schemas/drive-query-schema.js +84 -0
- package/dist/esm/schemas/drive-query-schema.js.map +1 -0
- package/dist/esm/schemas/drive-validation.d.ts +48 -0
- package/dist/esm/schemas/drive-validation.js +73 -0
- package/dist/esm/schemas/drive-validation.js.map +1 -0
- package/dist/esm/schemas/index.d.ts +2 -0
- package/dist/esm/schemas/index.js +2 -0
- package/dist/esm/schemas/index.js.map +1 -0
- package/dist/esm/setup/config.d.ts +44 -0
- package/dist/esm/setup/config.js +151 -0
- package/dist/esm/setup/config.js.map +1 -0
- package/dist/esm/setup/http.d.ts +8 -0
- package/dist/esm/setup/http.js +54 -0
- package/dist/esm/setup/http.js.map +1 -0
- package/dist/esm/setup/index.d.ts +5 -0
- package/dist/esm/setup/index.js +5 -0
- package/dist/esm/setup/index.js.map +1 -0
- package/dist/esm/setup/oauth-google.d.ts +64 -0
- package/dist/esm/setup/oauth-google.js +168 -0
- package/dist/esm/setup/oauth-google.js.map +1 -0
- package/dist/esm/setup/runtime.d.ts +10 -0
- package/dist/esm/setup/runtime.js +84 -0
- package/dist/esm/setup/runtime.js.map +1 -0
- package/dist/esm/setup/stdio.d.ts +7 -0
- package/dist/esm/setup/stdio.js +38 -0
- package/dist/esm/setup/stdio.js.map +1 -0
- package/dist/esm/types.d.ts +45 -0
- package/dist/esm/types.js +1 -0
- package/dist/esm/types.js.map +1 -0
- package/package.json +108 -0
|
@@ -0,0 +1,147 @@
|
|
|
1
|
+
import { schemas } from '@mcp-z/oauth-google';
|
|
2
|
+
const { AuthRequiredBranchSchema } = schemas;
|
|
3
|
+
import { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';
|
|
4
|
+
import { google } from 'googleapis';
|
|
5
|
+
import { z } from 'zod';
|
|
6
|
+
const inputSchema = z.object({
|
|
7
|
+
folderId: z.string().min(1).describe('Folder ID to get path for (or "root")')
|
|
8
|
+
});
|
|
9
|
+
// Success branch schema - uses items: for consistency with standard vocabulary
|
|
10
|
+
const successBranchSchema = z.object({
|
|
11
|
+
type: z.literal('success'),
|
|
12
|
+
path: z.string().describe('Full path from root (e.g., /Work/Projects/2024)'),
|
|
13
|
+
items: z.array(z.object({
|
|
14
|
+
id: z.string().describe('Folder ID'),
|
|
15
|
+
name: z.string().describe('Folder name')
|
|
16
|
+
})).describe('Path items from root to target folder')
|
|
17
|
+
});
|
|
18
|
+
// Output schema with auth_required support
|
|
19
|
+
const outputSchema = z.discriminatedUnion('type', [
|
|
20
|
+
successBranchSchema,
|
|
21
|
+
AuthRequiredBranchSchema
|
|
22
|
+
]);
|
|
23
|
+
const config = {
|
|
24
|
+
title: 'Get Folder Path',
|
|
25
|
+
description: 'Get full path from folder to root. Returns human-readable path and items with IDs.',
|
|
26
|
+
inputSchema: inputSchema,
|
|
27
|
+
outputSchema: z.object({
|
|
28
|
+
result: outputSchema
|
|
29
|
+
})
|
|
30
|
+
};
|
|
31
|
+
/**
|
|
32
|
+
* Resolves the full path for a folder by walking up the parent chain.
|
|
33
|
+
* Returns both the path string and structured segments with IDs and names.
|
|
34
|
+
*/ async function resolveFolderPath(drive, folderId, logger) {
|
|
35
|
+
// Handle root specially
|
|
36
|
+
if (folderId === 'root') {
|
|
37
|
+
return {
|
|
38
|
+
path: '/',
|
|
39
|
+
segments: [
|
|
40
|
+
{
|
|
41
|
+
id: 'root',
|
|
42
|
+
name: 'My Drive'
|
|
43
|
+
}
|
|
44
|
+
]
|
|
45
|
+
};
|
|
46
|
+
}
|
|
47
|
+
const segments = [];
|
|
48
|
+
let currentId = folderId;
|
|
49
|
+
const visited = new Set();
|
|
50
|
+
// Walk up the parent chain
|
|
51
|
+
while(currentId && currentId !== 'root'){
|
|
52
|
+
// Prevent infinite loops
|
|
53
|
+
if (visited.has(currentId)) {
|
|
54
|
+
logger.info('Circular folder reference detected', {
|
|
55
|
+
folderId: currentId
|
|
56
|
+
});
|
|
57
|
+
break;
|
|
58
|
+
}
|
|
59
|
+
visited.add(currentId);
|
|
60
|
+
// Fetch folder metadata
|
|
61
|
+
try {
|
|
62
|
+
const response = await drive.files.get({
|
|
63
|
+
fileId: currentId,
|
|
64
|
+
fields: 'id,name,parents'
|
|
65
|
+
});
|
|
66
|
+
const id = response.data.id;
|
|
67
|
+
const name = response.data.name || id;
|
|
68
|
+
const parents = response.data.parents;
|
|
69
|
+
// Add to segments at beginning (we're walking from child to root)
|
|
70
|
+
segments.unshift({
|
|
71
|
+
id,
|
|
72
|
+
name
|
|
73
|
+
});
|
|
74
|
+
// Move to parent
|
|
75
|
+
currentId = (parents && parents.length > 0 ? parents[0] : '') || '';
|
|
76
|
+
} catch (e) {
|
|
77
|
+
logger.info('Failed to resolve folder path', {
|
|
78
|
+
folderId: currentId,
|
|
79
|
+
error: e
|
|
80
|
+
});
|
|
81
|
+
break;
|
|
82
|
+
}
|
|
83
|
+
}
|
|
84
|
+
// Add root if we reached it
|
|
85
|
+
if (currentId === 'root') {
|
|
86
|
+
segments.unshift({
|
|
87
|
+
id: 'root',
|
|
88
|
+
name: 'My Drive'
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
// Build path string
|
|
92
|
+
const pathParts = segments.slice(1).map((seg)=>seg.name); // Skip root
|
|
93
|
+
const path = pathParts.length > 0 ? `/${pathParts.join('/')}` : '/';
|
|
94
|
+
return {
|
|
95
|
+
path,
|
|
96
|
+
segments
|
|
97
|
+
};
|
|
98
|
+
}
|
|
99
|
+
async function handler({ folderId }, extra) {
|
|
100
|
+
const logger = extra.logger;
|
|
101
|
+
logger.info('drive.folder.path called', {
|
|
102
|
+
folderId
|
|
103
|
+
});
|
|
104
|
+
try {
|
|
105
|
+
const drive = google.drive({
|
|
106
|
+
version: 'v3',
|
|
107
|
+
auth: extra.authContext.auth
|
|
108
|
+
});
|
|
109
|
+
const pathResult = await resolveFolderPath(drive, folderId, logger);
|
|
110
|
+
logger.info('drive.folder.path returning', {
|
|
111
|
+
path: pathResult.path,
|
|
112
|
+
segmentCount: pathResult.segments.length
|
|
113
|
+
});
|
|
114
|
+
const result = {
|
|
115
|
+
type: 'success',
|
|
116
|
+
path: pathResult.path,
|
|
117
|
+
items: pathResult.segments
|
|
118
|
+
};
|
|
119
|
+
return {
|
|
120
|
+
content: [
|
|
121
|
+
{
|
|
122
|
+
type: 'text',
|
|
123
|
+
text: JSON.stringify(result)
|
|
124
|
+
}
|
|
125
|
+
],
|
|
126
|
+
structuredContent: {
|
|
127
|
+
result
|
|
128
|
+
}
|
|
129
|
+
};
|
|
130
|
+
} catch (error) {
|
|
131
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
132
|
+
logger.error('drive.folder.path error', {
|
|
133
|
+
error: message
|
|
134
|
+
});
|
|
135
|
+
// Throw McpError
|
|
136
|
+
throw new McpError(ErrorCode.InternalError, `Error getting folder path: ${message}`, {
|
|
137
|
+
stack: error instanceof Error ? error.stack : undefined
|
|
138
|
+
});
|
|
139
|
+
}
|
|
140
|
+
}
|
|
141
|
+
export default function createTool() {
|
|
142
|
+
return {
|
|
143
|
+
name: 'folder-path',
|
|
144
|
+
config,
|
|
145
|
+
handler
|
|
146
|
+
};
|
|
147
|
+
}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["/Users/kevin/Dev/Projects/ai/mcp-z/servers/mcp-drive/src/mcp/tools/folder-path.ts"],"sourcesContent":["import type { EnrichedExtra } from '@mcp-z/oauth-google';\nimport { schemas } from '@mcp-z/oauth-google';\n\nconst { AuthRequiredBranchSchema } = schemas;\n\nimport { type CallToolResult, ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';\nimport type { drive_v3 } from 'googleapis';\nimport { google } from 'googleapis';\nimport { z } from 'zod';\nimport type { Logger } from '../../types.js';\n\nconst inputSchema = z.object({\n folderId: z.string().min(1).describe('Folder ID to get path for (or \"root\")'),\n});\n\n// Success branch schema - uses items: for consistency with standard vocabulary\nconst successBranchSchema = z.object({\n type: z.literal('success'),\n path: z.string().describe('Full path from root (e.g., /Work/Projects/2024)'),\n items: z\n .array(\n z.object({\n id: z.string().describe('Folder ID'),\n name: z.string().describe('Folder name'),\n })\n )\n .describe('Path items from root to target folder'),\n});\n\n// Output schema with auth_required support\nconst outputSchema = z.discriminatedUnion('type', [successBranchSchema, AuthRequiredBranchSchema]);\n\nconst config = {\n title: 'Get Folder Path',\n description: 'Get full path from folder to root. Returns human-readable path and items with IDs.',\n inputSchema: inputSchema,\n outputSchema: z.object({\n result: outputSchema,\n }),\n} as const;\n\nexport type Input = z.infer<typeof inputSchema>;\nexport type Output = z.infer<typeof outputSchema>;\n\n/**\n * Resolves the full path for a folder by walking up the parent chain.\n * Returns both the path string and structured segments with IDs and names.\n */\nasync function resolveFolderPath(drive: drive_v3.Drive, folderId: string, logger: Logger): Promise<{ path: string; segments: Array<{ id: string; name: string }> }> {\n // Handle root specially\n if (folderId === 'root') {\n return {\n path: '/',\n segments: [{ id: 'root', name: 'My Drive' }],\n };\n }\n\n const segments: Array<{ id: string; name: string }> = [];\n let currentId = folderId;\n const visited = new Set<string>();\n\n // Walk up the parent chain\n while (currentId && currentId !== 'root') {\n // Prevent infinite loops\n if (visited.has(currentId)) {\n logger.info('Circular folder reference detected', {\n folderId: currentId,\n });\n break;\n }\n visited.add(currentId);\n\n // Fetch folder metadata\n try {\n const response = await drive.files.get({\n fileId: currentId,\n fields: 'id,name,parents',\n });\n\n const id = response.data.id as string;\n const name = (response.data.name as string) || id;\n const parents = response.data.parents as string[] | undefined;\n\n // Add to segments at beginning (we're walking from child to root)\n segments.unshift({ id, name });\n\n // Move to parent\n currentId = (parents && parents.length > 0 ? parents[0] : '') || '';\n } catch (e) {\n logger.info('Failed to resolve folder path', {\n folderId: currentId,\n error: e,\n });\n break;\n }\n }\n\n // Add root if we reached it\n if (currentId === 'root') {\n segments.unshift({ id: 'root', name: 'My Drive' });\n }\n\n // Build path string\n const pathParts = segments.slice(1).map((seg) => seg.name); // Skip root\n const path = pathParts.length > 0 ? `/${pathParts.join('/')}` : '/';\n\n return { path, segments };\n}\n\nasync function handler({ folderId }: Input, extra: EnrichedExtra): Promise<CallToolResult> {\n const logger = extra.logger;\n logger.info('drive.folder.path called', { folderId });\n\n try {\n const drive = google.drive({ version: 'v3', auth: extra.authContext.auth });\n\n const pathResult = await resolveFolderPath(drive, folderId, logger);\n\n logger.info('drive.folder.path returning', {\n path: pathResult.path,\n segmentCount: pathResult.segments.length,\n });\n\n const result: Output = {\n type: 'success' as const,\n path: pathResult.path,\n items: pathResult.segments,\n };\n\n return {\n content: [\n {\n type: 'text' as const,\n text: JSON.stringify(result),\n },\n ],\n structuredContent: { result },\n };\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error);\n logger.error('drive.folder.path error', { error: message });\n\n // Throw McpError\n throw new McpError(ErrorCode.InternalError, `Error getting folder path: ${message}`, {\n stack: error instanceof Error ? error.stack : undefined,\n });\n }\n}\n\nexport default function createTool() {\n return {\n name: 'folder-path' as const,\n config,\n handler,\n };\n}\n"],"names":["schemas","AuthRequiredBranchSchema","ErrorCode","McpError","google","z","inputSchema","object","folderId","string","min","describe","successBranchSchema","type","literal","path","items","array","id","name","outputSchema","discriminatedUnion","config","title","description","result","resolveFolderPath","drive","logger","segments","currentId","visited","Set","has","info","add","response","files","get","fileId","fields","data","parents","unshift","length","e","error","pathParts","slice","map","seg","join","handler","extra","version","auth","authContext","pathResult","segmentCount","content","text","JSON","stringify","structuredContent","message","Error","String","InternalError","stack","undefined","createTool"],"mappings":"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"}
|
|
@@ -0,0 +1,139 @@
|
|
|
1
|
+
import type { EnrichedExtra } from '@mcp-z/oauth-google';
|
|
2
|
+
import { type CallToolResult } from '@modelcontextprotocol/sdk/types.js';
|
|
3
|
+
import { z } from 'zod';
|
|
4
|
+
declare const inputSchema: z.ZodObject<{
|
|
5
|
+
shape: z.ZodDefault<z.ZodOptional<z.ZodEnum<{
|
|
6
|
+
objects: "objects";
|
|
7
|
+
arrays: "arrays";
|
|
8
|
+
}>>>;
|
|
9
|
+
pageSize: z.ZodDefault<z.ZodOptional<z.ZodCoercedNumber<unknown>>>;
|
|
10
|
+
pageToken: z.ZodOptional<z.ZodString>;
|
|
11
|
+
query: z.ZodOptional<z.ZodUnion<readonly [z.ZodString, z.ZodType<import("../../types.js").DriveQueryObject, unknown, z.core.$ZodTypeInternals<import("../../types.js").DriveQueryObject, unknown>>]>>;
|
|
12
|
+
fields: z.ZodOptional<z.ZodString>;
|
|
13
|
+
resolvePaths: z.ZodOptional<z.ZodBoolean>;
|
|
14
|
+
}, z.core.$strip>;
|
|
15
|
+
declare const outputSchema: z.ZodUnion<readonly [z.ZodObject<{
|
|
16
|
+
type: z.ZodLiteral<"success">;
|
|
17
|
+
shape: z.ZodLiteral<"objects">;
|
|
18
|
+
items: z.ZodArray<z.ZodObject<{
|
|
19
|
+
id: z.ZodOptional<z.ZodString>;
|
|
20
|
+
name: z.ZodOptional<z.ZodString>;
|
|
21
|
+
mimeType: z.ZodOptional<z.ZodString>;
|
|
22
|
+
webViewLink: z.ZodOptional<z.ZodString>;
|
|
23
|
+
webContentLink: z.ZodOptional<z.ZodString>;
|
|
24
|
+
modifiedTime: z.ZodOptional<z.ZodString>;
|
|
25
|
+
createdTime: z.ZodOptional<z.ZodString>;
|
|
26
|
+
size: z.ZodOptional<z.ZodString>;
|
|
27
|
+
version: z.ZodOptional<z.ZodString>;
|
|
28
|
+
shared: z.ZodOptional<z.ZodBoolean>;
|
|
29
|
+
starred: z.ZodOptional<z.ZodBoolean>;
|
|
30
|
+
trashed: z.ZodOptional<z.ZodBoolean>;
|
|
31
|
+
parents: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
32
|
+
id: z.ZodString;
|
|
33
|
+
name: z.ZodString;
|
|
34
|
+
}, z.core.$strip>>>;
|
|
35
|
+
owners: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
36
|
+
displayName: z.ZodOptional<z.ZodString>;
|
|
37
|
+
emailAddress: z.ZodOptional<z.ZodString>;
|
|
38
|
+
kind: z.ZodOptional<z.ZodString>;
|
|
39
|
+
me: z.ZodOptional<z.ZodBoolean>;
|
|
40
|
+
permissionId: z.ZodOptional<z.ZodString>;
|
|
41
|
+
photoLink: z.ZodOptional<z.ZodString>;
|
|
42
|
+
}, z.core.$strip>>>;
|
|
43
|
+
permissions: z.ZodOptional<z.ZodObject<{
|
|
44
|
+
canEdit: z.ZodOptional<z.ZodBoolean>;
|
|
45
|
+
canComment: z.ZodOptional<z.ZodBoolean>;
|
|
46
|
+
canShare: z.ZodOptional<z.ZodBoolean>;
|
|
47
|
+
}, z.core.$strip>>;
|
|
48
|
+
path: z.ZodOptional<z.ZodString>;
|
|
49
|
+
}, z.core.$strip>>;
|
|
50
|
+
count: z.ZodNumber;
|
|
51
|
+
nextPageToken: z.ZodOptional<z.ZodString>;
|
|
52
|
+
}, z.core.$strip>, z.ZodObject<{
|
|
53
|
+
type: z.ZodLiteral<"success">;
|
|
54
|
+
shape: z.ZodLiteral<"arrays">;
|
|
55
|
+
columns: z.ZodArray<z.ZodString>;
|
|
56
|
+
rows: z.ZodArray<z.ZodArray<z.ZodUnknown>>;
|
|
57
|
+
count: z.ZodNumber;
|
|
58
|
+
nextPageToken: z.ZodOptional<z.ZodString>;
|
|
59
|
+
}, z.core.$strip>, z.ZodObject<{
|
|
60
|
+
type: z.ZodLiteral<"auth_required">;
|
|
61
|
+
provider: z.ZodString;
|
|
62
|
+
message: z.ZodString;
|
|
63
|
+
url: z.ZodOptional<z.ZodString>;
|
|
64
|
+
}, z.core.$strip>]>;
|
|
65
|
+
export type Input = z.infer<typeof inputSchema>;
|
|
66
|
+
export type Output = z.infer<typeof outputSchema>;
|
|
67
|
+
declare function handler({ query, resolvePaths, pageSize, pageToken, fields, shape }: Input, extra: EnrichedExtra): Promise<CallToolResult>;
|
|
68
|
+
export default function createTool(): {
|
|
69
|
+
name: "folder-search";
|
|
70
|
+
config: {
|
|
71
|
+
readonly title: "Search Folders";
|
|
72
|
+
readonly description: "Search Google Drive folders with flexible field selection and optional path resolution.";
|
|
73
|
+
readonly inputSchema: z.ZodObject<{
|
|
74
|
+
shape: z.ZodDefault<z.ZodOptional<z.ZodEnum<{
|
|
75
|
+
objects: "objects";
|
|
76
|
+
arrays: "arrays";
|
|
77
|
+
}>>>;
|
|
78
|
+
pageSize: z.ZodDefault<z.ZodOptional<z.ZodCoercedNumber<unknown>>>;
|
|
79
|
+
pageToken: z.ZodOptional<z.ZodString>;
|
|
80
|
+
query: z.ZodOptional<z.ZodUnion<readonly [z.ZodString, z.ZodType<import("../../types.js").DriveQueryObject, unknown, z.core.$ZodTypeInternals<import("../../types.js").DriveQueryObject, unknown>>]>>;
|
|
81
|
+
fields: z.ZodOptional<z.ZodString>;
|
|
82
|
+
resolvePaths: z.ZodOptional<z.ZodBoolean>;
|
|
83
|
+
}, z.core.$strip>;
|
|
84
|
+
readonly outputSchema: z.ZodObject<{
|
|
85
|
+
result: z.ZodUnion<readonly [z.ZodObject<{
|
|
86
|
+
type: z.ZodLiteral<"success">;
|
|
87
|
+
shape: z.ZodLiteral<"objects">;
|
|
88
|
+
items: z.ZodArray<z.ZodObject<{
|
|
89
|
+
id: z.ZodOptional<z.ZodString>;
|
|
90
|
+
name: z.ZodOptional<z.ZodString>;
|
|
91
|
+
mimeType: z.ZodOptional<z.ZodString>;
|
|
92
|
+
webViewLink: z.ZodOptional<z.ZodString>;
|
|
93
|
+
webContentLink: z.ZodOptional<z.ZodString>;
|
|
94
|
+
modifiedTime: z.ZodOptional<z.ZodString>;
|
|
95
|
+
createdTime: z.ZodOptional<z.ZodString>;
|
|
96
|
+
size: z.ZodOptional<z.ZodString>;
|
|
97
|
+
version: z.ZodOptional<z.ZodString>;
|
|
98
|
+
shared: z.ZodOptional<z.ZodBoolean>;
|
|
99
|
+
starred: z.ZodOptional<z.ZodBoolean>;
|
|
100
|
+
trashed: z.ZodOptional<z.ZodBoolean>;
|
|
101
|
+
parents: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
102
|
+
id: z.ZodString;
|
|
103
|
+
name: z.ZodString;
|
|
104
|
+
}, z.core.$strip>>>;
|
|
105
|
+
owners: z.ZodOptional<z.ZodArray<z.ZodObject<{
|
|
106
|
+
displayName: z.ZodOptional<z.ZodString>;
|
|
107
|
+
emailAddress: z.ZodOptional<z.ZodString>;
|
|
108
|
+
kind: z.ZodOptional<z.ZodString>;
|
|
109
|
+
me: z.ZodOptional<z.ZodBoolean>;
|
|
110
|
+
permissionId: z.ZodOptional<z.ZodString>;
|
|
111
|
+
photoLink: z.ZodOptional<z.ZodString>;
|
|
112
|
+
}, z.core.$strip>>>;
|
|
113
|
+
permissions: z.ZodOptional<z.ZodObject<{
|
|
114
|
+
canEdit: z.ZodOptional<z.ZodBoolean>;
|
|
115
|
+
canComment: z.ZodOptional<z.ZodBoolean>;
|
|
116
|
+
canShare: z.ZodOptional<z.ZodBoolean>;
|
|
117
|
+
}, z.core.$strip>>;
|
|
118
|
+
path: z.ZodOptional<z.ZodString>;
|
|
119
|
+
}, z.core.$strip>>;
|
|
120
|
+
count: z.ZodNumber;
|
|
121
|
+
nextPageToken: z.ZodOptional<z.ZodString>;
|
|
122
|
+
}, z.core.$strip>, z.ZodObject<{
|
|
123
|
+
type: z.ZodLiteral<"success">;
|
|
124
|
+
shape: z.ZodLiteral<"arrays">;
|
|
125
|
+
columns: z.ZodArray<z.ZodString>;
|
|
126
|
+
rows: z.ZodArray<z.ZodArray<z.ZodUnknown>>;
|
|
127
|
+
count: z.ZodNumber;
|
|
128
|
+
nextPageToken: z.ZodOptional<z.ZodString>;
|
|
129
|
+
}, z.core.$strip>, z.ZodObject<{
|
|
130
|
+
type: z.ZodLiteral<"auth_required">;
|
|
131
|
+
provider: z.ZodString;
|
|
132
|
+
message: z.ZodString;
|
|
133
|
+
url: z.ZodOptional<z.ZodString>;
|
|
134
|
+
}, z.core.$strip>]>;
|
|
135
|
+
}, z.core.$strip>;
|
|
136
|
+
};
|
|
137
|
+
handler: typeof handler;
|
|
138
|
+
};
|
|
139
|
+
export {};
|
|
@@ -0,0 +1,343 @@
|
|
|
1
|
+
import { schemas } from '@mcp-z/oauth-google';
|
|
2
|
+
const { AuthRequiredBranchSchema } = schemas;
|
|
3
|
+
import { createFieldsSchema, createPaginationSchema, createShapeSchema, filterFields, parseFields, toColumnarFormat } from '@mcp-z/server';
|
|
4
|
+
import { ErrorCode, McpError } from '@modelcontextprotocol/sdk/types.js';
|
|
5
|
+
import { google } from 'googleapis';
|
|
6
|
+
import { z } from 'zod';
|
|
7
|
+
import { toDriveQuery } from '../../lib/query-builder.js';
|
|
8
|
+
import { DRIVE_FILE_COMMON_PATTERNS, DRIVE_FILE_FIELD_DESCRIPTIONS, DRIVE_FILE_FIELDS, DriveFileSchema, DriveQuerySchema } from '../../schemas/index.js';
|
|
9
|
+
const inputSchema = z.object({
|
|
10
|
+
query: DriveQuerySchema.optional().describe('Drive query object with structured search fields. See DriveQuerySchema for detailed query syntax and examples.'),
|
|
11
|
+
fields: createFieldsSchema({
|
|
12
|
+
availableFields: [
|
|
13
|
+
...DRIVE_FILE_FIELDS,
|
|
14
|
+
'path'
|
|
15
|
+
],
|
|
16
|
+
fieldDescriptions: {
|
|
17
|
+
...DRIVE_FILE_FIELD_DESCRIPTIONS,
|
|
18
|
+
path: 'Full folder path like /Work/Projects (requires resolvePaths=true)'
|
|
19
|
+
},
|
|
20
|
+
commonPatterns: DRIVE_FILE_COMMON_PATTERNS,
|
|
21
|
+
resourceName: 'Drive folder'
|
|
22
|
+
}),
|
|
23
|
+
resolvePaths: z.boolean().optional().describe('Resolve full folder paths like /Work/Projects/2024 (requires additional API calls per result)'),
|
|
24
|
+
...createPaginationSchema({
|
|
25
|
+
defaultPageSize: 50,
|
|
26
|
+
maxPageSize: 1000,
|
|
27
|
+
provider: 'drive'
|
|
28
|
+
}).shape,
|
|
29
|
+
shape: createShapeSchema()
|
|
30
|
+
});
|
|
31
|
+
// Success branch schemas for different shapes
|
|
32
|
+
const successObjectsBranchSchema = z.object({
|
|
33
|
+
type: z.literal('success'),
|
|
34
|
+
shape: z.literal('objects'),
|
|
35
|
+
items: z.array(DriveFileSchema.extend({
|
|
36
|
+
path: z.string().optional().describe('Full folder path (if resolvePaths=true)')
|
|
37
|
+
})).describe('Matching Drive folders'),
|
|
38
|
+
count: z.number().describe('Number of folders in this page'),
|
|
39
|
+
nextPageToken: z.string().optional().describe('Token for fetching next page of results')
|
|
40
|
+
});
|
|
41
|
+
const successArraysBranchSchema = z.object({
|
|
42
|
+
type: z.literal('success'),
|
|
43
|
+
shape: z.literal('arrays'),
|
|
44
|
+
columns: z.array(z.string()).describe('Column names in canonical order'),
|
|
45
|
+
rows: z.array(z.array(z.unknown())).describe('Row data matching column order'),
|
|
46
|
+
count: z.number().describe('Number of folders in this page'),
|
|
47
|
+
nextPageToken: z.string().optional().describe('Token for fetching next page of results')
|
|
48
|
+
});
|
|
49
|
+
// Output schema with auth_required support
|
|
50
|
+
// Using z.union instead of discriminatedUnion since we have two success branches with different shapes
|
|
51
|
+
const outputSchema = z.union([
|
|
52
|
+
successObjectsBranchSchema,
|
|
53
|
+
successArraysBranchSchema,
|
|
54
|
+
AuthRequiredBranchSchema
|
|
55
|
+
]);
|
|
56
|
+
const config = {
|
|
57
|
+
title: 'Search Folders',
|
|
58
|
+
description: 'Search Google Drive folders with flexible field selection and optional path resolution.',
|
|
59
|
+
inputSchema: inputSchema,
|
|
60
|
+
outputSchema: z.object({
|
|
61
|
+
result: outputSchema
|
|
62
|
+
})
|
|
63
|
+
};
|
|
64
|
+
/**
|
|
65
|
+
* Resolves the full path for a folder by walking up the parent chain
|
|
66
|
+
* Caches folder names to reduce redundant API calls
|
|
67
|
+
*/ async function resolveFolderPath(drive, folderId, folderCache, logger) {
|
|
68
|
+
if (folderId === 'root') return '/';
|
|
69
|
+
const pathParts = [];
|
|
70
|
+
let currentId = folderId;
|
|
71
|
+
const visited = new Set();
|
|
72
|
+
while(currentId && currentId !== 'root'){
|
|
73
|
+
// Prevent infinite loops
|
|
74
|
+
if (visited.has(currentId)) {
|
|
75
|
+
logger.info('Circular folder reference detected', {
|
|
76
|
+
folderId: currentId
|
|
77
|
+
});
|
|
78
|
+
break;
|
|
79
|
+
}
|
|
80
|
+
visited.add(currentId);
|
|
81
|
+
// Check cache first
|
|
82
|
+
if (folderCache.has(currentId)) {
|
|
83
|
+
const cachedName = folderCache.get(currentId);
|
|
84
|
+
if (cachedName) {
|
|
85
|
+
pathParts.unshift(cachedName);
|
|
86
|
+
}
|
|
87
|
+
// Get parent of cached folder
|
|
88
|
+
try {
|
|
89
|
+
const response = await drive.files.get({
|
|
90
|
+
fileId: currentId,
|
|
91
|
+
fields: 'parents'
|
|
92
|
+
});
|
|
93
|
+
const parents = response.data.parents;
|
|
94
|
+
currentId = (parents && parents.length > 0 ? parents[0] : '') || '';
|
|
95
|
+
} catch (_e) {
|
|
96
|
+
logger.info('Failed to get parent for cached folder', {
|
|
97
|
+
folderId: currentId
|
|
98
|
+
});
|
|
99
|
+
break;
|
|
100
|
+
}
|
|
101
|
+
} else {
|
|
102
|
+
// Fetch folder metadata
|
|
103
|
+
try {
|
|
104
|
+
const response = await drive.files.get({
|
|
105
|
+
fileId: currentId,
|
|
106
|
+
fields: 'name,parents'
|
|
107
|
+
});
|
|
108
|
+
const folderName = response.data.name;
|
|
109
|
+
const parents = response.data.parents;
|
|
110
|
+
if (folderName) {
|
|
111
|
+
folderCache.set(currentId, folderName);
|
|
112
|
+
pathParts.unshift(folderName);
|
|
113
|
+
}
|
|
114
|
+
currentId = (parents && parents.length > 0 ? parents[0] : '') || '';
|
|
115
|
+
} catch (e) {
|
|
116
|
+
logger.info('Failed to resolve folder path', {
|
|
117
|
+
folderId: currentId,
|
|
118
|
+
error: e
|
|
119
|
+
});
|
|
120
|
+
break;
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
return `/${pathParts.join('/')}`;
|
|
125
|
+
}
|
|
126
|
+
async function handler({ query, resolvePaths = false, pageSize = 50, pageToken, fields, shape = 'arrays' }, extra) {
|
|
127
|
+
const logger = extra.logger;
|
|
128
|
+
const requestedFields = parseFields(fields, [
|
|
129
|
+
...DRIVE_FILE_FIELDS,
|
|
130
|
+
'path'
|
|
131
|
+
]);
|
|
132
|
+
// Validate and clamp pageSize to Google Drive API limits (1-1000)
|
|
133
|
+
const validPageSize = Math.max(1, Math.min(1000, Math.floor(pageSize || 50)));
|
|
134
|
+
logger.info('drive.folder.search called', {
|
|
135
|
+
query,
|
|
136
|
+
resolvePaths,
|
|
137
|
+
pageSize: validPageSize,
|
|
138
|
+
pageToken: pageToken ? '[provided]' : undefined,
|
|
139
|
+
fields: fields || 'all'
|
|
140
|
+
});
|
|
141
|
+
try {
|
|
142
|
+
const drive = google.drive({
|
|
143
|
+
version: 'v3',
|
|
144
|
+
auth: extra.authContext.auth
|
|
145
|
+
});
|
|
146
|
+
const folderMimeType = 'application/vnd.google-apps.folder';
|
|
147
|
+
let qStr;
|
|
148
|
+
if (typeof query === 'string') {
|
|
149
|
+
// String query - treat as raw Drive query
|
|
150
|
+
qStr = `(${query}) and mimeType='${folderMimeType}' and trashed = false`;
|
|
151
|
+
} else if (query && typeof query === 'object' && 'rawDriveQuery' in query && query.rawDriveQuery) {
|
|
152
|
+
// Object with rawDriveQuery field - use it directly
|
|
153
|
+
qStr = `(${query.rawDriveQuery}) and mimeType='${folderMimeType}' and trashed = false`;
|
|
154
|
+
} else if (query) {
|
|
155
|
+
// Structured query object - convert to Drive query string
|
|
156
|
+
const { q } = toDriveQuery(query);
|
|
157
|
+
qStr = q ? `(${q}) and mimeType='${folderMimeType}' and trashed = false` : `mimeType='${folderMimeType}' and trashed = false`;
|
|
158
|
+
} else {
|
|
159
|
+
// No query - return all folders
|
|
160
|
+
qStr = `mimeType='${folderMimeType}' and trashed = false`;
|
|
161
|
+
}
|
|
162
|
+
const listOptions = {
|
|
163
|
+
q: qStr,
|
|
164
|
+
pageSize: validPageSize,
|
|
165
|
+
fields: 'files(id,name,mimeType,webViewLink,modifiedTime,parents,shared,starred,owners),nextPageToken',
|
|
166
|
+
orderBy: 'modifiedTime desc'
|
|
167
|
+
};
|
|
168
|
+
if (pageToken && pageToken.trim().length > 0) {
|
|
169
|
+
listOptions.pageToken = pageToken;
|
|
170
|
+
}
|
|
171
|
+
const response = await drive.files.list(listOptions);
|
|
172
|
+
const res = response.data;
|
|
173
|
+
const folders = Array.isArray(res === null || res === void 0 ? void 0 : res.files) ? res.files : [];
|
|
174
|
+
const parentIds = new Set();
|
|
175
|
+
for (const f of folders){
|
|
176
|
+
if ((f === null || f === void 0 ? void 0 : f.parents) && f.parents.length > 0) {
|
|
177
|
+
for (const parentId of f.parents){
|
|
178
|
+
if (parentId && parentId !== 'root') {
|
|
179
|
+
parentIds.add(parentId);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
const parentNameMap = new Map();
|
|
185
|
+
if (parentIds.size > 0) {
|
|
186
|
+
logger.info('Fetching parent names', {
|
|
187
|
+
count: parentIds.size
|
|
188
|
+
});
|
|
189
|
+
const parentFetches = Array.from(parentIds).map(async (parentId)=>{
|
|
190
|
+
try {
|
|
191
|
+
const parentRes = await drive.files.get({
|
|
192
|
+
fileId: parentId,
|
|
193
|
+
fields: 'id,name'
|
|
194
|
+
});
|
|
195
|
+
const parentName = parentRes.data.name || parentId;
|
|
196
|
+
parentNameMap.set(parentId, parentName);
|
|
197
|
+
} catch (e) {
|
|
198
|
+
logger.info('Failed to fetch parent name', {
|
|
199
|
+
parentId,
|
|
200
|
+
error: e
|
|
201
|
+
});
|
|
202
|
+
parentNameMap.set(parentId, parentId); // Fallback to ID
|
|
203
|
+
}
|
|
204
|
+
});
|
|
205
|
+
await Promise.all(parentFetches);
|
|
206
|
+
}
|
|
207
|
+
// Cache for folder names to reduce API calls during path resolution
|
|
208
|
+
const folderCache = new Map();
|
|
209
|
+
const items = await Promise.all(folders.map(async (f)=>{
|
|
210
|
+
const id = (f === null || f === void 0 ? void 0 : f.id) ? String(f.id) : 'unknown';
|
|
211
|
+
const name = (f === null || f === void 0 ? void 0 : f.name) || id;
|
|
212
|
+
const result = {
|
|
213
|
+
id,
|
|
214
|
+
name
|
|
215
|
+
};
|
|
216
|
+
// Only include properties that have actual values
|
|
217
|
+
if (f === null || f === void 0 ? void 0 : f.mimeType) result.mimeType = f.mimeType;
|
|
218
|
+
if (f === null || f === void 0 ? void 0 : f.webViewLink) result.webViewLink = f.webViewLink;
|
|
219
|
+
if (f === null || f === void 0 ? void 0 : f.modifiedTime) result.modifiedTime = f.modifiedTime;
|
|
220
|
+
// Build parent objects with names
|
|
221
|
+
if ((f === null || f === void 0 ? void 0 : f.parents) && f.parents.length > 0) {
|
|
222
|
+
result.parents = f.parents.map((parentId)=>{
|
|
223
|
+
if (parentId === 'root') {
|
|
224
|
+
return {
|
|
225
|
+
id: 'root',
|
|
226
|
+
name: 'My Drive'
|
|
227
|
+
};
|
|
228
|
+
}
|
|
229
|
+
const parentName = parentNameMap.get(parentId) || parentId;
|
|
230
|
+
return {
|
|
231
|
+
id: parentId,
|
|
232
|
+
name: parentName
|
|
233
|
+
};
|
|
234
|
+
});
|
|
235
|
+
}
|
|
236
|
+
if ((f === null || f === void 0 ? void 0 : f.shared) !== undefined) result.shared = f.shared;
|
|
237
|
+
if ((f === null || f === void 0 ? void 0 : f.starred) !== undefined) result.starred = f.starred;
|
|
238
|
+
if ((f === null || f === void 0 ? void 0 : f.owners) && f.owners.length > 0) {
|
|
239
|
+
result.owners = f.owners.map((o)=>{
|
|
240
|
+
const owner = {};
|
|
241
|
+
if (o === null || o === void 0 ? void 0 : o.displayName) owner.displayName = o.displayName;
|
|
242
|
+
if (o === null || o === void 0 ? void 0 : o.emailAddress) owner.emailAddress = o.emailAddress;
|
|
243
|
+
if (o === null || o === void 0 ? void 0 : o.kind) owner.kind = o.kind;
|
|
244
|
+
if ((o === null || o === void 0 ? void 0 : o.me) !== undefined) owner.me = o.me;
|
|
245
|
+
if (o === null || o === void 0 ? void 0 : o.permissionId) owner.permissionId = o.permissionId;
|
|
246
|
+
if (o === null || o === void 0 ? void 0 : o.photoLink) owner.photoLink = o.photoLink;
|
|
247
|
+
return owner;
|
|
248
|
+
});
|
|
249
|
+
}
|
|
250
|
+
// Resolve path if requested
|
|
251
|
+
if (resolvePaths && id !== 'unknown') {
|
|
252
|
+
result.path = await resolveFolderPath(drive, id, folderCache, logger);
|
|
253
|
+
}
|
|
254
|
+
return result;
|
|
255
|
+
}));
|
|
256
|
+
const filteredItems = items.map((item)=>filterFields(item, requestedFields));
|
|
257
|
+
logger.info('drive.folder.search returning', {
|
|
258
|
+
query,
|
|
259
|
+
pageSize,
|
|
260
|
+
resultCount: filteredItems.length,
|
|
261
|
+
resolvePaths,
|
|
262
|
+
fields: fields || 'all'
|
|
263
|
+
});
|
|
264
|
+
const nextPageToken = res.nextPageToken && res.nextPageToken.trim().length > 0 ? res.nextPageToken : undefined;
|
|
265
|
+
// Build result based on shape
|
|
266
|
+
const result = shape === 'arrays' ? {
|
|
267
|
+
type: 'success',
|
|
268
|
+
shape: 'arrays',
|
|
269
|
+
...toColumnarFormat(filteredItems, requestedFields, [
|
|
270
|
+
...DRIVE_FILE_FIELDS,
|
|
271
|
+
'path'
|
|
272
|
+
]),
|
|
273
|
+
count: filteredItems.length,
|
|
274
|
+
...nextPageToken && {
|
|
275
|
+
nextPageToken
|
|
276
|
+
}
|
|
277
|
+
} : {
|
|
278
|
+
type: 'success',
|
|
279
|
+
shape: 'objects',
|
|
280
|
+
items: filteredItems,
|
|
281
|
+
count: filteredItems.length,
|
|
282
|
+
...nextPageToken && {
|
|
283
|
+
nextPageToken
|
|
284
|
+
}
|
|
285
|
+
};
|
|
286
|
+
return {
|
|
287
|
+
content: [
|
|
288
|
+
{
|
|
289
|
+
type: 'text',
|
|
290
|
+
text: JSON.stringify(result)
|
|
291
|
+
}
|
|
292
|
+
],
|
|
293
|
+
structuredContent: {
|
|
294
|
+
result
|
|
295
|
+
}
|
|
296
|
+
};
|
|
297
|
+
} catch (error) {
|
|
298
|
+
const message = error instanceof Error ? error.message : String(error);
|
|
299
|
+
logger.error('drive.folder.search error', {
|
|
300
|
+
error: message
|
|
301
|
+
});
|
|
302
|
+
// Check if this is a Drive API validation error (invalid query, invalid pageToken, etc.)
|
|
303
|
+
// These should return empty results rather than throw
|
|
304
|
+
const isDriveValidationError = message.includes('Invalid Value') || message.includes('Invalid value') || message.includes('File not found') || message.includes('Bad Request');
|
|
305
|
+
if (isDriveValidationError) {
|
|
306
|
+
// Return empty result set for validation errors
|
|
307
|
+
const result = shape === 'arrays' ? {
|
|
308
|
+
type: 'success',
|
|
309
|
+
shape: 'arrays',
|
|
310
|
+
columns: [],
|
|
311
|
+
rows: [],
|
|
312
|
+
count: 0
|
|
313
|
+
} : {
|
|
314
|
+
type: 'success',
|
|
315
|
+
shape: 'objects',
|
|
316
|
+
items: [],
|
|
317
|
+
count: 0
|
|
318
|
+
};
|
|
319
|
+
return {
|
|
320
|
+
content: [
|
|
321
|
+
{
|
|
322
|
+
type: 'text',
|
|
323
|
+
text: JSON.stringify(result)
|
|
324
|
+
}
|
|
325
|
+
],
|
|
326
|
+
structuredContent: {
|
|
327
|
+
result
|
|
328
|
+
}
|
|
329
|
+
};
|
|
330
|
+
}
|
|
331
|
+
// Throw McpError for other errors
|
|
332
|
+
throw new McpError(ErrorCode.InternalError, `Error searching folders: ${message}`, {
|
|
333
|
+
stack: error instanceof Error ? error.stack : undefined
|
|
334
|
+
});
|
|
335
|
+
}
|
|
336
|
+
}
|
|
337
|
+
export default function createTool() {
|
|
338
|
+
return {
|
|
339
|
+
name: 'folder-search',
|
|
340
|
+
config,
|
|
341
|
+
handler
|
|
342
|
+
};
|
|
343
|
+
}
|