@payloadcms/plugin-mcp 4.0.0-internal.38b7f1d → 4.0.0-internal.c2b57ce

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 (328) hide show
  1. package/bin.js +39 -0
  2. package/dist/collection/getAccessField.d.ts +12 -0
  3. package/dist/collection/getAccessField.d.ts.map +1 -0
  4. package/dist/collection/getAccessField.js +57 -0
  5. package/dist/collection/getAccessField.js.map +1 -0
  6. package/dist/collection/index.d.ts +6 -0
  7. package/dist/collection/index.d.ts.map +1 -0
  8. package/dist/collection/index.js +59 -0
  9. package/dist/collection/index.js.map +1 -0
  10. package/dist/components/AccessField/index.client.d.ts +10 -0
  11. package/dist/components/AccessField/index.client.d.ts.map +1 -0
  12. package/dist/components/AccessField/index.client.js +305 -0
  13. package/dist/components/AccessField/index.client.js.map +1 -0
  14. package/dist/components/AccessField/index.css +93 -0
  15. package/dist/defineTool.d.ts +26 -0
  16. package/dist/defineTool.d.ts.map +1 -0
  17. package/dist/defineTool.js +37 -0
  18. package/dist/defineTool.js.map +1 -0
  19. package/dist/endpoint/access.d.ts +10 -0
  20. package/dist/endpoint/access.d.ts.map +1 -0
  21. package/dist/endpoint/access.js +106 -0
  22. package/dist/endpoint/access.js.map +1 -0
  23. package/dist/endpoint/index.d.ts +3 -0
  24. package/dist/endpoint/index.d.ts.map +1 -0
  25. package/dist/endpoint/index.js +36 -0
  26. package/dist/endpoint/index.js.map +1 -0
  27. package/dist/exports/client.d.ts +2 -0
  28. package/dist/exports/client.d.ts.map +1 -0
  29. package/dist/exports/client.js +4 -0
  30. package/dist/exports/client.js.map +1 -0
  31. package/dist/index.d.ts +4 -7
  32. package/dist/index.d.ts.map +1 -1
  33. package/dist/index.js +44 -67
  34. package/dist/index.js.map +1 -1
  35. package/dist/mcp/buildMcpServer.d.ts +19 -0
  36. package/dist/mcp/buildMcpServer.d.ts.map +1 -0
  37. package/dist/mcp/buildMcpServer.js +163 -0
  38. package/dist/mcp/buildMcpServer.js.map +1 -0
  39. package/dist/mcp/builtin/collections/authTools.d.ts +7 -0
  40. package/dist/mcp/builtin/collections/authTools.d.ts.map +1 -0
  41. package/dist/mcp/builtin/collections/authTools.js +250 -0
  42. package/dist/mcp/builtin/collections/authTools.js.map +1 -0
  43. package/dist/mcp/builtin/collections/createTool.d.ts +2 -0
  44. package/dist/mcp/builtin/collections/createTool.d.ts.map +1 -0
  45. package/dist/mcp/builtin/collections/createTool.js +87 -0
  46. package/dist/mcp/builtin/collections/createTool.js.map +1 -0
  47. package/dist/mcp/builtin/collections/deleteTool.d.ts +2 -0
  48. package/dist/mcp/builtin/collections/deleteTool.d.ts.map +1 -0
  49. package/dist/mcp/builtin/collections/deleteTool.js +117 -0
  50. package/dist/mcp/builtin/collections/deleteTool.js.map +1 -0
  51. package/dist/mcp/builtin/collections/findTool.d.ts +2 -0
  52. package/dist/mcp/builtin/collections/findTool.d.ts.map +1 -0
  53. package/dist/mcp/builtin/collections/findTool.js +159 -0
  54. package/dist/mcp/builtin/collections/findTool.js.map +1 -0
  55. package/dist/mcp/builtin/collections/updateTool.d.ts +2 -0
  56. package/dist/mcp/builtin/collections/updateTool.d.ts.map +1 -0
  57. package/dist/mcp/builtin/collections/updateTool.js +187 -0
  58. package/dist/mcp/builtin/collections/updateTool.js.map +1 -0
  59. package/dist/mcp/builtin/globals/findTool.d.ts +2 -0
  60. package/dist/mcp/builtin/globals/findTool.d.ts.map +1 -0
  61. package/dist/mcp/builtin/globals/findTool.js +76 -0
  62. package/dist/mcp/builtin/globals/findTool.js.map +1 -0
  63. package/dist/mcp/builtin/globals/updateTool.d.ts +2 -0
  64. package/dist/mcp/builtin/globals/updateTool.d.ts.map +1 -0
  65. package/dist/mcp/builtin/globals/updateTool.js +92 -0
  66. package/dist/mcp/builtin/globals/updateTool.js.map +1 -0
  67. package/dist/mcp/builtinTools.d.ts +37 -0
  68. package/dist/mcp/builtinTools.d.ts.map +1 -0
  69. package/dist/mcp/builtinTools.js +64 -0
  70. package/dist/mcp/builtinTools.js.map +1 -0
  71. package/dist/mcp/sanitizeMCPConfig.d.ts +17 -0
  72. package/dist/mcp/sanitizeMCPConfig.d.ts.map +1 -0
  73. package/dist/mcp/sanitizeMCPConfig.js +167 -0
  74. package/dist/mcp/sanitizeMCPConfig.js.map +1 -0
  75. package/dist/stdio.d.ts +8 -0
  76. package/dist/stdio.d.ts.map +1 -0
  77. package/dist/stdio.js +89 -0
  78. package/dist/stdio.js.map +1 -0
  79. package/dist/types.d.ts +262 -455
  80. package/dist/types.d.ts.map +1 -1
  81. package/dist/types.js +6 -1
  82. package/dist/types.js.map +1 -1
  83. package/dist/utils/camelCase.d.ts.map +1 -1
  84. package/dist/utils/getLogger.d.ts +10 -0
  85. package/dist/utils/getLogger.d.ts.map +1 -0
  86. package/dist/utils/getLogger.js +22 -0
  87. package/dist/utils/getLogger.js.map +1 -0
  88. package/dist/utils/getPluginConfig.d.ts +12 -0
  89. package/dist/utils/getPluginConfig.d.ts.map +1 -0
  90. package/dist/utils/getPluginConfig.js +15 -0
  91. package/dist/utils/getPluginConfig.js.map +1 -0
  92. package/dist/utils/localAPIDefaults.d.ts +20 -0
  93. package/dist/utils/localAPIDefaults.d.ts.map +1 -0
  94. package/dist/utils/localAPIDefaults.js +19 -0
  95. package/dist/utils/localAPIDefaults.js.map +1 -0
  96. package/dist/utils/resolveProjectRoot.d.ts +7 -0
  97. package/dist/utils/resolveProjectRoot.d.ts.map +1 -0
  98. package/dist/utils/resolveProjectRoot.js +15 -0
  99. package/dist/utils/resolveProjectRoot.js.map +1 -0
  100. package/dist/utils/schemaConversion/prepareCollectionSchema.d.ts +7 -0
  101. package/dist/utils/schemaConversion/prepareCollectionSchema.d.ts.map +1 -0
  102. package/dist/utils/schemaConversion/prepareCollectionSchema.js +37 -0
  103. package/dist/utils/schemaConversion/prepareCollectionSchema.js.map +1 -0
  104. package/dist/utils/schemaConversion/removeVirtualFieldsFromSchema.d.ts +2 -2
  105. package/dist/utils/schemaConversion/removeVirtualFieldsFromSchema.d.ts.map +1 -1
  106. package/dist/utils/schemaConversion/removeVirtualFieldsFromSchema.js.map +1 -1
  107. package/dist/utils/schemaConversion/sanitizeJsonSchema.d.ts +2 -2
  108. package/dist/utils/schemaConversion/sanitizeJsonSchema.d.ts.map +1 -1
  109. package/dist/utils/schemaConversion/sanitizeJsonSchema.js.map +1 -1
  110. package/dist/utils/schemaConversion/simplifyRelationshipFields.d.ts +2 -2
  111. package/dist/utils/schemaConversion/simplifyRelationshipFields.d.ts.map +1 -1
  112. package/dist/utils/schemaConversion/simplifyRelationshipFields.js +7 -4
  113. package/dist/utils/schemaConversion/simplifyRelationshipFields.js.map +1 -1
  114. package/dist/utils/schemaConversion/transformPointFields.d.ts +2 -2
  115. package/dist/utils/schemaConversion/transformPointFields.d.ts.map +1 -1
  116. package/dist/utils/schemaConversion/transformPointFields.js +7 -1
  117. package/dist/utils/schemaConversion/transformPointFields.js.map +1 -1
  118. package/dist/utils/toStandardSchema.d.ts +5 -0
  119. package/dist/utils/toStandardSchema.d.ts.map +1 -0
  120. package/dist/utils/toStandardSchema.js +4 -0
  121. package/dist/utils/toStandardSchema.js.map +1 -0
  122. package/package.json +35 -10
  123. package/src/collection/getAccessField.ts +64 -0
  124. package/src/collection/index.ts +63 -0
  125. package/src/components/AccessField/index.client.tsx +344 -0
  126. package/src/components/AccessField/index.css +93 -0
  127. package/src/defineTool.ts +44 -0
  128. package/src/endpoint/access.ts +132 -0
  129. package/src/endpoint/index.ts +35 -0
  130. package/src/exports/client.ts +2 -0
  131. package/src/index.ts +35 -85
  132. package/src/mcp/buildMcpServer.ts +224 -0
  133. package/src/mcp/builtin/collections/authTools.ts +233 -0
  134. package/src/mcp/builtin/collections/createTool.ts +116 -0
  135. package/src/mcp/builtin/collections/deleteTool.ts +123 -0
  136. package/src/mcp/builtin/collections/findTool.ts +187 -0
  137. package/src/mcp/builtin/collections/updateTool.ts +205 -0
  138. package/src/mcp/builtin/globals/findTool.ts +96 -0
  139. package/src/mcp/builtin/globals/updateTool.ts +118 -0
  140. package/src/mcp/builtinTools.ts +84 -0
  141. package/src/mcp/sanitizeMCPConfig.ts +239 -0
  142. package/src/stdio.ts +98 -0
  143. package/src/types.ts +295 -490
  144. package/src/utils/getLogger.ts +22 -0
  145. package/src/utils/getPluginConfig.ts +24 -0
  146. package/src/utils/localAPIDefaults.ts +22 -0
  147. package/src/utils/resolveProjectRoot.ts +17 -0
  148. package/src/utils/schemaConversion/prepareCollectionSchema.ts +39 -0
  149. package/src/utils/schemaConversion/removeVirtualFieldsFromSchema.ts +3 -3
  150. package/src/utils/schemaConversion/sanitizeJsonSchema.ts +4 -4
  151. package/src/utils/schemaConversion/simplifyRelationshipFields.ts +11 -6
  152. package/src/utils/schemaConversion/transformPointFields.ts +6 -5
  153. package/src/utils/toStandardSchema.ts +9 -0
  154. package/dist/collections/createApiKeysCollection.d.ts +0 -7
  155. package/dist/collections/createApiKeysCollection.d.ts.map +0 -1
  156. package/dist/collections/createApiKeysCollection.js +0 -317
  157. package/dist/collections/createApiKeysCollection.js.map +0 -1
  158. package/dist/defaults.d.ts +0 -4
  159. package/dist/defaults.d.ts.map +0 -1
  160. package/dist/defaults.js +0 -5
  161. package/dist/defaults.js.map +0 -1
  162. package/dist/endpoints/mcp.d.ts +0 -4
  163. package/dist/endpoints/mcp.d.ts.map +0 -1
  164. package/dist/endpoints/mcp.js +0 -71
  165. package/dist/endpoints/mcp.js.map +0 -1
  166. package/dist/mcp/createRequest.d.ts +0 -3
  167. package/dist/mcp/createRequest.d.ts.map +0 -1
  168. package/dist/mcp/createRequest.js +0 -14
  169. package/dist/mcp/createRequest.js.map +0 -1
  170. package/dist/mcp/getMcpHandler.d.ts +0 -4
  171. package/dist/mcp/getMcpHandler.d.ts.map +0 -1
  172. package/dist/mcp/getMcpHandler.js +0 -231
  173. package/dist/mcp/getMcpHandler.js.map +0 -1
  174. package/dist/mcp/helpers/config.d.ts +0 -22
  175. package/dist/mcp/helpers/config.d.ts.map +0 -1
  176. package/dist/mcp/helpers/config.js +0 -153
  177. package/dist/mcp/helpers/config.js.map +0 -1
  178. package/dist/mcp/helpers/fields.d.ts +0 -19
  179. package/dist/mcp/helpers/fields.d.ts.map +0 -1
  180. package/dist/mcp/helpers/fields.js +0 -102
  181. package/dist/mcp/helpers/fields.js.map +0 -1
  182. package/dist/mcp/helpers/fileValidation.d.ts +0 -67
  183. package/dist/mcp/helpers/fileValidation.d.ts.map +0 -1
  184. package/dist/mcp/helpers/fileValidation.js +0 -267
  185. package/dist/mcp/helpers/fileValidation.js.map +0 -1
  186. package/dist/mcp/registerTool.d.ts +0 -6
  187. package/dist/mcp/registerTool.d.ts.map +0 -1
  188. package/dist/mcp/registerTool.js +0 -18
  189. package/dist/mcp/registerTool.js.map +0 -1
  190. package/dist/mcp/tools/auth/auth.d.ts +0 -4
  191. package/dist/mcp/tools/auth/auth.d.ts.map +0 -1
  192. package/dist/mcp/tools/auth/auth.js +0 -57
  193. package/dist/mcp/tools/auth/auth.js.map +0 -1
  194. package/dist/mcp/tools/auth/forgotPassword.d.ts +0 -4
  195. package/dist/mcp/tools/auth/forgotPassword.d.ts.map +0 -1
  196. package/dist/mcp/tools/auth/forgotPassword.js +0 -48
  197. package/dist/mcp/tools/auth/forgotPassword.js.map +0 -1
  198. package/dist/mcp/tools/auth/login.d.ts +0 -4
  199. package/dist/mcp/tools/auth/login.d.ts.map +0 -1
  200. package/dist/mcp/tools/auth/login.js +0 -51
  201. package/dist/mcp/tools/auth/login.js.map +0 -1
  202. package/dist/mcp/tools/auth/resetPassword.d.ts +0 -4
  203. package/dist/mcp/tools/auth/resetPassword.d.ts.map +0 -1
  204. package/dist/mcp/tools/auth/resetPassword.js +0 -49
  205. package/dist/mcp/tools/auth/resetPassword.js.map +0 -1
  206. package/dist/mcp/tools/auth/unlock.d.ts +0 -4
  207. package/dist/mcp/tools/auth/unlock.d.ts.map +0 -1
  208. package/dist/mcp/tools/auth/unlock.js +0 -48
  209. package/dist/mcp/tools/auth/unlock.js.map +0 -1
  210. package/dist/mcp/tools/auth/verify.d.ts +0 -4
  211. package/dist/mcp/tools/auth/verify.d.ts.map +0 -1
  212. package/dist/mcp/tools/auth/verify.js +0 -45
  213. package/dist/mcp/tools/auth/verify.js.map +0 -1
  214. package/dist/mcp/tools/collection/create.d.ts +0 -10
  215. package/dist/mcp/tools/collection/create.d.ts.map +0 -1
  216. package/dist/mcp/tools/collection/create.js +0 -139
  217. package/dist/mcp/tools/collection/create.js.map +0 -1
  218. package/dist/mcp/tools/collection/delete.d.ts +0 -10
  219. package/dist/mcp/tools/collection/delete.d.ts.map +0 -1
  220. package/dist/mcp/tools/collection/delete.js +0 -154
  221. package/dist/mcp/tools/collection/delete.js.map +0 -1
  222. package/dist/mcp/tools/collection/find.d.ts +0 -10
  223. package/dist/mcp/tools/collection/find.d.ts.map +0 -1
  224. package/dist/mcp/tools/collection/find.js +0 -165
  225. package/dist/mcp/tools/collection/find.js.map +0 -1
  226. package/dist/mcp/tools/collection/update.d.ts +0 -10
  227. package/dist/mcp/tools/collection/update.d.ts.map +0 -1
  228. package/dist/mcp/tools/collection/update.js +0 -209
  229. package/dist/mcp/tools/collection/update.js.map +0 -1
  230. package/dist/mcp/tools/config/find.d.ts +0 -10
  231. package/dist/mcp/tools/config/find.d.ts.map +0 -1
  232. package/dist/mcp/tools/config/find.js +0 -97
  233. package/dist/mcp/tools/config/find.js.map +0 -1
  234. package/dist/mcp/tools/config/update.d.ts +0 -10
  235. package/dist/mcp/tools/config/update.d.ts.map +0 -1
  236. package/dist/mcp/tools/config/update.js +0 -215
  237. package/dist/mcp/tools/config/update.js.map +0 -1
  238. package/dist/mcp/tools/global/find.d.ts +0 -5
  239. package/dist/mcp/tools/global/find.d.ts.map +0 -1
  240. package/dist/mcp/tools/global/find.js +0 -82
  241. package/dist/mcp/tools/global/find.js.map +0 -1
  242. package/dist/mcp/tools/global/update.d.ts +0 -6
  243. package/dist/mcp/tools/global/update.d.ts.map +0 -1
  244. package/dist/mcp/tools/global/update.js +0 -124
  245. package/dist/mcp/tools/global/update.js.map +0 -1
  246. package/dist/mcp/tools/job/create.d.ts +0 -10
  247. package/dist/mcp/tools/job/create.d.ts.map +0 -1
  248. package/dist/mcp/tools/job/create.js +0 -293
  249. package/dist/mcp/tools/job/create.js.map +0 -1
  250. package/dist/mcp/tools/job/run.d.ts +0 -10
  251. package/dist/mcp/tools/job/run.d.ts.map +0 -1
  252. package/dist/mcp/tools/job/run.js +0 -129
  253. package/dist/mcp/tools/job/run.js.map +0 -1
  254. package/dist/mcp/tools/job/update.d.ts +0 -11
  255. package/dist/mcp/tools/job/update.d.ts.map +0 -1
  256. package/dist/mcp/tools/job/update.js +0 -186
  257. package/dist/mcp/tools/job/update.js.map +0 -1
  258. package/dist/mcp/tools/resource/create.d.ts +0 -6
  259. package/dist/mcp/tools/resource/create.d.ts.map +0 -1
  260. package/dist/mcp/tools/resource/create.js +0 -124
  261. package/dist/mcp/tools/resource/create.js.map +0 -1
  262. package/dist/mcp/tools/resource/delete.d.ts +0 -5
  263. package/dist/mcp/tools/resource/delete.d.ts.map +0 -1
  264. package/dist/mcp/tools/resource/delete.js +0 -151
  265. package/dist/mcp/tools/resource/delete.js.map +0 -1
  266. package/dist/mcp/tools/resource/find.d.ts +0 -5
  267. package/dist/mcp/tools/resource/find.d.ts.map +0 -1
  268. package/dist/mcp/tools/resource/find.js +0 -170
  269. package/dist/mcp/tools/resource/find.js.map +0 -1
  270. package/dist/mcp/tools/resource/update.d.ts +0 -6
  271. package/dist/mcp/tools/resource/update.d.ts.map +0 -1
  272. package/dist/mcp/tools/resource/update.js +0 -256
  273. package/dist/mcp/tools/resource/update.js.map +0 -1
  274. package/dist/mcp/tools/schemas.d.ts +0 -457
  275. package/dist/mcp/tools/schemas.d.ts.map +0 -1
  276. package/dist/mcp/tools/schemas.js +0 -243
  277. package/dist/mcp/tools/schemas.js.map +0 -1
  278. package/dist/utils/adminEntitySettings.d.ts +0 -17
  279. package/dist/utils/adminEntitySettings.d.ts.map +0 -1
  280. package/dist/utils/adminEntitySettings.js +0 -41
  281. package/dist/utils/adminEntitySettings.js.map +0 -1
  282. package/dist/utils/createApiKeyFields.d.ts +0 -15
  283. package/dist/utils/createApiKeyFields.d.ts.map +0 -1
  284. package/dist/utils/createApiKeyFields.js +0 -57
  285. package/dist/utils/createApiKeyFields.js.map +0 -1
  286. package/dist/utils/getEnabledSlugs.d.ts +0 -13
  287. package/dist/utils/getEnabledSlugs.d.ts.map +0 -1
  288. package/dist/utils/getEnabledSlugs.js +0 -32
  289. package/dist/utils/getEnabledSlugs.js.map +0 -1
  290. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.d.ts +0 -3
  291. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.d.ts.map +0 -1
  292. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.js +0 -43
  293. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.js.map +0 -1
  294. package/src/collections/createApiKeysCollection.ts +0 -373
  295. package/src/defaults.ts +0 -3
  296. package/src/endpoints/mcp.ts +0 -91
  297. package/src/mcp/createRequest.ts +0 -13
  298. package/src/mcp/getMcpHandler.ts +0 -545
  299. package/src/mcp/helpers/config.ts +0 -213
  300. package/src/mcp/helpers/fields.ts +0 -154
  301. package/src/mcp/helpers/fileValidation.ts +0 -362
  302. package/src/mcp/registerTool.ts +0 -22
  303. package/src/mcp/tools/auth/auth.ts +0 -71
  304. package/src/mcp/tools/auth/forgotPassword.ts +0 -70
  305. package/src/mcp/tools/auth/login.ts +0 -72
  306. package/src/mcp/tools/auth/resetPassword.ts +0 -61
  307. package/src/mcp/tools/auth/unlock.ts +0 -64
  308. package/src/mcp/tools/auth/verify.ts +0 -57
  309. package/src/mcp/tools/collection/create.ts +0 -210
  310. package/src/mcp/tools/collection/delete.ts +0 -211
  311. package/src/mcp/tools/collection/find.ts +0 -224
  312. package/src/mcp/tools/collection/update.ts +0 -290
  313. package/src/mcp/tools/config/find.ts +0 -128
  314. package/src/mcp/tools/config/update.ts +0 -280
  315. package/src/mcp/tools/global/find.ts +0 -128
  316. package/src/mcp/tools/global/update.ts +0 -207
  317. package/src/mcp/tools/job/create.ts +0 -416
  318. package/src/mcp/tools/job/run.ts +0 -167
  319. package/src/mcp/tools/job/update.ts +0 -274
  320. package/src/mcp/tools/resource/create.ts +0 -211
  321. package/src/mcp/tools/resource/delete.ts +0 -218
  322. package/src/mcp/tools/resource/find.ts +0 -246
  323. package/src/mcp/tools/resource/update.ts +0 -383
  324. package/src/mcp/tools/schemas.ts +0 -520
  325. package/src/utils/adminEntitySettings.ts +0 -40
  326. package/src/utils/createApiKeyFields.ts +0 -72
  327. package/src/utils/getEnabledSlugs.ts +0 -42
  328. package/src/utils/schemaConversion/convertCollectionSchemaToZod.ts +0 -52
