@payloadcms/plugin-mcp 3.86.0-internal.ac46214 → 4.0.0-canary.1

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 (368) hide show
  1. package/README.md +2 -2
  2. package/bin.js +39 -0
  3. package/dist/collection/getAccessField.d.ts +12 -0
  4. package/dist/collection/getAccessField.d.ts.map +1 -0
  5. package/dist/collection/getAccessField.js +57 -0
  6. package/dist/collection/getAccessField.js.map +1 -0
  7. package/dist/collection/index.d.ts +6 -0
  8. package/dist/collection/index.d.ts.map +1 -0
  9. package/dist/collection/index.js +60 -0
  10. package/dist/collection/index.js.map +1 -0
  11. package/dist/components/AccessField/index.client.d.ts +10 -0
  12. package/dist/components/AccessField/index.client.d.ts.map +1 -0
  13. package/dist/components/AccessField/index.client.js +305 -0
  14. package/dist/components/AccessField/index.client.js.map +1 -0
  15. package/dist/components/AccessField/index.css +93 -0
  16. package/dist/defineTool.d.ts +26 -0
  17. package/dist/defineTool.d.ts.map +1 -0
  18. package/dist/defineTool.js +37 -0
  19. package/dist/defineTool.js.map +1 -0
  20. package/dist/endpoint/access.d.ts +10 -0
  21. package/dist/endpoint/access.d.ts.map +1 -0
  22. package/dist/endpoint/access.js +106 -0
  23. package/dist/endpoint/access.js.map +1 -0
  24. package/dist/endpoint/index.d.ts +3 -0
  25. package/dist/endpoint/index.d.ts.map +1 -0
  26. package/dist/endpoint/index.js +36 -0
  27. package/dist/endpoint/index.js.map +1 -0
  28. package/dist/exports/client.d.ts +2 -0
  29. package/dist/exports/client.d.ts.map +1 -0
  30. package/dist/exports/client.js +4 -0
  31. package/dist/exports/client.js.map +1 -0
  32. package/dist/index.d.ts +4 -7
  33. package/dist/index.d.ts.map +1 -1
  34. package/dist/index.js +44 -67
  35. package/dist/index.js.map +1 -1
  36. package/dist/mcp/buildMcpServer.d.ts +19 -0
  37. package/dist/mcp/buildMcpServer.d.ts.map +1 -0
  38. package/dist/mcp/buildMcpServer.js +199 -0
  39. package/dist/mcp/buildMcpServer.js.map +1 -0
  40. package/dist/mcp/builtin/collections/authTools.d.ts +7 -0
  41. package/dist/mcp/builtin/collections/authTools.d.ts.map +1 -0
  42. package/dist/mcp/builtin/collections/authTools.js +250 -0
  43. package/dist/mcp/builtin/collections/authTools.js.map +1 -0
  44. package/dist/mcp/builtin/collections/createTool.d.ts +2 -0
  45. package/dist/mcp/builtin/collections/createTool.d.ts.map +1 -0
  46. package/dist/mcp/builtin/collections/createTool.js +94 -0
  47. package/dist/mcp/builtin/collections/createTool.js.map +1 -0
  48. package/dist/mcp/builtin/collections/deleteTool.d.ts +2 -0
  49. package/dist/mcp/builtin/collections/deleteTool.d.ts.map +1 -0
  50. package/dist/mcp/builtin/collections/deleteTool.js +102 -0
  51. package/dist/mcp/builtin/collections/deleteTool.js.map +1 -0
  52. package/dist/mcp/builtin/collections/findTool.d.ts +2 -0
  53. package/dist/mcp/builtin/collections/findTool.d.ts.map +1 -0
  54. package/dist/mcp/builtin/collections/findTool.js +144 -0
  55. package/dist/mcp/builtin/collections/findTool.js.map +1 -0
  56. package/dist/mcp/builtin/collections/formatCollectionError.d.ts +9 -0
  57. package/dist/mcp/builtin/collections/formatCollectionError.d.ts.map +1 -0
  58. package/dist/mcp/builtin/collections/formatCollectionError.js +60 -0
  59. package/dist/mcp/builtin/collections/formatCollectionError.js.map +1 -0
  60. package/dist/mcp/builtin/collections/getCollectionSchemaTool.d.ts +2 -0
  61. package/dist/mcp/builtin/collections/getCollectionSchemaTool.d.ts.map +1 -0
  62. package/dist/mcp/builtin/collections/getCollectionSchemaTool.js +35 -0
  63. package/dist/mcp/builtin/collections/getCollectionSchemaTool.js.map +1 -0
  64. package/dist/mcp/builtin/collections/updateTool.d.ts +2 -0
  65. package/dist/mcp/builtin/collections/updateTool.d.ts.map +1 -0
  66. package/dist/mcp/builtin/collections/updateTool.js +199 -0
  67. package/dist/mcp/builtin/collections/updateTool.js.map +1 -0
  68. package/dist/mcp/builtin/getConfigInfoTool.d.ts +2 -0
  69. package/dist/mcp/builtin/getConfigInfoTool.d.ts.map +1 -0
  70. package/dist/mcp/builtin/getConfigInfoTool.js +49 -0
  71. package/dist/mcp/builtin/getConfigInfoTool.js.map +1 -0
  72. package/dist/mcp/builtin/globals/findTool.d.ts +2 -0
  73. package/dist/mcp/builtin/globals/findTool.d.ts.map +1 -0
  74. package/dist/mcp/builtin/globals/findTool.js +76 -0
  75. package/dist/mcp/builtin/globals/findTool.js.map +1 -0
  76. package/dist/mcp/builtin/globals/getGlobalSchemaTool.d.ts +2 -0
  77. package/dist/mcp/builtin/globals/getGlobalSchemaTool.d.ts.map +1 -0
  78. package/dist/mcp/builtin/globals/getGlobalSchemaTool.js +35 -0
  79. package/dist/mcp/builtin/globals/getGlobalSchemaTool.js.map +1 -0
  80. package/dist/mcp/builtin/globals/updateTool.d.ts +2 -0
  81. package/dist/mcp/builtin/globals/updateTool.d.ts.map +1 -0
  82. package/dist/mcp/builtin/globals/updateTool.js +94 -0
  83. package/dist/mcp/builtin/globals/updateTool.js.map +1 -0
  84. package/dist/mcp/builtin/validateEntityData.d.ts +14 -0
  85. package/dist/mcp/builtin/validateEntityData.d.ts.map +1 -0
  86. package/dist/mcp/builtin/validateEntityData.js +82 -0
  87. package/dist/mcp/builtin/validateEntityData.js.map +1 -0
  88. package/dist/mcp/builtinTools.d.ts +105 -0
  89. package/dist/mcp/builtinTools.d.ts.map +1 -0
  90. package/dist/mcp/builtinTools.js +107 -0
  91. package/dist/mcp/builtinTools.js.map +1 -0
  92. package/dist/mcp/sanitizeMCPConfig.d.ts +17 -0
  93. package/dist/mcp/sanitizeMCPConfig.d.ts.map +1 -0
  94. package/dist/mcp/sanitizeMCPConfig.js +188 -0
  95. package/dist/mcp/sanitizeMCPConfig.js.map +1 -0
  96. package/dist/stdio.d.ts +8 -0
  97. package/dist/stdio.d.ts.map +1 -0
  98. package/dist/stdio.js +89 -0
  99. package/dist/stdio.js.map +1 -0
  100. package/dist/types.d.ts +251 -455
  101. package/dist/types.d.ts.map +1 -1
  102. package/dist/types.js +6 -1
  103. package/dist/types.js.map +1 -1
  104. package/dist/utils/camelCase.d.ts.map +1 -1
  105. package/dist/utils/getLogger.d.ts +10 -0
  106. package/dist/utils/getLogger.d.ts.map +1 -0
  107. package/dist/utils/getLogger.js +22 -0
  108. package/dist/utils/getLogger.js.map +1 -0
  109. package/dist/utils/getPluginConfig.d.ts +12 -0
  110. package/dist/utils/getPluginConfig.d.ts.map +1 -0
  111. package/dist/utils/getPluginConfig.js +15 -0
  112. package/dist/utils/getPluginConfig.js.map +1 -0
  113. package/dist/utils/localAPIDefaults.d.ts +20 -0
  114. package/dist/utils/localAPIDefaults.d.ts.map +1 -0
  115. package/dist/utils/localAPIDefaults.js +19 -0
  116. package/dist/utils/localAPIDefaults.js.map +1 -0
  117. package/dist/utils/resolveProjectRoot.d.ts +7 -0
  118. package/dist/utils/resolveProjectRoot.d.ts.map +1 -0
  119. package/dist/utils/resolveProjectRoot.js +15 -0
  120. package/dist/utils/resolveProjectRoot.js.map +1 -0
  121. package/dist/utils/schemaConversion/getEntityInputSchema.d.ts +11 -0
  122. package/dist/utils/schemaConversion/getEntityInputSchema.d.ts.map +1 -0
  123. package/dist/utils/schemaConversion/getEntityInputSchema.js +34 -0
  124. package/dist/utils/schemaConversion/getEntityInputSchema.js.map +1 -0
  125. package/dist/utils/schemaConversion/removeVirtualFieldsFromSchema.d.ts +2 -2
  126. package/dist/utils/schemaConversion/removeVirtualFieldsFromSchema.d.ts.map +1 -1
  127. package/dist/utils/schemaConversion/removeVirtualFieldsFromSchema.js.map +1 -1
  128. package/dist/utils/schemaConversion/sanitizeEntitySchema.d.ts +15 -0
  129. package/dist/utils/schemaConversion/sanitizeEntitySchema.d.ts.map +1 -0
  130. package/dist/utils/schemaConversion/sanitizeEntitySchema.js +464 -0
  131. package/dist/utils/schemaConversion/sanitizeEntitySchema.js.map +1 -0
  132. package/dist/utils/schemaConversion/sanitizeEntitySchema.spec.js +158 -0
  133. package/dist/utils/schemaConversion/sanitizeEntitySchema.spec.js.map +1 -0
  134. package/dist/utils/toStandardSchema.d.ts +5 -0
  135. package/dist/utils/toStandardSchema.d.ts.map +1 -0
  136. package/dist/utils/toStandardSchema.js +4 -0
  137. package/dist/utils/toStandardSchema.js.map +1 -0
  138. package/dist/utils/whereSchema.d.ts +9 -0
  139. package/dist/utils/whereSchema.d.ts.map +1 -0
  140. package/dist/utils/whereSchema.js +13 -0
  141. package/dist/utils/whereSchema.js.map +1 -0
  142. package/package.json +35 -10
  143. package/src/collection/getAccessField.ts +64 -0
  144. package/src/collection/index.ts +64 -0
  145. package/src/components/AccessField/index.client.tsx +347 -0
  146. package/src/components/AccessField/index.css +93 -0
  147. package/src/defineTool.ts +44 -0
  148. package/src/endpoint/access.ts +132 -0
  149. package/src/endpoint/index.ts +35 -0
  150. package/src/exports/client.ts +2 -0
  151. package/src/index.ts +35 -85
  152. package/src/mcp/buildMcpServer.ts +257 -0
  153. package/src/mcp/builtin/collections/authTools.ts +233 -0
  154. package/src/mcp/builtin/collections/createTool.ts +112 -0
  155. package/src/mcp/builtin/collections/deleteTool.ts +116 -0
  156. package/src/mcp/builtin/collections/findTool.ts +177 -0
  157. package/src/mcp/builtin/collections/formatCollectionError.ts +84 -0
  158. package/src/mcp/builtin/collections/getCollectionSchemaTool.ts +28 -0
  159. package/src/mcp/builtin/collections/updateTool.ts +211 -0
  160. package/src/mcp/builtin/getConfigInfoTool.ts +44 -0
  161. package/src/mcp/builtin/globals/findTool.ts +96 -0
  162. package/src/mcp/builtin/globals/getGlobalSchemaTool.ts +28 -0
  163. package/src/mcp/builtin/globals/updateTool.ts +115 -0
  164. package/src/mcp/builtin/validateEntityData.ts +132 -0
  165. package/src/mcp/builtinTools.ts +98 -0
  166. package/src/mcp/sanitizeMCPConfig.ts +260 -0
  167. package/src/stdio.ts +98 -0
  168. package/src/types.ts +290 -490
  169. package/src/utils/getLogger.ts +22 -0
  170. package/src/utils/getPluginConfig.ts +24 -0
  171. package/src/utils/localAPIDefaults.ts +22 -0
  172. package/src/utils/resolveProjectRoot.ts +17 -0
  173. package/src/utils/schemaConversion/getEntityInputSchema.ts +78 -0
  174. package/src/utils/schemaConversion/removeVirtualFieldsFromSchema.ts +3 -3
  175. package/src/utils/schemaConversion/sanitizeEntitySchema.spec.ts +103 -0
  176. package/src/utils/schemaConversion/sanitizeEntitySchema.ts +529 -0
  177. package/src/utils/toStandardSchema.ts +9 -0
  178. package/src/utils/whereSchema.ts +24 -0
  179. package/dist/collections/createApiKeysCollection.d.ts +0 -7
  180. package/dist/collections/createApiKeysCollection.d.ts.map +0 -1
  181. package/dist/collections/createApiKeysCollection.js +0 -317
  182. package/dist/collections/createApiKeysCollection.js.map +0 -1
  183. package/dist/defaults.d.ts +0 -4
  184. package/dist/defaults.d.ts.map +0 -1
  185. package/dist/defaults.js +0 -5
  186. package/dist/defaults.js.map +0 -1
  187. package/dist/endpoints/mcp.d.ts +0 -4
  188. package/dist/endpoints/mcp.d.ts.map +0 -1
  189. package/dist/endpoints/mcp.js +0 -71
  190. package/dist/endpoints/mcp.js.map +0 -1
  191. package/dist/mcp/createRequest.d.ts +0 -3
  192. package/dist/mcp/createRequest.d.ts.map +0 -1
  193. package/dist/mcp/createRequest.js +0 -14
  194. package/dist/mcp/createRequest.js.map +0 -1
  195. package/dist/mcp/getMcpHandler.d.ts +0 -4
  196. package/dist/mcp/getMcpHandler.d.ts.map +0 -1
  197. package/dist/mcp/getMcpHandler.js +0 -231
  198. package/dist/mcp/getMcpHandler.js.map +0 -1
  199. package/dist/mcp/helpers/config.d.ts +0 -22
  200. package/dist/mcp/helpers/config.d.ts.map +0 -1
  201. package/dist/mcp/helpers/config.js +0 -153
  202. package/dist/mcp/helpers/config.js.map +0 -1
  203. package/dist/mcp/helpers/fields.d.ts +0 -19
  204. package/dist/mcp/helpers/fields.d.ts.map +0 -1
  205. package/dist/mcp/helpers/fields.js +0 -102
  206. package/dist/mcp/helpers/fields.js.map +0 -1
  207. package/dist/mcp/helpers/fileValidation.d.ts +0 -67
  208. package/dist/mcp/helpers/fileValidation.d.ts.map +0 -1
  209. package/dist/mcp/helpers/fileValidation.js +0 -267
  210. package/dist/mcp/helpers/fileValidation.js.map +0 -1
  211. package/dist/mcp/registerTool.d.ts +0 -6
  212. package/dist/mcp/registerTool.d.ts.map +0 -1
  213. package/dist/mcp/registerTool.js +0 -18
  214. package/dist/mcp/registerTool.js.map +0 -1
  215. package/dist/mcp/tools/auth/auth.d.ts +0 -4
  216. package/dist/mcp/tools/auth/auth.d.ts.map +0 -1
  217. package/dist/mcp/tools/auth/auth.js +0 -57
  218. package/dist/mcp/tools/auth/auth.js.map +0 -1
  219. package/dist/mcp/tools/auth/forgotPassword.d.ts +0 -4
  220. package/dist/mcp/tools/auth/forgotPassword.d.ts.map +0 -1
  221. package/dist/mcp/tools/auth/forgotPassword.js +0 -48
  222. package/dist/mcp/tools/auth/forgotPassword.js.map +0 -1
  223. package/dist/mcp/tools/auth/login.d.ts +0 -4
  224. package/dist/mcp/tools/auth/login.d.ts.map +0 -1
  225. package/dist/mcp/tools/auth/login.js +0 -51
  226. package/dist/mcp/tools/auth/login.js.map +0 -1
  227. package/dist/mcp/tools/auth/resetPassword.d.ts +0 -4
  228. package/dist/mcp/tools/auth/resetPassword.d.ts.map +0 -1
  229. package/dist/mcp/tools/auth/resetPassword.js +0 -49
  230. package/dist/mcp/tools/auth/resetPassword.js.map +0 -1
  231. package/dist/mcp/tools/auth/unlock.d.ts +0 -4
  232. package/dist/mcp/tools/auth/unlock.d.ts.map +0 -1
  233. package/dist/mcp/tools/auth/unlock.js +0 -48
  234. package/dist/mcp/tools/auth/unlock.js.map +0 -1
  235. package/dist/mcp/tools/auth/verify.d.ts +0 -4
  236. package/dist/mcp/tools/auth/verify.d.ts.map +0 -1
  237. package/dist/mcp/tools/auth/verify.js +0 -45
  238. package/dist/mcp/tools/auth/verify.js.map +0 -1
  239. package/dist/mcp/tools/collection/create.d.ts +0 -10
  240. package/dist/mcp/tools/collection/create.d.ts.map +0 -1
  241. package/dist/mcp/tools/collection/create.js +0 -139
  242. package/dist/mcp/tools/collection/create.js.map +0 -1
  243. package/dist/mcp/tools/collection/delete.d.ts +0 -10
  244. package/dist/mcp/tools/collection/delete.d.ts.map +0 -1
  245. package/dist/mcp/tools/collection/delete.js +0 -154
  246. package/dist/mcp/tools/collection/delete.js.map +0 -1
  247. package/dist/mcp/tools/collection/find.d.ts +0 -10
  248. package/dist/mcp/tools/collection/find.d.ts.map +0 -1
  249. package/dist/mcp/tools/collection/find.js +0 -165
  250. package/dist/mcp/tools/collection/find.js.map +0 -1
  251. package/dist/mcp/tools/collection/update.d.ts +0 -10
  252. package/dist/mcp/tools/collection/update.d.ts.map +0 -1
  253. package/dist/mcp/tools/collection/update.js +0 -209
  254. package/dist/mcp/tools/collection/update.js.map +0 -1
  255. package/dist/mcp/tools/config/find.d.ts +0 -10
  256. package/dist/mcp/tools/config/find.d.ts.map +0 -1
  257. package/dist/mcp/tools/config/find.js +0 -97
  258. package/dist/mcp/tools/config/find.js.map +0 -1
  259. package/dist/mcp/tools/config/update.d.ts +0 -10
  260. package/dist/mcp/tools/config/update.d.ts.map +0 -1
  261. package/dist/mcp/tools/config/update.js +0 -215
  262. package/dist/mcp/tools/config/update.js.map +0 -1
  263. package/dist/mcp/tools/global/find.d.ts +0 -5
  264. package/dist/mcp/tools/global/find.d.ts.map +0 -1
  265. package/dist/mcp/tools/global/find.js +0 -82
  266. package/dist/mcp/tools/global/find.js.map +0 -1
  267. package/dist/mcp/tools/global/update.d.ts +0 -6
  268. package/dist/mcp/tools/global/update.d.ts.map +0 -1
  269. package/dist/mcp/tools/global/update.js +0 -124
  270. package/dist/mcp/tools/global/update.js.map +0 -1
  271. package/dist/mcp/tools/job/create.d.ts +0 -10
  272. package/dist/mcp/tools/job/create.d.ts.map +0 -1
  273. package/dist/mcp/tools/job/create.js +0 -293
  274. package/dist/mcp/tools/job/create.js.map +0 -1
  275. package/dist/mcp/tools/job/run.d.ts +0 -10
  276. package/dist/mcp/tools/job/run.d.ts.map +0 -1
  277. package/dist/mcp/tools/job/run.js +0 -129
  278. package/dist/mcp/tools/job/run.js.map +0 -1
  279. package/dist/mcp/tools/job/update.d.ts +0 -11
  280. package/dist/mcp/tools/job/update.d.ts.map +0 -1
  281. package/dist/mcp/tools/job/update.js +0 -186
  282. package/dist/mcp/tools/job/update.js.map +0 -1
  283. package/dist/mcp/tools/resource/create.d.ts +0 -6
  284. package/dist/mcp/tools/resource/create.d.ts.map +0 -1
  285. package/dist/mcp/tools/resource/create.js +0 -124
  286. package/dist/mcp/tools/resource/create.js.map +0 -1
  287. package/dist/mcp/tools/resource/delete.d.ts +0 -5
  288. package/dist/mcp/tools/resource/delete.d.ts.map +0 -1
  289. package/dist/mcp/tools/resource/delete.js +0 -151
  290. package/dist/mcp/tools/resource/delete.js.map +0 -1
  291. package/dist/mcp/tools/resource/find.d.ts +0 -5
  292. package/dist/mcp/tools/resource/find.d.ts.map +0 -1
  293. package/dist/mcp/tools/resource/find.js +0 -170
  294. package/dist/mcp/tools/resource/find.js.map +0 -1
  295. package/dist/mcp/tools/resource/update.d.ts +0 -6
  296. package/dist/mcp/tools/resource/update.d.ts.map +0 -1
  297. package/dist/mcp/tools/resource/update.js +0 -256
  298. package/dist/mcp/tools/resource/update.js.map +0 -1
  299. package/dist/mcp/tools/schemas.d.ts +0 -457
  300. package/dist/mcp/tools/schemas.d.ts.map +0 -1
  301. package/dist/mcp/tools/schemas.js +0 -243
  302. package/dist/mcp/tools/schemas.js.map +0 -1
  303. package/dist/utils/adminEntitySettings.d.ts +0 -17
  304. package/dist/utils/adminEntitySettings.d.ts.map +0 -1
  305. package/dist/utils/adminEntitySettings.js +0 -41
  306. package/dist/utils/adminEntitySettings.js.map +0 -1
  307. package/dist/utils/createApiKeyFields.d.ts +0 -15
  308. package/dist/utils/createApiKeyFields.d.ts.map +0 -1
  309. package/dist/utils/createApiKeyFields.js +0 -57
  310. package/dist/utils/createApiKeyFields.js.map +0 -1
  311. package/dist/utils/getEnabledSlugs.d.ts +0 -13
  312. package/dist/utils/getEnabledSlugs.d.ts.map +0 -1
  313. package/dist/utils/getEnabledSlugs.js +0 -32
  314. package/dist/utils/getEnabledSlugs.js.map +0 -1
  315. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.d.ts +0 -3
  316. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.d.ts.map +0 -1
  317. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.js +0 -43
  318. package/dist/utils/schemaConversion/convertCollectionSchemaToZod.js.map +0 -1
  319. package/dist/utils/schemaConversion/sanitizeJsonSchema.d.ts +0 -13
  320. package/dist/utils/schemaConversion/sanitizeJsonSchema.d.ts.map +0 -1
  321. package/dist/utils/schemaConversion/sanitizeJsonSchema.js +0 -56
  322. package/dist/utils/schemaConversion/sanitizeJsonSchema.js.map +0 -1
  323. package/dist/utils/schemaConversion/simplifyRelationshipFields.d.ts +0 -20
  324. package/dist/utils/schemaConversion/simplifyRelationshipFields.d.ts.map +0 -1
  325. package/dist/utils/schemaConversion/simplifyRelationshipFields.js +0 -53
  326. package/dist/utils/schemaConversion/simplifyRelationshipFields.js.map +0 -1
  327. package/dist/utils/schemaConversion/transformPointFields.d.ts +0 -3
  328. package/dist/utils/schemaConversion/transformPointFields.d.ts.map +0 -1
  329. package/dist/utils/schemaConversion/transformPointFields.js +0 -51
  330. package/dist/utils/schemaConversion/transformPointFields.js.map +0 -1
  331. package/src/collections/createApiKeysCollection.ts +0 -373
  332. package/src/defaults.ts +0 -3
  333. package/src/endpoints/mcp.ts +0 -91
  334. package/src/mcp/createRequest.ts +0 -13
  335. package/src/mcp/getMcpHandler.ts +0 -545
  336. package/src/mcp/helpers/config.ts +0 -213
  337. package/src/mcp/helpers/fields.ts +0 -154
  338. package/src/mcp/helpers/fileValidation.ts +0 -362
  339. package/src/mcp/registerTool.ts +0 -22
  340. package/src/mcp/tools/auth/auth.ts +0 -71
  341. package/src/mcp/tools/auth/forgotPassword.ts +0 -70
  342. package/src/mcp/tools/auth/login.ts +0 -72
  343. package/src/mcp/tools/auth/resetPassword.ts +0 -61
  344. package/src/mcp/tools/auth/unlock.ts +0 -64
  345. package/src/mcp/tools/auth/verify.ts +0 -57
  346. package/src/mcp/tools/collection/create.ts +0 -210
  347. package/src/mcp/tools/collection/delete.ts +0 -211
  348. package/src/mcp/tools/collection/find.ts +0 -224
  349. package/src/mcp/tools/collection/update.ts +0 -290
  350. package/src/mcp/tools/config/find.ts +0 -128
  351. package/src/mcp/tools/config/update.ts +0 -280
  352. package/src/mcp/tools/global/find.ts +0 -128
  353. package/src/mcp/tools/global/update.ts +0 -207
  354. package/src/mcp/tools/job/create.ts +0 -416
  355. package/src/mcp/tools/job/run.ts +0 -167
  356. package/src/mcp/tools/job/update.ts +0 -274
  357. package/src/mcp/tools/resource/create.ts +0 -211
  358. package/src/mcp/tools/resource/delete.ts +0 -218
  359. package/src/mcp/tools/resource/find.ts +0 -246
  360. package/src/mcp/tools/resource/update.ts +0 -383
  361. package/src/mcp/tools/schemas.ts +0 -520
  362. package/src/utils/adminEntitySettings.ts +0 -40
  363. package/src/utils/createApiKeyFields.ts +0 -72
  364. package/src/utils/getEnabledSlugs.ts +0 -42
  365. package/src/utils/schemaConversion/convertCollectionSchemaToZod.ts +0 -52
  366. package/src/utils/schemaConversion/sanitizeJsonSchema.ts +0 -62
  367. package/src/utils/schemaConversion/simplifyRelationshipFields.ts +0 -65
  368. package/src/utils/schemaConversion/transformPointFields.ts +0 -55
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,257 @@
1
+ import { McpServer, type ServerContext } from '@modelcontextprotocol/server'
2
+ import { APIError, type PayloadRequest } from 'payload'
3
+ import { z } from 'zod'
4
+
5
+ import type {
6
+ AuthorizedMCP,
7
+ CollectionMCPItem,
8
+ GlobalMCPItem,
9
+ JsonSchemaType,
10
+ MCPResponseOverride,
11
+ MCPToolResponse,
12
+ SanitizedMCPPluginConfig,
13
+ ToolInputSchema,
14
+ } from '../types.js'
15
+
16
+ import { getLogger } from '../utils/getLogger.js'
17
+ import { toStandardSchema } from '../utils/toStandardSchema.js'
18
+
19
+ /**
20
+ * Transport-agnostic core: registers every authorized MCP item onto a fresh
21
+ * `McpServer` and returns it. The caller is responsible for picking a transport
22
+ * (`WebStandardStreamableHTTPServerTransport`, `StdioServerTransport`, …) and
23
+ * calling `server.connect(transport)`.
24
+ *
25
+ * `req` is the request context handlers see. For HTTP it's the live
26
+ * `PayloadRequest` derived from the incoming HTTP request; for stdio it's a
27
+ * synthesized one built via `createLocalReq`.
28
+ */
29
+ export const buildMcpServer = ({
30
+ authorizedMCP,
31
+ pluginConfig,
32
+ req,
33
+ }: {
34
+ authorizedMCP: AuthorizedMCP
35
+ pluginConfig: SanitizedMCPPluginConfig
36
+ req: PayloadRequest
37
+ }): McpServer => {
38
+ const serverOptions = pluginConfig.mcp?.serverOptions || {}
39
+ const server = new McpServer(
40
+ { name: 'Payload MCP Server', version: '1.0.0', ...serverOptions.serverInfo },
41
+ serverOptions.options,
42
+ )
43
+
44
+ const logger = getLogger({ payload: req.payload })
45
+
46
+ /**
47
+ * Wrap a tool handler's response with the tool's `overrideResponse`, then
48
+ * strip the internal `doc` field so it doesn't leak onto the wire.
49
+ */
50
+ const finalizeToolResponse = (
51
+ response: MCPToolResponse,
52
+ overrideResponse?: MCPResponseOverride,
53
+ ): MCPToolResponse => {
54
+ const overridden = overrideResponse?.(response, response.doc ?? {}, req) ?? response
55
+ const { doc: _doc, ...rest } = overridden
56
+ return rest
57
+ }
58
+
59
+ /**
60
+ * Runs a collection/global tool call:
61
+ * - reads `collectionSlug` / `globalSlug` from the input
62
+ * - runs access control: errors if `authorizedMCP.items` has no entry for this tool + slug
63
+ * - runs the tool handler and finalizes its response
64
+ */
65
+ const callEntityTool = async ({
66
+ input,
67
+ item,
68
+ serverContext,
69
+ }: {
70
+ input: unknown
71
+ item: CollectionMCPItem | GlobalMCPItem
72
+ serverContext: ServerContext
73
+ }): Promise<MCPToolResponse> => {
74
+ const entity = item.type === 'collectionTool' ? 'collection' : 'global'
75
+ const slugKey = item.type === 'collectionTool' ? 'collectionSlug' : 'globalSlug'
76
+ const toolInput = (input ?? {}) as Record<string, unknown>
77
+ const slug = toolInput[slugKey] as string | undefined
78
+
79
+ if (!slug) {
80
+ return {
81
+ content: [
82
+ {
83
+ type: 'text',
84
+ text: `Error: "${item.mcpName}" requires ${slugKey}. Use getConfigInfo to inspect ${entity} slugs.`,
85
+ },
86
+ ],
87
+ isError: true,
88
+ }
89
+ }
90
+
91
+ const match = authorizedMCP.items.find(
92
+ (candidate): candidate is CollectionMCPItem | GlobalMCPItem =>
93
+ candidate.type === item.type &&
94
+ candidate.mcpName === item.mcpName &&
95
+ (candidate.type === 'collectionTool'
96
+ ? candidate.collectionSlug === slug
97
+ : candidate.type === 'globalTool' && candidate.globalSlug === slug),
98
+ )
99
+
100
+ if (!match) {
101
+ return {
102
+ content: [
103
+ {
104
+ type: 'text',
105
+ text: `Error: MCP access to "${item.mcpName}" is not enabled for ${entity} "${slug}"`,
106
+ },
107
+ ],
108
+ isError: true,
109
+ }
110
+ }
111
+
112
+ const handlerArgs = { authorizedMCP, input: toolInput, req, serverContext }
113
+ const response = await (match.type === 'collectionTool'
114
+ ? match.tool.handler({ ...handlerArgs, collectionSlug: slug })
115
+ : match.tool.handler({ ...handlerArgs, globalSlug: slug }))
116
+
117
+ return finalizeToolResponse(response, match.tool.overrideResponse)
118
+ }
119
+
120
+ try {
121
+ const registeredEntityTools = new Set<string>()
122
+
123
+ for (const item of authorizedMCP.items) {
124
+ switch (item.type) {
125
+ case 'collectionTool':
126
+ case 'globalTool': {
127
+ if (registeredEntityTools.has(item.mcpName)) {
128
+ break
129
+ }
130
+ registeredEntityTools.add(item.mcpName)
131
+
132
+ const inputSchema = withSlugInput({
133
+ name: item.type === 'collectionTool' ? 'collectionSlug' : 'globalSlug',
134
+ input: item.tool.input,
135
+ })
136
+
137
+ server.registerTool(
138
+ item.mcpName,
139
+ {
140
+ description: item.tool.description,
141
+ inputSchema: toStandardSchema(inputSchema),
142
+ },
143
+ async (input: unknown, ctx: ServerContext) =>
144
+ callEntityTool({ input, item, serverContext: ctx }),
145
+ )
146
+ logger.info(`✅ Tool: ${item.mcpName} Registered.`)
147
+ break
148
+ }
149
+ case 'prompt': {
150
+ const prompt = item.prompt
151
+ server.registerPrompt(
152
+ item.mcpName,
153
+ {
154
+ argsSchema: prompt.argsSchema ? toStandardSchema(prompt.argsSchema) : undefined,
155
+ description: prompt.description,
156
+ title: prompt.title,
157
+ },
158
+ async (input: unknown, ctx: ServerContext) =>
159
+ prompt.handler({
160
+ input: (input ?? {}) as Record<string, unknown>,
161
+ req,
162
+ serverContext: ctx,
163
+ }),
164
+ )
165
+ logger.info(`✅ Prompt: ${prompt.title} Registered.`)
166
+ break
167
+ }
168
+ case 'resource': {
169
+ const resource = item.resource
170
+ server.registerResource(
171
+ item.mcpName,
172
+ // @ts-expect-error - Overload type ambiguity (string OR ResourceTemplate is valid)
173
+ resource.uri,
174
+ {
175
+ description: resource.description,
176
+ mimeType: resource.mimeType,
177
+ title: resource.title,
178
+ },
179
+ // Static URIs call (uri, ctx); ResourceTemplates call (uri, params, ctx).
180
+ // The rest-params shape lets us collect either signature uniformly.
181
+ async (...sdkArgs: unknown[]) => {
182
+ const ctx = sdkArgs[sdkArgs.length - 1] as ServerContext
183
+ const uri = sdkArgs[0] as URL
184
+ const params = (sdkArgs.length > 2 ? sdkArgs[1] : {}) as Record<string, string>
185
+ return resource.handler({ params, req, serverContext: ctx, uri })
186
+ },
187
+ )
188
+ logger.info(`✅ Resource: ${resource.title} Registered.`)
189
+ break
190
+ }
191
+ case 'tool': {
192
+ const tool = item.tool
193
+ server.registerTool(
194
+ item.mcpName,
195
+ {
196
+ description: tool.description,
197
+ inputSchema: tool.input ? toStandardSchema(tool.input) : undefined,
198
+ },
199
+ async (input: unknown, ctx: ServerContext) =>
200
+ finalizeToolResponse(
201
+ await tool.handler({
202
+ authorizedMCP,
203
+ input: (input ?? {}) as Record<string, unknown>,
204
+ req,
205
+ serverContext: ctx,
206
+ }),
207
+ tool.overrideResponse,
208
+ ),
209
+ )
210
+ logger.info(`✅ Tool: ${item.mcpName} Registered.`)
211
+ break
212
+ }
213
+ }
214
+ }
215
+ } catch (error) {
216
+ throw new APIError(`Error initializing MCP handler: ${String(error)}`, 500)
217
+ }
218
+
219
+ return server
220
+ }
221
+
222
+ const withSlugInput = ({
223
+ name,
224
+ input,
225
+ }: {
226
+ input?: ToolInputSchema
227
+ name: 'collectionSlug' | 'globalSlug'
228
+ }): ToolInputSchema => {
229
+ const description = name === 'collectionSlug' ? 'The collection slug' : 'The global slug'
230
+ const slugSchema = z.string().describe(description)
231
+
232
+ if (!input) {
233
+ return z.object({ [name]: slugSchema })
234
+ }
235
+
236
+ if (input instanceof z.ZodObject) {
237
+ return input.extend({ [name]: slugSchema })
238
+ }
239
+
240
+ const schema = input as {
241
+ properties?: Record<string, JsonSchemaType>
242
+ required?: string[]
243
+ } & JsonSchemaType
244
+
245
+ return {
246
+ ...schema,
247
+ type: 'object',
248
+ properties: {
249
+ ...schema.properties,
250
+ [name]: {
251
+ type: 'string',
252
+ description,
253
+ },
254
+ },
255
+ required: Array.from(new Set([name, ...(schema.required ?? [])])),
256
+ }
257
+ }
@@ -0,0 +1,233 @@
1
+ import { z } from 'zod'
2
+
3
+ import type { MCPToolResponse } from '../../../types.js'
4
+
5
+ import { defineCollectionTool } from '../../../defineTool.js'
6
+ import { getLogger } from '../../../utils/getLogger.js'
7
+
8
+ /**
9
+ * Auth tools surfaced under `collections.<auth-collection>.tools`. Opt-in: they
10
+ * default off (an LLM with `login` access can probe passwords; not something
11
+ * to expose by default). Enable in plugin config via `tools: { login: true }`
12
+ * (or `{ description: '...' }`).
13
+ *
14
+ * `auth` (check current token) doesn't actually depend on the collection at
15
+ * runtime — `payload.auth({ headers })` is global — but it's grouped under the
16
+ * users collection so all auth-shaped tools live in one place.
17
+ */
18
+
19
+ const emailSchema = z.string().email().describe('The user email address')
20
+
21
+ const wrapError =
22
+ (name: string) =>
23
+ ({ collectionSlug, message }: { collectionSlug: string; message: string }): MCPToolResponse => {
24
+ return {
25
+ content: [
26
+ {
27
+ type: 'text',
28
+ text: `❌ **Error in ${name}** on ${collectionSlug}: ${message}`,
29
+ },
30
+ ],
31
+ }
32
+ }
33
+
34
+ export const authCollectionTool = defineCollectionTool({
35
+ description: 'Checks authentication status for the current user.',
36
+ input: z.object({
37
+ headers: z
38
+ .string()
39
+ .describe(
40
+ 'Optional JSON string containing custom headers to send with the authentication request',
41
+ )
42
+ .optional(),
43
+ }),
44
+ }).handler(async ({ collectionSlug, input, req }) => {
45
+ const logger = getLogger({ payload: req.payload })
46
+ try {
47
+ let authHeaders = new Headers()
48
+ if (input.headers) {
49
+ try {
50
+ authHeaders = new Headers(JSON.parse(input.headers) as Record<string, string>)
51
+ } catch {
52
+ logger.warn(`Invalid headers JSON for auth: ${input.headers}, using empty headers`)
53
+ }
54
+ }
55
+ const result = await req.payload.auth({ headers: authHeaders })
56
+ return {
57
+ content: [
58
+ {
59
+ type: 'text',
60
+ text: `# Authentication Status\n\n\`\`\`json\n${JSON.stringify(result)}\n\`\`\``,
61
+ },
62
+ ],
63
+ doc: result as unknown as Record<string, unknown>,
64
+ }
65
+ } catch (error) {
66
+ const errorMessage = error instanceof Error ? error.message : 'Unknown error'
67
+ logger.error(`Error in auth tool on ${collectionSlug}: ${errorMessage}`)
68
+ return wrapError('auth')({ collectionSlug, message: errorMessage })
69
+ }
70
+ })
71
+
72
+ export const forgotPasswordCollectionTool = defineCollectionTool({
73
+ description: 'Sends a password reset email to a user.',
74
+ input: z.object({
75
+ disableEmail: z
76
+ .boolean()
77
+ .describe('Whether to disable sending the email (for testing)')
78
+ .optional()
79
+ .default(false),
80
+ email: emailSchema,
81
+ }),
82
+ }).handler(async ({ collectionSlug, input, req }) => {
83
+ const logger = getLogger({ payload: req.payload })
84
+ try {
85
+ const result = await req.payload.forgotPassword({
86
+ collection: collectionSlug,
87
+ data: { email: input.email },
88
+ disableEmail: input.disableEmail,
89
+ })
90
+ return {
91
+ content: [
92
+ {
93
+ type: 'text',
94
+ text: `# Forgot Password Email Sent\n\n**Collection:** ${collectionSlug}\n**Email:** ${input.email}\n\n\`\`\`json\n${JSON.stringify(result)}\n\`\`\``,
95
+ },
96
+ ],
97
+ doc: { result } as Record<string, unknown>,
98
+ }
99
+ } catch (error) {
100
+ const errorMessage = error instanceof Error ? error.message : 'Unknown error'
101
+ logger.error(`Error in forgotPassword tool on ${collectionSlug}: ${errorMessage}`)
102
+ return wrapError('forgotPassword')({ collectionSlug, message: errorMessage })
103
+ }
104
+ })
105
+
106
+ export const loginCollectionTool = defineCollectionTool({
107
+ description: 'Authenticates a user with email and password.',
108
+ input: z.object({
109
+ depth: z
110
+ .number()
111
+ .int()
112
+ .min(0)
113
+ .max(10)
114
+ .describe('Depth of population for relationships')
115
+ .optional()
116
+ .default(0),
117
+ email: emailSchema,
118
+ password: z.string().describe('The user password'),
119
+ showHiddenFields: z
120
+ .boolean()
121
+ .describe('Whether to show hidden fields in the response')
122
+ .optional()
123
+ .default(false),
124
+ }),
125
+ }).handler(async ({ collectionSlug, input, req }) => {
126
+ const logger = getLogger({ payload: req.payload })
127
+ try {
128
+ const result = await req.payload.login({
129
+ collection: collectionSlug,
130
+ data: { email: input.email, password: input.password },
131
+ depth: input.depth,
132
+ showHiddenFields: input.showHiddenFields,
133
+ })
134
+ return {
135
+ content: [
136
+ {
137
+ type: 'text',
138
+ text: `# Login Successful\n\n**User:** ${input.email}\n**Collection:** ${collectionSlug}\n\n\`\`\`json\n${JSON.stringify(result)}\n\`\`\``,
139
+ },
140
+ ],
141
+ doc: result as unknown as Record<string, unknown>,
142
+ }
143
+ } catch (error) {
144
+ const errorMessage = error instanceof Error ? error.message : 'Unknown error'
145
+ logger.error(`Error in login tool on ${collectionSlug}: ${errorMessage}`)
146
+ return wrapError('login')({ collectionSlug, message: errorMessage })
147
+ }
148
+ })
149
+
150
+ export const resetPasswordCollectionTool = defineCollectionTool({
151
+ description: 'Resets a user password with a reset token.',
152
+ input: z.object({
153
+ password: z.string().describe('The new password for the user'),
154
+ token: z.string().describe('The password reset token sent to the user email'),
155
+ }),
156
+ }).handler(async ({ collectionSlug, input, req }) => {
157
+ const logger = getLogger({ payload: req.payload })
158
+ try {
159
+ const result = await req.payload.resetPassword({
160
+ collection: collectionSlug,
161
+ data: { password: input.password, token: input.token },
162
+ overrideAccess: true,
163
+ })
164
+ return {
165
+ content: [
166
+ {
167
+ type: 'text',
168
+ text: `# Password Reset Successful\n\n**Collection:** ${collectionSlug}\n\n\`\`\`json\n${JSON.stringify(result)}\n\`\`\``,
169
+ },
170
+ ],
171
+ doc: result as unknown as Record<string, unknown>,
172
+ }
173
+ } catch (error) {
174
+ const errorMessage = error instanceof Error ? error.message : 'Unknown error'
175
+ logger.error(`Error in resetPassword tool on ${collectionSlug}: ${errorMessage}`)
176
+ return wrapError('resetPassword')({ collectionSlug, message: errorMessage })
177
+ }
178
+ })
179
+
180
+ export const unlockCollectionTool = defineCollectionTool({
181
+ description: 'Unlocks a user account that has been locked due to failed login attempts.',
182
+ input: z.object({ email: emailSchema }),
183
+ }).handler(async ({ collectionSlug, input, req }) => {
184
+ const logger = getLogger({ payload: req.payload })
185
+ try {
186
+ const result = await req.payload.unlock({
187
+ collection: collectionSlug,
188
+ data: { email: input.email },
189
+ overrideAccess: true,
190
+ })
191
+ return {
192
+ content: [
193
+ {
194
+ type: 'text',
195
+ text: `# Account Unlocked\n\n**Collection:** ${collectionSlug}\n**Email:** ${input.email}\n**Result:** ${result ? 'Success' : 'Failed'}`,
196
+ },
197
+ ],
198
+ doc: { result } as Record<string, unknown>,
199
+ }
200
+ } catch (error) {
201
+ const errorMessage = error instanceof Error ? error.message : 'Unknown error'
202
+ logger.error(`Error in unlock tool on ${collectionSlug}: ${errorMessage}`)
203
+ return wrapError('unlock')({ collectionSlug, message: errorMessage })
204
+ }
205
+ })
206
+
207
+ export const verifyCollectionTool = defineCollectionTool({
208
+ description: 'Verifies a user email with a verification token.',
209
+ input: z.object({
210
+ token: z.string().describe('The verification token sent to the user email'),
211
+ }),
212
+ }).handler(async ({ collectionSlug, input, req }) => {
213
+ const logger = getLogger({ payload: req.payload })
214
+ try {
215
+ const result = await req.payload.verifyEmail({
216
+ collection: collectionSlug,
217
+ token: input.token,
218
+ })
219
+ return {
220
+ content: [
221
+ {
222
+ type: 'text',
223
+ text: `# Email Verification Successful\n\n**Collection:** ${collectionSlug}\n**Result:** ${result ? 'Success' : 'Failed'}`,
224
+ },
225
+ ],
226
+ doc: { result } as Record<string, unknown>,
227
+ }
228
+ } catch (error) {
229
+ const errorMessage = error instanceof Error ? error.message : 'Unknown error'
230
+ logger.error(`Error in verify tool on ${collectionSlug}: ${errorMessage}`)
231
+ return wrapError('verify')({ collectionSlug, message: errorMessage })
232
+ }
233
+ })