@payloadcms/plugin-mcp 3.65.0-canary.6 → 3.65.0-canary.8

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 (39) hide show
  1. package/dist/endpoints/mcp.js.map +1 -1
  2. package/dist/mcp/getMcpHandler.d.ts.map +1 -1
  3. package/dist/mcp/getMcpHandler.js +14 -3
  4. package/dist/mcp/getMcpHandler.js.map +1 -1
  5. package/dist/mcp/helpers/fileValidation.js +22 -22
  6. package/dist/mcp/helpers/fileValidation.js.map +1 -1
  7. package/dist/mcp/tools/resource/create.d.ts.map +1 -1
  8. package/dist/mcp/tools/resource/create.js +22 -6
  9. package/dist/mcp/tools/resource/create.js.map +1 -1
  10. package/dist/mcp/tools/resource/delete.d.ts.map +1 -1
  11. package/dist/mcp/tools/resource/delete.js +11 -5
  12. package/dist/mcp/tools/resource/delete.js.map +1 -1
  13. package/dist/mcp/tools/resource/find.d.ts.map +1 -1
  14. package/dist/mcp/tools/resource/find.js +18 -6
  15. package/dist/mcp/tools/resource/find.js.map +1 -1
  16. package/dist/mcp/tools/resource/update.d.ts.map +1 -1
  17. package/dist/mcp/tools/resource/update.js +24 -6
  18. package/dist/mcp/tools/resource/update.js.map +1 -1
  19. package/dist/mcp/tools/schemas.d.ts +33 -9
  20. package/dist/mcp/tools/schemas.d.ts.map +1 -1
  21. package/dist/mcp/tools/schemas.js +21 -4
  22. package/dist/mcp/tools/schemas.js.map +1 -1
  23. package/dist/types.d.ts +39 -4
  24. package/dist/types.d.ts.map +1 -1
  25. package/dist/types.js.map +1 -1
  26. package/dist/utils/convertCollectionSchemaToZod.d.ts.map +1 -1
  27. package/dist/utils/convertCollectionSchemaToZod.js +1 -2
  28. package/dist/utils/convertCollectionSchemaToZod.js.map +1 -1
  29. package/package.json +3 -3
  30. package/src/endpoints/mcp.ts +1 -1
  31. package/src/mcp/getMcpHandler.ts +31 -4
  32. package/src/mcp/helpers/fileValidation.ts +22 -22
  33. package/src/mcp/tools/resource/create.ts +40 -4
  34. package/src/mcp/tools/resource/delete.ts +8 -4
  35. package/src/mcp/tools/resource/find.ts +10 -4
  36. package/src/mcp/tools/resource/update.ts +26 -5
  37. package/src/mcp/tools/schemas.ts +49 -3
  38. package/src/types.ts +58 -9
  39. package/src/utils/convertCollectionSchemaToZod.ts +1 -2
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/endpoints/mcp.ts"],"sourcesContent":["import crypto from 'crypto'\nimport { type PayloadHandler, UnauthorizedError, type Where } from 'payload'\n\nimport type { MCPAccessSettings, PluginMCPServerConfig } from '../types.js'\n\nimport { createRequestFromPayloadRequest } from '../mcp/createRequest.js'\nimport { getMCPHandler } from '../mcp/getMcpHandler.js'\n\nexport const initializeMCPHandler = (pluginOptions: PluginMCPServerConfig) => {\n const mcpHandler: PayloadHandler = async (req) => {\n const { payload } = req\n const MCPOptions = pluginOptions.mcp || {}\n const MCPHandlerOptions = MCPOptions.handlerOptions || {}\n const useVerboseLogs = MCPHandlerOptions.verboseLogs ?? false\n\n req.payloadAPI = 'MCP' as const\n\n const getDefaultMcpAccessSettings = async (overrideApiKey?: null | string) => {\n const apiKey =\n (overrideApiKey ?? req.headers.get('Authorization')?.startsWith('Bearer '))\n ? req.headers.get('Authorization')?.replace('Bearer ', '').trim()\n : null\n\n if (apiKey === null) {\n throw new UnauthorizedError()\n }\n\n const sha256APIKeyIndex = crypto\n .createHmac('sha256', payload.secret)\n .update(apiKey || '')\n .digest('hex')\n\n const apiKeyConstraints = [\n {\n apiKeyIndex: {\n equals: sha256APIKeyIndex,\n },\n },\n ]\n\n const where: Where = {\n or: apiKeyConstraints,\n }\n\n const { docs } = await payload.find({\n collection: 'payload-mcp-api-keys',\n limit: 1,\n pagination: false,\n where,\n })\n\n if (docs.length === 0) {\n throw new UnauthorizedError()\n }\n\n if (useVerboseLogs) {\n payload.logger.info('[payload-mcp] API Key is valid')\n }\n\n return docs[0] as MCPAccessSettings\n }\n\n const mcpAccessSettings = pluginOptions.overrideAuth\n ? await pluginOptions.overrideAuth(req, getDefaultMcpAccessSettings)\n : await getDefaultMcpAccessSettings()\n\n const handler = getMCPHandler(pluginOptions, mcpAccessSettings, req)\n const request = createRequestFromPayloadRequest(req)\n return await handler(request)\n }\n return mcpHandler\n}\n"],"names":["crypto","UnauthorizedError","createRequestFromPayloadRequest","getMCPHandler","initializeMCPHandler","pluginOptions","mcpHandler","req","payload","MCPOptions","mcp","MCPHandlerOptions","handlerOptions","useVerboseLogs","verboseLogs","payloadAPI","getDefaultMcpAccessSettings","overrideApiKey","apiKey","headers","get","startsWith","replace","trim","sha256APIKeyIndex","createHmac","secret","update","digest","apiKeyConstraints","apiKeyIndex","equals","where","or","docs","find","collection","limit","pagination","length","logger","info","mcpAccessSettings","overrideAuth","handler","request"],"mappings":"AAAA,OAAOA,YAAY,SAAQ;AAC3B,SAA8BC,iBAAiB,QAAoB,UAAS;AAI5E,SAASC,+BAA+B,QAAQ,0BAAyB;AACzE,SAASC,aAAa,QAAQ,0BAAyB;AAEvD,OAAO,MAAMC,uBAAuB,CAACC;IACnC,MAAMC,aAA6B,OAAOC;QACxC,MAAM,EAAEC,OAAO,EAAE,GAAGD;QACpB,MAAME,aAAaJ,cAAcK,GAAG,IAAI,CAAC;QACzC,MAAMC,oBAAoBF,WAAWG,cAAc,IAAI,CAAC;QACxD,MAAMC,iBAAiBF,kBAAkBG,WAAW,IAAI;QAExDP,IAAIQ,UAAU,GAAG;QAEjB,MAAMC,8BAA8B,OAAOC;YACzC,MAAMC,SACJ,AAACD,kBAAkBV,IAAIY,OAAO,CAACC,GAAG,CAAC,kBAAkBC,WAAW,aAC5Dd,IAAIY,OAAO,CAACC,GAAG,CAAC,kBAAkBE,QAAQ,WAAW,IAAIC,SACzD;YAEN,IAAIL,WAAW,MAAM;gBACnB,MAAM,IAAIjB;YACZ;YAEA,MAAMuB,oBAAoBxB,OACvByB,UAAU,CAAC,UAAUjB,QAAQkB,MAAM,EACnCC,MAAM,CAACT,UAAU,IACjBU,MAAM,CAAC;YAEV,MAAMC,oBAAoB;gBACxB;oBACEC,aAAa;wBACXC,QAAQP;oBACV;gBACF;aACD;YAED,MAAMQ,QAAe;gBACnBC,IAAIJ;YACN;YAEA,MAAM,EAAEK,IAAI,EAAE,GAAG,MAAM1B,QAAQ2B,IAAI,CAAC;gBAClCC,YAAY;gBACZC,OAAO;gBACPC,YAAY;gBACZN;YACF;YAEA,IAAIE,KAAKK,MAAM,KAAK,GAAG;gBACrB,MAAM,IAAItC;YACZ;YAEA,IAAIY,gBAAgB;gBAClBL,QAAQgC,MAAM,CAACC,IAAI,CAAC;YACtB;YAEA,OAAOP,IAAI,CAAC,EAAE;QAChB;QAEA,MAAMQ,oBAAoBrC,cAAcsC,YAAY,GAChD,MAAMtC,cAAcsC,YAAY,CAACpC,KAAKS,+BACtC,MAAMA;QAEV,MAAM4B,UAAUzC,cAAcE,eAAeqC,mBAAmBnC;QAChE,MAAMsC,UAAU3C,gCAAgCK;QAChD,OAAO,MAAMqC,QAAQC;IACvB;IACA,OAAOvC;AACT,EAAC"}
1
+ {"version":3,"sources":["../../src/endpoints/mcp.ts"],"sourcesContent":["import crypto from 'crypto'\nimport { type PayloadHandler, UnauthorizedError, type Where } from 'payload'\n\nimport type { MCPAccessSettings, PluginMCPServerConfig } from '../types.js'\n\nimport { createRequestFromPayloadRequest } from '../mcp/createRequest.js'\nimport { getMCPHandler } from '../mcp/getMcpHandler.js'\n\nexport const initializeMCPHandler = (pluginOptions: PluginMCPServerConfig) => {\n const mcpHandler: PayloadHandler = async (req) => {\n const { payload } = req\n const MCPOptions = pluginOptions.mcp || {}\n const MCPHandlerOptions = MCPOptions.handlerOptions || {}\n const useVerboseLogs = MCPHandlerOptions.verboseLogs ?? false\n\n req.payloadAPI = 'MCP' as const\n\n const getDefaultMcpAccessSettings = async (overrideApiKey?: null | string) => {\n const apiKey =\n (overrideApiKey ?? req.headers.get('Authorization')?.startsWith('Bearer '))\n ? req.headers.get('Authorization')?.replace('Bearer ', '').trim()\n : null\n\n if (apiKey === null) {\n throw new UnauthorizedError()\n }\n\n const sha256APIKeyIndex = crypto\n .createHmac('sha256', payload.secret)\n .update(apiKey || '')\n .digest('hex')\n\n const apiKeyConstraints = [\n {\n apiKeyIndex: {\n equals: sha256APIKeyIndex,\n },\n },\n ]\n\n const where: Where = {\n or: apiKeyConstraints,\n }\n\n const { docs } = await payload.find({\n collection: 'payload-mcp-api-keys',\n limit: 1,\n pagination: false,\n where,\n })\n\n if (docs.length === 0) {\n throw new UnauthorizedError()\n }\n\n if (useVerboseLogs) {\n payload.logger.info('[payload-mcp] API Key is valid')\n }\n\n return docs[0] as unknown as MCPAccessSettings\n }\n\n const mcpAccessSettings = pluginOptions.overrideAuth\n ? await pluginOptions.overrideAuth(req, getDefaultMcpAccessSettings)\n : await getDefaultMcpAccessSettings()\n\n const handler = getMCPHandler(pluginOptions, mcpAccessSettings, req)\n const request = createRequestFromPayloadRequest(req)\n return await handler(request)\n }\n return mcpHandler\n}\n"],"names":["crypto","UnauthorizedError","createRequestFromPayloadRequest","getMCPHandler","initializeMCPHandler","pluginOptions","mcpHandler","req","payload","MCPOptions","mcp","MCPHandlerOptions","handlerOptions","useVerboseLogs","verboseLogs","payloadAPI","getDefaultMcpAccessSettings","overrideApiKey","apiKey","headers","get","startsWith","replace","trim","sha256APIKeyIndex","createHmac","secret","update","digest","apiKeyConstraints","apiKeyIndex","equals","where","or","docs","find","collection","limit","pagination","length","logger","info","mcpAccessSettings","overrideAuth","handler","request"],"mappings":"AAAA,OAAOA,YAAY,SAAQ;AAC3B,SAA8BC,iBAAiB,QAAoB,UAAS;AAI5E,SAASC,+BAA+B,QAAQ,0BAAyB;AACzE,SAASC,aAAa,QAAQ,0BAAyB;AAEvD,OAAO,MAAMC,uBAAuB,CAACC;IACnC,MAAMC,aAA6B,OAAOC;QACxC,MAAM,EAAEC,OAAO,EAAE,GAAGD;QACpB,MAAME,aAAaJ,cAAcK,GAAG,IAAI,CAAC;QACzC,MAAMC,oBAAoBF,WAAWG,cAAc,IAAI,CAAC;QACxD,MAAMC,iBAAiBF,kBAAkBG,WAAW,IAAI;QAExDP,IAAIQ,UAAU,GAAG;QAEjB,MAAMC,8BAA8B,OAAOC;YACzC,MAAMC,SACJ,AAACD,kBAAkBV,IAAIY,OAAO,CAACC,GAAG,CAAC,kBAAkBC,WAAW,aAC5Dd,IAAIY,OAAO,CAACC,GAAG,CAAC,kBAAkBE,QAAQ,WAAW,IAAIC,SACzD;YAEN,IAAIL,WAAW,MAAM;gBACnB,MAAM,IAAIjB;YACZ;YAEA,MAAMuB,oBAAoBxB,OACvByB,UAAU,CAAC,UAAUjB,QAAQkB,MAAM,EACnCC,MAAM,CAACT,UAAU,IACjBU,MAAM,CAAC;YAEV,MAAMC,oBAAoB;gBACxB;oBACEC,aAAa;wBACXC,QAAQP;oBACV;gBACF;aACD;YAED,MAAMQ,QAAe;gBACnBC,IAAIJ;YACN;YAEA,MAAM,EAAEK,IAAI,EAAE,GAAG,MAAM1B,QAAQ2B,IAAI,CAAC;gBAClCC,YAAY;gBACZC,OAAO;gBACPC,YAAY;gBACZN;YACF;YAEA,IAAIE,KAAKK,MAAM,KAAK,GAAG;gBACrB,MAAM,IAAItC;YACZ;YAEA,IAAIY,gBAAgB;gBAClBL,QAAQgC,MAAM,CAACC,IAAI,CAAC;YACtB;YAEA,OAAOP,IAAI,CAAC,EAAE;QAChB;QAEA,MAAMQ,oBAAoBrC,cAAcsC,YAAY,GAChD,MAAMtC,cAAcsC,YAAY,CAACpC,KAAKS,+BACtC,MAAMA;QAEV,MAAM4B,UAAUzC,cAAcE,eAAeqC,mBAAmBnC;QAChE,MAAMsC,UAAU3C,gCAAgCK;QAChD,OAAO,MAAMqC,QAAQC;IACvB;IACA,OAAOvC;AACT,EAAC"}
@@ -1 +1 @@
1
- {"version":3,"file":"getMcpHandler.d.ts","sourceRoot":"","sources":["../../src/mcp/getMcpHandler.ts"],"names":[],"mappings":"AAIA,OAAO,EAAgC,KAAK,cAAc,EAAkB,MAAM,SAAS,CAAA;AAE3F,OAAO,KAAK,EAAE,iBAAiB,EAAE,qBAAqB,EAAE,MAAM,aAAa,CAAA;AA+B3E,eAAO,MAAM,aAAa,kBACT,qBAAqB,qBACjB,iBAAiB,OAC/B,cAAc,4CA+ZpB,CAAA"}
1
+ {"version":3,"file":"getMcpHandler.d.ts","sourceRoot":"","sources":["../../src/mcp/getMcpHandler.ts"],"names":[],"mappings":"AAIA,OAAO,EAAgC,KAAK,cAAc,EAAkB,MAAM,SAAS,CAAA;AAE3F,OAAO,KAAK,EAAE,iBAAiB,EAAE,qBAAqB,EAAE,MAAM,aAAa,CAAA;AA+B3E,eAAO,MAAM,aAAa,kBACT,qBAAqB,qBACjB,iBAAiB,OAC/B,cAAc,4CA0bpB,CAAA"}
@@ -29,6 +29,17 @@ import { updateJobTool } from './tools/job/update.js';
29
29
  export const getMCPHandler = (pluginOptions, mcpAccessSettings, req)=>{
30
30
  const { payload } = req;
31
31
  const configSchema = configToJSONSchema(payload.config);
32
+ // Handler wrapper that injects req before the _extra argument
33
+ const wrapHandler = (handler)=>{
34
+ return async (...args)=>{
35
+ const _extra = args[args.length - 1];
36
+ const handlerArgs = args.slice(0, -1);
37
+ return await handler(...handlerArgs, req, _extra);
38
+ };
39
+ };
40
+ const payloadToolHandler = (handler)=>wrapHandler(handler);
41
+ const payloadPromptHandler = (handler)=>wrapHandler(handler);
42
+ const payloadResourceHandler = (handler)=>wrapHandler(handler);
32
43
  // User
33
44
  const user = mcpAccessSettings.user;
34
45
  // MCP Server and Handler Options
@@ -87,7 +98,7 @@ export const getMCPHandler = (pluginOptions, mcpAccessSettings, req)=>{
87
98
  customMCPTools.forEach((tool)=>{
88
99
  const camelCasedToolName = toCamelCase(tool.name);
89
100
  const isToolEnabled = mcpAccessSettings['payload-mcp-tool']?.[camelCasedToolName] ?? false;
90
- registerTool(isToolEnabled, tool.name, ()=>server.tool(tool.name, tool.description, tool.parameters, tool.handler), payload, useVerboseLogs);
101
+ registerTool(isToolEnabled, tool.name, ()=>server.tool(tool.name, tool.description, tool.parameters, payloadToolHandler(tool.handler)), payload, useVerboseLogs);
91
102
  });
92
103
  // Custom prompts
93
104
  customMCPPrompts.forEach((prompt)=>{
@@ -98,7 +109,7 @@ export const getMCPHandler = (pluginOptions, mcpAccessSettings, req)=>{
98
109
  argsSchema: prompt.argsSchema,
99
110
  description: prompt.description,
100
111
  title: prompt.title
101
- }, prompt.handler);
112
+ }, payloadPromptHandler(prompt.handler));
102
113
  if (useVerboseLogs) {
103
114
  payload.logger.info(`[payload-mcp] ✅ Prompt: ${prompt.title} Registered.`);
104
115
  }
@@ -116,7 +127,7 @@ export const getMCPHandler = (pluginOptions, mcpAccessSettings, req)=>{
116
127
  description: resource.description,
117
128
  mimeType: resource.mimeType,
118
129
  title: resource.title
119
- }, resource.handler);
130
+ }, payloadResourceHandler(resource.handler));
120
131
  if (useVerboseLogs) {
121
132
  payload.logger.info(`[payload-mcp] ✅ Resource: ${resource.title} Registered.`);
122
133
  }
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/mcp/getMcpHandler.ts"],"sourcesContent":["import type { JSONSchema4 } from 'json-schema'\n\nimport { createMcpHandler } from '@vercel/mcp-adapter'\nimport { join } from 'path'\nimport { APIError, configToJSONSchema, type PayloadRequest, type TypedUser } from 'payload'\n\nimport type { MCPAccessSettings, PluginMCPServerConfig } from '../types.js'\n\nimport { toCamelCase } from '../utils/camelCase.js'\nimport { registerTool } from './registerTool.js'\n\n// Tools\nimport { createResourceTool } from './tools/resource/create.js'\nimport { deleteResourceTool } from './tools/resource/delete.js'\nimport { findResourceTool } from './tools/resource/find.js'\nimport { updateResourceTool } from './tools/resource/update.js'\n\n// Experimental Tools\n/**\n * @experimental This tools are experimental and may change or be removed in the future.\n */\nimport { authTool } from './tools/auth/auth.js'\nimport { forgotPasswordTool } from './tools/auth/forgotPassword.js'\nimport { loginTool } from './tools/auth/login.js'\nimport { resetPasswordTool } from './tools/auth/resetPassword.js'\nimport { unlockTool } from './tools/auth/unlock.js'\nimport { verifyTool } from './tools/auth/verify.js'\nimport { createCollectionTool } from './tools/collection/create.js'\nimport { deleteCollectionTool } from './tools/collection/delete.js'\nimport { findCollectionTool } from './tools/collection/find.js'\nimport { updateCollectionTool } from './tools/collection/update.js'\nimport { findConfigTool } from './tools/config/find.js'\nimport { updateConfigTool } from './tools/config/update.js'\nimport { createJobTool } from './tools/job/create.js'\nimport { runJobTool } from './tools/job/run.js'\nimport { updateJobTool } from './tools/job/update.js'\n\nexport const getMCPHandler = (\n pluginOptions: PluginMCPServerConfig,\n mcpAccessSettings: MCPAccessSettings,\n req: PayloadRequest,\n) => {\n const { payload } = req\n const configSchema = configToJSONSchema(payload.config)\n\n // User\n const user = mcpAccessSettings.user as TypedUser\n\n // MCP Server and Handler Options\n const MCPOptions = pluginOptions.mcp || {}\n const customMCPTools = MCPOptions.tools || []\n const customMCPPrompts = MCPOptions.prompts || []\n const customMCPResources = MCPOptions.resources || []\n const MCPHandlerOptions = MCPOptions.handlerOptions || {}\n const serverOptions = MCPOptions.serverOptions || {}\n const useVerboseLogs = MCPHandlerOptions.verboseLogs ?? false\n\n // Experimental MCP Tool Requirements\n const isDevelopment = process.env.NODE_ENV === 'development'\n const experimentalTools: NonNullable<PluginMCPServerConfig['experimental']>['tools'] =\n pluginOptions?.experimental?.tools || {}\n const collectionsPluginConfig = pluginOptions.collections || {}\n const collectionsDirPath =\n experimentalTools && experimentalTools.collections?.collectionsDirPath\n ? experimentalTools.collections.collectionsDirPath\n : join(process.cwd(), 'src/collections')\n const configFilePath =\n experimentalTools && experimentalTools.config?.configFilePath\n ? experimentalTools.config.configFilePath\n : join(process.cwd(), 'src/payload.config.ts')\n const jobsDirPath =\n experimentalTools && experimentalTools.jobs?.jobsDirPath\n ? experimentalTools.jobs.jobsDirPath\n : join(process.cwd(), 'src/jobs')\n\n try {\n return createMcpHandler(\n (server) => {\n const enabledCollectionSlugs = Object.keys(collectionsPluginConfig || {}).filter(\n (collection) => {\n const fullyEnabled =\n typeof collectionsPluginConfig?.[collection]?.enabled === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled\n\n if (fullyEnabled) {\n return true\n }\n\n const partiallyEnabled =\n typeof collectionsPluginConfig?.[collection]?.enabled !== 'boolean' &&\n ((typeof collectionsPluginConfig?.[collection]?.enabled?.find === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.find === true) ||\n (typeof collectionsPluginConfig?.[collection]?.enabled?.create === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.create === true) ||\n (typeof collectionsPluginConfig?.[collection]?.enabled?.update === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.update === true) ||\n (typeof collectionsPluginConfig?.[collection]?.enabled?.delete === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.delete === true))\n\n if (partiallyEnabled) {\n return true\n }\n },\n )\n\n // Collection Operation Tools\n enabledCollectionSlugs.forEach((enabledCollectionSlug) => {\n try {\n const schema = configSchema.definitions?.[enabledCollectionSlug] as JSONSchema4\n\n const toolCapabilities = mcpAccessSettings?.[\n `${toCamelCase(enabledCollectionSlug)}`\n ] as Record<string, unknown>\n const allowCreate: boolean | undefined = toolCapabilities?.create as boolean\n const allowUpdate: boolean | undefined = toolCapabilities?.update as boolean\n const allowFind: boolean | undefined = toolCapabilities?.find as boolean\n const allowDelete: boolean | undefined = toolCapabilities?.delete as boolean\n\n if (allowCreate) {\n registerTool(\n allowCreate,\n `Create ${enabledCollectionSlug}`,\n () =>\n createResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n schema,\n ),\n payload,\n useVerboseLogs,\n )\n }\n if (allowUpdate) {\n registerTool(\n allowUpdate,\n `Update ${enabledCollectionSlug}`,\n () =>\n updateResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n schema,\n ),\n payload,\n useVerboseLogs,\n )\n }\n if (allowFind) {\n registerTool(\n allowFind,\n `Find ${enabledCollectionSlug}`,\n () =>\n findResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n ),\n payload,\n useVerboseLogs,\n )\n }\n if (allowDelete) {\n registerTool(\n allowDelete,\n `Delete ${enabledCollectionSlug}`,\n () =>\n deleteResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n ),\n payload,\n useVerboseLogs,\n )\n }\n } catch (error) {\n throw new APIError(\n `Error registering tools for collection ${enabledCollectionSlug}: ${String(error)}`,\n 500,\n )\n }\n })\n\n // Custom tools\n customMCPTools.forEach((tool) => {\n const camelCasedToolName = toCamelCase(tool.name)\n const isToolEnabled = mcpAccessSettings['payload-mcp-tool']?.[camelCasedToolName] ?? false\n\n registerTool(\n isToolEnabled,\n tool.name,\n () => server.tool(tool.name, tool.description, tool.parameters, tool.handler),\n payload,\n useVerboseLogs,\n )\n })\n\n // Custom prompts\n customMCPPrompts.forEach((prompt) => {\n const camelCasedPromptName = toCamelCase(prompt.name)\n const isPromptEnabled =\n mcpAccessSettings['payload-mcp-prompt']?.[camelCasedPromptName] ?? false\n\n if (isPromptEnabled) {\n server.registerPrompt(\n prompt.name,\n {\n argsSchema: prompt.argsSchema,\n description: prompt.description,\n title: prompt.title,\n },\n prompt.handler,\n )\n if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ✅ Prompt: ${prompt.title} Registered.`)\n }\n } else if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ⏭️ Prompt: ${prompt.title} Skipped.`)\n }\n })\n\n // Custom resources\n customMCPResources.forEach((resource) => {\n const camelCasedResourceName = toCamelCase(resource.name)\n const isResourceEnabled =\n mcpAccessSettings['payload-mcp-resource']?.[camelCasedResourceName] ?? false\n\n if (isResourceEnabled) {\n server.registerResource(\n resource.name,\n // @ts-expect-error - Overload type is not working however -- ResourceTemplate OR String is a valid type\n resource.uri,\n {\n description: resource.description,\n mimeType: resource.mimeType,\n title: resource.title,\n },\n resource.handler,\n )\n\n if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ✅ Resource: ${resource.title} Registered.`)\n }\n } else if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ⏭️ Resource: ${resource.title} Skipped.`)\n }\n })\n\n // Experimental - Collection Schema Modfication Tools\n if (\n mcpAccessSettings.collections?.create &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.create,\n 'Create Collection',\n () =>\n createCollectionTool(server, req, useVerboseLogs, collectionsDirPath, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n if (\n mcpAccessSettings.collections?.delete &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.delete,\n 'Delete Collection',\n () =>\n deleteCollectionTool(server, req, useVerboseLogs, collectionsDirPath, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (\n mcpAccessSettings.collections?.find &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.find,\n 'Find Collection',\n () => findCollectionTool(server, req, useVerboseLogs, collectionsDirPath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (\n mcpAccessSettings.collections?.update &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.update,\n 'Update Collection',\n () =>\n updateCollectionTool(server, req, useVerboseLogs, collectionsDirPath, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n // Experimental - Payload Config Modification Tools\n if (mcpAccessSettings.config?.find && experimentalTools.config?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.config.find,\n 'Find Config',\n () => findConfigTool(server, req, useVerboseLogs, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (\n mcpAccessSettings.config?.update &&\n experimentalTools.config?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.config.update,\n 'Update Config',\n () => updateConfigTool(server, req, useVerboseLogs, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n // Experimental - Job Modification Tools\n if (mcpAccessSettings.jobs?.create && experimentalTools.jobs?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.jobs.create,\n 'Create Job',\n () => createJobTool(server, req, useVerboseLogs, jobsDirPath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.jobs?.update && experimentalTools.jobs?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.jobs.update,\n 'Update Job',\n () => updateJobTool(server, req, useVerboseLogs, jobsDirPath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.jobs?.run && experimentalTools.jobs?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.jobs.run,\n 'Run Job',\n () => runJobTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n // Experimental - Auth Modification Tools\n if (mcpAccessSettings.auth?.auth && experimentalTools.auth?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.auth.auth,\n 'Auth',\n () => authTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.login && experimentalTools.auth?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.auth.login,\n 'Login',\n () => loginTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.verify && experimentalTools.auth?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.auth.verify,\n 'Verify',\n () => verifyTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.resetPassword && experimentalTools.auth?.enabled) {\n registerTool(\n mcpAccessSettings.auth.resetPassword,\n 'Reset Password',\n () => resetPasswordTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.forgotPassword && experimentalTools.auth?.enabled) {\n registerTool(\n mcpAccessSettings.auth.forgotPassword,\n 'Forgot Password',\n () => forgotPasswordTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.unlock && experimentalTools.auth?.enabled) {\n registerTool(\n mcpAccessSettings.auth.unlock,\n 'Unlock',\n () => unlockTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (useVerboseLogs) {\n payload.logger.info('[payload-mcp] 🚀 MCP Server Ready.')\n }\n },\n {\n serverInfo: serverOptions.serverInfo,\n },\n {\n basePath: MCPHandlerOptions.basePath || '/api',\n maxDuration: MCPHandlerOptions.maxDuration || 60,\n // INFO: Disabled until developer clarity is reached for server side streaming and we have an auth pattern for all SSE patterns\n // redisUrl: MCPHandlerOptions.redisUrl || process.env.REDIS_URL,\n verboseLogs: useVerboseLogs,\n },\n )\n } catch (error) {\n throw new APIError(`Error initializing MCP handler: ${String(error)}`, 500)\n }\n}\n"],"names":["createMcpHandler","join","APIError","configToJSONSchema","toCamelCase","registerTool","createResourceTool","deleteResourceTool","findResourceTool","updateResourceTool","authTool","forgotPasswordTool","loginTool","resetPasswordTool","unlockTool","verifyTool","createCollectionTool","deleteCollectionTool","findCollectionTool","updateCollectionTool","findConfigTool","updateConfigTool","createJobTool","runJobTool","updateJobTool","getMCPHandler","pluginOptions","mcpAccessSettings","req","payload","configSchema","config","user","MCPOptions","mcp","customMCPTools","tools","customMCPPrompts","prompts","customMCPResources","resources","MCPHandlerOptions","handlerOptions","serverOptions","useVerboseLogs","verboseLogs","isDevelopment","process","env","NODE_ENV","experimentalTools","experimental","collectionsPluginConfig","collections","collectionsDirPath","cwd","configFilePath","jobsDirPath","jobs","server","enabledCollectionSlugs","Object","keys","filter","collection","fullyEnabled","enabled","partiallyEnabled","find","create","update","delete","forEach","enabledCollectionSlug","schema","definitions","toolCapabilities","allowCreate","allowUpdate","allowFind","allowDelete","error","String","tool","camelCasedToolName","name","isToolEnabled","description","parameters","handler","prompt","camelCasedPromptName","isPromptEnabled","registerPrompt","argsSchema","title","logger","info","resource","camelCasedResourceName","isResourceEnabled","registerResource","uri","mimeType","run","auth","login","verify","resetPassword","forgotPassword","unlock","serverInfo","basePath","maxDuration"],"mappings":"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"}
1
+ {"version":3,"sources":["../../src/mcp/getMcpHandler.ts"],"sourcesContent":["import type { JSONSchema4 } from 'json-schema'\n\nimport { createMcpHandler } from '@vercel/mcp-adapter'\nimport { join } from 'path'\nimport { APIError, configToJSONSchema, type PayloadRequest, type TypedUser } from 'payload'\n\nimport type { MCPAccessSettings, PluginMCPServerConfig } from '../types.js'\n\nimport { toCamelCase } from '../utils/camelCase.js'\nimport { registerTool } from './registerTool.js'\n\n// Tools\nimport { createResourceTool } from './tools/resource/create.js'\nimport { deleteResourceTool } from './tools/resource/delete.js'\nimport { findResourceTool } from './tools/resource/find.js'\nimport { updateResourceTool } from './tools/resource/update.js'\n\n// Experimental Tools\n/**\n * @experimental This tools are experimental and may change or be removed in the future.\n */\nimport { authTool } from './tools/auth/auth.js'\nimport { forgotPasswordTool } from './tools/auth/forgotPassword.js'\nimport { loginTool } from './tools/auth/login.js'\nimport { resetPasswordTool } from './tools/auth/resetPassword.js'\nimport { unlockTool } from './tools/auth/unlock.js'\nimport { verifyTool } from './tools/auth/verify.js'\nimport { createCollectionTool } from './tools/collection/create.js'\nimport { deleteCollectionTool } from './tools/collection/delete.js'\nimport { findCollectionTool } from './tools/collection/find.js'\nimport { updateCollectionTool } from './tools/collection/update.js'\nimport { findConfigTool } from './tools/config/find.js'\nimport { updateConfigTool } from './tools/config/update.js'\nimport { createJobTool } from './tools/job/create.js'\nimport { runJobTool } from './tools/job/run.js'\nimport { updateJobTool } from './tools/job/update.js'\n\nexport const getMCPHandler = (\n pluginOptions: PluginMCPServerConfig,\n mcpAccessSettings: MCPAccessSettings,\n req: PayloadRequest,\n) => {\n const { payload } = req\n const configSchema = configToJSONSchema(payload.config)\n\n // Handler wrapper that injects req before the _extra argument\n const wrapHandler = (handler: (...args: any[]) => any) => {\n return async (...args: any[]) => {\n const _extra = args[args.length - 1]\n const handlerArgs = args.slice(0, -1)\n return await handler(...handlerArgs, req, _extra)\n }\n }\n\n const payloadToolHandler = (\n handler: NonNullable<NonNullable<PluginMCPServerConfig['mcp']>['tools']>[number]['handler'],\n ) => wrapHandler(handler)\n\n const payloadPromptHandler = (\n handler: NonNullable<NonNullable<PluginMCPServerConfig['mcp']>['prompts']>[number]['handler'],\n ) => wrapHandler(handler)\n\n const payloadResourceHandler = (\n handler: NonNullable<NonNullable<PluginMCPServerConfig['mcp']>['resources']>[number]['handler'],\n ) => wrapHandler(handler)\n\n // User\n const user = mcpAccessSettings.user\n\n // MCP Server and Handler Options\n const MCPOptions = pluginOptions.mcp || {}\n const customMCPTools = MCPOptions.tools || []\n const customMCPPrompts = MCPOptions.prompts || []\n const customMCPResources = MCPOptions.resources || []\n const MCPHandlerOptions = MCPOptions.handlerOptions || {}\n const serverOptions = MCPOptions.serverOptions || {}\n const useVerboseLogs = MCPHandlerOptions.verboseLogs ?? false\n\n // Experimental MCP Tool Requirements\n const isDevelopment = process.env.NODE_ENV === 'development'\n const experimentalTools: NonNullable<PluginMCPServerConfig['experimental']>['tools'] =\n pluginOptions?.experimental?.tools || {}\n const collectionsPluginConfig = pluginOptions.collections || {}\n const collectionsDirPath =\n experimentalTools && experimentalTools.collections?.collectionsDirPath\n ? experimentalTools.collections.collectionsDirPath\n : join(process.cwd(), 'src/collections')\n const configFilePath =\n experimentalTools && experimentalTools.config?.configFilePath\n ? experimentalTools.config.configFilePath\n : join(process.cwd(), 'src/payload.config.ts')\n const jobsDirPath =\n experimentalTools && experimentalTools.jobs?.jobsDirPath\n ? experimentalTools.jobs.jobsDirPath\n : join(process.cwd(), 'src/jobs')\n\n try {\n return createMcpHandler(\n (server) => {\n const enabledCollectionSlugs = Object.keys(collectionsPluginConfig || {}).filter(\n (collection) => {\n const fullyEnabled =\n typeof collectionsPluginConfig?.[collection]?.enabled === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled\n\n if (fullyEnabled) {\n return true\n }\n\n const partiallyEnabled =\n typeof collectionsPluginConfig?.[collection]?.enabled !== 'boolean' &&\n ((typeof collectionsPluginConfig?.[collection]?.enabled?.find === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.find === true) ||\n (typeof collectionsPluginConfig?.[collection]?.enabled?.create === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.create === true) ||\n (typeof collectionsPluginConfig?.[collection]?.enabled?.update === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.update === true) ||\n (typeof collectionsPluginConfig?.[collection]?.enabled?.delete === 'boolean' &&\n collectionsPluginConfig?.[collection]?.enabled?.delete === true))\n\n if (partiallyEnabled) {\n return true\n }\n },\n )\n\n // Collection Operation Tools\n enabledCollectionSlugs.forEach((enabledCollectionSlug) => {\n try {\n const schema = configSchema.definitions?.[enabledCollectionSlug] as JSONSchema4\n\n const toolCapabilities = mcpAccessSettings?.[\n `${toCamelCase(enabledCollectionSlug)}`\n ] as Record<string, unknown>\n const allowCreate: boolean | undefined = toolCapabilities?.create as boolean\n const allowUpdate: boolean | undefined = toolCapabilities?.update as boolean\n const allowFind: boolean | undefined = toolCapabilities?.find as boolean\n const allowDelete: boolean | undefined = toolCapabilities?.delete as boolean\n\n if (allowCreate) {\n registerTool(\n allowCreate,\n `Create ${enabledCollectionSlug}`,\n () =>\n createResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n schema,\n ),\n payload,\n useVerboseLogs,\n )\n }\n if (allowUpdate) {\n registerTool(\n allowUpdate,\n `Update ${enabledCollectionSlug}`,\n () =>\n updateResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n schema,\n ),\n payload,\n useVerboseLogs,\n )\n }\n if (allowFind) {\n registerTool(\n allowFind,\n `Find ${enabledCollectionSlug}`,\n () =>\n findResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n ),\n payload,\n useVerboseLogs,\n )\n }\n if (allowDelete) {\n registerTool(\n allowDelete,\n `Delete ${enabledCollectionSlug}`,\n () =>\n deleteResourceTool(\n server,\n req,\n user,\n useVerboseLogs,\n enabledCollectionSlug,\n collectionsPluginConfig,\n ),\n payload,\n useVerboseLogs,\n )\n }\n } catch (error) {\n throw new APIError(\n `Error registering tools for collection ${enabledCollectionSlug}: ${String(error)}`,\n 500,\n )\n }\n })\n\n // Custom tools\n customMCPTools.forEach((tool) => {\n const camelCasedToolName = toCamelCase(tool.name)\n const isToolEnabled = mcpAccessSettings['payload-mcp-tool']?.[camelCasedToolName] ?? false\n\n registerTool(\n isToolEnabled,\n tool.name,\n () =>\n server.tool(\n tool.name,\n tool.description,\n tool.parameters,\n payloadToolHandler(tool.handler),\n ),\n payload,\n useVerboseLogs,\n )\n })\n\n // Custom prompts\n customMCPPrompts.forEach((prompt) => {\n const camelCasedPromptName = toCamelCase(prompt.name)\n const isPromptEnabled =\n mcpAccessSettings['payload-mcp-prompt']?.[camelCasedPromptName] ?? false\n\n if (isPromptEnabled) {\n server.registerPrompt(\n prompt.name,\n {\n argsSchema: prompt.argsSchema,\n description: prompt.description,\n title: prompt.title,\n },\n payloadPromptHandler(prompt.handler),\n )\n if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ✅ Prompt: ${prompt.title} Registered.`)\n }\n } else if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ⏭️ Prompt: ${prompt.title} Skipped.`)\n }\n })\n\n // Custom resources\n customMCPResources.forEach((resource) => {\n const camelCasedResourceName = toCamelCase(resource.name)\n const isResourceEnabled =\n mcpAccessSettings['payload-mcp-resource']?.[camelCasedResourceName] ?? false\n\n if (isResourceEnabled) {\n server.registerResource(\n resource.name,\n // @ts-expect-error - Overload type is not working however -- ResourceTemplate OR String is a valid type\n resource.uri,\n {\n description: resource.description,\n mimeType: resource.mimeType,\n title: resource.title,\n },\n payloadResourceHandler(resource.handler),\n )\n\n if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ✅ Resource: ${resource.title} Registered.`)\n }\n } else if (useVerboseLogs) {\n payload.logger.info(`[payload-mcp] ⏭️ Resource: ${resource.title} Skipped.`)\n }\n })\n\n // Experimental - Collection Schema Modfication Tools\n if (\n mcpAccessSettings.collections?.create &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.create,\n 'Create Collection',\n () =>\n createCollectionTool(server, req, useVerboseLogs, collectionsDirPath, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n if (\n mcpAccessSettings.collections?.delete &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.delete,\n 'Delete Collection',\n () =>\n deleteCollectionTool(server, req, useVerboseLogs, collectionsDirPath, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (\n mcpAccessSettings.collections?.find &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.find,\n 'Find Collection',\n () => findCollectionTool(server, req, useVerboseLogs, collectionsDirPath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (\n mcpAccessSettings.collections?.update &&\n experimentalTools.collections?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.collections.update,\n 'Update Collection',\n () =>\n updateCollectionTool(server, req, useVerboseLogs, collectionsDirPath, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n // Experimental - Payload Config Modification Tools\n if (mcpAccessSettings.config?.find && experimentalTools.config?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.config.find,\n 'Find Config',\n () => findConfigTool(server, req, useVerboseLogs, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (\n mcpAccessSettings.config?.update &&\n experimentalTools.config?.enabled &&\n isDevelopment\n ) {\n registerTool(\n mcpAccessSettings.config.update,\n 'Update Config',\n () => updateConfigTool(server, req, useVerboseLogs, configFilePath),\n payload,\n useVerboseLogs,\n )\n }\n\n // Experimental - Job Modification Tools\n if (mcpAccessSettings.jobs?.create && experimentalTools.jobs?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.jobs.create,\n 'Create Job',\n () => createJobTool(server, req, useVerboseLogs, jobsDirPath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.jobs?.update && experimentalTools.jobs?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.jobs.update,\n 'Update Job',\n () => updateJobTool(server, req, useVerboseLogs, jobsDirPath),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.jobs?.run && experimentalTools.jobs?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.jobs.run,\n 'Run Job',\n () => runJobTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n // Experimental - Auth Modification Tools\n if (mcpAccessSettings.auth?.auth && experimentalTools.auth?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.auth.auth,\n 'Auth',\n () => authTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.login && experimentalTools.auth?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.auth.login,\n 'Login',\n () => loginTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.verify && experimentalTools.auth?.enabled && isDevelopment) {\n registerTool(\n mcpAccessSettings.auth.verify,\n 'Verify',\n () => verifyTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.resetPassword && experimentalTools.auth?.enabled) {\n registerTool(\n mcpAccessSettings.auth.resetPassword,\n 'Reset Password',\n () => resetPasswordTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.forgotPassword && experimentalTools.auth?.enabled) {\n registerTool(\n mcpAccessSettings.auth.forgotPassword,\n 'Forgot Password',\n () => forgotPasswordTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (mcpAccessSettings.auth?.unlock && experimentalTools.auth?.enabled) {\n registerTool(\n mcpAccessSettings.auth.unlock,\n 'Unlock',\n () => unlockTool(server, req, useVerboseLogs),\n payload,\n useVerboseLogs,\n )\n }\n\n if (useVerboseLogs) {\n payload.logger.info('[payload-mcp] 🚀 MCP Server Ready.')\n }\n },\n {\n serverInfo: serverOptions.serverInfo,\n },\n {\n basePath: MCPHandlerOptions.basePath || '/api',\n maxDuration: MCPHandlerOptions.maxDuration || 60,\n // INFO: Disabled until developer clarity is reached for server side streaming and we have an auth pattern for all SSE patterns\n // redisUrl: MCPHandlerOptions.redisUrl || process.env.REDIS_URL,\n verboseLogs: useVerboseLogs,\n },\n )\n } catch (error) {\n throw new APIError(`Error initializing MCP handler: ${String(error)}`, 500)\n }\n}\n"],"names":["createMcpHandler","join","APIError","configToJSONSchema","toCamelCase","registerTool","createResourceTool","deleteResourceTool","findResourceTool","updateResourceTool","authTool","forgotPasswordTool","loginTool","resetPasswordTool","unlockTool","verifyTool","createCollectionTool","deleteCollectionTool","findCollectionTool","updateCollectionTool","findConfigTool","updateConfigTool","createJobTool","runJobTool","updateJobTool","getMCPHandler","pluginOptions","mcpAccessSettings","req","payload","configSchema","config","wrapHandler","handler","args","_extra","length","handlerArgs","slice","payloadToolHandler","payloadPromptHandler","payloadResourceHandler","user","MCPOptions","mcp","customMCPTools","tools","customMCPPrompts","prompts","customMCPResources","resources","MCPHandlerOptions","handlerOptions","serverOptions","useVerboseLogs","verboseLogs","isDevelopment","process","env","NODE_ENV","experimentalTools","experimental","collectionsPluginConfig","collections","collectionsDirPath","cwd","configFilePath","jobsDirPath","jobs","server","enabledCollectionSlugs","Object","keys","filter","collection","fullyEnabled","enabled","partiallyEnabled","find","create","update","delete","forEach","enabledCollectionSlug","schema","definitions","toolCapabilities","allowCreate","allowUpdate","allowFind","allowDelete","error","String","tool","camelCasedToolName","name","isToolEnabled","description","parameters","prompt","camelCasedPromptName","isPromptEnabled","registerPrompt","argsSchema","title","logger","info","resource","camelCasedResourceName","isResourceEnabled","registerResource","uri","mimeType","run","auth","login","verify","resetPassword","forgotPassword","unlock","serverInfo","basePath","maxDuration"],"mappings":"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"}
@@ -112,13 +112,13 @@ import { join } from 'path';
112
112
  /**
113
113
  * Validate collection configuration structure
114
114
  */ function validateCollectionConfig(config) {
115
- if (!config || typeof config !== 'object') {
115
+ if (!config) {
116
116
  return {
117
117
  error: 'Collection config is not a valid object',
118
118
  success: false
119
119
  };
120
120
  }
121
- if (!config.slug || typeof config.slug !== 'string') {
121
+ if (!config.slug) {
122
122
  return {
123
123
  error: 'Collection config must have a valid slug property',
124
124
  success: false
@@ -128,14 +128,14 @@ import { join } from 'path';
128
128
  if (config.fields) {
129
129
  for(let i = 0; i < config.fields.length; i++){
130
130
  const field = config.fields[i];
131
- if (!field || typeof field !== 'object') {
131
+ if (!field) {
132
132
  return {
133
133
  error: `Field at index ${i} is not a valid object`,
134
134
  success: false
135
135
  };
136
136
  }
137
137
  // Check if field has type property
138
- if ('type' in field && field.type && typeof field.type !== 'string') {
138
+ if ('type' in field && field.type) {
139
139
  return {
140
140
  error: `Field at index ${i} has invalid type property`,
141
141
  success: false
@@ -151,26 +151,26 @@ import { join } from 'path';
151
151
  /**
152
152
  * Validate task configuration structure
153
153
  */ function validateTaskConfig(config) {
154
- if (!config || typeof config !== 'object') {
154
+ if (!config) {
155
155
  return {
156
156
  error: 'Task config is not a valid object',
157
157
  success: false
158
158
  };
159
159
  }
160
- if (!config.slug || typeof config.slug !== 'string') {
160
+ if (!config.slug) {
161
161
  return {
162
162
  error: 'Task config must have a valid slug property',
163
163
  success: false
164
164
  };
165
165
  }
166
- if (!config.handler || typeof config.handler !== 'function') {
166
+ if (!config.handler) {
167
167
  return {
168
168
  error: 'Task config must have a valid handler function',
169
169
  success: false
170
170
  };
171
171
  }
172
172
  // Validate optional properties
173
- if (config.retries !== undefined && (typeof config.retries !== 'number' || config.retries < 0)) {
173
+ if (config.retries !== undefined && config.retries < 0) {
174
174
  return {
175
175
  error: 'Task config retries must be a non-negative number',
176
176
  success: false
@@ -180,19 +180,19 @@ import { join } from 'path';
180
180
  if (config.inputSchema && Array.isArray(config.inputSchema)) {
181
181
  for(let i = 0; i < config.inputSchema.length; i++){
182
182
  const field = config.inputSchema[i];
183
- if (!field || typeof field !== 'object') {
183
+ if (!field) {
184
184
  return {
185
185
  error: `Input schema field at index ${i} is not a valid object`,
186
186
  success: false
187
187
  };
188
188
  }
189
- if (!field.name || typeof field.name !== 'string') {
189
+ if (!field.name) {
190
190
  return {
191
191
  error: `Input schema field at index ${i} must have a valid name property`,
192
192
  success: false
193
193
  };
194
194
  }
195
- if (!field.type || typeof field.type !== 'string') {
195
+ if (!field.type) {
196
196
  return {
197
197
  error: `Input schema field at index ${i} must have a valid type property`,
198
198
  success: false
@@ -203,19 +203,19 @@ import { join } from 'path';
203
203
  if (config.outputSchema && Array.isArray(config.outputSchema)) {
204
204
  for(let i = 0; i < config.outputSchema.length; i++){
205
205
  const field = config.outputSchema[i];
206
- if (!field || typeof field !== 'object') {
206
+ if (!field) {
207
207
  return {
208
208
  error: `Output schema field at index ${i} is not a valid object`,
209
209
  success: false
210
210
  };
211
211
  }
212
- if (!field.name || typeof field.name !== 'string') {
212
+ if (!field.name) {
213
213
  return {
214
214
  error: `Output schema field at index ${i} must have a valid name property`,
215
215
  success: false
216
216
  };
217
217
  }
218
- if (!field.type || typeof field.type !== 'string') {
218
+ if (!field.type) {
219
219
  return {
220
220
  error: `Output schema field at index ${i} must have a valid type property`,
221
221
  success: false
@@ -231,32 +231,32 @@ import { join } from 'path';
231
231
  /**
232
232
  * Validate workflow configuration structure
233
233
  */ function validateWorkflowConfig(config) {
234
- if (!config || typeof config !== 'object') {
234
+ if (!config) {
235
235
  return {
236
236
  error: 'Workflow config is not a valid object',
237
237
  success: false
238
238
  };
239
239
  }
240
- if (!config.slug || typeof config.slug !== 'string') {
240
+ if (!config.slug) {
241
241
  return {
242
242
  error: 'Workflow config must have a valid slug property',
243
243
  success: false
244
244
  };
245
245
  }
246
- if (!config.handler || typeof config.handler !== 'function') {
246
+ if (!config.handler) {
247
247
  return {
248
248
  error: 'Workflow config must have a valid handler function',
249
249
  success: false
250
250
  };
251
251
  }
252
252
  // Validate optional properties
253
- if (config.queue && typeof config.queue !== 'string') {
253
+ if (config.queue) {
254
254
  return {
255
255
  error: 'Workflow config queue must be a string',
256
256
  success: false
257
257
  };
258
258
  }
259
- if (config.retries !== undefined && (typeof config.retries !== 'number' || config.retries < 0)) {
259
+ if (config.retries !== undefined && config.retries < 0) {
260
260
  return {
261
261
  error: 'Workflow config retries must be a non-negative number',
262
262
  success: false
@@ -266,19 +266,19 @@ import { join } from 'path';
266
266
  if (config.inputSchema && Array.isArray(config.inputSchema)) {
267
267
  for(let i = 0; i < config.inputSchema.length; i++){
268
268
  const field = config.inputSchema[i];
269
- if (!field || typeof field !== 'object') {
269
+ if (!field) {
270
270
  return {
271
271
  error: `Input schema field at index ${i} is not a valid object`,
272
272
  success: false
273
273
  };
274
274
  }
275
- if (!field.name || typeof field.name !== 'string') {
275
+ if (!field.name) {
276
276
  return {
277
277
  error: `Input schema field at index ${i} must have a valid name property`,
278
278
  success: false
279
279
  };
280
280
  }
281
- if (!field.type || typeof field.type !== 'string') {
281
+ if (!field.type) {
282
282
  return {
283
283
  error: `Input schema field at index ${i} must have a valid type property`,
284
284
  success: false
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../src/mcp/helpers/fileValidation.ts"],"sourcesContent":["import type { CollectionConfig } from 'payload'\n\nimport { existsSync } from 'fs'\nimport { join } from 'path'\n\nexport type ValidationType = 'collection' | 'task' | 'workflow'\n\nexport interface ValidationResult<T = unknown> {\n config?: T\n error?: string\n success: boolean\n}\n\n// Custom task config interface that matches what we're creating\nexport interface TaskConfig {\n handler: (args: {\n input: Record<string, unknown>\n job: Record<string, unknown>\n tasks: Record<string, unknown>\n }) => Record<string, unknown>\n inputSchema?: Array<{\n label?: string\n name: string\n options?: Array<{ label: string; value: string }>\n required?: boolean\n type: string\n }>\n label?: string\n outputSchema?: Array<{\n label?: string\n name: string\n options?: Array<{ label: string; value: string }>\n required?: boolean\n type: string\n }>\n retries?: number\n slug: string\n}\n\n// Custom workflow config interface that matches what we're creating\nexport interface WorkflowConfig {\n handler: (args: {\n input: Record<string, unknown>\n job: Record<string, unknown>\n tasks: Record<string, unknown>\n }) => void\n inputSchema?: Array<{\n label?: string\n name: string\n options?: Array<{ label: string; value: string }>\n required?: boolean\n type: string\n }>\n label?: string\n queue?: string\n retries?: number\n slug: string\n}\n\n/**\n * Generic validation function for Payload configuration files\n * @param fileName - The name of the file (e.g., 'Users.ts', 'my-task.ts')\n * @param type - The type of validation to perform ('collection', 'task', or 'workflow')\n * @returns Object containing success status and any error messages\n */\nexport const validatePayloadFile = async <T = CollectionConfig | TaskConfig | WorkflowConfig>(\n fileName: string,\n type: ValidationType,\n): Promise<ValidationResult<T>> => {\n try {\n const basePath = type === 'collection' ? 'collections' : type === 'task' ? 'tasks' : 'workflows'\n const fullPath = join(process.cwd(), 'src', basePath)\n const filePath = join(fullPath, fileName)\n\n // Check if file exists\n if (!existsSync(filePath)) {\n return {\n error: `${type} file does not exist: ${fileName}`,\n success: false,\n }\n }\n\n // Clear require cache to ensure fresh import\n delete require.cache[filePath]\n\n // Use relative path for webpack compatibility\n const moduleName = fileName.replace('.ts', '')\n const relativePath = `../${basePath}/${moduleName}`\n\n // Dynamic import with relative path\n const importedModule = await import(/* webpackIgnore: true */ relativePath)\n\n // Get the configuration based on type\n let config: T | undefined\n\n if (type === 'collection') {\n config = getCollectionConfig(importedModule, moduleName) as T\n } else if (type === 'task') {\n config = getTaskConfig(importedModule) as T\n } else if (type === 'workflow') {\n config = getWorkflowConfig(importedModule) as T\n }\n\n if (!config) {\n return {\n error: `${type} file does not export a valid ${type} config`,\n success: false,\n }\n }\n\n // Validate the configuration\n let validationResult: ValidationResult<unknown>\n if (type === 'collection') {\n validationResult = validateCollectionConfig(config as unknown as CollectionConfig)\n } else if (type === 'task') {\n validationResult = validateTaskConfig(config as unknown as TaskConfig)\n } else if (type === 'workflow') {\n validationResult = validateWorkflowConfig(config as unknown as WorkflowConfig)\n } else {\n return {\n error: `Unknown validation type: ${type}`,\n success: false,\n }\n }\n\n if (!validationResult.success) {\n return validationResult as ValidationResult<T>\n }\n\n return {\n config,\n success: true,\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error during validation'\n return {\n error: `Failed to validate ${type} file: ${errorMessage}`,\n success: false,\n }\n }\n}\n\n/**\n * Extract collection configuration from module exports\n */\nfunction getCollectionConfig(\n importedModule: Record<string, unknown>,\n moduleName: string,\n): CollectionConfig | undefined {\n if (importedModule.default) {\n return importedModule.default as CollectionConfig\n }\n\n if (importedModule[moduleName]) {\n return importedModule[moduleName] as CollectionConfig\n }\n\n return undefined\n}\n\n/**\n * Extract task configuration from module exports\n */\nfunction getTaskConfig(importedModule: Record<string, unknown>): TaskConfig | undefined {\n // First check for default export\n if (importedModule.default) {\n return importedModule.default as TaskConfig\n }\n\n // Look for named exports ending with \"Task\"\n const exportNames = Object.keys(importedModule)\n const taskExport = exportNames.find((name) => name.endsWith('Task'))\n\n if (taskExport) {\n return importedModule[taskExport] as TaskConfig\n }\n\n return undefined\n}\n\n/**\n * Extract workflow configuration from module exports\n */\nfunction getWorkflowConfig(importedModule: Record<string, unknown>): undefined | WorkflowConfig {\n // First check for default export\n if (importedModule.default) {\n return importedModule.default as WorkflowConfig\n }\n\n // Look for named exports ending with \"Workflow\"\n const exportNames = Object.keys(importedModule)\n const workflowExport = exportNames.find((name) => name.endsWith('Workflow'))\n\n if (workflowExport) {\n return importedModule[workflowExport] as WorkflowConfig\n }\n\n return undefined\n}\n\n/**\n * Validate collection configuration structure\n */\nfunction validateCollectionConfig(config: CollectionConfig): ValidationResult<CollectionConfig> {\n if (!config || typeof config !== 'object') {\n return {\n error: 'Collection config is not a valid object',\n success: false,\n }\n }\n\n if (!config.slug || typeof config.slug !== 'string') {\n return {\n error: 'Collection config must have a valid slug property',\n success: false,\n }\n }\n\n // Validate each field has required properties\n if (config.fields) {\n for (let i = 0; i < config.fields.length; i++) {\n const field = config.fields[i] as Record<string, unknown>\n if (!field || typeof field !== 'object') {\n return {\n error: `Field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n // Check if field has type property\n if ('type' in field && field.type && typeof field.type !== 'string') {\n return {\n error: `Field at index ${i} has invalid type property`,\n success: false,\n }\n }\n }\n }\n\n return { config, success: true }\n}\n\n/**\n * Validate task configuration structure\n */\nfunction validateTaskConfig(config: TaskConfig): ValidationResult<TaskConfig> {\n if (!config || typeof config !== 'object') {\n return {\n error: 'Task config is not a valid object',\n success: false,\n }\n }\n\n if (!config.slug || typeof config.slug !== 'string') {\n return {\n error: 'Task config must have a valid slug property',\n success: false,\n }\n }\n\n if (!config.handler || typeof config.handler !== 'function') {\n return {\n error: 'Task config must have a valid handler function',\n success: false,\n }\n }\n\n // Validate optional properties\n if (config.retries !== undefined && (typeof config.retries !== 'number' || config.retries < 0)) {\n return {\n error: 'Task config retries must be a non-negative number',\n success: false,\n }\n }\n\n // Validate schemas if present\n if (config.inputSchema && Array.isArray(config.inputSchema)) {\n for (let i = 0; i < config.inputSchema.length; i++) {\n const field = config.inputSchema[i]\n if (!field || typeof field !== 'object') {\n return {\n error: `Input schema field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n if (!field.name || typeof field.name !== 'string') {\n return {\n error: `Input schema field at index ${i} must have a valid name property`,\n success: false,\n }\n }\n\n if (!field.type || typeof field.type !== 'string') {\n return {\n error: `Input schema field at index ${i} must have a valid type property`,\n success: false,\n }\n }\n }\n }\n\n if (config.outputSchema && Array.isArray(config.outputSchema)) {\n for (let i = 0; i < config.outputSchema.length; i++) {\n const field = config.outputSchema[i]\n if (!field || typeof field !== 'object') {\n return {\n error: `Output schema field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n if (!field.name || typeof field.name !== 'string') {\n return {\n error: `Output schema field at index ${i} must have a valid name property`,\n success: false,\n }\n }\n\n if (!field.type || typeof field.type !== 'string') {\n return {\n error: `Output schema field at index ${i} must have a valid type property`,\n success: false,\n }\n }\n }\n }\n\n return { config, success: true }\n}\n\n/**\n * Validate workflow configuration structure\n */\nfunction validateWorkflowConfig(config: WorkflowConfig): ValidationResult<WorkflowConfig> {\n if (!config || typeof config !== 'object') {\n return {\n error: 'Workflow config is not a valid object',\n success: false,\n }\n }\n\n if (!config.slug || typeof config.slug !== 'string') {\n return {\n error: 'Workflow config must have a valid slug property',\n success: false,\n }\n }\n\n if (!config.handler || typeof config.handler !== 'function') {\n return {\n error: 'Workflow config must have a valid handler function',\n success: false,\n }\n }\n\n // Validate optional properties\n if (config.queue && typeof config.queue !== 'string') {\n return {\n error: 'Workflow config queue must be a string',\n success: false,\n }\n }\n\n if (config.retries !== undefined && (typeof config.retries !== 'number' || config.retries < 0)) {\n return {\n error: 'Workflow config retries must be a non-negative number',\n success: false,\n }\n }\n\n // Validate schema if present\n if (config.inputSchema && Array.isArray(config.inputSchema)) {\n for (let i = 0; i < config.inputSchema.length; i++) {\n const field = config.inputSchema[i]\n if (!field || typeof field !== 'object') {\n return {\n error: `Input schema field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n if (!field.name || typeof field.name !== 'string') {\n return {\n error: `Input schema field at index ${i} must have a valid name property`,\n success: false,\n }\n }\n\n if (!field.type || typeof field.type !== 'string') {\n return {\n error: `Input schema field at index ${i} must have a valid type property`,\n success: false,\n }\n }\n }\n }\n\n return { config, success: true }\n}\n\n// Convenience functions for backward compatibility\nexport const validateCollectionFile = async (\n fileName: string,\n): Promise<ValidationResult<CollectionConfig>> => {\n return validatePayloadFile<CollectionConfig>(fileName, 'collection')\n}\n\nexport const validateTaskFile = async (fileName: string): Promise<ValidationResult<TaskConfig>> => {\n return validatePayloadFile<TaskConfig>(fileName, 'task')\n}\n\nexport const validateWorkflowFile = async (\n fileName: string,\n): Promise<ValidationResult<WorkflowConfig>> => {\n return validatePayloadFile<WorkflowConfig>(fileName, 'workflow')\n}\n"],"names":["existsSync","join","validatePayloadFile","fileName","type","basePath","fullPath","process","cwd","filePath","error","success","require","cache","moduleName","replace","relativePath","importedModule","config","getCollectionConfig","getTaskConfig","getWorkflowConfig","validationResult","validateCollectionConfig","validateTaskConfig","validateWorkflowConfig","errorMessage","Error","message","default","undefined","exportNames","Object","keys","taskExport","find","name","endsWith","workflowExport","slug","fields","i","length","field","handler","retries","inputSchema","Array","isArray","outputSchema","queue","validateCollectionFile","validateTaskFile","validateWorkflowFile"],"mappings":"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"}
1
+ {"version":3,"sources":["../../../src/mcp/helpers/fileValidation.ts"],"sourcesContent":["import type { CollectionConfig } from 'payload'\n\nimport { existsSync } from 'fs'\nimport { join } from 'path'\n\nexport type ValidationType = 'collection' | 'task' | 'workflow'\n\nexport interface ValidationResult<T = unknown> {\n config?: T\n error?: string\n success: boolean\n}\n\n// Custom task config interface that matches what we're creating\nexport interface TaskConfig {\n handler: (args: {\n input: Record<string, unknown>\n job: Record<string, unknown>\n tasks: Record<string, unknown>\n }) => Record<string, unknown>\n inputSchema?: Array<{\n label?: string\n name: string\n options?: Array<{ label: string; value: string }>\n required?: boolean\n type: string\n }>\n label?: string\n outputSchema?: Array<{\n label?: string\n name: string\n options?: Array<{ label: string; value: string }>\n required?: boolean\n type: string\n }>\n retries?: number\n slug: string\n}\n\n// Custom workflow config interface that matches what we're creating\nexport interface WorkflowConfig {\n handler: (args: {\n input: Record<string, unknown>\n job: Record<string, unknown>\n tasks: Record<string, unknown>\n }) => void\n inputSchema?: Array<{\n label?: string\n name: string\n options?: Array<{ label: string; value: string }>\n required?: boolean\n type: string\n }>\n label?: string\n queue?: string\n retries?: number\n slug: string\n}\n\n/**\n * Generic validation function for Payload configuration files\n * @param fileName - The name of the file (e.g., 'Users.ts', 'my-task.ts')\n * @param type - The type of validation to perform ('collection', 'task', or 'workflow')\n * @returns Object containing success status and any error messages\n */\nexport const validatePayloadFile = async <T = CollectionConfig | TaskConfig | WorkflowConfig>(\n fileName: string,\n type: ValidationType,\n): Promise<ValidationResult<T>> => {\n try {\n const basePath = type === 'collection' ? 'collections' : type === 'task' ? 'tasks' : 'workflows'\n const fullPath = join(process.cwd(), 'src', basePath)\n const filePath = join(fullPath, fileName)\n\n // Check if file exists\n if (!existsSync(filePath)) {\n return {\n error: `${type} file does not exist: ${fileName}`,\n success: false,\n }\n }\n\n // Clear require cache to ensure fresh import\n delete require.cache[filePath]\n\n // Use relative path for webpack compatibility\n const moduleName = fileName.replace('.ts', '')\n const relativePath = `../${basePath}/${moduleName}`\n\n // Dynamic import with relative path\n const importedModule = await import(/* webpackIgnore: true */ relativePath)\n\n // Get the configuration based on type\n let config: T | undefined\n\n if (type === 'collection') {\n config = getCollectionConfig(importedModule, moduleName) as T\n } else if (type === 'task') {\n config = getTaskConfig(importedModule) as T\n } else if (type === 'workflow') {\n config = getWorkflowConfig(importedModule) as T\n }\n\n if (!config) {\n return {\n error: `${type} file does not export a valid ${type} config`,\n success: false,\n }\n }\n\n // Validate the configuration\n let validationResult: ValidationResult<unknown>\n if (type === 'collection') {\n validationResult = validateCollectionConfig(config as unknown as CollectionConfig)\n } else if (type === 'task') {\n validationResult = validateTaskConfig(config as unknown as TaskConfig)\n } else if (type === 'workflow') {\n validationResult = validateWorkflowConfig(config as unknown as WorkflowConfig)\n } else {\n return {\n error: `Unknown validation type: ${type}`,\n success: false,\n }\n }\n\n if (!validationResult.success) {\n return validationResult as ValidationResult<T>\n }\n\n return {\n config,\n success: true,\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error during validation'\n return {\n error: `Failed to validate ${type} file: ${errorMessage}`,\n success: false,\n }\n }\n}\n\n/**\n * Extract collection configuration from module exports\n */\nfunction getCollectionConfig(\n importedModule: Record<string, unknown>,\n moduleName: string,\n): CollectionConfig | undefined {\n if (importedModule.default) {\n return importedModule.default as CollectionConfig\n }\n\n if (importedModule[moduleName]) {\n return importedModule[moduleName] as CollectionConfig\n }\n\n return undefined\n}\n\n/**\n * Extract task configuration from module exports\n */\nfunction getTaskConfig(importedModule: Record<string, unknown>): TaskConfig | undefined {\n // First check for default export\n if (importedModule.default) {\n return importedModule.default as TaskConfig\n }\n\n // Look for named exports ending with \"Task\"\n const exportNames = Object.keys(importedModule)\n const taskExport = exportNames.find((name) => name.endsWith('Task'))\n\n if (taskExport) {\n return importedModule[taskExport] as TaskConfig\n }\n\n return undefined\n}\n\n/**\n * Extract workflow configuration from module exports\n */\nfunction getWorkflowConfig(importedModule: Record<string, unknown>): undefined | WorkflowConfig {\n // First check for default export\n if (importedModule.default) {\n return importedModule.default as WorkflowConfig\n }\n\n // Look for named exports ending with \"Workflow\"\n const exportNames = Object.keys(importedModule)\n const workflowExport = exportNames.find((name) => name.endsWith('Workflow'))\n\n if (workflowExport) {\n return importedModule[workflowExport] as WorkflowConfig\n }\n\n return undefined\n}\n\n/**\n * Validate collection configuration structure\n */\nfunction validateCollectionConfig(config: CollectionConfig): ValidationResult<CollectionConfig> {\n if (!config) {\n return {\n error: 'Collection config is not a valid object',\n success: false,\n }\n }\n\n if (!config.slug) {\n return {\n error: 'Collection config must have a valid slug property',\n success: false,\n }\n }\n\n // Validate each field has required properties\n if (config.fields) {\n for (let i = 0; i < config.fields.length; i++) {\n const field = config.fields[i] as Record<string, unknown>\n if (!field) {\n return {\n error: `Field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n // Check if field has type property\n if ('type' in field && field.type) {\n return {\n error: `Field at index ${i} has invalid type property`,\n success: false,\n }\n }\n }\n }\n\n return { config, success: true }\n}\n\n/**\n * Validate task configuration structure\n */\nfunction validateTaskConfig(config: TaskConfig): ValidationResult<TaskConfig> {\n if (!config) {\n return {\n error: 'Task config is not a valid object',\n success: false,\n }\n }\n\n if (!config.slug) {\n return {\n error: 'Task config must have a valid slug property',\n success: false,\n }\n }\n\n if (!config.handler) {\n return {\n error: 'Task config must have a valid handler function',\n success: false,\n }\n }\n\n // Validate optional properties\n if (config.retries !== undefined && config.retries < 0) {\n return {\n error: 'Task config retries must be a non-negative number',\n success: false,\n }\n }\n\n // Validate schemas if present\n if (config.inputSchema && Array.isArray(config.inputSchema)) {\n for (let i = 0; i < config.inputSchema.length; i++) {\n const field = config.inputSchema[i]\n if (!field) {\n return {\n error: `Input schema field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n if (!field.name) {\n return {\n error: `Input schema field at index ${i} must have a valid name property`,\n success: false,\n }\n }\n\n if (!field.type) {\n return {\n error: `Input schema field at index ${i} must have a valid type property`,\n success: false,\n }\n }\n }\n }\n\n if (config.outputSchema && Array.isArray(config.outputSchema)) {\n for (let i = 0; i < config.outputSchema.length; i++) {\n const field = config.outputSchema[i]\n if (!field) {\n return {\n error: `Output schema field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n if (!field.name) {\n return {\n error: `Output schema field at index ${i} must have a valid name property`,\n success: false,\n }\n }\n\n if (!field.type) {\n return {\n error: `Output schema field at index ${i} must have a valid type property`,\n success: false,\n }\n }\n }\n }\n\n return { config, success: true }\n}\n\n/**\n * Validate workflow configuration structure\n */\nfunction validateWorkflowConfig(config: WorkflowConfig): ValidationResult<WorkflowConfig> {\n if (!config) {\n return {\n error: 'Workflow config is not a valid object',\n success: false,\n }\n }\n\n if (!config.slug) {\n return {\n error: 'Workflow config must have a valid slug property',\n success: false,\n }\n }\n\n if (!config.handler) {\n return {\n error: 'Workflow config must have a valid handler function',\n success: false,\n }\n }\n\n // Validate optional properties\n if (config.queue) {\n return {\n error: 'Workflow config queue must be a string',\n success: false,\n }\n }\n\n if (config.retries !== undefined && config.retries < 0) {\n return {\n error: 'Workflow config retries must be a non-negative number',\n success: false,\n }\n }\n\n // Validate schema if present\n if (config.inputSchema && Array.isArray(config.inputSchema)) {\n for (let i = 0; i < config.inputSchema.length; i++) {\n const field = config.inputSchema[i]\n if (!field) {\n return {\n error: `Input schema field at index ${i} is not a valid object`,\n success: false,\n }\n }\n\n if (!field.name) {\n return {\n error: `Input schema field at index ${i} must have a valid name property`,\n success: false,\n }\n }\n\n if (!field.type) {\n return {\n error: `Input schema field at index ${i} must have a valid type property`,\n success: false,\n }\n }\n }\n }\n\n return { config, success: true }\n}\n\n// Convenience functions for backward compatibility\nexport const validateCollectionFile = async (\n fileName: string,\n): Promise<ValidationResult<CollectionConfig>> => {\n return validatePayloadFile<CollectionConfig>(fileName, 'collection')\n}\n\nexport const validateTaskFile = async (fileName: string): Promise<ValidationResult<TaskConfig>> => {\n return validatePayloadFile<TaskConfig>(fileName, 'task')\n}\n\nexport const validateWorkflowFile = async (\n fileName: string,\n): Promise<ValidationResult<WorkflowConfig>> => {\n return validatePayloadFile<WorkflowConfig>(fileName, 'workflow')\n}\n"],"names":["existsSync","join","validatePayloadFile","fileName","type","basePath","fullPath","process","cwd","filePath","error","success","require","cache","moduleName","replace","relativePath","importedModule","config","getCollectionConfig","getTaskConfig","getWorkflowConfig","validationResult","validateCollectionConfig","validateTaskConfig","validateWorkflowConfig","errorMessage","Error","message","default","undefined","exportNames","Object","keys","taskExport","find","name","endsWith","workflowExport","slug","fields","i","length","field","handler","retries","inputSchema","Array","isArray","outputSchema","queue","validateCollectionFile","validateTaskFile","validateWorkflowFile"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"create.d.ts","sourceRoot":"","sources":["../../../../src/mcp/tools/resource/create.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AACxE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAC9C,OAAO,KAAK,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,SAAS,CAAA;AAExD,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAA;AAK9D,eAAO,MAAM,kBAAkB,WACrB,SAAS,OACZ,cAAc,QACb,SAAS,eACF,OAAO,kBACJ,MAAM,eACT,qBAAqB,CAAC,aAAa,CAAC,UACzC,WAAW,SAyGpB,CAAA"}
1
+ {"version":3,"file":"create.d.ts","sourceRoot":"","sources":["../../../../src/mcp/tools/resource/create.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AACxE,OAAO,KAAK,EAAE,WAAW,EAAE,MAAM,aAAa,CAAA;AAC9C,OAAO,KAAK,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,SAAS,CAAA;AAIxD,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAA;AAK9D,eAAO,MAAM,kBAAkB,WACrB,SAAS,OACZ,cAAc,QACb,SAAS,eACF,OAAO,kBACJ,MAAM,eACT,qBAAqB,CAAC,aAAa,CAAC,UACzC,WAAW,SA2IpB,CAAA"}
@@ -1,11 +1,12 @@
1
+ import { z } from 'zod';
1
2
  import { toCamelCase } from '../../../utils/camelCase.js';
2
3
  import { convertCollectionSchemaToZod } from '../../../utils/convertCollectionSchemaToZod.js';
3
4
  import { toolSchemas } from '../schemas.js';
4
5
  export const createResourceTool = (server, req, user, verboseLogs, collectionSlug, collections, schema)=>{
5
- const tool = async (data)=>{
6
+ const tool = async (data, draft, locale, fallbackLocale)=>{
6
7
  const payload = req.payload;
7
8
  if (verboseLogs) {
8
- payload.logger.info(`[payload-mcp] Creating resource in collection: ${collectionSlug}`);
9
+ payload.logger.info(`[payload-mcp] Creating resource in collection: ${collectionSlug}${locale ? ` with locale: ${locale}` : ''}`);
9
10
  }
10
11
  try {
11
12
  // Parse the data JSON
@@ -30,9 +31,16 @@ export const createResourceTool = (server, req, user, verboseLogs, collectionSlu
30
31
  const result = await payload.create({
31
32
  collection: collectionSlug,
32
33
  data: parsedData,
34
+ draft,
33
35
  overrideAccess: false,
34
36
  req,
35
- user
37
+ user,
38
+ ...locale && {
39
+ locale
40
+ },
41
+ ...fallbackLocale && {
42
+ fallbackLocale
43
+ }
36
44
  });
37
45
  if (verboseLogs) {
38
46
  payload.logger.info(`[payload-mcp] Successfully created resource in ${collectionSlug} with ID: ${result.id}`);
@@ -66,9 +74,17 @@ ${JSON.stringify(result, null, 2)}
66
74
  };
67
75
  if (collections?.[collectionSlug]?.enabled) {
68
76
  const convertedFields = convertCollectionSchemaToZod(schema);
69
- server.tool(`create${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`, `${collections?.[collectionSlug]?.description || toolSchemas.createResource.description.trim()}`, convertedFields.shape, async (params)=>{
70
- const data = JSON.stringify(params);
71
- return await tool(data);
77
+ // Create a new schema that combines the converted fields with create-specific parameters
78
+ const createResourceSchema = z.object({
79
+ ...convertedFields.shape,
80
+ draft: z.boolean().optional().default(false).describe('Whether to create the document as a draft'),
81
+ fallbackLocale: z.string().optional().describe('Optional: fallback locale code to use when requested locale is not available'),
82
+ locale: z.string().optional().describe('Optional: locale code to create the document in (e.g., "en", "es"). Defaults to the default locale')
83
+ });
84
+ server.tool(`create${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`, `${collections?.[collectionSlug]?.description || toolSchemas.createResource.description.trim()}`, createResourceSchema.shape, async (params)=>{
85
+ const { draft, fallbackLocale, locale, ...fieldData } = params;
86
+ const data = JSON.stringify(fieldData);
87
+ return await tool(data, draft, locale, fallbackLocale);
72
88
  });
73
89
  }
74
90
  };
@@ -1 +1 @@
1
- {"version":3,"sources":["../../../../src/mcp/tools/resource/create.ts"],"sourcesContent":["import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\nimport type { JSONSchema4 } from 'json-schema'\nimport type { PayloadRequest, TypedUser } from 'payload'\n\nimport type { PluginMCPServerConfig } from '../../../types.js'\n\nimport { toCamelCase } from '../../../utils/camelCase.js'\nimport { convertCollectionSchemaToZod } from '../../../utils/convertCollectionSchemaToZod.js'\nimport { toolSchemas } from '../schemas.js'\nexport const createResourceTool = (\n server: McpServer,\n req: PayloadRequest,\n user: TypedUser,\n verboseLogs: boolean,\n collectionSlug: string,\n collections: PluginMCPServerConfig['collections'],\n schema: JSONSchema4,\n) => {\n const tool = async (\n data: string,\n ): Promise<{\n content: Array<{\n text: string\n type: 'text'\n }>\n }> => {\n const payload = req.payload\n\n if (verboseLogs) {\n payload.logger.info(`[payload-mcp] Creating resource in collection: ${collectionSlug}`)\n }\n\n try {\n // Parse the data JSON\n let parsedData: Record<string, unknown>\n try {\n parsedData = JSON.parse(data)\n if (verboseLogs) {\n payload.logger.info(\n `[payload-mcp] Parsed data for ${collectionSlug}: ${JSON.stringify(parsedData)}`,\n )\n }\n } catch (_parseError) {\n payload.logger.error(`[payload-mcp] Invalid JSON data provided: ${data}`)\n return {\n content: [{ type: 'text' as const, text: 'Error: Invalid JSON data provided' }],\n }\n }\n\n // Create the resource\n const result = await payload.create({\n collection: collectionSlug,\n data: parsedData,\n overrideAccess: false,\n req,\n user,\n })\n\n if (verboseLogs) {\n payload.logger.info(\n `[payload-mcp] Successfully created resource in ${collectionSlug} with ID: ${result.id}`,\n )\n }\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Resource created successfully in collection \"${collectionSlug}\"!\nCreated resource:\n\\`\\`\\`json\n${JSON.stringify(result, null, 2)}\n\\`\\`\\``,\n },\n ],\n }\n\n return (collections?.[collectionSlug]?.overrideResponse?.(response, result, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error'\n payload.logger.error(\n `[payload-mcp] Error creating resource in ${collectionSlug}: ${errorMessage}`,\n )\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Error creating resource in collection \"${collectionSlug}\": ${errorMessage}`,\n },\n ],\n }\n\n return (collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n }\n\n if (collections?.[collectionSlug]?.enabled) {\n const convertedFields = convertCollectionSchemaToZod(schema)\n\n server.tool(\n `create${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`,\n `${collections?.[collectionSlug]?.description || toolSchemas.createResource.description.trim()}`,\n convertedFields.shape,\n async (params: Record<string, unknown>) => {\n const data = JSON.stringify(params)\n return await tool(data)\n },\n )\n }\n}\n"],"names":["toCamelCase","convertCollectionSchemaToZod","toolSchemas","createResourceTool","server","req","user","verboseLogs","collectionSlug","collections","schema","tool","data","payload","logger","info","parsedData","JSON","parse","stringify","_parseError","error","content","type","text","result","create","collection","overrideAccess","id","response","overrideResponse","errorMessage","Error","message","enabled","convertedFields","charAt","toUpperCase","slice","description","createResource","trim","shape","params"],"mappings":"AAMA,SAASA,WAAW,QAAQ,8BAA6B;AACzD,SAASC,4BAA4B,QAAQ,iDAAgD;AAC7F,SAASC,WAAW,QAAQ,gBAAe;AAC3C,OAAO,MAAMC,qBAAqB,CAChCC,QACAC,KACAC,MACAC,aACAC,gBACAC,aACAC;IAEA,MAAMC,OAAO,OACXC;QAOA,MAAMC,UAAUR,IAAIQ,OAAO;QAE3B,IAAIN,aAAa;YACfM,QAAQC,MAAM,CAACC,IAAI,CAAC,CAAC,+CAA+C,EAAEP,gBAAgB;QACxF;QAEA,IAAI;YACF,sBAAsB;YACtB,IAAIQ;YACJ,IAAI;gBACFA,aAAaC,KAAKC,KAAK,CAACN;gBACxB,IAAIL,aAAa;oBACfM,QAAQC,MAAM,CAACC,IAAI,CACjB,CAAC,8BAA8B,EAAEP,eAAe,EAAE,EAAES,KAAKE,SAAS,CAACH,aAAa;gBAEpF;YACF,EAAE,OAAOI,aAAa;gBACpBP,QAAQC,MAAM,CAACO,KAAK,CAAC,CAAC,0CAA0C,EAAET,MAAM;gBACxE,OAAO;oBACLU,SAAS;wBAAC;4BAAEC,MAAM;4BAAiBC,MAAM;wBAAoC;qBAAE;gBACjF;YACF;YAEA,sBAAsB;YACtB,MAAMC,SAAS,MAAMZ,QAAQa,MAAM,CAAC;gBAClCC,YAAYnB;gBACZI,MAAMI;gBACNY,gBAAgB;gBAChBvB;gBACAC;YACF;YAEA,IAAIC,aAAa;gBACfM,QAAQC,MAAM,CAACC,IAAI,CACjB,CAAC,+CAA+C,EAAEP,eAAe,UAAU,EAAEiB,OAAOI,EAAE,EAAE;YAE5F;YAEA,MAAMC,WAAW;gBACfR,SAAS;oBACP;wBACEC,MAAM;wBACNC,MAAM,CAAC,6CAA6C,EAAEhB,eAAe;;;AAGjF,EAAES,KAAKE,SAAS,CAACM,QAAQ,MAAM,GAAG;MAC5B,CAAC;oBACG;iBACD;YACH;YAEA,OAAQhB,aAAa,CAACD,eAAe,EAAEuB,mBAAmBD,UAAUL,QAAQpB,QAC1EyB;QAMJ,EAAE,OAAOT,OAAO;YACd,MAAMW,eAAeX,iBAAiBY,QAAQZ,MAAMa,OAAO,GAAG;YAC9DrB,QAAQC,MAAM,CAACO,KAAK,CAClB,CAAC,yCAAyC,EAAEb,eAAe,EAAE,EAAEwB,cAAc;YAG/E,MAAMF,WAAW;gBACfR,SAAS;oBACP;wBACEC,MAAM;wBACNC,MAAM,CAAC,uCAAuC,EAAEhB,eAAe,GAAG,EAAEwB,cAAc;oBACpF;iBACD;YACH;YAEA,OAAQvB,aAAa,CAACD,eAAe,EAAEuB,mBAAmBD,UAAU,CAAC,GAAGzB,QAAQyB;QAMlF;IACF;IAEA,IAAIrB,aAAa,CAACD,eAAe,EAAE2B,SAAS;QAC1C,MAAMC,kBAAkBnC,6BAA6BS;QAErDN,OAAOO,IAAI,CACT,CAAC,MAAM,EAAEH,eAAe6B,MAAM,CAAC,GAAGC,WAAW,KAAKtC,YAAYQ,gBAAgB+B,KAAK,CAAC,IAAI,EACxF,GAAG9B,aAAa,CAACD,eAAe,EAAEgC,eAAetC,YAAYuC,cAAc,CAACD,WAAW,CAACE,IAAI,IAAI,EAChGN,gBAAgBO,KAAK,EACrB,OAAOC;YACL,MAAMhC,OAAOK,KAAKE,SAAS,CAACyB;YAC5B,OAAO,MAAMjC,KAAKC;QACpB;IAEJ;AACF,EAAC"}
1
+ {"version":3,"sources":["../../../../src/mcp/tools/resource/create.ts"],"sourcesContent":["import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\nimport type { JSONSchema4 } from 'json-schema'\nimport type { PayloadRequest, TypedUser } from 'payload'\n\nimport { z } from 'zod'\n\nimport type { PluginMCPServerConfig } from '../../../types.js'\n\nimport { toCamelCase } from '../../../utils/camelCase.js'\nimport { convertCollectionSchemaToZod } from '../../../utils/convertCollectionSchemaToZod.js'\nimport { toolSchemas } from '../schemas.js'\nexport const createResourceTool = (\n server: McpServer,\n req: PayloadRequest,\n user: TypedUser,\n verboseLogs: boolean,\n collectionSlug: string,\n collections: PluginMCPServerConfig['collections'],\n schema: JSONSchema4,\n) => {\n const tool = async (\n data: string,\n draft: boolean,\n locale?: string,\n fallbackLocale?: string,\n ): Promise<{\n content: Array<{\n text: string\n type: 'text'\n }>\n }> => {\n const payload = req.payload\n\n if (verboseLogs) {\n payload.logger.info(\n `[payload-mcp] Creating resource in collection: ${collectionSlug}${locale ? ` with locale: ${locale}` : ''}`,\n )\n }\n\n try {\n // Parse the data JSON\n let parsedData: Record<string, unknown>\n try {\n parsedData = JSON.parse(data)\n if (verboseLogs) {\n payload.logger.info(\n `[payload-mcp] Parsed data for ${collectionSlug}: ${JSON.stringify(parsedData)}`,\n )\n }\n } catch (_parseError) {\n payload.logger.error(`[payload-mcp] Invalid JSON data provided: ${data}`)\n return {\n content: [{ type: 'text' as const, text: 'Error: Invalid JSON data provided' }],\n }\n }\n\n // Create the resource\n const result = await payload.create({\n collection: collectionSlug,\n data: parsedData,\n draft,\n overrideAccess: false,\n req,\n user,\n ...(locale && { locale }),\n ...(fallbackLocale && { fallbackLocale }),\n })\n\n if (verboseLogs) {\n payload.logger.info(\n `[payload-mcp] Successfully created resource in ${collectionSlug} with ID: ${result.id}`,\n )\n }\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Resource created successfully in collection \"${collectionSlug}\"!\nCreated resource:\n\\`\\`\\`json\n${JSON.stringify(result, null, 2)}\n\\`\\`\\``,\n },\n ],\n }\n\n return (collections?.[collectionSlug]?.overrideResponse?.(response, result, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error'\n payload.logger.error(\n `[payload-mcp] Error creating resource in ${collectionSlug}: ${errorMessage}`,\n )\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Error creating resource in collection \"${collectionSlug}\": ${errorMessage}`,\n },\n ],\n }\n\n return (collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n }\n\n if (collections?.[collectionSlug]?.enabled) {\n const convertedFields = convertCollectionSchemaToZod(schema)\n\n // Create a new schema that combines the converted fields with create-specific parameters\n const createResourceSchema = z.object({\n ...convertedFields.shape,\n draft: z\n .boolean()\n .optional()\n .default(false)\n .describe('Whether to create the document as a draft'),\n fallbackLocale: z\n .string()\n .optional()\n .describe('Optional: fallback locale code to use when requested locale is not available'),\n locale: z\n .string()\n .optional()\n .describe(\n 'Optional: locale code to create the document in (e.g., \"en\", \"es\"). Defaults to the default locale',\n ),\n })\n\n server.tool(\n `create${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`,\n `${collections?.[collectionSlug]?.description || toolSchemas.createResource.description.trim()}`,\n createResourceSchema.shape,\n async (params: Record<string, unknown>) => {\n const { draft, fallbackLocale, locale, ...fieldData } = params\n const data = JSON.stringify(fieldData)\n return await tool(\n data,\n draft as boolean,\n locale as string | undefined,\n fallbackLocale as string | undefined,\n )\n },\n )\n }\n}\n"],"names":["z","toCamelCase","convertCollectionSchemaToZod","toolSchemas","createResourceTool","server","req","user","verboseLogs","collectionSlug","collections","schema","tool","data","draft","locale","fallbackLocale","payload","logger","info","parsedData","JSON","parse","stringify","_parseError","error","content","type","text","result","create","collection","overrideAccess","id","response","overrideResponse","errorMessage","Error","message","enabled","convertedFields","createResourceSchema","object","shape","boolean","optional","default","describe","string","charAt","toUpperCase","slice","description","createResource","trim","params","fieldData"],"mappings":"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"}
@@ -1 +1 @@
1
- {"version":3,"file":"delete.d.ts","sourceRoot":"","sources":["../../../../src/mcp/tools/resource/delete.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AACxE,OAAO,KAAK,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,SAAS,CAAA;AAExD,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAA;AAK9D,eAAO,MAAM,kBAAkB,WACrB,SAAS,OACZ,cAAc,QACb,SAAS,eACF,OAAO,kBACJ,MAAM,eACT,qBAAqB,CAAC,aAAa,CAAC,SAqMlD,CAAA"}
1
+ {"version":3,"file":"delete.d.ts","sourceRoot":"","sources":["../../../../src/mcp/tools/resource/delete.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAA;AACxE,OAAO,KAAK,EAAE,cAAc,EAAE,SAAS,EAAE,MAAM,SAAS,CAAA;AAExD,OAAO,KAAK,EAAE,qBAAqB,EAAE,MAAM,mBAAmB,CAAA;AAK9D,eAAO,MAAM,kBAAkB,WACrB,SAAS,OACZ,cAAc,QACb,SAAS,eACF,OAAO,kBACJ,MAAM,eACT,qBAAqB,CAAC,aAAa,CAAC,SAyMlD,CAAA"}