@@ -0,0 +1,44 @@
1
+ import type { CollectionTool, GlobalTool, Prompt, Tool, ToolInputSchema } from './types.js'
2
+
3
+ /**
4
+ * Two-stage builder: pass the schema/metadata first, then chain `.handler(fn)`. Splitting the
5
+ * call lets TypeScript resolve `TSchema` from `input` (call 1) before contextually typing the
6
+ * handler (call 2). A single-call API hit a TS limitation where property order in the literal
7
+ * decided whether `TSchema` was inferred or pinned to its default.
8
+ *
9
+ * defineCollectionTool({ description, input })
10
+ * .handler(({ input }) => …) // ← input is fully typed here regardless of order
11
+ *
12
+ * Config and handler signatures are derived from `Tool` / `CollectionTool` / `GlobalTool` /
13
+ * `Prompt` via `Omit` + indexed access so there's no duplication with `types.ts`.
14
+ */
15
+
16
+ export const defineTool = <
17
+ TSchema extends ToolInputSchema | undefined = ToolInputSchema | undefined,
18
+ >(
19
+ args: Omit<Tool<TSchema>, 'handler'>,
20
+ ): { handler: (fn: Tool<TSchema>['handler']) => Tool } => ({
21
+ handler: (fn) => ({ ...args, handler: fn }) as unknown as Tool,
22
+ })
23
+
24
+ export const defineCollectionTool = <
25
+ TSchema extends ToolInputSchema | undefined = ToolInputSchema | undefined,
26
+ >(
27
+ args: Omit<CollectionTool<TSchema>, 'handler'>,
28
+ ): { handler: (fn: CollectionTool<TSchema>['handler']) => CollectionTool } => ({
29
+ handler: (fn) => ({ ...args, handler: fn }) as unknown as CollectionTool,
30
+ })
31
+
32
+ export const defineGlobalTool = <
33
+ TSchema extends ToolInputSchema | undefined = ToolInputSchema | undefined,
34
+ >(
35
+ args: Omit<GlobalTool<TSchema>, 'handler'>,
36
+ ): { handler: (fn: GlobalTool<TSchema>['handler']) => GlobalTool } => ({
37
+ handler: (fn) => ({ ...args, handler: fn }) as unknown as GlobalTool,
38
+ })
39
+
40
+ export const definePrompt = <TSchema extends ToolInputSchema = ToolInputSchema>(
41
+ args: Omit<Prompt<TSchema>, 'handler'>,
42
+ ): { handler: (fn: Prompt<TSchema>['handler']) => Prompt } => ({
43
+ handler: (fn) => ({ ...args, handler: fn }) as unknown as Prompt,
44
+ })
@@ -0,0 +1,132 @@
1
+ import type { DefaultDocumentIDType, PayloadRequest, TypedUser } from 'payload'
2
+
3
+ import crypto from 'crypto'
4
+ import { UnauthorizedError } from 'payload'
5
+
6
+ import type { AuthorizedMCP, MCPAPIKeysDoc } from '../types.js'
7
+
8
+ import { getLogger } from '../utils/getLogger.js'
9
+ import { getPluginConfig } from '../utils/getPluginConfig.js'
10
+
11
+ /**
12
+ * Resolves the API key (or dev-mode session) and returns the items the caller
13
+ * may use. Denied items are dropped from the array.
14
+ */
15
+ export const getAuthorizedMCP: (args: { req: PayloadRequest }) => Promise<AuthorizedMCP> = async ({
16
+ req,
17
+ }) => {
18
+ const logger = getLogger({ payload: req.payload })
19
+ const pluginConfig = getPluginConfig({ config: req.payload.config })
20
+
21
+ const authHeader = req.headers.get('Authorization')
22
+ const hasBearerToken = authHeader?.startsWith('Bearer ')
23
+
24
+ const buildAuthorized = (apiKeyDoc: MCPAPIKeysDoc): AuthorizedMCP => ({
25
+ items: pluginConfig.items.filter((item) => {
26
+ switch (item.type) {
27
+ case 'collectionTool':
28
+ return apiKeyDoc.access.collections?.[item.collectionSlug]?.[item.key] !== false
29
+ case 'globalTool':
30
+ return apiKeyDoc.access.globals?.[item.globalSlug]?.[item.key] !== false
31
+ case 'prompt':
32
+ return apiKeyDoc.access.prompts?.[item.key] !== false
33
+ case 'resource':
34
+ return apiKeyDoc.access.resources?.[item.key] !== false
35
+ case 'tool':
36
+ return apiKeyDoc.access.tools?.[item.key] !== false
37
+ }
38
+ }),
39
+ overrideAccess:
40
+ typeof apiKeyDoc.overrideAccess === 'boolean' ? apiKeyDoc.overrideAccess : false,
41
+ user: apiKeyDoc.user,
42
+ })
43
+
44
+ if (pluginConfig.overrideAuth) {
45
+ return await pluginConfig.overrideAuth({
46
+ getAPIKeyDoc: (overrideApiKey) => getAPIKeyDoc({ logger, overrideApiKey, pluginConfig, req }),
47
+ getAuthorizedMCP: ({ apiKeyDoc }) => buildAuthorized(apiKeyDoc),
48
+ pluginConfig,
49
+ req,
50
+ })
51
+ }
52
+
53
+ if (process.env.NODE_ENV === 'development' && !hasBearerToken) {
54
+ logger.info('Dev mode: skipping API key check, using session user')
55
+ return buildAuthorized({
56
+ id: -1,
57
+ access: {},
58
+ overrideAccess: true,
59
+ user: req.user ?? null,
60
+ })
61
+ }
62
+
63
+ return buildAuthorized(await getAPIKeyDoc({ logger, pluginConfig, req }))
64
+ }
65
+
66
+ const getAPIKeyDoc = async ({
67
+ logger,
68
+ overrideApiKey,
69
+ pluginConfig,
70
+ req,
71
+ }: {
72
+ logger: ReturnType<typeof getLogger>
73
+ overrideApiKey?: string
74
+ pluginConfig: ReturnType<typeof getPluginConfig>
75
+ req: PayloadRequest
76
+ }): Promise<MCPAPIKeysDoc> => {
77
+ const authHeader = req.headers.get('Authorization')
78
+ const hasBearerToken = authHeader?.startsWith('Bearer ')
79
+
80
+ const apiKey =
81
+ overrideApiKey ?? (hasBearerToken ? authHeader?.replace('Bearer ', '').trim() || null : null)
82
+
83
+ if (!apiKey) {
84
+ throw new UnauthorizedError()
85
+ }
86
+
87
+ const sha256APIKeyIndex = crypto
88
+ .createHmac('sha256', req.payload.secret)
89
+ .update(apiKey)
90
+ .digest('hex')
91
+
92
+ const doc = await req.payload.db.findOne<MCPAPIKeysDoc>({
93
+ collection: 'payload-mcp-api-keys',
94
+ req,
95
+ where: {
96
+ apiKeyIndex: { equals: sha256APIKeyIndex },
97
+ },
98
+ })
99
+
100
+ if (!doc || !doc.user) {
101
+ throw new UnauthorizedError()
102
+ }
103
+
104
+ logger.info('API Key is valid')
105
+
106
+ const userRef = doc.user
107
+ const userID =
108
+ typeof userRef === 'object' && userRef !== null && 'id' in userRef
109
+ ? userRef.id
110
+ : (userRef as unknown as DefaultDocumentIDType)
111
+
112
+ const user = (await req.payload.findByID({
113
+ id: userID,
114
+ collection: pluginConfig.userCollection,
115
+ depth: 0,
116
+ disableErrors: true,
117
+ req,
118
+ })) as null | TypedUser
119
+
120
+ if (!user) {
121
+ throw new UnauthorizedError()
122
+ }
123
+
124
+ return {
125
+ ...doc,
126
+ user: {
127
+ ...user,
128
+ _strategy: 'mcp-api-key' as const,
129
+ collection: pluginConfig.userCollection,
130
+ },
131
+ }
132
+ }
@@ -0,0 +1,35 @@
1
+ import { WebStandardStreamableHTTPServerTransport } from '@modelcontextprotocol/server'
2
+ import { AuthenticationError, type PayloadHandler } from 'payload'
3
+
4
+ import { buildMcpServer } from '../mcp/buildMcpServer.js'
5
+ import { getPluginConfig } from '../utils/getPluginConfig.js'
6
+ import { getAuthorizedMCP } from './access.js'
7
+
8
+ export const mcpEndpoint: PayloadHandler = async (req) => {
9
+ if (!req.url) {
10
+ throw new AuthenticationError()
11
+ }
12
+
13
+ req.payloadAPI = 'MCP' as const
14
+
15
+ const pluginConfig = getPluginConfig({ config: req.payload.config })
16
+ const authorizedMCP = await getAuthorizedMCP({ req })
17
+
18
+ const server = buildMcpServer({ authorizedMCP, pluginConfig, req })
19
+
20
+ const transport = new WebStandardStreamableHTTPServerTransport({
21
+ enableJsonResponse: true,
22
+ sessionIdGenerator: undefined, // stateless mode
23
+ })
24
+
25
+ await server.connect(transport)
26
+
27
+ const mcpRequest = new Request(req.url, {
28
+ body: req.body,
29
+ duplex: 'half',
30
+ headers: req.headers,
31
+ method: req.method,
32
+ } as { duplex: 'half' } & RequestInit)
33
+
34
+ return await transport.handleRequest(mcpRequest)
35
+ }
@@ -0,0 +1,2 @@
1
+ 'use client'
2
+ export { AccessField } from '../components/AccessField/index.client.js'
package/src/index.ts CHANGED
@@ -1,113 +1,63 @@
1
- import { definePlugin } from 'payload'
1
+ import { defaultUserCollection, definePlugin } from 'payload'
2
2
 
