@payloadcms/plugin-mcp 0.0.1-alpha.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE.md +22 -0
- package/README.md +7 -0
- package/dist/collections/createApiKeysCollection.d.ts +7 -0
- package/dist/collections/createApiKeysCollection.d.ts.map +1 -0
- package/dist/collections/createApiKeysCollection.js +315 -0
- package/dist/collections/createApiKeysCollection.js.map +1 -0
- package/dist/endpoints/mcp.d.ts +4 -0
- package/dist/endpoints/mcp.d.ts.map +1 -0
- package/dist/endpoints/mcp.js +44 -0
- package/dist/endpoints/mcp.js.map +1 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +67 -0
- package/dist/index.js.map +1 -0
- package/dist/mcp/createRequest.d.ts +3 -0
- package/dist/mcp/createRequest.d.ts.map +1 -0
- package/dist/mcp/createRequest.js +14 -0
- package/dist/mcp/createRequest.js.map +1 -0
- package/dist/mcp/getMcpHandler.d.ts +4 -0
- package/dist/mcp/getMcpHandler.d.ts.map +1 -0
- package/dist/mcp/getMcpHandler.js +179 -0
- package/dist/mcp/getMcpHandler.js.map +1 -0
- package/dist/mcp/helpers/config.d.ts +30 -0
- package/dist/mcp/helpers/config.d.ts.map +1 -0
- package/dist/mcp/helpers/config.js +217 -0
- package/dist/mcp/helpers/config.js.map +1 -0
- package/dist/mcp/helpers/conversion.d.ts +2 -0
- package/dist/mcp/helpers/conversion.d.ts.map +1 -0
- package/dist/mcp/helpers/conversion.js +5 -0
- package/dist/mcp/helpers/conversion.js.map +1 -0
- package/dist/mcp/helpers/fields.d.ts +38 -0
- package/dist/mcp/helpers/fields.d.ts.map +1 -0
- package/dist/mcp/helpers/fields.js +96 -0
- package/dist/mcp/helpers/fields.js.map +1 -0
- package/dist/mcp/helpers/fileValidation.d.ts +69 -0
- package/dist/mcp/helpers/fileValidation.d.ts.map +1 -0
- package/dist/mcp/helpers/fileValidation.js +305 -0
- package/dist/mcp/helpers/fileValidation.js.map +1 -0
- package/dist/mcp/helpers/validation.d.ts +9 -0
- package/dist/mcp/helpers/validation.d.ts.map +1 -0
- package/dist/mcp/helpers/validation.js +22 -0
- package/dist/mcp/helpers/validation.js.map +1 -0
- package/dist/mcp/registerTool.d.ts +6 -0
- package/dist/mcp/registerTool.d.ts.map +1 -0
- package/dist/mcp/registerTool.js +18 -0
- package/dist/mcp/registerTool.js.map +1 -0
- package/dist/mcp/tools/auth/auth.d.ts +4 -0
- package/dist/mcp/tools/auth/auth.d.ts.map +1 -0
- package/dist/mcp/tools/auth/auth.js +54 -0
- package/dist/mcp/tools/auth/auth.js.map +1 -0
- package/dist/mcp/tools/auth/forgotPassword.d.ts +4 -0
- package/dist/mcp/tools/auth/forgotPassword.d.ts.map +1 -0
- package/dist/mcp/tools/auth/forgotPassword.js +45 -0
- package/dist/mcp/tools/auth/forgotPassword.js.map +1 -0
- package/dist/mcp/tools/auth/login.d.ts +4 -0
- package/dist/mcp/tools/auth/login.d.ts.map +1 -0
- package/dist/mcp/tools/auth/login.js +48 -0
- package/dist/mcp/tools/auth/login.js.map +1 -0
- package/dist/mcp/tools/auth/resetPassword.d.ts +4 -0
- package/dist/mcp/tools/auth/resetPassword.d.ts.map +1 -0
- package/dist/mcp/tools/auth/resetPassword.js +46 -0
- package/dist/mcp/tools/auth/resetPassword.js.map +1 -0
- package/dist/mcp/tools/auth/unlock.d.ts +4 -0
- package/dist/mcp/tools/auth/unlock.d.ts.map +1 -0
- package/dist/mcp/tools/auth/unlock.js +45 -0
- package/dist/mcp/tools/auth/unlock.js.map +1 -0
- package/dist/mcp/tools/auth/verify.d.ts +4 -0
- package/dist/mcp/tools/auth/verify.d.ts.map +1 -0
- package/dist/mcp/tools/auth/verify.js +42 -0
- package/dist/mcp/tools/auth/verify.js.map +1 -0
- package/dist/mcp/tools/collection/create.d.ts +10 -0
- package/dist/mcp/tools/collection/create.d.ts.map +1 -0
- package/dist/mcp/tools/collection/create.js +159 -0
- package/dist/mcp/tools/collection/create.js.map +1 -0
- package/dist/mcp/tools/collection/delete.d.ts +10 -0
- package/dist/mcp/tools/collection/delete.d.ts.map +1 -0
- package/dist/mcp/tools/collection/delete.js +162 -0
- package/dist/mcp/tools/collection/delete.js.map +1 -0
- package/dist/mcp/tools/collection/find.d.ts +10 -0
- package/dist/mcp/tools/collection/find.d.ts.map +1 -0
- package/dist/mcp/tools/collection/find.js +162 -0
- package/dist/mcp/tools/collection/find.js.map +1 -0
- package/dist/mcp/tools/collection/update.d.ts +10 -0
- package/dist/mcp/tools/collection/update.d.ts.map +1 -0
- package/dist/mcp/tools/collection/update.js +206 -0
- package/dist/mcp/tools/collection/update.js.map +1 -0
- package/dist/mcp/tools/config/find.d.ts +10 -0
- package/dist/mcp/tools/config/find.d.ts.map +1 -0
- package/dist/mcp/tools/config/find.js +94 -0
- package/dist/mcp/tools/config/find.js.map +1 -0
- package/dist/mcp/tools/config/update.d.ts +10 -0
- package/dist/mcp/tools/config/update.d.ts.map +1 -0
- package/dist/mcp/tools/config/update.js +212 -0
- package/dist/mcp/tools/config/update.js.map +1 -0
- package/dist/mcp/tools/job/create.d.ts +10 -0
- package/dist/mcp/tools/job/create.d.ts.map +1 -0
- package/dist/mcp/tools/job/create.js +293 -0
- package/dist/mcp/tools/job/create.js.map +1 -0
- package/dist/mcp/tools/job/run.d.ts +10 -0
- package/dist/mcp/tools/job/run.d.ts.map +1 -0
- package/dist/mcp/tools/job/run.js +147 -0
- package/dist/mcp/tools/job/run.js.map +1 -0
- package/dist/mcp/tools/job/update.d.ts +11 -0
- package/dist/mcp/tools/job/update.d.ts.map +1 -0
- package/dist/mcp/tools/job/update.js +211 -0
- package/dist/mcp/tools/job/update.js.map +1 -0
- package/dist/mcp/tools/resource/create.d.ts +6 -0
- package/dist/mcp/tools/resource/create.d.ts.map +1 -0
- package/dist/mcp/tools/resource/create.js +75 -0
- package/dist/mcp/tools/resource/create.js.map +1 -0
- package/dist/mcp/tools/resource/delete.d.ts +5 -0
- package/dist/mcp/tools/resource/delete.d.ts.map +1 -0
- package/dist/mcp/tools/resource/delete.js +140 -0
- package/dist/mcp/tools/resource/delete.js.map +1 -0
- package/dist/mcp/tools/resource/find.d.ts +5 -0
- package/dist/mcp/tools/resource/find.d.ts.map +1 -0
- package/dist/mcp/tools/resource/find.js +119 -0
- package/dist/mcp/tools/resource/find.js.map +1 -0
- package/dist/mcp/tools/resource/update.d.ts +6 -0
- package/dist/mcp/tools/resource/update.d.ts.map +1 -0
- package/dist/mcp/tools/resource/update.js +201 -0
- package/dist/mcp/tools/resource/update.js.map +1 -0
- package/dist/mcp/tools/schemas.d.ts +374 -0
- package/dist/mcp/tools/schemas.d.ts.map +1 -0
- package/dist/mcp/tools/schemas.js +201 -0
- package/dist/mcp/tools/schemas.js.map +1 -0
- package/dist/types.d.ts +379 -0
- package/dist/types.d.ts.map +1 -0
- package/dist/types.js +3 -0
- package/dist/types.js.map +1 -0
- package/dist/utils/camelCase.d.ts +9 -0
- package/dist/utils/camelCase.d.ts.map +1 -0
- package/dist/utils/camelCase.js +11 -0
- package/dist/utils/camelCase.js.map +1 -0
- package/dist/utils/convertCollectionSchemaToZod.d.ts +3 -0
- package/dist/utils/convertCollectionSchemaToZod.d.ts.map +1 -0
- package/dist/utils/convertCollectionSchemaToZod.js +30 -0
- package/dist/utils/convertCollectionSchemaToZod.js.map +1 -0
- package/package.json +64 -0
- package/src/collections/createApiKeysCollection.ts +393 -0
- package/src/endpoints/mcp.ts +60 -0
- package/src/index.ts +86 -0
- package/src/mcp/createRequest.ts +13 -0
- package/src/mcp/getMcpHandler.ts +433 -0
- package/src/mcp/helpers/config.ts +326 -0
- package/src/mcp/helpers/conversion.ts +3 -0
- package/src/mcp/helpers/fields.ts +158 -0
- package/src/mcp/helpers/fileValidation.ts +417 -0
- package/src/mcp/helpers/validation.ts +32 -0
- package/src/mcp/registerTool.ts +22 -0
- package/src/mcp/tools/auth/auth.ts +69 -0
- package/src/mcp/tools/auth/forgotPassword.ts +68 -0
- package/src/mcp/tools/auth/login.ts +70 -0
- package/src/mcp/tools/auth/resetPassword.ts +59 -0
- package/src/mcp/tools/auth/unlock.ts +62 -0
- package/src/mcp/tools/auth/verify.ts +55 -0
- package/src/mcp/tools/collection/create.ts +236 -0
- package/src/mcp/tools/collection/delete.ts +227 -0
- package/src/mcp/tools/collection/find.ts +222 -0
- package/src/mcp/tools/collection/update.ts +288 -0
- package/src/mcp/tools/config/find.ts +126 -0
- package/src/mcp/tools/config/update.ts +282 -0
- package/src/mcp/tools/job/create.ts +420 -0
- package/src/mcp/tools/job/run.ts +189 -0
- package/src/mcp/tools/job/update.ts +319 -0
- package/src/mcp/tools/resource/create.ts +121 -0
- package/src/mcp/tools/resource/delete.ts +210 -0
- package/src/mcp/tools/resource/find.ts +194 -0
- package/src/mcp/tools/resource/update.ts +314 -0
- package/src/mcp/tools/schemas.ts +373 -0
- package/src/types.ts +405 -0
- package/src/utils/camelCase.ts +12 -0
- package/src/utils/convertCollectionSchemaToZod.ts +35 -0
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
import { toCamelCase } from '../../../utils/camelCase.js';
|
|
2
|
+
import { toolSchemas } from '../schemas.js';
|
|
3
|
+
export const findResourceTool = (server, req, user, verboseLogs, collectionSlug, collections)=>{
|
|
4
|
+
const tool = async (id, limit = 10, page = 1, sort, where)=>{
|
|
5
|
+
const payload = req.payload;
|
|
6
|
+
if (verboseLogs) {
|
|
7
|
+
payload.logger.info(`[payload-mcp] Reading resource from collection: ${collectionSlug}${id ? ` with ID: ${id}` : ''}, limit: ${limit}, page: ${page}`);
|
|
8
|
+
}
|
|
9
|
+
try {
|
|
10
|
+
// Parse where clause if provided
|
|
11
|
+
let whereClause = {};
|
|
12
|
+
if (where) {
|
|
13
|
+
try {
|
|
14
|
+
whereClause = JSON.parse(where);
|
|
15
|
+
if (verboseLogs) {
|
|
16
|
+
payload.logger.info(`[payload-mcp] Using where clause: ${where}`);
|
|
17
|
+
}
|
|
18
|
+
} catch (_parseError) {
|
|
19
|
+
payload.logger.warn(`[payload-mcp] Invalid where clause JSON: ${where}`);
|
|
20
|
+
const response = {
|
|
21
|
+
content: [
|
|
22
|
+
{
|
|
23
|
+
type: 'text',
|
|
24
|
+
text: 'Error: Invalid JSON in where clause'
|
|
25
|
+
}
|
|
26
|
+
]
|
|
27
|
+
};
|
|
28
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response;
|
|
29
|
+
}
|
|
30
|
+
}
|
|
31
|
+
// If ID is provided, use findByID
|
|
32
|
+
if (id) {
|
|
33
|
+
try {
|
|
34
|
+
const doc = await payload.findByID({
|
|
35
|
+
id,
|
|
36
|
+
collection: collectionSlug,
|
|
37
|
+
user
|
|
38
|
+
});
|
|
39
|
+
if (verboseLogs) {
|
|
40
|
+
payload.logger.info(`[payload-mcp] Found document with ID: ${id}`);
|
|
41
|
+
}
|
|
42
|
+
const response = {
|
|
43
|
+
content: [
|
|
44
|
+
{
|
|
45
|
+
type: 'text',
|
|
46
|
+
text: `Resource from collection "${collectionSlug}":
|
|
47
|
+
${JSON.stringify(doc, null, 2)}`
|
|
48
|
+
}
|
|
49
|
+
]
|
|
50
|
+
};
|
|
51
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, doc, req) || response;
|
|
52
|
+
} catch (_findError) {
|
|
53
|
+
payload.logger.warn(`[payload-mcp] Document not found with ID: ${id} in collection: ${collectionSlug}`);
|
|
54
|
+
const response = {
|
|
55
|
+
content: [
|
|
56
|
+
{
|
|
57
|
+
type: 'text',
|
|
58
|
+
text: `Error: Document with ID "${id}" not found in collection "${collectionSlug}"`
|
|
59
|
+
}
|
|
60
|
+
]
|
|
61
|
+
};
|
|
62
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
// Otherwise, use find to get multiple documents
|
|
66
|
+
const findOptions = {
|
|
67
|
+
collection: collectionSlug,
|
|
68
|
+
limit,
|
|
69
|
+
page,
|
|
70
|
+
user
|
|
71
|
+
};
|
|
72
|
+
if (sort) {
|
|
73
|
+
findOptions.sort = sort;
|
|
74
|
+
}
|
|
75
|
+
if (Object.keys(whereClause).length > 0) {
|
|
76
|
+
findOptions.where = whereClause;
|
|
77
|
+
}
|
|
78
|
+
const result = await payload.find(findOptions);
|
|
79
|
+
if (verboseLogs) {
|
|
80
|
+
payload.logger.info(`[payload-mcp] Found ${result.docs.length} documents in collection: ${collectionSlug}`);
|
|
81
|
+
}
|
|
82
|
+
let responseText = `Collection: "${collectionSlug}"
|
|
83
|
+
Total: ${result.totalDocs} documents
|
|
84
|
+
Page: ${result.page} of ${result.totalPages}
|
|
85
|
+
`;
|
|
86
|
+
for (const doc of result.docs){
|
|
87
|
+
responseText += `\n\`\`\`json\n${JSON.stringify(doc, null, 2)}\n\`\`\``;
|
|
88
|
+
}
|
|
89
|
+
const response = {
|
|
90
|
+
content: [
|
|
91
|
+
{
|
|
92
|
+
type: 'text',
|
|
93
|
+
text: responseText
|
|
94
|
+
}
|
|
95
|
+
]
|
|
96
|
+
};
|
|
97
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, result, req) || response;
|
|
98
|
+
} catch (error) {
|
|
99
|
+
const errorMessage = error instanceof Error ? error.message : 'Unknown error';
|
|
100
|
+
payload.logger.error(`[payload-mcp] Error reading resources from collection ${collectionSlug}: ${errorMessage}`);
|
|
101
|
+
const response = {
|
|
102
|
+
content: [
|
|
103
|
+
{
|
|
104
|
+
type: 'text',
|
|
105
|
+
text: `❌ **Error reading resources from collection "${collectionSlug}":** ${errorMessage}`
|
|
106
|
+
}
|
|
107
|
+
]
|
|
108
|
+
};
|
|
109
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response;
|
|
110
|
+
}
|
|
111
|
+
};
|
|
112
|
+
if (collections?.[collectionSlug]?.enabled) {
|
|
113
|
+
server.tool(`find${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`, `${toolSchemas.findResources.description.trim()}\n\n${collections?.[collectionSlug]?.description || ''}`, toolSchemas.findResources.parameters.shape, async ({ id, limit, page, sort, where })=>{
|
|
114
|
+
return await tool(id, limit, page, sort, where);
|
|
115
|
+
});
|
|
116
|
+
}
|
|
117
|
+
};
|
|
118
|
+
|
|
119
|
+
//# sourceMappingURL=find.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../../src/mcp/tools/resource/find.ts"],"sourcesContent":["import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js'\nimport type { PayloadRequest, TypedUser } from 'payload'\n\nimport type { PluginMCPServerConfig } from '../../../types.js'\n\nimport { toCamelCase } from '../../../utils/camelCase.js'\nimport { toolSchemas } from '../schemas.js'\n\nexport const findResourceTool = (\n server: McpServer,\n req: PayloadRequest,\n user: TypedUser,\n verboseLogs: boolean,\n collectionSlug: string,\n collections: PluginMCPServerConfig['collections'],\n) => {\n const tool = async (\n id?: string,\n limit: number = 10,\n page: number = 1,\n sort?: string,\n where?: 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] Reading resource from collection: ${collectionSlug}${id ? ` with ID: ${id}` : ''}, limit: ${limit}, page: ${page}`,\n )\n }\n\n try {\n // Parse where clause if provided\n let whereClause = {}\n if (where) {\n try {\n whereClause = JSON.parse(where)\n if (verboseLogs) {\n payload.logger.info(`[payload-mcp] Using where clause: ${where}`)\n }\n } catch (_parseError) {\n payload.logger.warn(`[payload-mcp] Invalid where clause JSON: ${where}`)\n const response = {\n content: [{ type: 'text' as const, text: 'Error: Invalid JSON in where clause' }],\n }\n return (collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n }\n\n // If ID is provided, use findByID\n if (id) {\n try {\n const doc = await payload.findByID({\n id,\n collection: collectionSlug,\n user,\n })\n\n if (verboseLogs) {\n payload.logger.info(`[payload-mcp] Found document with ID: ${id}`)\n }\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Resource from collection \"${collectionSlug}\":\n${JSON.stringify(doc, null, 2)}`,\n },\n ],\n }\n\n return (collections?.[collectionSlug]?.overrideResponse?.(response, doc, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n } catch (_findError) {\n payload.logger.warn(\n `[payload-mcp] Document not found with ID: ${id} in collection: ${collectionSlug}`,\n )\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Error: Document with ID \"${id}\" not found in collection \"${collectionSlug}\"`,\n },\n ],\n }\n return (collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n }\n\n // Otherwise, use find to get multiple documents\n const findOptions: Parameters<typeof payload.find>[0] = {\n collection: collectionSlug,\n limit,\n page,\n user,\n }\n\n if (sort) {\n findOptions.sort = sort\n }\n\n if (Object.keys(whereClause).length > 0) {\n findOptions.where = whereClause\n }\n\n const result = await payload.find(findOptions)\n\n if (verboseLogs) {\n payload.logger.info(\n `[payload-mcp] Found ${result.docs.length} documents in collection: ${collectionSlug}`,\n )\n }\n\n let responseText = `Collection: \"${collectionSlug}\"\nTotal: ${result.totalDocs} documents\nPage: ${result.page} of ${result.totalPages}\n`\n\n for (const doc of result.docs) {\n responseText += `\\n\\`\\`\\`json\\n${JSON.stringify(doc, null, 2)}\\n\\`\\`\\``\n }\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: responseText,\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 reading resources from collection ${collectionSlug}: ${errorMessage}`,\n )\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `❌ **Error reading resources from collection \"${collectionSlug}\":** ${errorMessage}`,\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 server.tool(\n `find${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`,\n `${toolSchemas.findResources.description.trim()}\\n\\n${collections?.[collectionSlug]?.description || ''}`,\n toolSchemas.findResources.parameters.shape,\n async ({ id, limit, page, sort, where }) => {\n return await tool(id, limit, page, sort, where)\n },\n )\n }\n}\n"],"names":["toCamelCase","toolSchemas","findResourceTool","server","req","user","verboseLogs","collectionSlug","collections","tool","id","limit","page","sort","where","payload","logger","info","whereClause","JSON","parse","_parseError","warn","response","content","type","text","overrideResponse","doc","findByID","collection","stringify","_findError","findOptions","Object","keys","length","result","find","docs","responseText","totalDocs","totalPages","error","errorMessage","Error","message","enabled","charAt","toUpperCase","slice","findResources","description","trim","parameters","shape"],"mappings":"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"}
|
|
@@ -0,0 +1,6 @@
|
|
|
1
|
+
import type { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
2
|
+
import type { JSONSchema4 } from 'json-schema';
|
|
3
|
+
import type { PayloadRequest, TypedUser } from 'payload';
|
|
4
|
+
import type { PluginMCPServerConfig } from '../../../types.js';
|
|
5
|
+
export declare const updateResourceTool: (server: McpServer, req: PayloadRequest, user: TypedUser, verboseLogs: boolean, collectionSlug: string, collections: PluginMCPServerConfig["collections"], schema: JSONSchema4) => void;
|
|
6
|
+
//# sourceMappingURL=update.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"update.d.ts","sourceRoot":"","sources":["../../../../src/mcp/tools/resource/update.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,SAuSpB,CAAA"}
|
|
@@ -0,0 +1,201 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
import { toCamelCase } from '../../../utils/camelCase.js';
|
|
3
|
+
import { convertCollectionSchemaToZod } from '../../../utils/convertCollectionSchemaToZod.js';
|
|
4
|
+
import { toolSchemas } from '../schemas.js';
|
|
5
|
+
export const updateResourceTool = (server, req, user, verboseLogs, collectionSlug, collections, schema)=>{
|
|
6
|
+
const tool = async (data, id, where, draft = false, depth = 0, overrideLock = true, filePath, overwriteExistingFiles = false)=>{
|
|
7
|
+
const payload = req.payload;
|
|
8
|
+
if (verboseLogs) {
|
|
9
|
+
payload.logger.info(`[payload-mcp] Updating resource in collection: ${collectionSlug}${id ? ` with ID: ${id}` : ' with where clause'}, draft: ${draft}`);
|
|
10
|
+
}
|
|
11
|
+
try {
|
|
12
|
+
// Parse the data JSON
|
|
13
|
+
let parsedData;
|
|
14
|
+
try {
|
|
15
|
+
parsedData = JSON.parse(data);
|
|
16
|
+
if (verboseLogs) {
|
|
17
|
+
payload.logger.info(`[payload-mcp] Parsed data for ${collectionSlug}: ${JSON.stringify(parsedData)}`);
|
|
18
|
+
}
|
|
19
|
+
} catch (_parseError) {
|
|
20
|
+
payload.logger.error(`[payload-mcp] Invalid JSON data provided: ${data}`);
|
|
21
|
+
const response = {
|
|
22
|
+
content: [
|
|
23
|
+
{
|
|
24
|
+
type: 'text',
|
|
25
|
+
text: 'Error: Invalid JSON data provided'
|
|
26
|
+
}
|
|
27
|
+
]
|
|
28
|
+
};
|
|
29
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response;
|
|
30
|
+
}
|
|
31
|
+
// Validate that either id or where is provided
|
|
32
|
+
if (!id && !where) {
|
|
33
|
+
payload.logger.error('[payload-mcp] Either id or where clause must be provided');
|
|
34
|
+
const response = {
|
|
35
|
+
content: [
|
|
36
|
+
{
|
|
37
|
+
type: 'text',
|
|
38
|
+
text: 'Error: Either id or where clause must be provided'
|
|
39
|
+
}
|
|
40
|
+
]
|
|
41
|
+
};
|
|
42
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response;
|
|
43
|
+
}
|
|
44
|
+
// Parse where clause if provided
|
|
45
|
+
let whereClause = {};
|
|
46
|
+
if (where) {
|
|
47
|
+
try {
|
|
48
|
+
whereClause = JSON.parse(where);
|
|
49
|
+
if (verboseLogs) {
|
|
50
|
+
payload.logger.info(`[payload-mcp] Using where clause: ${where}`);
|
|
51
|
+
}
|
|
52
|
+
} catch (_parseError) {
|
|
53
|
+
payload.logger.error(`[payload-mcp] Invalid where clause JSON: ${where}`);
|
|
54
|
+
const response = {
|
|
55
|
+
content: [
|
|
56
|
+
{
|
|
57
|
+
type: 'text',
|
|
58
|
+
text: 'Error: Invalid JSON in where clause'
|
|
59
|
+
}
|
|
60
|
+
]
|
|
61
|
+
};
|
|
62
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response;
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
// Update by ID or where clause
|
|
66
|
+
if (id) {
|
|
67
|
+
// Single document update
|
|
68
|
+
const updateOptions = {
|
|
69
|
+
id,
|
|
70
|
+
collection: collectionSlug,
|
|
71
|
+
data: parsedData,
|
|
72
|
+
depth,
|
|
73
|
+
draft,
|
|
74
|
+
overrideLock,
|
|
75
|
+
user,
|
|
76
|
+
...filePath && {
|
|
77
|
+
filePath
|
|
78
|
+
},
|
|
79
|
+
...overwriteExistingFiles && {
|
|
80
|
+
overwriteExistingFiles
|
|
81
|
+
}
|
|
82
|
+
};
|
|
83
|
+
if (verboseLogs) {
|
|
84
|
+
payload.logger.info(`[payload-mcp] Updating single document with ID: ${id}`);
|
|
85
|
+
}
|
|
86
|
+
const result = await payload.update({
|
|
87
|
+
...updateOptions,
|
|
88
|
+
data: collections?.[collectionSlug]?.override?.(parsedData, req) || parsedData
|
|
89
|
+
});
|
|
90
|
+
if (verboseLogs) {
|
|
91
|
+
payload.logger.info(`[payload-mcp] Successfully updated document with ID: ${id}`);
|
|
92
|
+
}
|
|
93
|
+
const response = {
|
|
94
|
+
content: [
|
|
95
|
+
{
|
|
96
|
+
type: 'text',
|
|
97
|
+
text: `Document updated successfully in collection "${collectionSlug}"!
|
|
98
|
+
Updated document:
|
|
99
|
+
\`\`\`json
|
|
100
|
+
${JSON.stringify(result, null, 2)}
|
|
101
|
+
\`\`\``
|
|
102
|
+
}
|
|
103
|
+
]
|
|
104
|
+
};
|
|
105
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, result, req) || response;
|
|
106
|
+
} else {
|
|
107
|
+
// Multiple documents update
|
|
108
|
+
const updateOptions = {
|
|
109
|
+
collection: collectionSlug,
|
|
110
|
+
data: parsedData,
|
|
111
|
+
depth,
|
|
112
|
+
draft,
|
|
113
|
+
overrideAccess: true,
|
|
114
|
+
overrideLock,
|
|
115
|
+
where: whereClause,
|
|
116
|
+
...filePath && {
|
|
117
|
+
filePath
|
|
118
|
+
},
|
|
119
|
+
...overwriteExistingFiles && {
|
|
120
|
+
overwriteExistingFiles
|
|
121
|
+
}
|
|
122
|
+
};
|
|
123
|
+
if (verboseLogs) {
|
|
124
|
+
payload.logger.info(`[payload-mcp] Updating multiple documents with where clause`);
|
|
125
|
+
}
|
|
126
|
+
const result = await payload.update({
|
|
127
|
+
...updateOptions,
|
|
128
|
+
data: collections?.[collectionSlug]?.override?.(parsedData, req) || parsedData
|
|
129
|
+
});
|
|
130
|
+
const bulkResult = result;
|
|
131
|
+
const docs = bulkResult.docs || [];
|
|
132
|
+
const errors = bulkResult.errors || [];
|
|
133
|
+
if (verboseLogs) {
|
|
134
|
+
payload.logger.info(`[payload-mcp] Successfully updated ${docs.length} documents, ${errors.length} errors`);
|
|
135
|
+
}
|
|
136
|
+
let responseText = `Multiple documents updated in collection "${collectionSlug}"!
|
|
137
|
+
Updated: ${docs.length} documents
|
|
138
|
+
Errors: ${errors.length}
|
|
139
|
+
---`;
|
|
140
|
+
if (docs.length > 0) {
|
|
141
|
+
responseText += `\n\nUpdated documents:
|
|
142
|
+
\`\`\`json
|
|
143
|
+
${JSON.stringify(docs, null, 2)}
|
|
144
|
+
\`\`\``;
|
|
145
|
+
}
|
|
146
|
+
if (errors.length > 0) {
|
|
147
|
+
responseText += `\n\nErrors:
|
|
148
|
+
\`\`\`json
|
|
149
|
+
${JSON.stringify(errors, null, 2)}
|
|
150
|
+
\`\`\``;
|
|
151
|
+
}
|
|
152
|
+
const response = {
|
|
153
|
+
content: [
|
|
154
|
+
{
|
|
155
|
+
type: 'text',
|
|
156
|
+
text: responseText
|
|
157
|
+
}
|
|
158
|
+
]
|
|
159
|
+
};
|
|
160
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {
|
|
161
|
+
docs,
|
|
162
|
+
errors
|
|
163
|
+
}, req) || response;
|
|
164
|
+
}
|
|
165
|
+
} catch (error) {
|
|
166
|
+
const errorMessage = error instanceof Error ? error.message : 'Unknown error';
|
|
167
|
+
payload.logger.error(`[payload-mcp] Error updating resource in ${collectionSlug}: ${errorMessage}`);
|
|
168
|
+
const response = {
|
|
169
|
+
content: [
|
|
170
|
+
{
|
|
171
|
+
type: 'text',
|
|
172
|
+
text: `Error updating resource in collection "${collectionSlug}": ${errorMessage}`
|
|
173
|
+
}
|
|
174
|
+
]
|
|
175
|
+
};
|
|
176
|
+
return collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) || response;
|
|
177
|
+
}
|
|
178
|
+
};
|
|
179
|
+
if (collections?.[collectionSlug]?.enabled) {
|
|
180
|
+
const convertedFields = convertCollectionSchemaToZod(schema);
|
|
181
|
+
// Create a new schema that combines the converted fields with update-specific parameters
|
|
182
|
+
const updateResourceSchema = z.object({
|
|
183
|
+
...convertedFields.shape,
|
|
184
|
+
id: z.string().optional().describe('The ID of the document to update'),
|
|
185
|
+
depth: z.number().optional().default(0).describe('How many levels deep to populate relationships'),
|
|
186
|
+
draft: z.boolean().optional().default(false).describe('Whether to update the document as a draft'),
|
|
187
|
+
filePath: z.string().optional().describe('File path for file uploads'),
|
|
188
|
+
overrideLock: z.boolean().optional().default(true).describe('Whether to override document locks'),
|
|
189
|
+
overwriteExistingFiles: z.boolean().optional().default(false).describe('Whether to overwrite existing files'),
|
|
190
|
+
where: z.string().optional().describe('JSON string for where clause to update multiple documents')
|
|
191
|
+
});
|
|
192
|
+
server.tool(`update${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`, `${toolSchemas.updateResource.description.trim()}\n\n${collections?.[collectionSlug]?.description || ''}`, updateResourceSchema.shape, async (params)=>{
|
|
193
|
+
const { id, depth, draft, filePath, overrideLock, overwriteExistingFiles, where, ...fieldData } = params;
|
|
194
|
+
// Convert field data back to JSON string format expected by the tool
|
|
195
|
+
const data = JSON.stringify(fieldData);
|
|
196
|
+
return await tool(data, id, where, draft, depth, overrideLock, filePath, overwriteExistingFiles);
|
|
197
|
+
});
|
|
198
|
+
}
|
|
199
|
+
};
|
|
200
|
+
|
|
201
|
+
//# sourceMappingURL=update.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../../../src/mcp/tools/resource/update.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 updateResourceTool = (\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 id?: string,\n where?: string,\n draft: boolean = false,\n depth: number = 0,\n overrideLock: boolean = true,\n filePath?: string,\n overwriteExistingFiles: boolean = false,\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] Updating resource in collection: ${collectionSlug}${id ? ` with ID: ${id}` : ' with where clause'}, draft: ${draft}`,\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 const response = {\n content: [{ type: 'text' as const, text: 'Error: Invalid JSON data provided' }],\n }\n return (collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n\n // Validate that either id or where is provided\n if (!id && !where) {\n payload.logger.error('[payload-mcp] Either id or where clause must be provided')\n const response = {\n content: [\n { type: 'text' as const, text: 'Error: Either id or where clause must be provided' },\n ],\n }\n return (collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n\n // Parse where clause if provided\n let whereClause = {}\n if (where) {\n try {\n whereClause = JSON.parse(where)\n if (verboseLogs) {\n payload.logger.info(`[payload-mcp] Using where clause: ${where}`)\n }\n } catch (_parseError) {\n payload.logger.error(`[payload-mcp] Invalid where clause JSON: ${where}`)\n const response = {\n content: [{ type: 'text' as const, text: 'Error: Invalid JSON in where clause' }],\n }\n return (collections?.[collectionSlug]?.overrideResponse?.(response, {}, req) ||\n response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n }\n\n // Update by ID or where clause\n if (id) {\n // Single document update\n const updateOptions = {\n id,\n collection: collectionSlug,\n data: parsedData,\n depth,\n draft,\n overrideLock,\n user,\n ...(filePath && { filePath }),\n ...(overwriteExistingFiles && { overwriteExistingFiles }),\n }\n\n if (verboseLogs) {\n payload.logger.info(`[payload-mcp] Updating single document with ID: ${id}`)\n }\n const result = await payload.update({\n ...updateOptions,\n data: collections?.[collectionSlug]?.override?.(parsedData, req) || parsedData,\n } as any)\n\n if (verboseLogs) {\n payload.logger.info(`[payload-mcp] Successfully updated document with ID: ${id}`)\n }\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Document updated successfully in collection \"${collectionSlug}\"!\nUpdated document:\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 } else {\n // Multiple documents update\n const updateOptions = {\n collection: collectionSlug,\n data: parsedData,\n depth,\n draft,\n overrideAccess: true,\n overrideLock,\n where: whereClause,\n ...(filePath && { filePath }),\n ...(overwriteExistingFiles && { overwriteExistingFiles }),\n }\n\n if (verboseLogs) {\n payload.logger.info(`[payload-mcp] Updating multiple documents with where clause`)\n }\n const result = await payload.update({\n ...updateOptions,\n data: collections?.[collectionSlug]?.override?.(parsedData, req) || parsedData,\n } as any)\n\n const bulkResult = result as { docs?: unknown[]; errors?: unknown[] }\n const docs = bulkResult.docs || []\n const errors = bulkResult.errors || []\n\n if (verboseLogs) {\n payload.logger.info(\n `[payload-mcp] Successfully updated ${docs.length} documents, ${errors.length} errors`,\n )\n }\n\n let responseText = `Multiple documents updated in collection \"${collectionSlug}\"!\nUpdated: ${docs.length} documents\nErrors: ${errors.length}\n---`\n\n if (docs.length > 0) {\n responseText += `\\n\\nUpdated documents:\n\\`\\`\\`json\n${JSON.stringify(docs, null, 2)}\n\\`\\`\\``\n }\n\n if (errors.length > 0) {\n responseText += `\\n\\nErrors:\n\\`\\`\\`json\n${JSON.stringify(errors, null, 2)}\n\\`\\`\\``\n }\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: responseText,\n },\n ],\n }\n\n return (collections?.[collectionSlug]?.overrideResponse?.(\n response,\n { docs, errors },\n req,\n ) || response) as {\n content: Array<{\n text: string\n type: 'text'\n }>\n }\n }\n } catch (error) {\n const errorMessage = error instanceof Error ? error.message : 'Unknown error'\n payload.logger.error(\n `[payload-mcp] Error updating resource in ${collectionSlug}: ${errorMessage}`,\n )\n\n const response = {\n content: [\n {\n type: 'text' as const,\n text: `Error updating 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 update-specific parameters\n const updateResourceSchema = z.object({\n ...convertedFields.shape,\n id: z.string().optional().describe('The ID of the document to update'),\n depth: z\n .number()\n .optional()\n .default(0)\n .describe('How many levels deep to populate relationships'),\n draft: z\n .boolean()\n .optional()\n .default(false)\n .describe('Whether to update the document as a draft'),\n filePath: z.string().optional().describe('File path for file uploads'),\n overrideLock: z\n .boolean()\n .optional()\n .default(true)\n .describe('Whether to override document locks'),\n overwriteExistingFiles: z\n .boolean()\n .optional()\n .default(false)\n .describe('Whether to overwrite existing files'),\n where: z\n .string()\n .optional()\n .describe('JSON string for where clause to update multiple documents'),\n })\n\n server.tool(\n `update${collectionSlug.charAt(0).toUpperCase() + toCamelCase(collectionSlug).slice(1)}`,\n `${toolSchemas.updateResource.description.trim()}\\n\\n${collections?.[collectionSlug]?.description || ''}`,\n updateResourceSchema.shape,\n async (params: Record<string, unknown>) => {\n const {\n id,\n depth,\n draft,\n filePath,\n overrideLock,\n overwriteExistingFiles,\n where,\n ...fieldData\n } = params\n // Convert field data back to JSON string format expected by the tool\n const data = JSON.stringify(fieldData)\n return await tool(\n data,\n id as string | undefined,\n where as string | undefined,\n draft as boolean,\n depth as number,\n overrideLock as boolean,\n filePath as string | undefined,\n overwriteExistingFiles as boolean,\n )\n },\n )\n }\n}\n"],"names":["z","toCamelCase","convertCollectionSchemaToZod","toolSchemas","updateResourceTool","server","req","user","verboseLogs","collectionSlug","collections","schema","tool","data","id","where","draft","depth","overrideLock","filePath","overwriteExistingFiles","payload","logger","info","parsedData","JSON","parse","stringify","_parseError","error","response","content","type","text","overrideResponse","whereClause","updateOptions","collection","result","update","override","overrideAccess","bulkResult","docs","errors","length","responseText","errorMessage","Error","message","enabled","convertedFields","updateResourceSchema","object","shape","string","optional","describe","number","default","boolean","charAt","toUpperCase","slice","updateResource","description","trim","params","fieldData"],"mappings":"AAIA,SAASA,CAAC,QAAQ,MAAK;AAIvB,SAASC,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,MACAC,IACAC,OACAC,QAAiB,KAAK,EACtBC,QAAgB,CAAC,EACjBC,eAAwB,IAAI,EAC5BC,UACAC,yBAAkC,KAAK;QAOvC,MAAMC,UAAUf,IAAIe,OAAO;QAE3B,IAAIb,aAAa;YACfa,QAAQC,MAAM,CAACC,IAAI,CACjB,CAAC,+CAA+C,EAAEd,iBAAiBK,KAAK,CAAC,UAAU,EAAEA,IAAI,GAAG,qBAAqB,SAAS,EAAEE,OAAO;QAEvI;QAEA,IAAI;YACF,sBAAsB;YACtB,IAAIQ;YACJ,IAAI;gBACFA,aAAaC,KAAKC,KAAK,CAACb;gBACxB,IAAIL,aAAa;oBACfa,QAAQC,MAAM,CAACC,IAAI,CACjB,CAAC,8BAA8B,EAAEd,eAAe,EAAE,EAAEgB,KAAKE,SAAS,CAACH,aAAa;gBAEpF;YACF,EAAE,OAAOI,aAAa;gBACpBP,QAAQC,MAAM,CAACO,KAAK,CAAC,CAAC,0CAA0C,EAAEhB,MAAM;gBACxE,MAAMiB,WAAW;oBACfC,SAAS;wBAAC;4BAAEC,MAAM;4BAAiBC,MAAM;wBAAoC;qBAAE;gBACjF;gBACA,OAAQvB,aAAa,CAACD,eAAe,EAAEyB,mBAAmBJ,UAAU,CAAC,GAAGxB,QACtEwB;YAMJ;YAEA,+CAA+C;YAC/C,IAAI,CAAChB,MAAM,CAACC,OAAO;gBACjBM,QAAQC,MAAM,CAACO,KAAK,CAAC;gBACrB,MAAMC,WAAW;oBACfC,SAAS;wBACP;4BAAEC,MAAM;4BAAiBC,MAAM;wBAAoD;qBACpF;gBACH;gBACA,OAAQvB,aAAa,CAACD,eAAe,EAAEyB,mBAAmBJ,UAAU,CAAC,GAAGxB,QACtEwB;YAMJ;YAEA,iCAAiC;YACjC,IAAIK,cAAc,CAAC;YACnB,IAAIpB,OAAO;gBACT,IAAI;oBACFoB,cAAcV,KAAKC,KAAK,CAACX;oBACzB,IAAIP,aAAa;wBACfa,QAAQC,MAAM,CAACC,IAAI,CAAC,CAAC,kCAAkC,EAAER,OAAO;oBAClE;gBACF,EAAE,OAAOa,aAAa;oBACpBP,QAAQC,MAAM,CAACO,KAAK,CAAC,CAAC,yCAAyC,EAAEd,OAAO;oBACxE,MAAMe,WAAW;wBACfC,SAAS;4BAAC;gCAAEC,MAAM;gCAAiBC,MAAM;4BAAsC;yBAAE;oBACnF;oBACA,OAAQvB,aAAa,CAACD,eAAe,EAAEyB,mBAAmBJ,UAAU,CAAC,GAAGxB,QACtEwB;gBAMJ;YACF;YAEA,+BAA+B;YAC/B,IAAIhB,IAAI;gBACN,yBAAyB;gBACzB,MAAMsB,gBAAgB;oBACpBtB;oBACAuB,YAAY5B;oBACZI,MAAMW;oBACNP;oBACAD;oBACAE;oBACAX;oBACA,GAAIY,YAAY;wBAAEA;oBAAS,CAAC;oBAC5B,GAAIC,0BAA0B;wBAAEA;oBAAuB,CAAC;gBAC1D;gBAEA,IAAIZ,aAAa;oBACfa,QAAQC,MAAM,CAACC,IAAI,CAAC,CAAC,gDAAgD,EAAET,IAAI;gBAC7E;gBACA,MAAMwB,SAAS,MAAMjB,QAAQkB,MAAM,CAAC;oBAClC,GAAGH,aAAa;oBAChBvB,MAAMH,aAAa,CAACD,eAAe,EAAE+B,WAAWhB,YAAYlB,QAAQkB;gBACtE;gBAEA,IAAIhB,aAAa;oBACfa,QAAQC,MAAM,CAACC,IAAI,CAAC,CAAC,qDAAqD,EAAET,IAAI;gBAClF;gBAEA,MAAMgB,WAAW;oBACfC,SAAS;wBACP;4BACEC,MAAM;4BACNC,MAAM,CAAC,6CAA6C,EAAExB,eAAe;;;AAGnF,EAAEgB,KAAKE,SAAS,CAACW,QAAQ,MAAM,GAAG;MAC5B,CAAC;wBACK;qBACD;gBACH;gBAEA,OAAQ5B,aAAa,CAACD,eAAe,EAAEyB,mBAAmBJ,UAAUQ,QAAQhC,QAC1EwB;YAMJ,OAAO;gBACL,4BAA4B;gBAC5B,MAAMM,gBAAgB;oBACpBC,YAAY5B;oBACZI,MAAMW;oBACNP;oBACAD;oBACAyB,gBAAgB;oBAChBvB;oBACAH,OAAOoB;oBACP,GAAIhB,YAAY;wBAAEA;oBAAS,CAAC;oBAC5B,GAAIC,0BAA0B;wBAAEA;oBAAuB,CAAC;gBAC1D;gBAEA,IAAIZ,aAAa;oBACfa,QAAQC,MAAM,CAACC,IAAI,CAAC,CAAC,2DAA2D,CAAC;gBACnF;gBACA,MAAMe,SAAS,MAAMjB,QAAQkB,MAAM,CAAC;oBAClC,GAAGH,aAAa;oBAChBvB,MAAMH,aAAa,CAACD,eAAe,EAAE+B,WAAWhB,YAAYlB,QAAQkB;gBACtE;gBAEA,MAAMkB,aAAaJ;gBACnB,MAAMK,OAAOD,WAAWC,IAAI,IAAI,EAAE;gBAClC,MAAMC,SAASF,WAAWE,MAAM,IAAI,EAAE;gBAEtC,IAAIpC,aAAa;oBACfa,QAAQC,MAAM,CAACC,IAAI,CACjB,CAAC,mCAAmC,EAAEoB,KAAKE,MAAM,CAAC,YAAY,EAAED,OAAOC,MAAM,CAAC,OAAO,CAAC;gBAE1F;gBAEA,IAAIC,eAAe,CAAC,0CAA0C,EAAErC,eAAe;SAC9E,EAAEkC,KAAKE,MAAM,CAAC;QACf,EAAED,OAAOC,MAAM,CAAC;GACrB,CAAC;gBAEI,IAAIF,KAAKE,MAAM,GAAG,GAAG;oBACnBC,gBAAgB,CAAC;;AAE3B,EAAErB,KAAKE,SAAS,CAACgB,MAAM,MAAM,GAAG;MAC1B,CAAC;gBACC;gBAEA,IAAIC,OAAOC,MAAM,GAAG,GAAG;oBACrBC,gBAAgB,CAAC;;AAE3B,EAAErB,KAAKE,SAAS,CAACiB,QAAQ,MAAM,GAAG;MAC5B,CAAC;gBACC;gBAEA,MAAMd,WAAW;oBACfC,SAAS;wBACP;4BACEC,MAAM;4BACNC,MAAMa;wBACR;qBACD;gBACH;gBAEA,OAAQpC,aAAa,CAACD,eAAe,EAAEyB,mBACrCJ,UACA;oBAAEa;oBAAMC;gBAAO,GACftC,QACGwB;YAMP;QACF,EAAE,OAAOD,OAAO;YACd,MAAMkB,eAAelB,iBAAiBmB,QAAQnB,MAAMoB,OAAO,GAAG;YAC9D5B,QAAQC,MAAM,CAACO,KAAK,CAClB,CAAC,yCAAyC,EAAEpB,eAAe,EAAE,EAAEsC,cAAc;YAG/E,MAAMjB,WAAW;gBACfC,SAAS;oBACP;wBACEC,MAAM;wBACNC,MAAM,CAAC,uCAAuC,EAAExB,eAAe,GAAG,EAAEsC,cAAc;oBACpF;iBACD;YACH;YAEA,OAAQrC,aAAa,CAACD,eAAe,EAAEyB,mBAAmBJ,UAAU,CAAC,GAAGxB,QAAQwB;QAMlF;IACF;IAEA,IAAIpB,aAAa,CAACD,eAAe,EAAEyC,SAAS;QAC1C,MAAMC,kBAAkBjD,6BAA6BS;QAErD,yFAAyF;QACzF,MAAMyC,uBAAuBpD,EAAEqD,MAAM,CAAC;YACpC,GAAGF,gBAAgBG,KAAK;YACxBxC,IAAId,EAAEuD,MAAM,GAAGC,QAAQ,GAAGC,QAAQ,CAAC;YACnCxC,OAAOjB,EACJ0D,MAAM,GACNF,QAAQ,GACRG,OAAO,CAAC,GACRF,QAAQ,CAAC;YACZzC,OAAOhB,EACJ4D,OAAO,GACPJ,QAAQ,GACRG,OAAO,CAAC,OACRF,QAAQ,CAAC;YACZtC,UAAUnB,EAAEuD,MAAM,GAAGC,QAAQ,GAAGC,QAAQ,CAAC;YACzCvC,cAAclB,EACX4D,OAAO,GACPJ,QAAQ,GACRG,OAAO,CAAC,MACRF,QAAQ,CAAC;YACZrC,wBAAwBpB,EACrB4D,OAAO,GACPJ,QAAQ,GACRG,OAAO,CAAC,OACRF,QAAQ,CAAC;YACZ1C,OAAOf,EACJuD,MAAM,GACNC,QAAQ,GACRC,QAAQ,CAAC;QACd;QAEApD,OAAOO,IAAI,CACT,CAAC,MAAM,EAAEH,eAAeoD,MAAM,CAAC,GAAGC,WAAW,KAAK7D,YAAYQ,gBAAgBsD,KAAK,CAAC,IAAI,EACxF,GAAG5D,YAAY6D,cAAc,CAACC,WAAW,CAACC,IAAI,GAAG,IAAI,EAAExD,aAAa,CAACD,eAAe,EAAEwD,eAAe,IAAI,EACzGb,qBAAqBE,KAAK,EAC1B,OAAOa;YACL,MAAM,EACJrD,EAAE,EACFG,KAAK,EACLD,KAAK,EACLG,QAAQ,EACRD,YAAY,EACZE,sBAAsB,EACtBL,KAAK,EACL,GAAGqD,WACJ,GAAGD;YACJ,qEAAqE;YACrE,MAAMtD,OAAOY,KAAKE,SAAS,CAACyC;YAC5B,OAAO,MAAMxD,KACXC,MACAC,IACAC,OACAC,OACAC,OACAC,cACAC,UACAC;QAEJ;IAEJ;AACF,EAAC"}
|