@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.
Files changed (202) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +163 -0
  3. package/bin/server.js +5 -0
  4. package/dist/cjs/constants.d.cts +7 -0
  5. package/dist/cjs/constants.d.ts +7 -0
  6. package/dist/cjs/constants.js +18 -0
  7. package/dist/cjs/constants.js.map +1 -0
  8. package/dist/cjs/index.d.cts +8 -0
  9. package/dist/cjs/index.d.ts +8 -0
  10. package/dist/cjs/index.js +314 -0
  11. package/dist/cjs/index.js.map +1 -0
  12. package/dist/cjs/lib/create-store.d.cts +2 -0
  13. package/dist/cjs/lib/create-store.d.ts +2 -0
  14. package/dist/cjs/lib/create-store.js +166 -0
  15. package/dist/cjs/lib/create-store.js.map +1 -0
  16. package/dist/cjs/lib/query-builder.d.cts +45 -0
  17. package/dist/cjs/lib/query-builder.d.ts +45 -0
  18. package/dist/cjs/lib/query-builder.js +219 -0
  19. package/dist/cjs/lib/query-builder.js.map +1 -0
  20. package/dist/cjs/mcp/index.d.cts +3 -0
  21. package/dist/cjs/mcp/index.d.ts +3 -0
  22. package/dist/cjs/mcp/index.js +66 -0
  23. package/dist/cjs/mcp/index.js.map +1 -0
  24. package/dist/cjs/mcp/prompts/index.d.cts +2 -0
  25. package/dist/cjs/mcp/prompts/index.d.ts +2 -0
  26. package/dist/cjs/mcp/prompts/index.js +26 -0
  27. package/dist/cjs/mcp/prompts/index.js.map +1 -0
  28. package/dist/cjs/mcp/prompts/organize-files.d.cts +16 -0
  29. package/dist/cjs/mcp/prompts/organize-files.d.ts +16 -0
  30. package/dist/cjs/mcp/prompts/organize-files.js +169 -0
  31. package/dist/cjs/mcp/prompts/organize-files.js.map +1 -0
  32. package/dist/cjs/mcp/prompts/query-syntax.d.cts +19 -0
  33. package/dist/cjs/mcp/prompts/query-syntax.d.ts +19 -0
  34. package/dist/cjs/mcp/prompts/query-syntax.js +169 -0
  35. package/dist/cjs/mcp/prompts/query-syntax.js.map +1 -0
  36. package/dist/cjs/mcp/resources/file.d.cts +9 -0
  37. package/dist/cjs/mcp/resources/file.d.ts +9 -0
  38. package/dist/cjs/mcp/resources/file.js +247 -0
  39. package/dist/cjs/mcp/resources/file.js.map +1 -0
  40. package/dist/cjs/mcp/resources/index.d.cts +1 -0
  41. package/dist/cjs/mcp/resources/index.d.ts +1 -0
  42. package/dist/cjs/mcp/resources/index.js +17 -0
  43. package/dist/cjs/mcp/resources/index.js.map +1 -0
  44. package/dist/cjs/mcp/tools/file-move-to-trash.d.cts +59 -0
  45. package/dist/cjs/mcp/tools/file-move-to-trash.d.ts +59 -0
  46. package/dist/cjs/mcp/tools/file-move-to-trash.js +334 -0
  47. package/dist/cjs/mcp/tools/file-move-to-trash.js.map +1 -0
  48. package/dist/cjs/mcp/tools/file-move.d.cts +73 -0
  49. package/dist/cjs/mcp/tools/file-move.d.ts +73 -0
  50. package/dist/cjs/mcp/tools/file-move.js +613 -0
  51. package/dist/cjs/mcp/tools/file-move.js.map +1 -0
  52. package/dist/cjs/mcp/tools/files-search.d.cts +135 -0
  53. package/dist/cjs/mcp/tools/files-search.d.ts +135 -0
  54. package/dist/cjs/mcp/tools/files-search.js +558 -0
  55. package/dist/cjs/mcp/tools/files-search.js.map +1 -0
  56. package/dist/cjs/mcp/tools/folder-contents.d.cts +139 -0
  57. package/dist/cjs/mcp/tools/folder-contents.d.ts +139 -0
  58. package/dist/cjs/mcp/tools/folder-contents.js +513 -0
  59. package/dist/cjs/mcp/tools/folder-contents.js.map +1 -0
  60. package/dist/cjs/mcp/tools/folder-create.d.cts +59 -0
  61. package/dist/cjs/mcp/tools/folder-create.d.ts +59 -0
  62. package/dist/cjs/mcp/tools/folder-create.js +368 -0
  63. package/dist/cjs/mcp/tools/folder-create.js.map +1 -0
  64. package/dist/cjs/mcp/tools/folder-path.d.cts +49 -0
  65. package/dist/cjs/mcp/tools/folder-path.d.ts +49 -0
  66. package/dist/cjs/mcp/tools/folder-path.js +367 -0
  67. package/dist/cjs/mcp/tools/folder-path.js.map +1 -0
  68. package/dist/cjs/mcp/tools/folder-search.d.cts +139 -0
  69. package/dist/cjs/mcp/tools/folder-search.d.ts +139 -0
  70. package/dist/cjs/mcp/tools/folder-search.js +760 -0
  71. package/dist/cjs/mcp/tools/folder-search.js.map +1 -0
  72. package/dist/cjs/mcp/tools/index.d.cts +7 -0
  73. package/dist/cjs/mcp/tools/index.d.ts +7 -0
  74. package/dist/cjs/mcp/tools/index.js +46 -0
  75. package/dist/cjs/mcp/tools/index.js.map +1 -0
  76. package/dist/cjs/package.json +1 -0
  77. package/dist/cjs/schemas/drive-query-schema.d.cts +40 -0
  78. package/dist/cjs/schemas/drive-query-schema.d.ts +40 -0
  79. package/dist/cjs/schemas/drive-query-schema.js +90 -0
  80. package/dist/cjs/schemas/drive-query-schema.js.map +1 -0
  81. package/dist/cjs/schemas/drive-validation.d.cts +48 -0
  82. package/dist/cjs/schemas/drive-validation.d.ts +48 -0
  83. package/dist/cjs/schemas/drive-validation.js +96 -0
  84. package/dist/cjs/schemas/drive-validation.js.map +1 -0
  85. package/dist/cjs/schemas/index.d.cts +2 -0
  86. package/dist/cjs/schemas/index.d.ts +2 -0
  87. package/dist/cjs/schemas/index.js +20 -0
  88. package/dist/cjs/schemas/index.js.map +1 -0
  89. package/dist/cjs/setup/config.d.cts +44 -0
  90. package/dist/cjs/setup/config.d.ts +44 -0
  91. package/dist/cjs/setup/config.js +201 -0
  92. package/dist/cjs/setup/config.js.map +1 -0
  93. package/dist/cjs/setup/http.d.cts +8 -0
  94. package/dist/cjs/setup/http.d.ts +8 -0
  95. package/dist/cjs/setup/http.js +260 -0
  96. package/dist/cjs/setup/http.js.map +1 -0
  97. package/dist/cjs/setup/index.d.cts +5 -0
  98. package/dist/cjs/setup/index.d.ts +5 -0
  99. package/dist/cjs/setup/index.js +46 -0
  100. package/dist/cjs/setup/index.js.map +1 -0
  101. package/dist/cjs/setup/oauth-google.d.cts +64 -0
  102. package/dist/cjs/setup/oauth-google.d.ts +64 -0
  103. package/dist/cjs/setup/oauth-google.js +347 -0
  104. package/dist/cjs/setup/oauth-google.js.map +1 -0
  105. package/dist/cjs/setup/runtime.d.cts +10 -0
  106. package/dist/cjs/setup/runtime.d.ts +10 -0
  107. package/dist/cjs/setup/runtime.js +353 -0
  108. package/dist/cjs/setup/runtime.js.map +1 -0
  109. package/dist/cjs/setup/stdio.d.cts +7 -0
  110. package/dist/cjs/setup/stdio.d.ts +7 -0
  111. package/dist/cjs/setup/stdio.js +239 -0
  112. package/dist/cjs/setup/stdio.js.map +1 -0
  113. package/dist/cjs/types.d.cts +45 -0
  114. package/dist/cjs/types.d.ts +45 -0
  115. package/dist/cjs/types.js +5 -0
  116. package/dist/cjs/types.js.map +1 -0
  117. package/dist/esm/constants.d.ts +7 -0
  118. package/dist/esm/constants.js +7 -0
  119. package/dist/esm/constants.js.map +1 -0
  120. package/dist/esm/index.d.ts +8 -0
  121. package/dist/esm/index.js +34 -0
  122. package/dist/esm/index.js.map +1 -0
  123. package/dist/esm/lib/create-store.d.ts +2 -0
  124. package/dist/esm/lib/create-store.js +6 -0
  125. package/dist/esm/lib/create-store.js.map +1 -0
  126. package/dist/esm/lib/query-builder.d.ts +45 -0
  127. package/dist/esm/lib/query-builder.js +184 -0
  128. package/dist/esm/lib/query-builder.js.map +1 -0
  129. package/dist/esm/mcp/index.d.ts +3 -0
  130. package/dist/esm/mcp/index.js +6 -0
  131. package/dist/esm/mcp/index.js.map +1 -0
  132. package/dist/esm/mcp/prompts/index.d.ts +2 -0
  133. package/dist/esm/mcp/prompts/index.js +2 -0
  134. package/dist/esm/mcp/prompts/index.js.map +1 -0
  135. package/dist/esm/mcp/prompts/organize-files.d.ts +16 -0
  136. package/dist/esm/mcp/prompts/organize-files.js +21 -0
  137. package/dist/esm/mcp/prompts/organize-files.js.map +1 -0
  138. package/dist/esm/mcp/prompts/query-syntax.d.ts +19 -0
  139. package/dist/esm/mcp/prompts/query-syntax.js +82 -0
  140. package/dist/esm/mcp/prompts/query-syntax.js.map +1 -0
  141. package/dist/esm/mcp/resources/file.d.ts +9 -0
  142. package/dist/esm/mcp/resources/file.js +77 -0
  143. package/dist/esm/mcp/resources/file.js.map +1 -0
  144. package/dist/esm/mcp/resources/index.d.ts +1 -0
  145. package/dist/esm/mcp/resources/index.js +1 -0
  146. package/dist/esm/mcp/resources/index.js.map +1 -0
  147. package/dist/esm/mcp/tools/file-move-to-trash.d.ts +59 -0
  148. package/dist/esm/mcp/tools/file-move-to-trash.js +118 -0
  149. package/dist/esm/mcp/tools/file-move-to-trash.js.map +1 -0
  150. package/dist/esm/mcp/tools/file-move.d.ts +73 -0
  151. package/dist/esm/mcp/tools/file-move.js +274 -0
  152. package/dist/esm/mcp/tools/file-move.js.map +1 -0
  153. package/dist/esm/mcp/tools/files-search.d.ts +135 -0
  154. package/dist/esm/mcp/tools/files-search.js +254 -0
  155. package/dist/esm/mcp/tools/files-search.js.map +1 -0
  156. package/dist/esm/mcp/tools/folder-contents.d.ts +139 -0
  157. package/dist/esm/mcp/tools/folder-contents.js +214 -0
  158. package/dist/esm/mcp/tools/folder-contents.js.map +1 -0
  159. package/dist/esm/mcp/tools/folder-create.d.ts +59 -0
  160. package/dist/esm/mcp/tools/folder-create.js +140 -0
  161. package/dist/esm/mcp/tools/folder-create.js.map +1 -0
  162. package/dist/esm/mcp/tools/folder-path.d.ts +49 -0
  163. package/dist/esm/mcp/tools/folder-path.js +147 -0
  164. package/dist/esm/mcp/tools/folder-path.js.map +1 -0
  165. package/dist/esm/mcp/tools/folder-search.d.ts +139 -0
  166. package/dist/esm/mcp/tools/folder-search.js +343 -0
  167. package/dist/esm/mcp/tools/folder-search.js.map +1 -0
  168. package/dist/esm/mcp/tools/index.d.ts +7 -0
  169. package/dist/esm/mcp/tools/index.js +7 -0
  170. package/dist/esm/mcp/tools/index.js.map +1 -0
  171. package/dist/esm/package.json +1 -0
  172. package/dist/esm/schemas/drive-query-schema.d.ts +40 -0
  173. package/dist/esm/schemas/drive-query-schema.js +84 -0
  174. package/dist/esm/schemas/drive-query-schema.js.map +1 -0
  175. package/dist/esm/schemas/drive-validation.d.ts +48 -0
  176. package/dist/esm/schemas/drive-validation.js +73 -0
  177. package/dist/esm/schemas/drive-validation.js.map +1 -0
  178. package/dist/esm/schemas/index.d.ts +2 -0
  179. package/dist/esm/schemas/index.js +2 -0
  180. package/dist/esm/schemas/index.js.map +1 -0
  181. package/dist/esm/setup/config.d.ts +44 -0
  182. package/dist/esm/setup/config.js +151 -0
  183. package/dist/esm/setup/config.js.map +1 -0
  184. package/dist/esm/setup/http.d.ts +8 -0
  185. package/dist/esm/setup/http.js +54 -0
  186. package/dist/esm/setup/http.js.map +1 -0
  187. package/dist/esm/setup/index.d.ts +5 -0
  188. package/dist/esm/setup/index.js +5 -0
  189. package/dist/esm/setup/index.js.map +1 -0
  190. package/dist/esm/setup/oauth-google.d.ts +64 -0
  191. package/dist/esm/setup/oauth-google.js +168 -0
  192. package/dist/esm/setup/oauth-google.js.map +1 -0
  193. package/dist/esm/setup/runtime.d.ts +10 -0
  194. package/dist/esm/setup/runtime.js +84 -0
  195. package/dist/esm/setup/runtime.js.map +1 -0
  196. package/dist/esm/setup/stdio.d.ts +7 -0
  197. package/dist/esm/setup/stdio.js +38 -0
  198. package/dist/esm/setup/stdio.js.map +1 -0
  199. package/dist/esm/types.d.ts +45 -0
  200. package/dist/esm/types.js +1 -0
  201. package/dist/esm/types.js.map +1 -0
  202. 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
+ }