3
- import type { MCPAccessSettings, MCPPluginConfig } from './types.js'
3
+ import type { AuthorizedMCP, MCPPluginConfig, SanitizedMCPPluginConfig } from './types.js'
4
4
 
5
- import { createAPIKeysCollection } from './collections/createApiKeysCollection.js'
6
- import { initializeMCPHandler } from './endpoints/mcp.js'
5
+ import { getAPIKeysCollection } from './collection/index.js'
6
+ import { mcpEndpoint } from './endpoint/index.js'
7
+ import { sanitizeMCPConfig } from './mcp/sanitizeMCPConfig.js'
7
8
 
8
9
  declare module 'payload' {
9
10
  export interface PayloadRequest {
10
11
  payloadAPI: 'GraphQL' | 'local' | 'MCP' | 'REST'
11
12
  }
12
13
  interface RegisteredPlugins {
14
+ /** After the plugin's `plugin` callback runs, `options` holds the sanitized config. */
13
15
  '@payloadcms/plugin-mcp': MCPPluginConfig
14
16
  }
15
17
  }
16
18
 
17
- import { defaults } from './defaults.js'
19
+ export type { AuthorizedMCP, MCPPluginConfig, SanitizedMCPPluginConfig }
18
20
 
19
- export type { MCPAccessSettings, MCPPluginConfig }
21
+ export { defineCollectionTool, defineGlobalTool, definePrompt, defineTool } from './defineTool.js'
20
22
 
