@strapi/content-manager 5.46.1 → 5.47.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.
- package/dist/admin/history/components/VersionInputRenderer.js +63 -25
- package/dist/admin/history/components/VersionInputRenderer.js.map +1 -1
- package/dist/admin/history/components/VersionInputRenderer.mjs +62 -26
- package/dist/admin/history/components/VersionInputRenderer.mjs.map +1 -1
- package/dist/admin/hooks/useDocument.js +82 -2
- package/dist/admin/hooks/useDocument.js.map +1 -1
- package/dist/admin/hooks/useDocument.mjs +82 -2
- package/dist/admin/hooks/useDocument.mjs.map +1 -1
- package/dist/admin/pages/EditView/EditViewPage.js +3 -2
- package/dist/admin/pages/EditView/EditViewPage.js.map +1 -1
- package/dist/admin/pages/EditView/EditViewPage.mjs +3 -2
- package/dist/admin/pages/EditView/EditViewPage.mjs.map +1 -1
- package/dist/admin/pages/EditView/components/DocumentActions.js +6 -3
- package/dist/admin/pages/EditView/components/DocumentActions.js.map +1 -1
- package/dist/admin/pages/EditView/components/DocumentActions.mjs +6 -3
- package/dist/admin/pages/EditView/components/DocumentActions.mjs.map +1 -1
- package/dist/admin/pages/EditView/components/FormInputs/Component/Repeatable.js +4 -1
- package/dist/admin/pages/EditView/components/FormInputs/Component/Repeatable.js.map +1 -1
- package/dist/admin/pages/EditView/components/FormInputs/Component/Repeatable.mjs +4 -1
- package/dist/admin/pages/EditView/components/FormInputs/Component/Repeatable.mjs.map +1 -1
- package/dist/admin/pages/EditView/components/FormInputs/Relations/Relations.js +4 -4
- package/dist/admin/pages/EditView/components/FormInputs/Relations/Relations.js.map +1 -1
- package/dist/admin/pages/EditView/components/FormInputs/Relations/Relations.mjs +4 -4
- package/dist/admin/pages/EditView/components/FormInputs/Relations/Relations.mjs.map +1 -1
- package/dist/admin/src/history/components/VersionInputRenderer.d.ts +27 -1
- package/dist/admin/translations/sk.json.js +175 -9
- package/dist/admin/translations/sk.json.js.map +1 -1
- package/dist/admin/translations/sk.json.mjs +175 -9
- package/dist/admin/translations/sk.json.mjs.map +1 -1
- package/dist/admin/utils/relations.js +4 -0
- package/dist/admin/utils/relations.js.map +1 -1
- package/dist/admin/utils/relations.mjs +4 -0
- package/dist/admin/utils/relations.mjs.map +1 -1
- package/dist/server/bootstrap.js +4 -0
- package/dist/server/bootstrap.js.map +1 -1
- package/dist/server/bootstrap.mjs +4 -0
- package/dist/server/bootstrap.mjs.map +1 -1
- package/dist/server/controllers/collection-types.js +9 -5
- package/dist/server/controllers/collection-types.js.map +1 -1
- package/dist/server/controllers/collection-types.mjs +10 -6
- package/dist/server/controllers/collection-types.mjs.map +1 -1
- package/dist/server/homepage/services/homepage.js +21 -16
- package/dist/server/homepage/services/homepage.js.map +1 -1
- package/dist/server/homepage/services/homepage.mjs +21 -16
- package/dist/server/homepage/services/homepage.mjs.map +1 -1
- package/dist/server/mcp/derive-content-type-mcp-tools.js +524 -0
- package/dist/server/mcp/derive-content-type-mcp-tools.js.map +1 -0
- package/dist/server/mcp/derive-content-type-mcp-tools.mjs +518 -0
- package/dist/server/mcp/derive-content-type-mcp-tools.mjs.map +1 -0
- package/dist/server/mcp/handlers/collection-handlers.js +404 -0
- package/dist/server/mcp/handlers/collection-handlers.js.map +1 -0
- package/dist/server/mcp/handlers/collection-handlers.mjs +395 -0
- package/dist/server/mcp/handlers/collection-handlers.mjs.map +1 -0
- package/dist/server/mcp/handlers/constants.js +10 -0
- package/dist/server/mcp/handlers/constants.js.map +1 -0
- package/dist/server/mcp/handlers/constants.mjs +6 -0
- package/dist/server/mcp/handlers/constants.mjs.map +1 -0
- package/dist/server/mcp/handlers/single-type-handlers.js +344 -0
- package/dist/server/mcp/handlers/single-type-handlers.js.map +1 -0
- package/dist/server/mcp/handlers/single-type-handlers.mjs +336 -0
- package/dist/server/mcp/handlers/single-type-handlers.mjs.map +1 -0
- package/dist/server/mcp/permissions.js +138 -0
- package/dist/server/mcp/permissions.js.map +1 -0
- package/dist/server/mcp/permissions.mjs +131 -0
- package/dist/server/mcp/permissions.mjs.map +1 -0
- package/dist/server/mcp/register-content-manager-mcp-tools.js +30 -0
- package/dist/server/mcp/register-content-manager-mcp-tools.js.map +1 -0
- package/dist/server/mcp/register-content-manager-mcp-tools.mjs +28 -0
- package/dist/server/mcp/register-content-manager-mcp-tools.mjs.map +1 -0
- package/dist/server/mcp/schemas/blocks-schema.js +124 -0
- package/dist/server/mcp/schemas/blocks-schema.js.map +1 -0
- package/dist/server/mcp/schemas/blocks-schema.mjs +122 -0
- package/dist/server/mcp/schemas/blocks-schema.mjs.map +1 -0
- package/dist/server/mcp/schemas/data-schema.js +252 -0
- package/dist/server/mcp/schemas/data-schema.js.map +1 -0
- package/dist/server/mcp/schemas/data-schema.mjs +248 -0
- package/dist/server/mcp/schemas/data-schema.mjs.map +1 -0
- package/dist/server/mcp/schemas/filters-schema.js +111 -0
- package/dist/server/mcp/schemas/filters-schema.js.map +1 -0
- package/dist/server/mcp/schemas/filters-schema.mjs +107 -0
- package/dist/server/mcp/schemas/filters-schema.mjs.map +1 -0
- package/dist/server/mcp/schemas/input-schemas.js +18 -0
- package/dist/server/mcp/schemas/input-schemas.js.map +1 -0
- package/dist/server/mcp/schemas/input-schemas.mjs +13 -0
- package/dist/server/mcp/schemas/input-schemas.mjs.map +1 -0
- package/dist/server/mcp/schemas/output-schemas.js +48 -0
- package/dist/server/mcp/schemas/output-schemas.js.map +1 -0
- package/dist/server/mcp/schemas/output-schemas.mjs +44 -0
- package/dist/server/mcp/schemas/output-schemas.mjs.map +1 -0
- package/dist/server/mcp/schemas/sort-schema.js +80 -0
- package/dist/server/mcp/schemas/sort-schema.js.map +1 -0
- package/dist/server/mcp/schemas/sort-schema.mjs +76 -0
- package/dist/server/mcp/schemas/sort-schema.mjs.map +1 -0
- package/dist/server/mcp/utils.js +43 -0
- package/dist/server/mcp/utils.js.map +1 -0
- package/dist/server/mcp/utils.mjs +39 -0
- package/dist/server/mcp/utils.mjs.map +1 -0
- package/dist/server/services/document-metadata.js +32 -3
- package/dist/server/services/document-metadata.js.map +1 -1
- package/dist/server/services/document-metadata.mjs +32 -3
- package/dist/server/services/document-metadata.mjs.map +1 -1
- package/dist/server/services/index.js +1 -1
- package/dist/server/services/index.js.map +1 -1
- package/dist/server/services/permission-checker.js +4 -1
- package/dist/server/services/permission-checker.js.map +1 -1
- package/dist/server/services/permission-checker.mjs +1 -1
- package/dist/server/services/permission-checker.mjs.map +1 -1
- package/dist/server/services/utils/populate.js +3 -3
- package/dist/server/services/utils/populate.js.map +1 -1
- package/dist/server/services/utils/populate.mjs +3 -3
- package/dist/server/services/utils/populate.mjs.map +1 -1
- package/dist/server/src/bootstrap.d.ts.map +1 -1
- package/dist/server/src/controllers/collection-types.d.ts.map +1 -1
- package/dist/server/src/homepage/services/homepage.d.ts.map +1 -1
- package/dist/server/src/index.d.ts +3 -3
- package/dist/server/src/mcp/derive-content-type-mcp-tools.d.ts +12 -0
- package/dist/server/src/mcp/derive-content-type-mcp-tools.d.ts.map +1 -0
- package/dist/server/src/mcp/handlers/collection-handlers.d.ts +69 -0
- package/dist/server/src/mcp/handlers/collection-handlers.d.ts.map +1 -0
- package/dist/server/src/mcp/handlers/constants.d.ts +4 -0
- package/dist/server/src/mcp/handlers/constants.d.ts.map +1 -0
- package/dist/server/src/mcp/handlers/index.d.ts +3 -0
- package/dist/server/src/mcp/handlers/index.d.ts.map +1 -0
- package/dist/server/src/mcp/handlers/single-type-handlers.d.ts +66 -0
- package/dist/server/src/mcp/handlers/single-type-handlers.d.ts.map +1 -0
- package/dist/server/src/mcp/permissions.d.ts +49 -0
- package/dist/server/src/mcp/permissions.d.ts.map +1 -0
- package/dist/server/src/mcp/register-content-manager-mcp-tools.d.ts +8 -0
- package/dist/server/src/mcp/register-content-manager-mcp-tools.d.ts.map +1 -0
- package/dist/server/src/mcp/schemas/blocks-schema.d.ts +8 -0
- package/dist/server/src/mcp/schemas/blocks-schema.d.ts.map +1 -0
- package/dist/server/src/mcp/schemas/data-schema.d.ts +36 -0
- package/dist/server/src/mcp/schemas/data-schema.d.ts.map +1 -0
- package/dist/server/src/mcp/schemas/filters-schema.d.ts +22 -0
- package/dist/server/src/mcp/schemas/filters-schema.d.ts.map +1 -0
- package/dist/server/src/mcp/schemas/index.d.ts +7 -0
- package/dist/server/src/mcp/schemas/index.d.ts.map +1 -0
- package/dist/server/src/mcp/schemas/input-schemas.d.ts +10 -0
- package/dist/server/src/mcp/schemas/input-schemas.d.ts.map +1 -0
- package/dist/server/src/mcp/schemas/output-schemas.d.ts +18 -0
- package/dist/server/src/mcp/schemas/output-schemas.d.ts.map +1 -0
- package/dist/server/src/mcp/schemas/sort-schema.d.ts +24 -0
- package/dist/server/src/mcp/schemas/sort-schema.d.ts.map +1 -0
- package/dist/server/src/mcp/types.d.ts +31 -0
- package/dist/server/src/mcp/types.d.ts.map +1 -0
- package/dist/server/src/mcp/utils.d.ts +21 -0
- package/dist/server/src/mcp/utils.d.ts.map +1 -0
- package/dist/server/src/services/document-metadata.d.ts +11 -1
- package/dist/server/src/services/document-metadata.d.ts.map +1 -1
- package/dist/server/src/services/index.d.ts +3 -3
- package/dist/server/src/services/permission-checker.d.ts +13 -3
- package/dist/server/src/services/permission-checker.d.ts.map +1 -1
- package/dist/server/src/services/utils/configuration/index.d.ts +2 -2
- package/dist/server/src/services/utils/configuration/layouts.d.ts +2 -2
- package/dist/server/src/services/utils/populate.d.ts.map +1 -1
- package/package.json +10 -8
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
'use strict';
|
|
2
|
+
|
|
3
|
+
/**
|
|
4
|
+
* Converts a Strapi content-type UID into a safe MCP tool-name segment.
|
|
5
|
+
* `api::article.article` → `article`; `plugin::i18n.locale` → `plugin-i18n_locale`.
|
|
6
|
+
*/ const slugifyUidForMcpToolName = (uid)=>{
|
|
7
|
+
const [namespace, modelName] = uid.split('::');
|
|
8
|
+
const modelNameParts = modelName.split('.').map((part)=>part.toLowerCase());
|
|
9
|
+
if (namespace === 'api') {
|
|
10
|
+
return `${modelNameParts[0]}`;
|
|
11
|
+
}
|
|
12
|
+
return `${namespace.toLowerCase()}_${modelNameParts[0]}`;
|
|
13
|
+
};
|
|
14
|
+
/** Wraps a plain object into the dual-representation MCP tool return value (text + structuredContent). */ const ok = (structuredContent)=>({
|
|
15
|
+
content: [
|
|
16
|
+
{
|
|
17
|
+
type: 'text',
|
|
18
|
+
text: JSON.stringify(structuredContent)
|
|
19
|
+
}
|
|
20
|
+
],
|
|
21
|
+
structuredContent
|
|
22
|
+
});
|
|
23
|
+
/**
|
|
24
|
+
* Generates the `title` and `description` metadata for a derived MCP tool.
|
|
25
|
+
* Appends operation-specific notes for write/publish/unpublish/discard_draft operations.
|
|
26
|
+
*/ const describeTool = (params)=>{
|
|
27
|
+
const { apiID, uid, operation } = params;
|
|
28
|
+
const operationNoteByType = {
|
|
29
|
+
write: ' Creates or updates the single-type document. If no document exists, creates one; otherwise updates the existing draft.',
|
|
30
|
+
publish: ' Operates on an existing document by documentId and may return a different numeric id for the published version row.',
|
|
31
|
+
unpublish: ' Operates on an existing document by documentId and may return a different numeric id for the draft version row.',
|
|
32
|
+
discard_draft: ' Operates on an existing document by documentId; treat documentId as the stable identity.'
|
|
33
|
+
};
|
|
34
|
+
return {
|
|
35
|
+
title: `Content: ${apiID} — ${operation}`,
|
|
36
|
+
description: `Content-manager ${operation} for ${uid}.${operationNoteByType[operation] ?? ''}`
|
|
37
|
+
};
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
exports.describeTool = describeTool;
|
|
41
|
+
exports.ok = ok;
|
|
42
|
+
exports.slugifyUidForMcpToolName = slugifyUidForMcpToolName;
|
|
43
|
+
//# sourceMappingURL=utils.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.js","sources":["../../../server/src/mcp/utils.ts"],"sourcesContent":["import type { Modules } from '@strapi/types';\n\n/**\n * Converts a Strapi content-type UID into a safe MCP tool-name segment.\n * `api::article.article` → `article`; `plugin::i18n.locale` → `plugin-i18n_locale`.\n */\nexport const slugifyUidForMcpToolName = (uid: string): string => {\n const [namespace, modelName] = uid.split('::');\n const modelNameParts = modelName.split('.').map((part) => part.toLowerCase());\n if (namespace === 'api') {\n return `${modelNameParts[0]}`;\n }\n return `${namespace.toLowerCase()}_${modelNameParts[0]}`;\n};\n\n/** Wraps a plain object into the dual-representation MCP tool return value (text + structuredContent). */\nexport const ok = (\n structuredContent: Record<string, unknown>\n): Modules.MCP.McpToolHandlerReturn => ({\n content: [{ type: 'text', text: JSON.stringify(structuredContent) }],\n structuredContent,\n});\n\n/**\n * Generates the `title` and `description` metadata for a derived MCP tool.\n * Appends operation-specific notes for write/publish/unpublish/discard_draft operations.\n */\nexport const describeTool = (params: {\n apiID: string;\n uid: string;\n operation: string;\n}): { title: string; description: string } => {\n const { apiID, uid, operation } = params;\n const operationNoteByType: Partial<Record<string, string>> = {\n write:\n ' Creates or updates the single-type document. If no document exists, creates one; otherwise updates the existing draft.',\n publish:\n ' Operates on an existing document by documentId and may return a different numeric id for the published version row.',\n unpublish:\n ' Operates on an existing document by documentId and may return a different numeric id for the draft version row.',\n discard_draft:\n ' Operates on an existing document by documentId; treat documentId as the stable identity.',\n };\n\n return {\n title: `Content: ${apiID} — ${operation}`,\n description: `Content-manager ${operation} for ${uid}.${operationNoteByType[operation] ?? ''}`,\n };\n};\n"],"names":["slugifyUidForMcpToolName","uid","namespace","modelName","split","modelNameParts","map","part","toLowerCase","ok","structuredContent","content","type","text","JSON","stringify","describeTool","params","apiID","operation","operationNoteByType","write","publish","unpublish","discard_draft","title","description"],"mappings":";;AAEA;;;IAIO,MAAMA,wBAAAA,GAA2B,CAACC,GAAAA,GAAAA;AACvC,IAAA,MAAM,CAACC,SAAAA,EAAWC,SAAAA,CAAU,GAAGF,GAAAA,CAAIG,KAAK,CAAC,IAAA,CAAA;IACzC,MAAMC,cAAAA,GAAiBF,SAAAA,CAAUC,KAAK,CAAC,GAAA,CAAA,CAAKE,GAAG,CAAC,CAACC,IAAAA,GAASA,IAAAA,CAAKC,WAAW,EAAA,CAAA;AAC1E,IAAA,IAAIN,cAAc,KAAA,EAAO;AACvB,QAAA,OAAO,CAAA,EAAGG,cAAc,CAAC,CAAA,CAAE,CAAA,CAAE;AAC/B,IAAA;IACA,OAAO,CAAA,EAAGH,UAAUM,WAAW,EAAA,CAAG,CAAC,EAAEH,cAAc,CAAC,CAAA,CAAE,CAAA,CAAE;AAC1D;AAEA,2GACO,MAAMI,EAAAA,GAAK,CAChBC,qBACsC;QACtCC,OAAAA,EAAS;AAAC,YAAA;gBAAEC,IAAAA,EAAM,MAAA;gBAAQC,IAAAA,EAAMC,IAAAA,CAAKC,SAAS,CAACL,iBAAAA;AAAmB;AAAE,SAAA;AACpEA,QAAAA;AACF,KAAA;AAEA;;;IAIO,MAAMM,YAAAA,GAAe,CAACC,MAAAA,GAAAA;AAK3B,IAAA,MAAM,EAAEC,KAAK,EAAEjB,GAAG,EAAEkB,SAAS,EAAE,GAAGF,MAAAA;AAClC,IAAA,MAAMG,mBAAAA,GAAuD;QAC3DC,KAAAA,EACE,yHAAA;QACFC,OAAAA,EACE,sHAAA;QACFC,SAAAA,EACE,kHAAA;QACFC,aAAAA,EACE;AACJ,KAAA;IAEA,OAAO;AACLC,QAAAA,KAAAA,EAAO,CAAC,SAAS,EAAEP,KAAAA,CAAM,GAAG,EAAEC,SAAAA,CAAAA,CAAW;AACzCO,QAAAA,WAAAA,EAAa,CAAC,gBAAgB,EAAEP,SAAAA,CAAU,KAAK,EAAElB,GAAAA,CAAI,CAAC,EAAEmB,mBAAmB,CAACD,SAAAA,CAAU,IAAI,EAAA,CAAA;AAC5F,KAAA;AACF;;;;;;"}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Converts a Strapi content-type UID into a safe MCP tool-name segment.
|
|
3
|
+
* `api::article.article` → `article`; `plugin::i18n.locale` → `plugin-i18n_locale`.
|
|
4
|
+
*/ const slugifyUidForMcpToolName = (uid)=>{
|
|
5
|
+
const [namespace, modelName] = uid.split('::');
|
|
6
|
+
const modelNameParts = modelName.split('.').map((part)=>part.toLowerCase());
|
|
7
|
+
if (namespace === 'api') {
|
|
8
|
+
return `${modelNameParts[0]}`;
|
|
9
|
+
}
|
|
10
|
+
return `${namespace.toLowerCase()}_${modelNameParts[0]}`;
|
|
11
|
+
};
|
|
12
|
+
/** Wraps a plain object into the dual-representation MCP tool return value (text + structuredContent). */ const ok = (structuredContent)=>({
|
|
13
|
+
content: [
|
|
14
|
+
{
|
|
15
|
+
type: 'text',
|
|
16
|
+
text: JSON.stringify(structuredContent)
|
|
17
|
+
}
|
|
18
|
+
],
|
|
19
|
+
structuredContent
|
|
20
|
+
});
|
|
21
|
+
/**
|
|
22
|
+
* Generates the `title` and `description` metadata for a derived MCP tool.
|
|
23
|
+
* Appends operation-specific notes for write/publish/unpublish/discard_draft operations.
|
|
24
|
+
*/ const describeTool = (params)=>{
|
|
25
|
+
const { apiID, uid, operation } = params;
|
|
26
|
+
const operationNoteByType = {
|
|
27
|
+
write: ' Creates or updates the single-type document. If no document exists, creates one; otherwise updates the existing draft.',
|
|
28
|
+
publish: ' Operates on an existing document by documentId and may return a different numeric id for the published version row.',
|
|
29
|
+
unpublish: ' Operates on an existing document by documentId and may return a different numeric id for the draft version row.',
|
|
30
|
+
discard_draft: ' Operates on an existing document by documentId; treat documentId as the stable identity.'
|
|
31
|
+
};
|
|
32
|
+
return {
|
|
33
|
+
title: `Content: ${apiID} — ${operation}`,
|
|
34
|
+
description: `Content-manager ${operation} for ${uid}.${operationNoteByType[operation] ?? ''}`
|
|
35
|
+
};
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
export { describeTool, ok, slugifyUidForMcpToolName };
|
|
39
|
+
//# sourceMappingURL=utils.mjs.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"utils.mjs","sources":["../../../server/src/mcp/utils.ts"],"sourcesContent":["import type { Modules } from '@strapi/types';\n\n/**\n * Converts a Strapi content-type UID into a safe MCP tool-name segment.\n * `api::article.article` → `article`; `plugin::i18n.locale` → `plugin-i18n_locale`.\n */\nexport const slugifyUidForMcpToolName = (uid: string): string => {\n const [namespace, modelName] = uid.split('::');\n const modelNameParts = modelName.split('.').map((part) => part.toLowerCase());\n if (namespace === 'api') {\n return `${modelNameParts[0]}`;\n }\n return `${namespace.toLowerCase()}_${modelNameParts[0]}`;\n};\n\n/** Wraps a plain object into the dual-representation MCP tool return value (text + structuredContent). */\nexport const ok = (\n structuredContent: Record<string, unknown>\n): Modules.MCP.McpToolHandlerReturn => ({\n content: [{ type: 'text', text: JSON.stringify(structuredContent) }],\n structuredContent,\n});\n\n/**\n * Generates the `title` and `description` metadata for a derived MCP tool.\n * Appends operation-specific notes for write/publish/unpublish/discard_draft operations.\n */\nexport const describeTool = (params: {\n apiID: string;\n uid: string;\n operation: string;\n}): { title: string; description: string } => {\n const { apiID, uid, operation } = params;\n const operationNoteByType: Partial<Record<string, string>> = {\n write:\n ' Creates or updates the single-type document. If no document exists, creates one; otherwise updates the existing draft.',\n publish:\n ' Operates on an existing document by documentId and may return a different numeric id for the published version row.',\n unpublish:\n ' Operates on an existing document by documentId and may return a different numeric id for the draft version row.',\n discard_draft:\n ' Operates on an existing document by documentId; treat documentId as the stable identity.',\n };\n\n return {\n title: `Content: ${apiID} — ${operation}`,\n description: `Content-manager ${operation} for ${uid}.${operationNoteByType[operation] ?? ''}`,\n };\n};\n"],"names":["slugifyUidForMcpToolName","uid","namespace","modelName","split","modelNameParts","map","part","toLowerCase","ok","structuredContent","content","type","text","JSON","stringify","describeTool","params","apiID","operation","operationNoteByType","write","publish","unpublish","discard_draft","title","description"],"mappings":"AAEA;;;IAIO,MAAMA,wBAAAA,GAA2B,CAACC,GAAAA,GAAAA;AACvC,IAAA,MAAM,CAACC,SAAAA,EAAWC,SAAAA,CAAU,GAAGF,GAAAA,CAAIG,KAAK,CAAC,IAAA,CAAA;IACzC,MAAMC,cAAAA,GAAiBF,SAAAA,CAAUC,KAAK,CAAC,GAAA,CAAA,CAAKE,GAAG,CAAC,CAACC,IAAAA,GAASA,IAAAA,CAAKC,WAAW,EAAA,CAAA;AAC1E,IAAA,IAAIN,cAAc,KAAA,EAAO;AACvB,QAAA,OAAO,CAAA,EAAGG,cAAc,CAAC,CAAA,CAAE,CAAA,CAAE;AAC/B,IAAA;IACA,OAAO,CAAA,EAAGH,UAAUM,WAAW,EAAA,CAAG,CAAC,EAAEH,cAAc,CAAC,CAAA,CAAE,CAAA,CAAE;AAC1D;AAEA,2GACO,MAAMI,EAAAA,GAAK,CAChBC,qBACsC;QACtCC,OAAAA,EAAS;AAAC,YAAA;gBAAEC,IAAAA,EAAM,MAAA;gBAAQC,IAAAA,EAAMC,IAAAA,CAAKC,SAAS,CAACL,iBAAAA;AAAmB;AAAE,SAAA;AACpEA,QAAAA;AACF,KAAA;AAEA;;;IAIO,MAAMM,YAAAA,GAAe,CAACC,MAAAA,GAAAA;AAK3B,IAAA,MAAM,EAAEC,KAAK,EAAEjB,GAAG,EAAEkB,SAAS,EAAE,GAAGF,MAAAA;AAClC,IAAA,MAAMG,mBAAAA,GAAuD;QAC3DC,KAAAA,EACE,yHAAA;QACFC,OAAAA,EACE,sHAAA;QACFC,SAAAA,EACE,kHAAA;QACFC,aAAAA,EACE;AACJ,KAAA;IAEA,OAAO;AACLC,QAAAA,KAAAA,EAAO,CAAC,SAAS,EAAEP,KAAAA,CAAM,GAAG,EAAEC,SAAAA,CAAAA,CAAW;AACzCO,QAAAA,WAAAA,EAAa,CAAC,gBAAgB,EAAEP,SAAAA,CAAU,KAAK,EAAElB,GAAAA,CAAI,CAAC,EAAEmB,mBAAmB,CAACD,SAAAA,CAAU,IAAI,EAAA,CAAA;AAC5F,KAAA;AACF;;;;"}
|
|
@@ -69,7 +69,17 @@ const CONTENT_MANAGER_STATUS = {
|
|
|
69
69
|
};
|
|
70
70
|
var documentMetadata = (({ strapi })=>({
|
|
71
71
|
/**
|
|
72
|
-
* Returns available locales of a document for the current status
|
|
72
|
+
* Returns available locales of a document for the current status.
|
|
73
|
+
*
|
|
74
|
+
* The result is sorted with the default locale (as defined by the i18n plugin)
|
|
75
|
+
* at index 0 when present. This is the canonical-source invariant relied on by
|
|
76
|
+
* `useDocument.getInitialFormValues` in the admin: when creating a new locale
|
|
77
|
+
* draft, non-localized scalar/media values are inherited from
|
|
78
|
+
* `availableLocales[0]`. Putting the default locale first means inheritance
|
|
79
|
+
* stays predictable when sibling locales have drifted on non-localized fields
|
|
80
|
+
* (which can happen because the server only syncs non-localized fields at
|
|
81
|
+
* locale-creation time, not on subsequent updates — see
|
|
82
|
+
* `copyNonLocalizedFields` in document-service/internationalization.ts).
|
|
73
83
|
*/ async getAvailableLocales (uid, version, allVersions) {
|
|
74
84
|
// Group all versions by locale
|
|
75
85
|
const versionsByLocale = fp.groupBy('locale', allVersions);
|
|
@@ -95,8 +105,27 @@ var documentMetadata = (({ strapi })=>({
|
|
|
95
105
|
status: this.getStatus(draftVersion, otherVersions)
|
|
96
106
|
};
|
|
97
107
|
});
|
|
98
|
-
|
|
99
|
-
|
|
108
|
+
const filtered = mappingResult.filter(Boolean);
|
|
109
|
+
// Sort the default locale first so `availableLocales[0]` is the canonical
|
|
110
|
+
// source for non-localized field inheritance in the admin. Guarded so that
|
|
111
|
+
// we no-op if the i18n plugin or its locales service is unavailable.
|
|
112
|
+
let defaultLocaleCode;
|
|
113
|
+
try {
|
|
114
|
+
defaultLocaleCode = await strapi.plugin('i18n')?.service('locales')?.getDefaultLocale();
|
|
115
|
+
} catch {
|
|
116
|
+
// i18n plugin disabled or service errored — leave order untouched.
|
|
117
|
+
}
|
|
118
|
+
if (!defaultLocaleCode) {
|
|
119
|
+
return filtered;
|
|
120
|
+
}
|
|
121
|
+
// Stable partition: move the default locale entry to the front, keep the
|
|
122
|
+
// rest of the order intact.
|
|
123
|
+
const defaultEntries = filtered.filter((entry)=>entry.locale === defaultLocaleCode);
|
|
124
|
+
const otherEntries = filtered.filter((entry)=>entry.locale !== defaultLocaleCode);
|
|
125
|
+
return [
|
|
126
|
+
...defaultEntries,
|
|
127
|
+
...otherEntries
|
|
128
|
+
];
|
|
100
129
|
},
|
|
101
130
|
/**
|
|
102
131
|
* Returns available status of a document for the current locale
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"document-metadata.js","sources":["../../../server/src/services/document-metadata.ts"],"sourcesContent":["import { groupBy, pick, uniq } from 'lodash/fp';\n\nimport { async, contentTypes } from '@strapi/utils';\nimport type { Core, UID, Modules } from '@strapi/types';\n\nimport type { DocumentMetadata } from '../../../shared/contracts/collection-types';\n\nconst { getScalarAttributes, getMediaAttributes } = contentTypes;\n\nexport interface DocumentVersion {\n id: string | number;\n documentId: Modules.Documents.ID;\n locale?: string;\n localizations?: DocumentVersion[];\n updatedAt?: string | null | Date;\n publishedAt?: string | null | Date;\n}\n\n// Scalar fields that can be used in a DB `select`\nconst AVAILABLE_STATUS_SCALAR_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n// Relation populate shared by both the fast path and the full path\nconst AVAILABLE_STATUS_POPULATE = {\n createdBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n updatedBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n};\n// All fields to pick from a hydrated result (scalars + populated relations + virtual)\nconst AVAILABLE_STATUS_FIELDS = [\n ...AVAILABLE_STATUS_SCALAR_FIELDS,\n 'createdBy',\n 'updatedBy',\n 'status',\n];\nconst AVAILABLE_LOCALES_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n\n/** Returns a DB filter that matches the opposite publish status. */\nconst oppositePublishStatus = (publishedAt: unknown) =>\n publishedAt !== null ? { $null: true } : { $notNull: true };\n\nconst CONTENT_MANAGER_STATUS = {\n PUBLISHED: 'published',\n DRAFT: 'draft',\n MODIFIED: 'modified',\n};\n\n/**\n * Controls the metadata properties to be returned\n *\n * If `availableLocales` is set to `true` (default), the returned metadata will include\n * the available locales of the document for its current status.\n *\n * If `availableStatus` is set to `true` (default), the returned metadata will include\n * the available status of the document for its current locale.\n */\nexport interface GetMetadataOptions {\n availableLocales?: boolean;\n availableStatus?: boolean;\n}\n\n/**\n * Checks if the provided document version has been modified after all other versions.\n */\nconst getIsVersionLatestModification = (\n version?: DocumentVersion,\n otherVersion?: DocumentVersion\n): boolean => {\n if (!version || !version.updatedAt) {\n return false;\n }\n\n const versionUpdatedAt = version?.updatedAt ? new Date(version.updatedAt).getTime() : 0;\n\n const otherUpdatedAt = otherVersion?.updatedAt ? new Date(otherVersion.updatedAt).getTime() : 0;\n\n return versionUpdatedAt > otherUpdatedAt;\n};\n\nexport default ({ strapi }: { strapi: Core.Strapi }) => ({\n /**\n * Returns available locales of a document for the current status\n */\n async getAvailableLocales(\n uid: UID.ContentType,\n version: DocumentVersion,\n allVersions: DocumentVersion[]\n ) {\n // Group all versions by locale\n const versionsByLocale = groupBy('locale', allVersions);\n\n // Delete the current locale\n if (version.locale) {\n delete versionsByLocale[version.locale];\n }\n\n // For each locale, get the ones with the same status\n // There will not be a draft and a version counterpart if the content\n // type does not have draft and publish\n const model = strapi.getModel(uid);\n\n const mappingResult = await async.map(\n Object.values(versionsByLocale),\n async (localeVersions: DocumentVersion[]) => {\n if (!contentTypes.hasDraftAndPublish(model)) {\n return localeVersions[0];\n }\n\n const draftVersion = localeVersions.find((v) => v.publishedAt === null);\n const otherVersions = localeVersions.filter((v) => v.id !== draftVersion?.id);\n\n if (!draftVersion) {\n return;\n }\n\n return {\n ...draftVersion,\n status: this.getStatus(draftVersion, otherVersions as any),\n };\n }\n );\n\n return (\n mappingResult\n // Filter just in case there is a document with no drafts\n .filter(Boolean) as unknown as DocumentMetadata['availableLocales']\n );\n },\n\n /**\n * Returns available status of a document for the current locale\n */\n getAvailableStatus(version: DocumentVersion, allVersions: DocumentVersion[]) {\n // Find the other status of the document\n const status =\n version.publishedAt !== null\n ? CONTENT_MANAGER_STATUS.DRAFT\n : CONTENT_MANAGER_STATUS.PUBLISHED;\n\n // Get version that match the current locale and not match the current status\n const availableStatus = allVersions.find((v) => {\n const matchLocale = v.locale === version.locale;\n const matchStatus = status === 'published' ? v.publishedAt !== null : v.publishedAt === null;\n return matchLocale && matchStatus;\n });\n\n if (!availableStatus) return availableStatus;\n\n // Pick status fields (at fields, status, by fields), use lodash fp\n return pick(AVAILABLE_STATUS_FIELDS, availableStatus);\n },\n\n /**\n * Get the available status of many documents, useful for batch operations\n * @param uid\n * @param documents\n * @returns\n */\n async getManyAvailableStatus(uid: UID.ContentType, documents: DocumentVersion[]) {\n if (!documents.length) return [];\n\n // The status and locale of all documents should be the same\n const status = documents[0].publishedAt !== null ? 'published' : 'draft';\n const locales = documents.map((d) => d.locale).filter(Boolean);\n\n const where: Record<string, any> = {\n documentId: { $in: documents.map((d) => d.documentId).filter(Boolean) },\n publishedAt: { $null: status === 'published' },\n };\n\n // If there is any locale to filter (if i18n is enabled)\n if (locales.length) {\n where.locale = { $in: locales };\n }\n\n return strapi.query(uid).findMany({\n where,\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n });\n },\n\n getStatus(version: DocumentVersion, otherDocumentStatuses?: DocumentMetadata['availableStatus']) {\n let draftVersion: DocumentVersion | undefined;\n let publishedVersion: DocumentVersion | undefined;\n\n if (version.publishedAt) {\n publishedVersion = version;\n } else {\n draftVersion = version;\n }\n\n const otherVersion = otherDocumentStatuses?.at(0);\n if (otherVersion?.publishedAt) {\n publishedVersion = otherVersion;\n } else if (otherVersion) {\n draftVersion = otherVersion;\n }\n\n if (!draftVersion) return CONTENT_MANAGER_STATUS.PUBLISHED;\n if (!publishedVersion) return CONTENT_MANAGER_STATUS.DRAFT;\n\n /*\n * The document is modified if the draft version has been updated more\n * recently than the published version.\n */\n const isDraftModified = getIsVersionLatestModification(draftVersion, publishedVersion);\n return isDraftModified ? CONTENT_MANAGER_STATUS.MODIFIED : CONTENT_MANAGER_STATUS.PUBLISHED;\n },\n\n // TODO is it necessary to return metadata on every page of the CM\n // We could refactor this so the locales are only loaded when they're\n // needed. e.g. in the bulk locale action modal.\n async getMetadata(\n uid: UID.ContentType,\n version: DocumentVersion,\n { availableLocales = true, availableStatus = true }: GetMetadataOptions = {}\n ) {\n const model = strapi.getModel(uid);\n const hasDnP = contentTypes.hasDraftAndPublish(model);\n const isLocalized = (model.pluginOptions?.i18n as any)?.localized === true;\n\n if (!availableLocales && !availableStatus) {\n // Nothing to compute.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n if (!isLocalized && !hasDnP) {\n // If there are no locales and no draft/publish, there's only ever 1 version of any document.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n\n const onlyStatusIsRelevant = hasDnP && (!isLocalized || !availableLocales);\n if (onlyStatusIsRelevant) {\n const otherVersion = availableStatus\n ? await strapi.db.query(uid).findOne({\n where: {\n documentId: version.documentId,\n ...(version.locale ? { locale: version.locale } : {}),\n publishedAt: oppositePublishStatus(version.publishedAt),\n },\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n populate: AVAILABLE_STATUS_POPULATE,\n })\n : null;\n return {\n availableLocales: [],\n availableStatus: otherVersion ? [pick(AVAILABLE_STATUS_FIELDS, otherVersion)] : [],\n versions: [] as DocumentVersion[],\n };\n }\n\n // Full path for localized content types\n // TODO: Ignore publishedAt if availableStatus=false, and ignore locale if\n // i18n is disabled\n\n // Include non-translatable scalar and media fields in availableLocales for i18n prefilling\n let nonLocalizedFields: string[] = [];\n let nonLocalizedMediaFields: string[] = [];\n try {\n const i18nPlugin = strapi.plugin('i18n');\n if (i18nPlugin) {\n const i18nService = i18nPlugin.service('content-types');\n if (i18nService?.getNonLocalizedAttributes) {\n if (model?.attributes) {\n const allNonLocalized = i18nService.getNonLocalizedAttributes(model);\n // Get scalar and media attributes separately\n const scalarAttrs = getScalarAttributes(model);\n const mediaAttrs = getMediaAttributes(model);\n\n // Separate scalar fields (can be in fields array) from media fields (need to be populated)\n nonLocalizedFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && scalarAttrs.includes(field)\n );\n nonLocalizedMediaFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && mediaAttrs.includes(field)\n );\n }\n }\n }\n } catch (error) {\n // i18n plugin might not be enabled or might error, ignore silently\n }\n\n // Build populate object for non-localized media fields\n const mediaPopulate = nonLocalizedMediaFields.reduce(\n (acc, field) => {\n acc[field] = {\n populate: {\n folder: true,\n },\n };\n return acc;\n },\n {} as Record<string, { populate: { folder: boolean } }>\n );\n\n const params = {\n populate: {\n ...mediaPopulate,\n ...AVAILABLE_STATUS_POPULATE,\n },\n fields: uniq([...AVAILABLE_LOCALES_FIELDS, ...nonLocalizedFields]),\n filters: {\n documentId: version.documentId,\n },\n };\n\n const dbParams = strapi.get('query-params').transform(uid, params);\n const versions = await strapi.db.query(uid).findMany(dbParams);\n\n // TODO: Remove use of available locales and use localizations instead\n const availableLocalesResult = availableLocales\n ? await this.getAvailableLocales(uid, version, versions)\n : [];\n\n const availableStatusResult = availableStatus\n ? this.getAvailableStatus(version, versions)\n : null;\n\n return {\n availableLocales: availableLocalesResult,\n availableStatus: availableStatusResult ? [availableStatusResult] : [],\n versions,\n };\n },\n\n /**\n * Returns associated metadata of a document:\n * - Available locales of the document for the current status\n * - Available status of the document for the current locale\n */\n async formatDocumentWithMetadata(\n uid: UID.ContentType,\n document: DocumentVersion,\n opts: GetMetadataOptions = {}\n ) {\n if (!document) {\n return {\n data: document,\n meta: {\n availableLocales: [],\n availableStatus: [],\n },\n };\n }\n\n const hasDraftAndPublish = contentTypes.hasDraftAndPublish(strapi.getModel(uid));\n\n // Ignore available status if the content type does not have draft and publish\n if (!hasDraftAndPublish) {\n opts.availableStatus = false;\n }\n\n const { versions, ...meta } = await this.getMetadata(uid, document, opts);\n\n // Populate localization statuses\n if (document.localizations?.length) {\n document.localizations = document.localizations.map((d) => {\n // Find the counterpart version (same documentId + locale, opposite publishedAt) from\n // the already-fetched versions array, avoiding an extra DB query.\n const counterpart = versions.find(\n (v) =>\n v.documentId === d.documentId &&\n v.locale === d.locale &&\n (d.publishedAt === null) !== (v.publishedAt === null)\n );\n return {\n ...d,\n status: this.getStatus(d, counterpart ? [counterpart] : []),\n };\n });\n }\n\n return {\n data: {\n ...document,\n // Add status to the document only if draft and publish is enabled\n status: hasDraftAndPublish\n ? this.getStatus(document, meta.availableStatus as any)\n : undefined,\n },\n meta,\n };\n },\n});\n"],"names":["getScalarAttributes","getMediaAttributes","contentTypes","AVAILABLE_STATUS_SCALAR_FIELDS","AVAILABLE_STATUS_POPULATE","createdBy","select","updatedBy","AVAILABLE_STATUS_FIELDS","AVAILABLE_LOCALES_FIELDS","oppositePublishStatus","publishedAt","$null","$notNull","CONTENT_MANAGER_STATUS","PUBLISHED","DRAFT","MODIFIED","getIsVersionLatestModification","version","otherVersion","updatedAt","versionUpdatedAt","Date","getTime","otherUpdatedAt","strapi","getAvailableLocales","uid","allVersions","versionsByLocale","groupBy","locale","model","getModel","mappingResult","async","map","Object","values","localeVersions","hasDraftAndPublish","draftVersion","find","v","otherVersions","filter","id","status","getStatus","Boolean","getAvailableStatus","availableStatus","matchLocale","matchStatus","pick","getManyAvailableStatus","documents","length","locales","d","where","documentId","$in","query","findMany","otherDocumentStatuses","publishedVersion","at","isDraftModified","getMetadata","availableLocales","hasDnP","isLocalized","pluginOptions","i18n","localized","versions","onlyStatusIsRelevant","db","findOne","populate","nonLocalizedFields","nonLocalizedMediaFields","i18nPlugin","plugin","i18nService","service","getNonLocalizedAttributes","attributes","allNonLocalized","scalarAttrs","mediaAttrs","field","includes","error","mediaPopulate","reduce","acc","folder","params","fields","uniq","filters","dbParams","get","transform","availableLocalesResult","availableStatusResult","formatDocumentWithMetadata","document","opts","data","meta","localizations","counterpart","undefined"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"document-metadata.js","sources":["../../../server/src/services/document-metadata.ts"],"sourcesContent":["import { groupBy, pick, uniq } from 'lodash/fp';\n\nimport { async, contentTypes } from '@strapi/utils';\nimport type { Core, UID, Modules } from '@strapi/types';\n\nimport type { DocumentMetadata } from '../../../shared/contracts/collection-types';\n\nconst { getScalarAttributes, getMediaAttributes } = contentTypes;\n\nexport interface DocumentVersion {\n id: string | number;\n documentId: Modules.Documents.ID;\n locale?: string;\n localizations?: DocumentVersion[];\n updatedAt?: string | null | Date;\n publishedAt?: string | null | Date;\n}\n\n// Scalar fields that can be used in a DB `select`\nconst AVAILABLE_STATUS_SCALAR_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n// Relation populate shared by both the fast path and the full path\nconst AVAILABLE_STATUS_POPULATE = {\n createdBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n updatedBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n};\n// All fields to pick from a hydrated result (scalars + populated relations + virtual)\nconst AVAILABLE_STATUS_FIELDS = [\n ...AVAILABLE_STATUS_SCALAR_FIELDS,\n 'createdBy',\n 'updatedBy',\n 'status',\n];\nconst AVAILABLE_LOCALES_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n\n/** Returns a DB filter that matches the opposite publish status. */\nconst oppositePublishStatus = (publishedAt: unknown) =>\n publishedAt !== null ? { $null: true } : { $notNull: true };\n\nconst CONTENT_MANAGER_STATUS = {\n PUBLISHED: 'published',\n DRAFT: 'draft',\n MODIFIED: 'modified',\n};\n\n/**\n * Controls the metadata properties to be returned\n *\n * If `availableLocales` is set to `true` (default), the returned metadata will include\n * the available locales of the document for its current status.\n *\n * If `availableStatus` is set to `true` (default), the returned metadata will include\n * the available status of the document for its current locale.\n */\nexport interface GetMetadataOptions {\n availableLocales?: boolean;\n availableStatus?: boolean;\n}\n\n/**\n * Checks if the provided document version has been modified after all other versions.\n */\nconst getIsVersionLatestModification = (\n version?: DocumentVersion,\n otherVersion?: DocumentVersion\n): boolean => {\n if (!version || !version.updatedAt) {\n return false;\n }\n\n const versionUpdatedAt = version?.updatedAt ? new Date(version.updatedAt).getTime() : 0;\n\n const otherUpdatedAt = otherVersion?.updatedAt ? new Date(otherVersion.updatedAt).getTime() : 0;\n\n return versionUpdatedAt > otherUpdatedAt;\n};\n\nexport default ({ strapi }: { strapi: Core.Strapi }) => ({\n /**\n * Returns available locales of a document for the current status.\n *\n * The result is sorted with the default locale (as defined by the i18n plugin)\n * at index 0 when present. This is the canonical-source invariant relied on by\n * `useDocument.getInitialFormValues` in the admin: when creating a new locale\n * draft, non-localized scalar/media values are inherited from\n * `availableLocales[0]`. Putting the default locale first means inheritance\n * stays predictable when sibling locales have drifted on non-localized fields\n * (which can happen because the server only syncs non-localized fields at\n * locale-creation time, not on subsequent updates — see\n * `copyNonLocalizedFields` in document-service/internationalization.ts).\n */\n async getAvailableLocales(\n uid: UID.ContentType,\n version: DocumentVersion,\n allVersions: DocumentVersion[]\n ) {\n // Group all versions by locale\n const versionsByLocale = groupBy('locale', allVersions);\n\n // Delete the current locale\n if (version.locale) {\n delete versionsByLocale[version.locale];\n }\n\n // For each locale, get the ones with the same status\n // There will not be a draft and a version counterpart if the content\n // type does not have draft and publish\n const model = strapi.getModel(uid);\n\n const mappingResult = await async.map(\n Object.values(versionsByLocale),\n async (localeVersions: DocumentVersion[]) => {\n if (!contentTypes.hasDraftAndPublish(model)) {\n return localeVersions[0];\n }\n\n const draftVersion = localeVersions.find((v) => v.publishedAt === null);\n const otherVersions = localeVersions.filter((v) => v.id !== draftVersion?.id);\n\n if (!draftVersion) {\n return;\n }\n\n return {\n ...draftVersion,\n status: this.getStatus(draftVersion, otherVersions as any),\n };\n }\n );\n\n const filtered = mappingResult.filter(Boolean) as unknown as NonNullable<\n DocumentMetadata['availableLocales']\n >;\n\n // Sort the default locale first so `availableLocales[0]` is the canonical\n // source for non-localized field inheritance in the admin. Guarded so that\n // we no-op if the i18n plugin or its locales service is unavailable.\n let defaultLocaleCode: string | undefined;\n try {\n defaultLocaleCode = await strapi.plugin('i18n')?.service('locales')?.getDefaultLocale();\n } catch {\n // i18n plugin disabled or service errored — leave order untouched.\n }\n\n if (!defaultLocaleCode) {\n return filtered as DocumentMetadata['availableLocales'];\n }\n\n // Stable partition: move the default locale entry to the front, keep the\n // rest of the order intact.\n const defaultEntries = filtered.filter((entry) => entry.locale === defaultLocaleCode);\n const otherEntries = filtered.filter((entry) => entry.locale !== defaultLocaleCode);\n\n return [...defaultEntries, ...otherEntries] as DocumentMetadata['availableLocales'];\n },\n\n /**\n * Returns available status of a document for the current locale\n */\n getAvailableStatus(version: DocumentVersion, allVersions: DocumentVersion[]) {\n // Find the other status of the document\n const status =\n version.publishedAt !== null\n ? CONTENT_MANAGER_STATUS.DRAFT\n : CONTENT_MANAGER_STATUS.PUBLISHED;\n\n // Get version that match the current locale and not match the current status\n const availableStatus = allVersions.find((v) => {\n const matchLocale = v.locale === version.locale;\n const matchStatus = status === 'published' ? v.publishedAt !== null : v.publishedAt === null;\n return matchLocale && matchStatus;\n });\n\n if (!availableStatus) return availableStatus;\n\n // Pick status fields (at fields, status, by fields), use lodash fp\n return pick(AVAILABLE_STATUS_FIELDS, availableStatus);\n },\n\n /**\n * Get the available status of many documents, useful for batch operations\n * @param uid\n * @param documents\n * @returns\n */\n async getManyAvailableStatus(uid: UID.ContentType, documents: DocumentVersion[]) {\n if (!documents.length) return [];\n\n // The status and locale of all documents should be the same\n const status = documents[0].publishedAt !== null ? 'published' : 'draft';\n const locales = documents.map((d) => d.locale).filter(Boolean);\n\n const where: Record<string, any> = {\n documentId: { $in: documents.map((d) => d.documentId).filter(Boolean) },\n publishedAt: { $null: status === 'published' },\n };\n\n // If there is any locale to filter (if i18n is enabled)\n if (locales.length) {\n where.locale = { $in: locales };\n }\n\n return strapi.query(uid).findMany({\n where,\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n });\n },\n\n getStatus(version: DocumentVersion, otherDocumentStatuses?: DocumentMetadata['availableStatus']) {\n let draftVersion: DocumentVersion | undefined;\n let publishedVersion: DocumentVersion | undefined;\n\n if (version.publishedAt) {\n publishedVersion = version;\n } else {\n draftVersion = version;\n }\n\n const otherVersion = otherDocumentStatuses?.at(0);\n if (otherVersion?.publishedAt) {\n publishedVersion = otherVersion;\n } else if (otherVersion) {\n draftVersion = otherVersion;\n }\n\n if (!draftVersion) return CONTENT_MANAGER_STATUS.PUBLISHED;\n if (!publishedVersion) return CONTENT_MANAGER_STATUS.DRAFT;\n\n /*\n * The document is modified if the draft version has been updated more\n * recently than the published version.\n */\n const isDraftModified = getIsVersionLatestModification(draftVersion, publishedVersion);\n return isDraftModified ? CONTENT_MANAGER_STATUS.MODIFIED : CONTENT_MANAGER_STATUS.PUBLISHED;\n },\n\n // TODO is it necessary to return metadata on every page of the CM\n // We could refactor this so the locales are only loaded when they're\n // needed. e.g. in the bulk locale action modal.\n async getMetadata(\n uid: UID.ContentType,\n version: DocumentVersion,\n { availableLocales = true, availableStatus = true }: GetMetadataOptions = {}\n ) {\n const model = strapi.getModel(uid);\n const hasDnP = contentTypes.hasDraftAndPublish(model);\n const isLocalized = (model.pluginOptions?.i18n as any)?.localized === true;\n\n if (!availableLocales && !availableStatus) {\n // Nothing to compute.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n if (!isLocalized && !hasDnP) {\n // If there are no locales and no draft/publish, there's only ever 1 version of any document.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n\n const onlyStatusIsRelevant = hasDnP && (!isLocalized || !availableLocales);\n if (onlyStatusIsRelevant) {\n const otherVersion = availableStatus\n ? await strapi.db.query(uid).findOne({\n where: {\n documentId: version.documentId,\n ...(version.locale ? { locale: version.locale } : {}),\n publishedAt: oppositePublishStatus(version.publishedAt),\n },\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n populate: AVAILABLE_STATUS_POPULATE,\n })\n : null;\n return {\n availableLocales: [],\n availableStatus: otherVersion ? [pick(AVAILABLE_STATUS_FIELDS, otherVersion)] : [],\n versions: [] as DocumentVersion[],\n };\n }\n\n // Full path for localized content types\n // TODO: Ignore publishedAt if availableStatus=false, and ignore locale if\n // i18n is disabled\n\n // Include non-translatable scalar and media fields in availableLocales for i18n prefilling\n let nonLocalizedFields: string[] = [];\n let nonLocalizedMediaFields: string[] = [];\n try {\n const i18nPlugin = strapi.plugin('i18n');\n if (i18nPlugin) {\n const i18nService = i18nPlugin.service('content-types');\n if (i18nService?.getNonLocalizedAttributes) {\n if (model?.attributes) {\n const allNonLocalized = i18nService.getNonLocalizedAttributes(model);\n // Get scalar and media attributes separately\n const scalarAttrs = getScalarAttributes(model);\n const mediaAttrs = getMediaAttributes(model);\n\n // Separate scalar fields (can be in fields array) from media fields (need to be populated)\n nonLocalizedFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && scalarAttrs.includes(field)\n );\n nonLocalizedMediaFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && mediaAttrs.includes(field)\n );\n }\n }\n }\n } catch (error) {\n // i18n plugin might not be enabled or might error, ignore silently\n }\n\n // Build populate object for non-localized media fields\n const mediaPopulate = nonLocalizedMediaFields.reduce(\n (acc, field) => {\n acc[field] = {\n populate: {\n folder: true,\n },\n };\n return acc;\n },\n {} as Record<string, { populate: { folder: boolean } }>\n );\n\n const params = {\n populate: {\n ...mediaPopulate,\n ...AVAILABLE_STATUS_POPULATE,\n },\n fields: uniq([...AVAILABLE_LOCALES_FIELDS, ...nonLocalizedFields]),\n filters: {\n documentId: version.documentId,\n },\n };\n\n const dbParams = strapi.get('query-params').transform(uid, params);\n const versions = await strapi.db.query(uid).findMany(dbParams);\n\n // TODO: Remove use of available locales and use localizations instead\n const availableLocalesResult = availableLocales\n ? await this.getAvailableLocales(uid, version, versions)\n : [];\n\n const availableStatusResult = availableStatus\n ? this.getAvailableStatus(version, versions)\n : null;\n\n return {\n availableLocales: availableLocalesResult,\n availableStatus: availableStatusResult ? [availableStatusResult] : [],\n versions,\n };\n },\n\n /**\n * Returns associated metadata of a document:\n * - Available locales of the document for the current status\n * - Available status of the document for the current locale\n */\n async formatDocumentWithMetadata(\n uid: UID.ContentType,\n document: DocumentVersion,\n opts: GetMetadataOptions = {}\n ) {\n if (!document) {\n return {\n data: document,\n meta: {\n availableLocales: [],\n availableStatus: [],\n },\n };\n }\n\n const hasDraftAndPublish = contentTypes.hasDraftAndPublish(strapi.getModel(uid));\n\n // Ignore available status if the content type does not have draft and publish\n if (!hasDraftAndPublish) {\n opts.availableStatus = false;\n }\n\n const { versions, ...meta } = await this.getMetadata(uid, document, opts);\n\n // Populate localization statuses\n if (document.localizations?.length) {\n document.localizations = document.localizations.map((d) => {\n // Find the counterpart version (same documentId + locale, opposite publishedAt) from\n // the already-fetched versions array, avoiding an extra DB query.\n const counterpart = versions.find(\n (v) =>\n v.documentId === d.documentId &&\n v.locale === d.locale &&\n (d.publishedAt === null) !== (v.publishedAt === null)\n );\n return {\n ...d,\n status: this.getStatus(d, counterpart ? [counterpart] : []),\n };\n });\n }\n\n return {\n data: {\n ...document,\n // Add status to the document only if draft and publish is enabled\n status: hasDraftAndPublish\n ? this.getStatus(document, meta.availableStatus as any)\n : undefined,\n },\n meta,\n };\n },\n});\n"],"names":["getScalarAttributes","getMediaAttributes","contentTypes","AVAILABLE_STATUS_SCALAR_FIELDS","AVAILABLE_STATUS_POPULATE","createdBy","select","updatedBy","AVAILABLE_STATUS_FIELDS","AVAILABLE_LOCALES_FIELDS","oppositePublishStatus","publishedAt","$null","$notNull","CONTENT_MANAGER_STATUS","PUBLISHED","DRAFT","MODIFIED","getIsVersionLatestModification","version","otherVersion","updatedAt","versionUpdatedAt","Date","getTime","otherUpdatedAt","strapi","getAvailableLocales","uid","allVersions","versionsByLocale","groupBy","locale","model","getModel","mappingResult","async","map","Object","values","localeVersions","hasDraftAndPublish","draftVersion","find","v","otherVersions","filter","id","status","getStatus","filtered","Boolean","defaultLocaleCode","plugin","service","getDefaultLocale","defaultEntries","entry","otherEntries","getAvailableStatus","availableStatus","matchLocale","matchStatus","pick","getManyAvailableStatus","documents","length","locales","d","where","documentId","$in","query","findMany","otherDocumentStatuses","publishedVersion","at","isDraftModified","getMetadata","availableLocales","hasDnP","isLocalized","pluginOptions","i18n","localized","versions","onlyStatusIsRelevant","db","findOne","populate","nonLocalizedFields","nonLocalizedMediaFields","i18nPlugin","i18nService","getNonLocalizedAttributes","attributes","allNonLocalized","scalarAttrs","mediaAttrs","field","includes","error","mediaPopulate","reduce","acc","folder","params","fields","uniq","filters","dbParams","get","transform","availableLocalesResult","availableStatusResult","formatDocumentWithMetadata","document","opts","data","meta","localizations","counterpart","undefined"],"mappings":"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"}
|
|
@@ -67,7 +67,17 @@ const CONTENT_MANAGER_STATUS = {
|
|
|
67
67
|
};
|
|
68
68
|
var documentMetadata = (({ strapi })=>({
|
|
69
69
|
/**
|
|
70
|
-
* Returns available locales of a document for the current status
|
|
70
|
+
* Returns available locales of a document for the current status.
|
|
71
|
+
*
|
|
72
|
+
* The result is sorted with the default locale (as defined by the i18n plugin)
|
|
73
|
+
* at index 0 when present. This is the canonical-source invariant relied on by
|
|
74
|
+
* `useDocument.getInitialFormValues` in the admin: when creating a new locale
|
|
75
|
+
* draft, non-localized scalar/media values are inherited from
|
|
76
|
+
* `availableLocales[0]`. Putting the default locale first means inheritance
|
|
77
|
+
* stays predictable when sibling locales have drifted on non-localized fields
|
|
78
|
+
* (which can happen because the server only syncs non-localized fields at
|
|
79
|
+
* locale-creation time, not on subsequent updates — see
|
|
80
|
+
* `copyNonLocalizedFields` in document-service/internationalization.ts).
|
|
71
81
|
*/ async getAvailableLocales (uid, version, allVersions) {
|
|
72
82
|
// Group all versions by locale
|
|
73
83
|
const versionsByLocale = groupBy('locale', allVersions);
|
|
@@ -93,8 +103,27 @@ var documentMetadata = (({ strapi })=>({
|
|
|
93
103
|
status: this.getStatus(draftVersion, otherVersions)
|
|
94
104
|
};
|
|
95
105
|
});
|
|
96
|
-
|
|
97
|
-
|
|
106
|
+
const filtered = mappingResult.filter(Boolean);
|
|
107
|
+
// Sort the default locale first so `availableLocales[0]` is the canonical
|
|
108
|
+
// source for non-localized field inheritance in the admin. Guarded so that
|
|
109
|
+
// we no-op if the i18n plugin or its locales service is unavailable.
|
|
110
|
+
let defaultLocaleCode;
|
|
111
|
+
try {
|
|
112
|
+
defaultLocaleCode = await strapi.plugin('i18n')?.service('locales')?.getDefaultLocale();
|
|
113
|
+
} catch {
|
|
114
|
+
// i18n plugin disabled or service errored — leave order untouched.
|
|
115
|
+
}
|
|
116
|
+
if (!defaultLocaleCode) {
|
|
117
|
+
return filtered;
|
|
118
|
+
}
|
|
119
|
+
// Stable partition: move the default locale entry to the front, keep the
|
|
120
|
+
// rest of the order intact.
|
|
121
|
+
const defaultEntries = filtered.filter((entry)=>entry.locale === defaultLocaleCode);
|
|
122
|
+
const otherEntries = filtered.filter((entry)=>entry.locale !== defaultLocaleCode);
|
|
123
|
+
return [
|
|
124
|
+
...defaultEntries,
|
|
125
|
+
...otherEntries
|
|
126
|
+
];
|
|
98
127
|
},
|
|
99
128
|
/**
|
|
100
129
|
* Returns available status of a document for the current locale
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"document-metadata.mjs","sources":["../../../server/src/services/document-metadata.ts"],"sourcesContent":["import { groupBy, pick, uniq } from 'lodash/fp';\n\nimport { async, contentTypes } from '@strapi/utils';\nimport type { Core, UID, Modules } from '@strapi/types';\n\nimport type { DocumentMetadata } from '../../../shared/contracts/collection-types';\n\nconst { getScalarAttributes, getMediaAttributes } = contentTypes;\n\nexport interface DocumentVersion {\n id: string | number;\n documentId: Modules.Documents.ID;\n locale?: string;\n localizations?: DocumentVersion[];\n updatedAt?: string | null | Date;\n publishedAt?: string | null | Date;\n}\n\n// Scalar fields that can be used in a DB `select`\nconst AVAILABLE_STATUS_SCALAR_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n// Relation populate shared by both the fast path and the full path\nconst AVAILABLE_STATUS_POPULATE = {\n createdBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n updatedBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n};\n// All fields to pick from a hydrated result (scalars + populated relations + virtual)\nconst AVAILABLE_STATUS_FIELDS = [\n ...AVAILABLE_STATUS_SCALAR_FIELDS,\n 'createdBy',\n 'updatedBy',\n 'status',\n];\nconst AVAILABLE_LOCALES_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n\n/** Returns a DB filter that matches the opposite publish status. */\nconst oppositePublishStatus = (publishedAt: unknown) =>\n publishedAt !== null ? { $null: true } : { $notNull: true };\n\nconst CONTENT_MANAGER_STATUS = {\n PUBLISHED: 'published',\n DRAFT: 'draft',\n MODIFIED: 'modified',\n};\n\n/**\n * Controls the metadata properties to be returned\n *\n * If `availableLocales` is set to `true` (default), the returned metadata will include\n * the available locales of the document for its current status.\n *\n * If `availableStatus` is set to `true` (default), the returned metadata will include\n * the available status of the document for its current locale.\n */\nexport interface GetMetadataOptions {\n availableLocales?: boolean;\n availableStatus?: boolean;\n}\n\n/**\n * Checks if the provided document version has been modified after all other versions.\n */\nconst getIsVersionLatestModification = (\n version?: DocumentVersion,\n otherVersion?: DocumentVersion\n): boolean => {\n if (!version || !version.updatedAt) {\n return false;\n }\n\n const versionUpdatedAt = version?.updatedAt ? new Date(version.updatedAt).getTime() : 0;\n\n const otherUpdatedAt = otherVersion?.updatedAt ? new Date(otherVersion.updatedAt).getTime() : 0;\n\n return versionUpdatedAt > otherUpdatedAt;\n};\n\nexport default ({ strapi }: { strapi: Core.Strapi }) => ({\n /**\n * Returns available locales of a document for the current status\n */\n async getAvailableLocales(\n uid: UID.ContentType,\n version: DocumentVersion,\n allVersions: DocumentVersion[]\n ) {\n // Group all versions by locale\n const versionsByLocale = groupBy('locale', allVersions);\n\n // Delete the current locale\n if (version.locale) {\n delete versionsByLocale[version.locale];\n }\n\n // For each locale, get the ones with the same status\n // There will not be a draft and a version counterpart if the content\n // type does not have draft and publish\n const model = strapi.getModel(uid);\n\n const mappingResult = await async.map(\n Object.values(versionsByLocale),\n async (localeVersions: DocumentVersion[]) => {\n if (!contentTypes.hasDraftAndPublish(model)) {\n return localeVersions[0];\n }\n\n const draftVersion = localeVersions.find((v) => v.publishedAt === null);\n const otherVersions = localeVersions.filter((v) => v.id !== draftVersion?.id);\n\n if (!draftVersion) {\n return;\n }\n\n return {\n ...draftVersion,\n status: this.getStatus(draftVersion, otherVersions as any),\n };\n }\n );\n\n return (\n mappingResult\n // Filter just in case there is a document with no drafts\n .filter(Boolean) as unknown as DocumentMetadata['availableLocales']\n );\n },\n\n /**\n * Returns available status of a document for the current locale\n */\n getAvailableStatus(version: DocumentVersion, allVersions: DocumentVersion[]) {\n // Find the other status of the document\n const status =\n version.publishedAt !== null\n ? CONTENT_MANAGER_STATUS.DRAFT\n : CONTENT_MANAGER_STATUS.PUBLISHED;\n\n // Get version that match the current locale and not match the current status\n const availableStatus = allVersions.find((v) => {\n const matchLocale = v.locale === version.locale;\n const matchStatus = status === 'published' ? v.publishedAt !== null : v.publishedAt === null;\n return matchLocale && matchStatus;\n });\n\n if (!availableStatus) return availableStatus;\n\n // Pick status fields (at fields, status, by fields), use lodash fp\n return pick(AVAILABLE_STATUS_FIELDS, availableStatus);\n },\n\n /**\n * Get the available status of many documents, useful for batch operations\n * @param uid\n * @param documents\n * @returns\n */\n async getManyAvailableStatus(uid: UID.ContentType, documents: DocumentVersion[]) {\n if (!documents.length) return [];\n\n // The status and locale of all documents should be the same\n const status = documents[0].publishedAt !== null ? 'published' : 'draft';\n const locales = documents.map((d) => d.locale).filter(Boolean);\n\n const where: Record<string, any> = {\n documentId: { $in: documents.map((d) => d.documentId).filter(Boolean) },\n publishedAt: { $null: status === 'published' },\n };\n\n // If there is any locale to filter (if i18n is enabled)\n if (locales.length) {\n where.locale = { $in: locales };\n }\n\n return strapi.query(uid).findMany({\n where,\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n });\n },\n\n getStatus(version: DocumentVersion, otherDocumentStatuses?: DocumentMetadata['availableStatus']) {\n let draftVersion: DocumentVersion | undefined;\n let publishedVersion: DocumentVersion | undefined;\n\n if (version.publishedAt) {\n publishedVersion = version;\n } else {\n draftVersion = version;\n }\n\n const otherVersion = otherDocumentStatuses?.at(0);\n if (otherVersion?.publishedAt) {\n publishedVersion = otherVersion;\n } else if (otherVersion) {\n draftVersion = otherVersion;\n }\n\n if (!draftVersion) return CONTENT_MANAGER_STATUS.PUBLISHED;\n if (!publishedVersion) return CONTENT_MANAGER_STATUS.DRAFT;\n\n /*\n * The document is modified if the draft version has been updated more\n * recently than the published version.\n */\n const isDraftModified = getIsVersionLatestModification(draftVersion, publishedVersion);\n return isDraftModified ? CONTENT_MANAGER_STATUS.MODIFIED : CONTENT_MANAGER_STATUS.PUBLISHED;\n },\n\n // TODO is it necessary to return metadata on every page of the CM\n // We could refactor this so the locales are only loaded when they're\n // needed. e.g. in the bulk locale action modal.\n async getMetadata(\n uid: UID.ContentType,\n version: DocumentVersion,\n { availableLocales = true, availableStatus = true }: GetMetadataOptions = {}\n ) {\n const model = strapi.getModel(uid);\n const hasDnP = contentTypes.hasDraftAndPublish(model);\n const isLocalized = (model.pluginOptions?.i18n as any)?.localized === true;\n\n if (!availableLocales && !availableStatus) {\n // Nothing to compute.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n if (!isLocalized && !hasDnP) {\n // If there are no locales and no draft/publish, there's only ever 1 version of any document.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n\n const onlyStatusIsRelevant = hasDnP && (!isLocalized || !availableLocales);\n if (onlyStatusIsRelevant) {\n const otherVersion = availableStatus\n ? await strapi.db.query(uid).findOne({\n where: {\n documentId: version.documentId,\n ...(version.locale ? { locale: version.locale } : {}),\n publishedAt: oppositePublishStatus(version.publishedAt),\n },\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n populate: AVAILABLE_STATUS_POPULATE,\n })\n : null;\n return {\n availableLocales: [],\n availableStatus: otherVersion ? [pick(AVAILABLE_STATUS_FIELDS, otherVersion)] : [],\n versions: [] as DocumentVersion[],\n };\n }\n\n // Full path for localized content types\n // TODO: Ignore publishedAt if availableStatus=false, and ignore locale if\n // i18n is disabled\n\n // Include non-translatable scalar and media fields in availableLocales for i18n prefilling\n let nonLocalizedFields: string[] = [];\n let nonLocalizedMediaFields: string[] = [];\n try {\n const i18nPlugin = strapi.plugin('i18n');\n if (i18nPlugin) {\n const i18nService = i18nPlugin.service('content-types');\n if (i18nService?.getNonLocalizedAttributes) {\n if (model?.attributes) {\n const allNonLocalized = i18nService.getNonLocalizedAttributes(model);\n // Get scalar and media attributes separately\n const scalarAttrs = getScalarAttributes(model);\n const mediaAttrs = getMediaAttributes(model);\n\n // Separate scalar fields (can be in fields array) from media fields (need to be populated)\n nonLocalizedFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && scalarAttrs.includes(field)\n );\n nonLocalizedMediaFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && mediaAttrs.includes(field)\n );\n }\n }\n }\n } catch (error) {\n // i18n plugin might not be enabled or might error, ignore silently\n }\n\n // Build populate object for non-localized media fields\n const mediaPopulate = nonLocalizedMediaFields.reduce(\n (acc, field) => {\n acc[field] = {\n populate: {\n folder: true,\n },\n };\n return acc;\n },\n {} as Record<string, { populate: { folder: boolean } }>\n );\n\n const params = {\n populate: {\n ...mediaPopulate,\n ...AVAILABLE_STATUS_POPULATE,\n },\n fields: uniq([...AVAILABLE_LOCALES_FIELDS, ...nonLocalizedFields]),\n filters: {\n documentId: version.documentId,\n },\n };\n\n const dbParams = strapi.get('query-params').transform(uid, params);\n const versions = await strapi.db.query(uid).findMany(dbParams);\n\n // TODO: Remove use of available locales and use localizations instead\n const availableLocalesResult = availableLocales\n ? await this.getAvailableLocales(uid, version, versions)\n : [];\n\n const availableStatusResult = availableStatus\n ? this.getAvailableStatus(version, versions)\n : null;\n\n return {\n availableLocales: availableLocalesResult,\n availableStatus: availableStatusResult ? [availableStatusResult] : [],\n versions,\n };\n },\n\n /**\n * Returns associated metadata of a document:\n * - Available locales of the document for the current status\n * - Available status of the document for the current locale\n */\n async formatDocumentWithMetadata(\n uid: UID.ContentType,\n document: DocumentVersion,\n opts: GetMetadataOptions = {}\n ) {\n if (!document) {\n return {\n data: document,\n meta: {\n availableLocales: [],\n availableStatus: [],\n },\n };\n }\n\n const hasDraftAndPublish = contentTypes.hasDraftAndPublish(strapi.getModel(uid));\n\n // Ignore available status if the content type does not have draft and publish\n if (!hasDraftAndPublish) {\n opts.availableStatus = false;\n }\n\n const { versions, ...meta } = await this.getMetadata(uid, document, opts);\n\n // Populate localization statuses\n if (document.localizations?.length) {\n document.localizations = document.localizations.map((d) => {\n // Find the counterpart version (same documentId + locale, opposite publishedAt) from\n // the already-fetched versions array, avoiding an extra DB query.\n const counterpart = versions.find(\n (v) =>\n v.documentId === d.documentId &&\n v.locale === d.locale &&\n (d.publishedAt === null) !== (v.publishedAt === null)\n );\n return {\n ...d,\n status: this.getStatus(d, counterpart ? [counterpart] : []),\n };\n });\n }\n\n return {\n data: {\n ...document,\n // Add status to the document only if draft and publish is enabled\n status: hasDraftAndPublish\n ? this.getStatus(document, meta.availableStatus as any)\n : undefined,\n },\n meta,\n };\n },\n});\n"],"names":["getScalarAttributes","getMediaAttributes","contentTypes","AVAILABLE_STATUS_SCALAR_FIELDS","AVAILABLE_STATUS_POPULATE","createdBy","select","updatedBy","AVAILABLE_STATUS_FIELDS","AVAILABLE_LOCALES_FIELDS","oppositePublishStatus","publishedAt","$null","$notNull","CONTENT_MANAGER_STATUS","PUBLISHED","DRAFT","MODIFIED","getIsVersionLatestModification","version","otherVersion","updatedAt","versionUpdatedAt","Date","getTime","otherUpdatedAt","strapi","getAvailableLocales","uid","allVersions","versionsByLocale","groupBy","locale","model","getModel","mappingResult","async","map","Object","values","localeVersions","hasDraftAndPublish","draftVersion","find","v","otherVersions","filter","id","status","getStatus","Boolean","getAvailableStatus","availableStatus","matchLocale","matchStatus","pick","getManyAvailableStatus","documents","length","locales","d","where","documentId","$in","query","findMany","otherDocumentStatuses","publishedVersion","at","isDraftModified","getMetadata","availableLocales","hasDnP","isLocalized","pluginOptions","i18n","localized","versions","onlyStatusIsRelevant","db","findOne","populate","nonLocalizedFields","nonLocalizedMediaFields","i18nPlugin","plugin","i18nService","service","getNonLocalizedAttributes","attributes","allNonLocalized","scalarAttrs","mediaAttrs","field","includes","error","mediaPopulate","reduce","acc","folder","params","fields","uniq","filters","dbParams","get","transform","availableLocalesResult","availableStatusResult","formatDocumentWithMetadata","document","opts","data","meta","localizations","counterpart","undefined"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"document-metadata.mjs","sources":["../../../server/src/services/document-metadata.ts"],"sourcesContent":["import { groupBy, pick, uniq } from 'lodash/fp';\n\nimport { async, contentTypes } from '@strapi/utils';\nimport type { Core, UID, Modules } from '@strapi/types';\n\nimport type { DocumentMetadata } from '../../../shared/contracts/collection-types';\n\nconst { getScalarAttributes, getMediaAttributes } = contentTypes;\n\nexport interface DocumentVersion {\n id: string | number;\n documentId: Modules.Documents.ID;\n locale?: string;\n localizations?: DocumentVersion[];\n updatedAt?: string | null | Date;\n publishedAt?: string | null | Date;\n}\n\n// Scalar fields that can be used in a DB `select`\nconst AVAILABLE_STATUS_SCALAR_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n// Relation populate shared by both the fast path and the full path\nconst AVAILABLE_STATUS_POPULATE = {\n createdBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n updatedBy: { select: ['id', 'firstname', 'lastname', 'email'] },\n};\n// All fields to pick from a hydrated result (scalars + populated relations + virtual)\nconst AVAILABLE_STATUS_FIELDS = [\n ...AVAILABLE_STATUS_SCALAR_FIELDS,\n 'createdBy',\n 'updatedBy',\n 'status',\n];\nconst AVAILABLE_LOCALES_FIELDS = [\n 'id',\n 'documentId',\n 'locale',\n 'updatedAt',\n 'createdAt',\n 'publishedAt',\n];\n\n/** Returns a DB filter that matches the opposite publish status. */\nconst oppositePublishStatus = (publishedAt: unknown) =>\n publishedAt !== null ? { $null: true } : { $notNull: true };\n\nconst CONTENT_MANAGER_STATUS = {\n PUBLISHED: 'published',\n DRAFT: 'draft',\n MODIFIED: 'modified',\n};\n\n/**\n * Controls the metadata properties to be returned\n *\n * If `availableLocales` is set to `true` (default), the returned metadata will include\n * the available locales of the document for its current status.\n *\n * If `availableStatus` is set to `true` (default), the returned metadata will include\n * the available status of the document for its current locale.\n */\nexport interface GetMetadataOptions {\n availableLocales?: boolean;\n availableStatus?: boolean;\n}\n\n/**\n * Checks if the provided document version has been modified after all other versions.\n */\nconst getIsVersionLatestModification = (\n version?: DocumentVersion,\n otherVersion?: DocumentVersion\n): boolean => {\n if (!version || !version.updatedAt) {\n return false;\n }\n\n const versionUpdatedAt = version?.updatedAt ? new Date(version.updatedAt).getTime() : 0;\n\n const otherUpdatedAt = otherVersion?.updatedAt ? new Date(otherVersion.updatedAt).getTime() : 0;\n\n return versionUpdatedAt > otherUpdatedAt;\n};\n\nexport default ({ strapi }: { strapi: Core.Strapi }) => ({\n /**\n * Returns available locales of a document for the current status.\n *\n * The result is sorted with the default locale (as defined by the i18n plugin)\n * at index 0 when present. This is the canonical-source invariant relied on by\n * `useDocument.getInitialFormValues` in the admin: when creating a new locale\n * draft, non-localized scalar/media values are inherited from\n * `availableLocales[0]`. Putting the default locale first means inheritance\n * stays predictable when sibling locales have drifted on non-localized fields\n * (which can happen because the server only syncs non-localized fields at\n * locale-creation time, not on subsequent updates — see\n * `copyNonLocalizedFields` in document-service/internationalization.ts).\n */\n async getAvailableLocales(\n uid: UID.ContentType,\n version: DocumentVersion,\n allVersions: DocumentVersion[]\n ) {\n // Group all versions by locale\n const versionsByLocale = groupBy('locale', allVersions);\n\n // Delete the current locale\n if (version.locale) {\n delete versionsByLocale[version.locale];\n }\n\n // For each locale, get the ones with the same status\n // There will not be a draft and a version counterpart if the content\n // type does not have draft and publish\n const model = strapi.getModel(uid);\n\n const mappingResult = await async.map(\n Object.values(versionsByLocale),\n async (localeVersions: DocumentVersion[]) => {\n if (!contentTypes.hasDraftAndPublish(model)) {\n return localeVersions[0];\n }\n\n const draftVersion = localeVersions.find((v) => v.publishedAt === null);\n const otherVersions = localeVersions.filter((v) => v.id !== draftVersion?.id);\n\n if (!draftVersion) {\n return;\n }\n\n return {\n ...draftVersion,\n status: this.getStatus(draftVersion, otherVersions as any),\n };\n }\n );\n\n const filtered = mappingResult.filter(Boolean) as unknown as NonNullable<\n DocumentMetadata['availableLocales']\n >;\n\n // Sort the default locale first so `availableLocales[0]` is the canonical\n // source for non-localized field inheritance in the admin. Guarded so that\n // we no-op if the i18n plugin or its locales service is unavailable.\n let defaultLocaleCode: string | undefined;\n try {\n defaultLocaleCode = await strapi.plugin('i18n')?.service('locales')?.getDefaultLocale();\n } catch {\n // i18n plugin disabled or service errored — leave order untouched.\n }\n\n if (!defaultLocaleCode) {\n return filtered as DocumentMetadata['availableLocales'];\n }\n\n // Stable partition: move the default locale entry to the front, keep the\n // rest of the order intact.\n const defaultEntries = filtered.filter((entry) => entry.locale === defaultLocaleCode);\n const otherEntries = filtered.filter((entry) => entry.locale !== defaultLocaleCode);\n\n return [...defaultEntries, ...otherEntries] as DocumentMetadata['availableLocales'];\n },\n\n /**\n * Returns available status of a document for the current locale\n */\n getAvailableStatus(version: DocumentVersion, allVersions: DocumentVersion[]) {\n // Find the other status of the document\n const status =\n version.publishedAt !== null\n ? CONTENT_MANAGER_STATUS.DRAFT\n : CONTENT_MANAGER_STATUS.PUBLISHED;\n\n // Get version that match the current locale and not match the current status\n const availableStatus = allVersions.find((v) => {\n const matchLocale = v.locale === version.locale;\n const matchStatus = status === 'published' ? v.publishedAt !== null : v.publishedAt === null;\n return matchLocale && matchStatus;\n });\n\n if (!availableStatus) return availableStatus;\n\n // Pick status fields (at fields, status, by fields), use lodash fp\n return pick(AVAILABLE_STATUS_FIELDS, availableStatus);\n },\n\n /**\n * Get the available status of many documents, useful for batch operations\n * @param uid\n * @param documents\n * @returns\n */\n async getManyAvailableStatus(uid: UID.ContentType, documents: DocumentVersion[]) {\n if (!documents.length) return [];\n\n // The status and locale of all documents should be the same\n const status = documents[0].publishedAt !== null ? 'published' : 'draft';\n const locales = documents.map((d) => d.locale).filter(Boolean);\n\n const where: Record<string, any> = {\n documentId: { $in: documents.map((d) => d.documentId).filter(Boolean) },\n publishedAt: { $null: status === 'published' },\n };\n\n // If there is any locale to filter (if i18n is enabled)\n if (locales.length) {\n where.locale = { $in: locales };\n }\n\n return strapi.query(uid).findMany({\n where,\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n });\n },\n\n getStatus(version: DocumentVersion, otherDocumentStatuses?: DocumentMetadata['availableStatus']) {\n let draftVersion: DocumentVersion | undefined;\n let publishedVersion: DocumentVersion | undefined;\n\n if (version.publishedAt) {\n publishedVersion = version;\n } else {\n draftVersion = version;\n }\n\n const otherVersion = otherDocumentStatuses?.at(0);\n if (otherVersion?.publishedAt) {\n publishedVersion = otherVersion;\n } else if (otherVersion) {\n draftVersion = otherVersion;\n }\n\n if (!draftVersion) return CONTENT_MANAGER_STATUS.PUBLISHED;\n if (!publishedVersion) return CONTENT_MANAGER_STATUS.DRAFT;\n\n /*\n * The document is modified if the draft version has been updated more\n * recently than the published version.\n */\n const isDraftModified = getIsVersionLatestModification(draftVersion, publishedVersion);\n return isDraftModified ? CONTENT_MANAGER_STATUS.MODIFIED : CONTENT_MANAGER_STATUS.PUBLISHED;\n },\n\n // TODO is it necessary to return metadata on every page of the CM\n // We could refactor this so the locales are only loaded when they're\n // needed. e.g. in the bulk locale action modal.\n async getMetadata(\n uid: UID.ContentType,\n version: DocumentVersion,\n { availableLocales = true, availableStatus = true }: GetMetadataOptions = {}\n ) {\n const model = strapi.getModel(uid);\n const hasDnP = contentTypes.hasDraftAndPublish(model);\n const isLocalized = (model.pluginOptions?.i18n as any)?.localized === true;\n\n if (!availableLocales && !availableStatus) {\n // Nothing to compute.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n if (!isLocalized && !hasDnP) {\n // If there are no locales and no draft/publish, there's only ever 1 version of any document.\n return { availableLocales: [], availableStatus: [], versions: [] as DocumentVersion[] };\n }\n\n const onlyStatusIsRelevant = hasDnP && (!isLocalized || !availableLocales);\n if (onlyStatusIsRelevant) {\n const otherVersion = availableStatus\n ? await strapi.db.query(uid).findOne({\n where: {\n documentId: version.documentId,\n ...(version.locale ? { locale: version.locale } : {}),\n publishedAt: oppositePublishStatus(version.publishedAt),\n },\n select: AVAILABLE_STATUS_SCALAR_FIELDS,\n populate: AVAILABLE_STATUS_POPULATE,\n })\n : null;\n return {\n availableLocales: [],\n availableStatus: otherVersion ? [pick(AVAILABLE_STATUS_FIELDS, otherVersion)] : [],\n versions: [] as DocumentVersion[],\n };\n }\n\n // Full path for localized content types\n // TODO: Ignore publishedAt if availableStatus=false, and ignore locale if\n // i18n is disabled\n\n // Include non-translatable scalar and media fields in availableLocales for i18n prefilling\n let nonLocalizedFields: string[] = [];\n let nonLocalizedMediaFields: string[] = [];\n try {\n const i18nPlugin = strapi.plugin('i18n');\n if (i18nPlugin) {\n const i18nService = i18nPlugin.service('content-types');\n if (i18nService?.getNonLocalizedAttributes) {\n if (model?.attributes) {\n const allNonLocalized = i18nService.getNonLocalizedAttributes(model);\n // Get scalar and media attributes separately\n const scalarAttrs = getScalarAttributes(model);\n const mediaAttrs = getMediaAttributes(model);\n\n // Separate scalar fields (can be in fields array) from media fields (need to be populated)\n nonLocalizedFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && scalarAttrs.includes(field)\n );\n nonLocalizedMediaFields = allNonLocalized.filter(\n (field: string) => field in model.attributes && mediaAttrs.includes(field)\n );\n }\n }\n }\n } catch (error) {\n // i18n plugin might not be enabled or might error, ignore silently\n }\n\n // Build populate object for non-localized media fields\n const mediaPopulate = nonLocalizedMediaFields.reduce(\n (acc, field) => {\n acc[field] = {\n populate: {\n folder: true,\n },\n };\n return acc;\n },\n {} as Record<string, { populate: { folder: boolean } }>\n );\n\n const params = {\n populate: {\n ...mediaPopulate,\n ...AVAILABLE_STATUS_POPULATE,\n },\n fields: uniq([...AVAILABLE_LOCALES_FIELDS, ...nonLocalizedFields]),\n filters: {\n documentId: version.documentId,\n },\n };\n\n const dbParams = strapi.get('query-params').transform(uid, params);\n const versions = await strapi.db.query(uid).findMany(dbParams);\n\n // TODO: Remove use of available locales and use localizations instead\n const availableLocalesResult = availableLocales\n ? await this.getAvailableLocales(uid, version, versions)\n : [];\n\n const availableStatusResult = availableStatus\n ? this.getAvailableStatus(version, versions)\n : null;\n\n return {\n availableLocales: availableLocalesResult,\n availableStatus: availableStatusResult ? [availableStatusResult] : [],\n versions,\n };\n },\n\n /**\n * Returns associated metadata of a document:\n * - Available locales of the document for the current status\n * - Available status of the document for the current locale\n */\n async formatDocumentWithMetadata(\n uid: UID.ContentType,\n document: DocumentVersion,\n opts: GetMetadataOptions = {}\n ) {\n if (!document) {\n return {\n data: document,\n meta: {\n availableLocales: [],\n availableStatus: [],\n },\n };\n }\n\n const hasDraftAndPublish = contentTypes.hasDraftAndPublish(strapi.getModel(uid));\n\n // Ignore available status if the content type does not have draft and publish\n if (!hasDraftAndPublish) {\n opts.availableStatus = false;\n }\n\n const { versions, ...meta } = await this.getMetadata(uid, document, opts);\n\n // Populate localization statuses\n if (document.localizations?.length) {\n document.localizations = document.localizations.map((d) => {\n // Find the counterpart version (same documentId + locale, opposite publishedAt) from\n // the already-fetched versions array, avoiding an extra DB query.\n const counterpart = versions.find(\n (v) =>\n v.documentId === d.documentId &&\n v.locale === d.locale &&\n (d.publishedAt === null) !== (v.publishedAt === null)\n );\n return {\n ...d,\n status: this.getStatus(d, counterpart ? [counterpart] : []),\n };\n });\n }\n\n return {\n data: {\n ...document,\n // Add status to the document only if draft and publish is enabled\n status: hasDraftAndPublish\n ? this.getStatus(document, meta.availableStatus as any)\n : undefined,\n },\n meta,\n };\n },\n});\n"],"names":["getScalarAttributes","getMediaAttributes","contentTypes","AVAILABLE_STATUS_SCALAR_FIELDS","AVAILABLE_STATUS_POPULATE","createdBy","select","updatedBy","AVAILABLE_STATUS_FIELDS","AVAILABLE_LOCALES_FIELDS","oppositePublishStatus","publishedAt","$null","$notNull","CONTENT_MANAGER_STATUS","PUBLISHED","DRAFT","MODIFIED","getIsVersionLatestModification","version","otherVersion","updatedAt","versionUpdatedAt","Date","getTime","otherUpdatedAt","strapi","getAvailableLocales","uid","allVersions","versionsByLocale","groupBy","locale","model","getModel","mappingResult","async","map","Object","values","localeVersions","hasDraftAndPublish","draftVersion","find","v","otherVersions","filter","id","status","getStatus","filtered","Boolean","defaultLocaleCode","plugin","service","getDefaultLocale","defaultEntries","entry","otherEntries","getAvailableStatus","availableStatus","matchLocale","matchStatus","pick","getManyAvailableStatus","documents","length","locales","d","where","documentId","$in","query","findMany","otherDocumentStatuses","publishedVersion","at","isDraftModified","getMetadata","availableLocales","hasDnP","isLocalized","pluginOptions","i18n","localized","versions","onlyStatusIsRelevant","db","findOne","populate","nonLocalizedFields","nonLocalizedMediaFields","i18nPlugin","i18nService","getNonLocalizedAttributes","attributes","allNonLocalized","scalarAttrs","mediaAttrs","field","includes","error","mediaPopulate","reduce","acc","folder","params","fields","uniq","filters","dbParams","get","transform","availableLocalesResult","availableStatusResult","formatDocumentWithMetadata","document","opts","data","meta","localizations","counterpart","undefined"],"mappings":"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"}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"index.js","sources":["../../../server/src/services/index.ts"],"sourcesContent":["import components from './components';\nimport contentTypes from './content-types';\nimport dataMapper from './data-mapper';\nimport fieldSizes from './field-sizes';\nimport metrics from './metrics';\nimport permissionChecker from './permission-checker';\nimport permission from './permission';\nimport populateBuilder from './populate-builder';\nimport uid from './uid';\nimport history from '../history';\nimport preview from '../preview';\nimport homepage from '../homepage';\nimport documentMetadata from './document-metadata';\nimport documentManager from './document-manager';\n\nexport default {\n components,\n 'content-types': contentTypes,\n 'data-mapper': dataMapper,\n 'document-metadata': documentMetadata,\n 'document-manager': documentManager,\n 'field-sizes': fieldSizes,\n metrics,\n 'permission-checker': permissionChecker,\n permission,\n 'populate-builder': populateBuilder,\n uid,\n ...(history.services ? history.services : {}),\n ...(preview.services ? preview.services : {}),\n ...homepage.services,\n};\n"],"names":["components","contentTypes","dataMapper","documentMetadata","documentManager","fieldSizes","metrics","permissionChecker","permission","populateBuilder","uid","history","services","preview","homepage"],"mappings":";;;;;;;;;;;;;;;;;AAeA,eAAe;AACbA,IAAAA,UAAAA;IACA,eAAA,EAAiBC,YAAAA;IACjB,aAAA,EAAeC,UAAAA;IACf,mBAAA,EAAqBC,gBAAAA;IACrB,kBAAA,EAAoBC,eAAAA;IACpB,aAAA,EAAeC,UAAAA;AACfC,IAAAA,OAAAA;IACA,oBAAA,EAAsBC,
|
|
1
|
+
{"version":3,"file":"index.js","sources":["../../../server/src/services/index.ts"],"sourcesContent":["import components from './components';\nimport contentTypes from './content-types';\nimport dataMapper from './data-mapper';\nimport fieldSizes from './field-sizes';\nimport metrics from './metrics';\nimport permissionChecker from './permission-checker';\nimport permission from './permission';\nimport populateBuilder from './populate-builder';\nimport uid from './uid';\nimport history from '../history';\nimport preview from '../preview';\nimport homepage from '../homepage';\nimport documentMetadata from './document-metadata';\nimport documentManager from './document-manager';\n\nexport default {\n components,\n 'content-types': contentTypes,\n 'data-mapper': dataMapper,\n 'document-metadata': documentMetadata,\n 'document-manager': documentManager,\n 'field-sizes': fieldSizes,\n metrics,\n 'permission-checker': permissionChecker,\n permission,\n 'populate-builder': populateBuilder,\n uid,\n ...(history.services ? history.services : {}),\n ...(preview.services ? preview.services : {}),\n ...homepage.services,\n};\n"],"names":["components","contentTypes","dataMapper","documentMetadata","documentManager","fieldSizes","metrics","permissionChecker","permission","populateBuilder","uid","history","services","preview","homepage"],"mappings":";;;;;;;;;;;;;;;;;AAeA,eAAe;AACbA,IAAAA,UAAAA;IACA,eAAA,EAAiBC,YAAAA;IACjB,aAAA,EAAeC,UAAAA;IACf,mBAAA,EAAqBC,gBAAAA;IACrB,kBAAA,EAAoBC,eAAAA;IACpB,aAAA,EAAeC,UAAAA;AACfC,IAAAA,OAAAA;IACA,oBAAA,EAAsBC,yBAAAA;AACtBC,IAAAA,UAAAA;IACA,kBAAA,EAAoBC,eAAAA;AACpBC,IAAAA,GAAAA;AACA,IAAA,GAAIC,MAAQC,QAAQ,GAAGD,MAAQC,QAAQ,GAAG,EAAE;AAC5C,IAAA,GAAIC,QAAQD,QAAQ,GAAGC,QAAQD,QAAQ,GAAG,EAAE;AAC5C,IAAA,GAAGE,QAASF;AACd,CAAA;;;;"}
|
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
|
+
Object.defineProperty(exports, '__esModule', { value: true });
|
|
4
|
+
|
|
3
5
|
var strapiUtils = require('@strapi/utils');
|
|
4
6
|
var fp = require('lodash/fp');
|
|
5
7
|
|
|
@@ -131,5 +133,6 @@ var permissionChecker = (({ strapi })=>({
|
|
|
131
133
|
create: createPermissionChecker(strapi)
|
|
132
134
|
}));
|
|
133
135
|
|
|
134
|
-
|
|
136
|
+
exports.ACTIONS = ACTIONS;
|
|
137
|
+
exports.default = permissionChecker;
|
|
135
138
|
//# sourceMappingURL=permission-checker.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"permission-checker.js","sources":["../../../server/src/services/permission-checker.ts"],"sourcesContent":["import { async } from '@strapi/utils';\nimport { isEmpty } from 'lodash/fp';\nimport type { Core, UID, Modules } from '@strapi/types';\n\nconst ACTIONS = {\n read: 'plugin::content-manager.explorer.read',\n create: 'plugin::content-manager.explorer.create',\n update: 'plugin::content-manager.explorer.update',\n delete: 'plugin::content-manager.explorer.delete',\n publish: 'plugin::content-manager.explorer.publish',\n unpublish: 'plugin::content-manager.explorer.publish',\n discard: 'plugin::content-manager.explorer.update',\n} as const;\n\ntype Entity = Modules.EntityService.Result<UID.ContentType>;\ntype Query = {\n page?: string;\n pageSize?: string;\n sort?: string;\n};\n\nconst createPermissionChecker =\n (strapi: Core.Strapi) =>\n ({ userAbility, model }: { userAbility: any; model: string }) => {\n const permissionsManager = strapi.service('admin::permission').createPermissionsManager({\n ability: userAbility,\n model,\n });\n\n const { actionProvider } = strapi.service('admin::permission');\n\n const toSubject = (entity?: Entity) => {\n return entity ? permissionsManager.toSubject(entity, model) : model;\n };\n\n // @ts-expect-error preserve the parameter order\n // eslint-disable-next-line @typescript-eslint/default-param-last\n const can = (action: string, entity?: Entity, field: string) => {\n const subject = toSubject(entity);\n const aliases = actionProvider.unstable_aliases(action, model) as string[];\n\n return (\n // Test the original action to see if it passes\n userAbility.can(action, subject, field) ||\n // Else try every known alias if at least one of them succeed, then the user \"can\"\n aliases.some((alias) => userAbility.can(alias, subject, field))\n );\n };\n\n // @ts-expect-error preserve the parameter order\n // eslint-disable-next-line @typescript-eslint/default-param-last\n const cannot = (action: string, entity?: Entity, field: string) => {\n const subject = toSubject(entity);\n const aliases = actionProvider.unstable_aliases(action, model) as string[];\n\n return (\n // Test both the original action\n userAbility.cannot(action, subject, field) &&\n // and every known alias, if all of them fail (cannot), then the user truly \"cannot\"\n aliases.every((alias) => userAbility.cannot(alias, subject, field))\n );\n };\n\n const sanitizeOutput = (data: Entity, { action = ACTIONS.read }: { action?: string } = {}) => {\n return permissionsManager.sanitizeOutput(data, { subject: toSubject(data), action });\n };\n\n const getRulesForAction = (action: string) => {\n if (typeof userAbility.rulesFor !== 'function') {\n return [];\n }\n\n const aliases = actionProvider.unstable_aliases(action, model) as string[];\n const actions = [action, ...aliases];\n\n return actions.flatMap((actionName) => userAbility.rulesFor(actionName, model));\n };\n\n // Tell callers if we need the full entity to check access\n const requiresEntity = (action: string) => {\n if (typeof userAbility.rulesFor !== 'function') {\n return true;\n }\n\n const rules = getRulesForAction(action);\n\n return rules.some((rule: any) => rule.conditions && !isEmpty(rule.conditions));\n };\n\n const sanitizeQuery = (query: Query, { action = ACTIONS.read }: { action?: string } = {}) => {\n return permissionsManager.sanitizeQuery(query, { subject: model, action });\n };\n\n const sanitizeInput = (action: string, data: any, entity?: Entity) => {\n return permissionsManager.sanitizeInput(data, {\n subject: entity ? toSubject(entity) : model,\n action,\n });\n };\n\n const validateQuery = (query: Query, { action = ACTIONS.read }: { action?: string } = {}) => {\n return permissionsManager.validateQuery(query, { subject: model, action });\n };\n\n const validateInput = (action: string, data: any, entity?: Entity) => {\n return permissionsManager.validateInput(data, {\n subject: entity ? toSubject(entity) : model,\n action,\n });\n };\n\n const sanitizeCreateInput = (data: any) => sanitizeInput(ACTIONS.create, data);\n const sanitizeUpdateInput = (entity: Entity) => (data: any) =>\n sanitizeInput(ACTIONS.update, data, entity);\n\n const buildPermissionQuery = (query: Query, action: { action?: string } = {}) => {\n return permissionsManager.addPermissionsQueryTo(query, action);\n };\n\n const sanitizedQuery = (query: Query, action: { action?: string } = {}) => {\n return async.pipe(\n (q: Query) => sanitizeQuery(q, action),\n (q: Query) => buildPermissionQuery(q, action)\n )(query);\n };\n\n // Sanitized queries shortcuts\n Object.keys(ACTIONS).forEach((action) => {\n // @ts-expect-error TODO\n sanitizedQuery[action] = (query: Query) => sanitizedQuery(query, ACTIONS[action]);\n });\n\n // Permission utils shortcuts\n Object.keys(ACTIONS).forEach((action) => {\n // @ts-expect-error TODO\n can[action] = (...args: any) => can(ACTIONS[action], ...args);\n // @ts-expect-error TODO\n cannot[action] = (...args: any) => cannot(ACTIONS[action], ...args);\n // @ts-expect-error TODO\n requiresEntity[action] = () => requiresEntity(ACTIONS[action]);\n });\n\n return {\n // Permission utils\n can, // check if you have the permission\n cannot, // check if you don't have the permission\n requiresEntity, // check if entity data is needed for permission evaluation\n // Sanitizers\n sanitizeOutput,\n sanitizeQuery,\n sanitizeCreateInput,\n sanitizeUpdateInput,\n // Validators\n validateQuery,\n validateInput,\n // Queries Builder\n sanitizedQuery,\n };\n };\n\nexport default ({ strapi }: { strapi: Core.Strapi }) => ({\n create: createPermissionChecker(strapi),\n});\n"],"names":["ACTIONS","read","create","update","delete","publish","unpublish","discard","createPermissionChecker","strapi","userAbility","model","permissionsManager","service","createPermissionsManager","ability","actionProvider","toSubject","entity","can","action","field","subject","aliases","unstable_aliases","some","alias","cannot","every","sanitizeOutput","data","getRulesForAction","rulesFor","actions","flatMap","actionName","requiresEntity","rules","rule","conditions","isEmpty","sanitizeQuery","query","sanitizeInput","validateQuery","validateInput","sanitizeCreateInput","sanitizeUpdateInput","buildPermissionQuery","addPermissionsQueryTo","sanitizedQuery","async","pipe","q","Object","keys","forEach","args"],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"permission-checker.js","sources":["../../../server/src/services/permission-checker.ts"],"sourcesContent":["import type { Ability } from '@casl/ability';\nimport { async } from '@strapi/utils';\nimport { isEmpty } from 'lodash/fp';\nimport type { Core, UID, Modules } from '@strapi/types';\n\nexport const ACTIONS = {\n read: 'plugin::content-manager.explorer.read',\n create: 'plugin::content-manager.explorer.create',\n update: 'plugin::content-manager.explorer.update',\n delete: 'plugin::content-manager.explorer.delete',\n publish: 'plugin::content-manager.explorer.publish',\n unpublish: 'plugin::content-manager.explorer.publish',\n discard: 'plugin::content-manager.explorer.update',\n} as const;\n\ntype Entity = Modules.EntityService.Result<UID.ContentType>;\ntype Query = {\n page?: string;\n pageSize?: string;\n sort?: string;\n};\n\nconst createPermissionChecker =\n (strapi: Core.Strapi) =>\n ({ userAbility, model }: { userAbility: Ability; model: string }) => {\n const permissionsManager = strapi.service('admin::permission').createPermissionsManager({\n ability: userAbility,\n model,\n });\n\n const { actionProvider } = strapi.service('admin::permission');\n\n const toSubject = (entity?: Entity) => {\n return entity ? permissionsManager.toSubject(entity, model) : model;\n };\n\n // @ts-expect-error preserve the parameter order\n // eslint-disable-next-line @typescript-eslint/default-param-last\n const can = (action: string, entity?: Entity, field: string) => {\n const subject = toSubject(entity);\n const aliases = actionProvider.unstable_aliases(action, model) as string[];\n\n return (\n // Test the original action to see if it passes\n userAbility.can(action, subject, field) ||\n // Else try every known alias if at least one of them succeed, then the user \"can\"\n aliases.some((alias) => userAbility.can(alias, subject, field))\n );\n };\n\n // @ts-expect-error preserve the parameter order\n // eslint-disable-next-line @typescript-eslint/default-param-last\n const cannot = (action: string, entity?: Entity, field: string) => {\n const subject = toSubject(entity);\n const aliases = actionProvider.unstable_aliases(action, model) as string[];\n\n return (\n // Test both the original action\n userAbility.cannot(action, subject, field) &&\n // and every known alias, if all of them fail (cannot), then the user truly \"cannot\"\n aliases.every((alias) => userAbility.cannot(alias, subject, field))\n );\n };\n\n const sanitizeOutput = (data: Entity, { action = ACTIONS.read }: { action?: string } = {}) => {\n return permissionsManager.sanitizeOutput(data, { subject: toSubject(data), action });\n };\n\n const getRulesForAction = (action: string) => {\n if (typeof userAbility.rulesFor !== 'function') {\n return [];\n }\n\n const aliases = actionProvider.unstable_aliases(action, model) as string[];\n const actions = [action, ...aliases];\n\n return actions.flatMap((actionName) => userAbility.rulesFor(actionName, model));\n };\n\n // Tell callers if we need the full entity to check access\n const requiresEntity = (action: string) => {\n if (typeof userAbility.rulesFor !== 'function') {\n return true;\n }\n\n const rules = getRulesForAction(action);\n\n return rules.some((rule: any) => rule.conditions && !isEmpty(rule.conditions));\n };\n\n const sanitizeQuery = (query: Query, { action = ACTIONS.read }: { action?: string } = {}) => {\n return permissionsManager.sanitizeQuery(query, { subject: model, action });\n };\n\n const sanitizeInput = (action: string, data: any, entity?: Entity) => {\n return permissionsManager.sanitizeInput(data, {\n subject: entity ? toSubject(entity) : model,\n action,\n });\n };\n\n const validateQuery = (query: Query, { action = ACTIONS.read }: { action?: string } = {}) => {\n return permissionsManager.validateQuery(query, { subject: model, action });\n };\n\n const validateInput = (action: string, data: any, entity?: Entity) => {\n return permissionsManager.validateInput(data, {\n subject: entity ? toSubject(entity) : model,\n action,\n });\n };\n\n const sanitizeCreateInput = (data: any) => sanitizeInput(ACTIONS.create, data);\n const sanitizeUpdateInput = (entity: Entity) => (data: any) =>\n sanitizeInput(ACTIONS.update, data, entity);\n\n const buildPermissionQuery = (query: Query, action: { action?: string } = {}) => {\n return permissionsManager.addPermissionsQueryTo(query, action);\n };\n\n const sanitizedQuery = (query: Query, action: { action?: string } = {}) => {\n return async.pipe(\n (q: Query) => sanitizeQuery(q, action),\n (q: Query) => buildPermissionQuery(q, action)\n )(query);\n };\n\n // Sanitized queries shortcuts\n Object.keys(ACTIONS).forEach((action) => {\n // @ts-expect-error TODO\n sanitizedQuery[action] = (query: Query) => sanitizedQuery(query, ACTIONS[action]);\n });\n\n // Permission utils shortcuts\n Object.keys(ACTIONS).forEach((action) => {\n // @ts-expect-error TODO\n can[action] = (...args: any) => can(ACTIONS[action], ...args);\n // @ts-expect-error TODO\n cannot[action] = (...args: any) => cannot(ACTIONS[action], ...args);\n // @ts-expect-error TODO\n requiresEntity[action] = () => requiresEntity(ACTIONS[action]);\n });\n\n return {\n // Permission utils\n can, // check if you have the permission\n cannot, // check if you don't have the permission\n requiresEntity, // check if entity data is needed for permission evaluation\n // Sanitizers\n sanitizeOutput,\n sanitizeQuery,\n sanitizeCreateInput,\n sanitizeUpdateInput,\n // Validators\n validateQuery,\n validateInput,\n // Queries Builder\n sanitizedQuery,\n };\n };\n\nexport default ({ strapi }: { strapi: Core.Strapi }) => ({\n create: createPermissionChecker(strapi),\n});\n"],"names":["ACTIONS","read","create","update","delete","publish","unpublish","discard","createPermissionChecker","strapi","userAbility","model","permissionsManager","service","createPermissionsManager","ability","actionProvider","toSubject","entity","can","action","field","subject","aliases","unstable_aliases","some","alias","cannot","every","sanitizeOutput","data","getRulesForAction","rulesFor","actions","flatMap","actionName","requiresEntity","rules","rule","conditions","isEmpty","sanitizeQuery","query","sanitizeInput","validateQuery","validateInput","sanitizeCreateInput","sanitizeUpdateInput","buildPermissionQuery","addPermissionsQueryTo","sanitizedQuery","async","pipe","q","Object","keys","forEach","args"],"mappings":"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"}
|