21
- /**
22
- * The MCP Plugin for Payload. This plugin allows you to add MCP capabilities to your Payload project.
23
- *
24
- * @param pluginOptions - The options for the MCP plugin.
25
- */
26
23
  export const mcpPlugin = definePlugin<MCPPluginConfig>({
27
24
  slug: '@payloadcms/plugin-mcp',
28
25
  order: 10,
29
- plugin: ({ config, plugins: _plugins, ...pluginOptions }) => {
30
- if (!config.collections) {
31
- config.collections = []
26
+ plugin: ({ config, plugins, ...rawConfig }) => {
27
+ // Our `payload-mcp-api-keys` is auth-enabled; if it'd be the only auth
28
+ // collection, Payload's later sanitize would pick it as `admin.user`.
29
+ // Pre-seed the default user collection to prevent that.
30
+ if (!config.admin?.user) {
31
+ const firstCollectionWithAuth = (config.collections ?? []).find(({ auth }) => Boolean(auth))
32
+ if (!firstCollectionWithAuth) {
33
+ ;(config.collections ??= []).push(defaultUserCollection)
34
+ }
32
35
  }
33
36
 
34
- // Collections
35
- const collections = pluginOptions.collections || {}
36
- // Globals
37
- const globals = pluginOptions.globals || {}
38
- // Extract custom tools for the global config
39
- const customTools =
40
- pluginOptions.mcp?.tools?.map((tool) => ({
41
- name: tool.name,
42
- description: tool.description,
43
- })) || []
37
+ const pluginConfig = sanitizeMCPConfig({ config, pluginConfig: rawConfig })
44
38
 
45
- // User Collection
46
- pluginOptions.userCollection =
47
- pluginOptions.userCollection ?? config?.admin?.user ?? defaults.userCollection
48
-
49
- const experimentalTools = pluginOptions?.experimental?.tools || {}
50
-
51
- /**
52
- * API Keys
53
- * --------
54
- * High resolution control over MCP capabilities is crucial when using Payload with LLMs.
55
- *
56
- * This API Keys collection has ways for admins to create API keys and allow or disallow the MCP capabilities.
57
- * This is useful when Admins want to allow or disallow the use of the MCP capabilities in real time.
58
- * For example:
59
- * - If a collection has all of its capabilities enabled, admins can allow or disallow the create, update, delete, and find capabilities on that collection.
60
- * - If a collection only has the find capability enabled, admins can only allow or disallow the find capability on that collection.
61
- * - If a global has all of its capabilities enabled, admins can allow or disallow the find and update capabilities on that global.
62
- * - If a custom tool has gone haywire, admins can disallow that tool.
63
- *
64
- */
65
- const apiKeyCollection = createAPIKeysCollection(
66
- collections,
67
- globals,
68
- customTools,
69
- experimentalTools,
70
- pluginOptions,
71
- )
72
- if (pluginOptions.overrideApiKeyCollection) {
73
- config.collections.push(pluginOptions.overrideApiKeyCollection(apiKeyCollection))
74
- } else {
75
- config.collections.push(apiKeyCollection)
39
+ // Stash the sanitized config on plugin options so `getPluginConfig()` reads it.
40
+ const registered = plugins['@payloadcms/plugin-mcp']
41
+ if (registered) {
42
+ // @ts-expect-error
43
+ registered.sanitizedOptions = pluginConfig as unknown as typeof registered.options
76
44
  }
77
45
 
78
- /**
79
- * If the plugin is disabled, we still want to keep added collections/fields so the database schema is consistent which is important for migrations.
80
- * If your plugin heavily modifies the database schema, you may want to remove this property.
81
- */
82
- if (pluginOptions.disabled) {
46
+ ;(config.collections ??= []).push(getAPIKeysCollection({ pluginConfig }))
47
+
48
+ // Keep the API-keys collection registered even when disabled, so DB schema
49
+ // and generated types don't drift between enabled/disabled environments.
50
+ if (pluginConfig.disabled) {
83
51
  return config
84
52
  }
85
53
 
86
- if (!config.endpoints) {
87
- config.endpoints = []
54
+ return {
55
+ ...config,
56
+ endpoints: [
57
+ ...(config.endpoints ?? []),
58
+ // Payload prefixes /api, so the full path is /api/mcp.
59
+ { handler: mcpEndpoint, method: 'post', path: '/mcp' },
60
+ ],
88
61
  }
89
-
90
- /**
91
- * This is the primary MCP Server Endpoint.
92
- * Payload will automatically add the /api prefix to the path, so the full path is `/api/mcp`
93
- * NOTE: This is only transport method until we add full support for SSE which will be another endpoint at `/api/sse`
94
- */
95
- config.endpoints.push({
96
- handler: initializeMCPHandler(pluginOptions),
97
- method: 'post',
98
- path: '/mcp',
99
- })
100
-
101
- /**
102
- * The GET response is always: {"jsonrpc":"2.0","error":{"code":-32000,"message":"Method not allowed."},"id":null} -- even with an API key
103
- * This is expected behavior and MCP clients should always use the POST endpoint.
104
- */
105
- config.endpoints.push({
106
- handler: initializeMCPHandler(pluginOptions),
107
- method: 'get',
108
- path: '/mcp',
109
- })
110
-
111
- return config
112
62
  },
113
63
  })
@@ -0,0 +1,224 @@
1
+ import { McpServer, type ServerContext } from '@modelcontextprotocol/server'
2
+ import { APIError, configToJSONSchema, type PayloadRequest } from 'payload'
3
+
4
+ import type {
5
+ AuthorizedMCP,
6
+ JsonSchemaType,
7
+ MCPResponseOverride,
8
+ MCPToolResponse,
9
+ SanitizedMCPPluginConfig,
10
+ } from '../types.js'
11
+
12
+ import { toCamelCase } from '../utils/camelCase.js'
13
+ import { getLogger } from '../utils/getLogger.js'
14
+ import {
15
+ getCollectionVirtualFieldNames,
16
+ getGlobalVirtualFieldNames,
17
+ } from '../utils/getVirtualFieldNames.js'
18
+ import { removeVirtualFieldsFromSchema } from '../utils/schemaConversion/removeVirtualFieldsFromSchema.js'
19
+ import { toStandardSchema } from '../utils/toStandardSchema.js'
20
+
21
+ /** `findPosts`, `updateSiteSettings` — auto-prefixed wire name for collection/global tools. */
22
+ const wireName = (key: string, slug: string): string => {
23
+ const camel = toCamelCase(slug)
24
+ return `${key}${camel.charAt(0).toUpperCase()}${camel.slice(1)}`
25
+ }
26
+
27
+ /**
28
+ * Transport-agnostic core: registers every authorized MCP item onto a fresh
29
+ * `McpServer` and returns it. The caller is responsible for picking a transport
30
+ * (`WebStandardStreamableHTTPServerTransport`, `StdioServerTransport`, …) and
31
+ * calling `server.connect(transport)`.
32
+ *
33
+ * `req` is the request context handlers see. For HTTP it's the live
34
+ * `PayloadRequest` derived from the incoming HTTP request; for stdio it's a
35
+ * synthesized one built via `createLocalReq`.
36
+ */
37
+ export const buildMcpServer = ({
38
+ authorizedMCP,
39
+ pluginConfig,
40
+ req,
41
+ }: {
42
+ authorizedMCP: AuthorizedMCP
43
+ pluginConfig: SanitizedMCPPluginConfig
44
+ req: PayloadRequest
45
+ }): McpServer => {
46
+ const serverOptions = pluginConfig.mcp?.serverOptions || {}
47
+ const server = new McpServer(
48
+ { name: 'Payload MCP Server', version: '1.0.0', ...serverOptions.serverInfo },
49
+ serverOptions.options,
50
+ )
51
+
52
+ const logger = getLogger({ payload: req.payload })
53
+
54
+ /**
55
+ * Wrap a tool handler's response with the tool's `overrideResponse`, then
56
+ * strip the internal `doc` field so it doesn't leak onto the wire.
57
+ */
58
+ const finalizeToolResponse = (
59
+ response: MCPToolResponse,
60
+ overrideResponse?: MCPResponseOverride,
61
+ ): MCPToolResponse => {
62
+ const overridden = overrideResponse?.(response, response.doc ?? {}, req) ?? response
63
+ const { doc: _doc, ...rest } = overridden
64
+ return rest
65
+ }
66
+
67
+ const configSchema = configToJSONSchema(
68
+ req.payload.config,
69
+ req.payload.db.defaultIDType,
70
+ req.i18n,
71
+ { forceInlineBlocks: true },
72
+ ) as JsonSchemaType
73
+
74
+ try {
75
+ for (const item of authorizedMCP.items) {
76
+ switch (item.type) {
77
+ case 'collectionTool': {
78
+ const tool = item.tool
79
+ const name = wireName(item.key, item.collectionSlug)
80
+ let inputSchema = tool.input
81
+ if (typeof inputSchema === 'function') {
82
+ const raw = configSchema.definitions?.[item.collectionSlug]
83
+ if (!raw) {
84
+ throw new APIError(
85
+ `Collection schema not found for slug: ${item.collectionSlug}`,
86
+ 500,
87
+ )
88
+ }
89
+ const collectionSchema = removeVirtualFieldsFromSchema(
90
+ JSON.parse(JSON.stringify(raw)) as JsonSchemaType,
91
+ getCollectionVirtualFieldNames(req.payload.config, item.collectionSlug),
92
+ )
93
+ inputSchema = inputSchema({ collectionSchema })
94
+ }
95
+ server.registerTool(
96
+ name,
97
+ {
98
+ description: tool.description,
99
+ inputSchema: inputSchema ? toStandardSchema(inputSchema) : undefined,
100
+ },
101
+ async (input: unknown, ctx: ServerContext) =>
102
+ finalizeToolResponse(
103
+ await tool.handler({
104
+ authorizedMCP,
105
+ collectionSlug: item.collectionSlug,
106
+ input: (input ?? {}) as Record<string, unknown>,
107
+ req,
108
+ serverContext: ctx,
109
+ }),
110
+ tool.overrideResponse,
111
+ ),
112
+ )
113
+ logger.info(`✅ Tool: ${name} Registered.`)
114
+ break
115
+ }
116
+ case 'globalTool': {
117
+ const tool = item.tool
118
+ const name = wireName(item.key, item.globalSlug)
119
+ let inputSchema = tool.input
120
+ if (typeof inputSchema === 'function') {
121
+ const raw = configSchema.definitions?.[item.globalSlug]
122
+ if (!raw) {
123
+ throw new APIError(`Global schema not found for slug: ${item.globalSlug}`, 500)
124
+ }
125
+ const globalSchema = removeVirtualFieldsFromSchema(
126
+ JSON.parse(JSON.stringify(raw)) as JsonSchemaType,
127
+ getGlobalVirtualFieldNames(req.payload.config, item.globalSlug),
128
+ )
129
+
130
+ inputSchema = inputSchema({ globalSchema })
131
+ }
132
+ server.registerTool(
133
+ name,
134
+ {
135
+ description: tool.description,
136
+ inputSchema: inputSchema ? toStandardSchema(inputSchema) : undefined,
137
+ },
138
+ async (input: unknown, ctx: ServerContext) =>
139
+ finalizeToolResponse(
140
+ await tool.handler({
141
+ authorizedMCP,
142
+ globalSlug: item.globalSlug,
143
+ input: (input ?? {}) as Record<string, unknown>,
144
+ req,
145
+ serverContext: ctx,
146
+ }),
147
+ tool.overrideResponse,
148
+ ),
149
+ )
150
+ logger.info(`✅ Tool: ${name} Registered.`)
151
+ break
152
+ }
153
+ case 'prompt': {
154
+ const prompt = item.prompt
155
+ server.registerPrompt(
156
+ item.key,
157
+ {
158
+ argsSchema: prompt.argsSchema ? toStandardSchema(prompt.argsSchema) : undefined,
159
+ description: prompt.description,
160
+ title: prompt.title,
161
+ },
162
+ async (input: unknown, ctx: ServerContext) =>
163
+ prompt.handler({
164
+ input: (input ?? {}) as Record<string, unknown>,
165
+ req,
166
+ serverContext: ctx,
167
+ }),
168
+ )
169
+ logger.info(`✅ Prompt: ${prompt.title} Registered.`)
170
+ break
171
+ }
172
+ case 'resource': {
173
+ const resource = item.resource
174
+ server.registerResource(
175
+ item.key,
176
+ // @ts-expect-error - Overload type ambiguity (string OR ResourceTemplate is valid)
177
+ resource.uri,
178
+ {
179
+ description: resource.description,
180
+ mimeType: resource.mimeType,
181
+ title: resource.title,
182
+ },
183
+ // Static URIs call (uri, ctx); ResourceTemplates call (uri, params, ctx).
184
+ // The rest-params shape lets us collect either signature uniformly.
185
+ async (...sdkArgs: unknown[]) => {
186
+ const ctx = sdkArgs[sdkArgs.length - 1] as ServerContext
187
+ const uri = sdkArgs[0] as URL
188
+ const params = (sdkArgs.length > 2 ? sdkArgs[1] : {}) as Record<string, string>
189
+ return resource.handler({ params, req, serverContext: ctx, uri })
190
+ },
191
+ )
192
+ logger.info(`✅ Resource: ${resource.title} Registered.`)
193
+ break
194
+ }
195
+ case 'tool': {
196
+ const tool = item.tool
197
+ server.registerTool(
198
+ item.key,
199
+ {
200
+ description: tool.description,
201
+ inputSchema: tool.input ? toStandardSchema(tool.input) : undefined,
202
+ },
203
+ async (input: unknown, ctx: ServerContext) =>
204
+ finalizeToolResponse(
205
+ await tool.handler({
206
+ authorizedMCP,
207
+ input: (input ?? {}) as Record<string, unknown>,
208
+ req,
209
+ serverContext: ctx,
210
+ }),
211
+ tool.overrideResponse,
212
+ ),
213
+ )
214
+ logger.info(`✅ Tool: ${item.key} Registered.`)
215
+ break
216
+ }
217
+ }
218
+ }
219
+ } catch (error) {
220
+ throw new APIError(`Error initializing MCP handler: ${String(error)}`, 500)
221
+ }
222
+
223
+ return server
224
+ }