@strapi/i18n 0.0.0-experimental.afa3b513b8f95459043f33fb94f4bac03af1474f → 0.0.0-experimental.b05633daea1bf090c66312b8ab30ec13bdb52f57
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/_chunks/{SettingsPage-BjxjwEOb.mjs → SettingsPage-BAx9nmep.mjs} +4 -4
- package/dist/_chunks/SettingsPage-BAx9nmep.mjs.map +1 -0
- package/dist/_chunks/{SettingsPage-CfTmCkup.js → SettingsPage-BTgjb2KS.js} +5 -6
- package/dist/_chunks/SettingsPage-BTgjb2KS.js.map +1 -0
- package/dist/_chunks/{en-DWpfm8h5.js → en-BKBz3tro.js} +5 -4
- package/dist/_chunks/en-BKBz3tro.js.map +1 -0
- package/dist/_chunks/{en-2xztdZE1.mjs → en-DlXfy6Gy.mjs} +5 -4
- package/dist/_chunks/en-DlXfy6Gy.mjs.map +1 -0
- package/dist/_chunks/{index-5XLZwzwx.js → index-3yyF237r.js} +94 -36
- package/dist/_chunks/index-3yyF237r.js.map +1 -0
- package/dist/_chunks/{index-D-qx3tz4.mjs → index-B0NijiBB.mjs} +94 -35
- package/dist/_chunks/index-B0NijiBB.mjs.map +1 -0
- package/dist/admin/index.js +1 -1
- package/dist/admin/index.mjs +1 -1
- package/dist/admin/src/components/CMHeaderActions.d.ts +5 -3
- package/dist/admin/src/components/CreateLocale.d.ts +6 -6
- package/dist/server/index.js +398 -421
- package/dist/server/index.js.map +1 -1
- package/dist/server/index.mjs +399 -421
- package/dist/server/index.mjs.map +1 -1
- package/dist/server/src/index.d.ts +6 -0
- package/dist/server/src/index.d.ts.map +1 -1
- package/dist/server/src/register.d.ts.map +1 -1
- package/dist/server/src/services/index.d.ts +6 -0
- package/dist/server/src/services/index.d.ts.map +1 -1
- package/dist/server/src/services/sanitize/index.d.ts +11 -0
- package/dist/server/src/services/sanitize/index.d.ts.map +1 -0
- package/dist/server/src/utils/index.d.ts +2 -0
- package/dist/server/src/utils/index.d.ts.map +1 -1
- package/package.json +13 -13
- package/dist/_chunks/SettingsPage-BjxjwEOb.mjs.map +0 -1
- package/dist/_chunks/SettingsPage-CfTmCkup.js.map +0 -1
- package/dist/_chunks/en-2xztdZE1.mjs.map +0 -1
- package/dist/_chunks/en-DWpfm8h5.js.map +0 -1
- package/dist/_chunks/index-5XLZwzwx.js.map +0 -1
- package/dist/_chunks/index-D-qx3tz4.mjs.map +0 -1
- package/dist/server/src/migrations/content-type/disable/index.d.ts +0 -3
- package/dist/server/src/migrations/content-type/disable/index.d.ts.map +0 -1
- package/dist/server/src/migrations/content-type/enable/index.d.ts +0 -3
- package/dist/server/src/migrations/content-type/enable/index.d.ts.map +0 -1
package/dist/server/index.mjs
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
import _ from "lodash";
|
2
|
-
import { get, identity, propEq, isArray, prop, getOr, isEmpty, reduce, cloneDeep, isNil, pipe, pick, has, map, flatten, uniq } from "lodash/fp";
|
2
|
+
import { get, identity, propEq, isArray, prop, getOr, isEmpty, reduce, cloneDeep, isNil, pipe, pick, has, curry, map, flatten, uniq } from "lodash/fp";
|
3
3
|
import * as utils from "@strapi/utils";
|
4
|
-
import { errors, async, contentTypes as contentTypes$2, yup, validateYupSchema } from "@strapi/utils";
|
4
|
+
import { errors, async, contentTypes as contentTypes$2, traverseEntity, yup, validateYupSchema } from "@strapi/utils";
|
5
5
|
const getCoreStore = () => {
|
6
6
|
return strapi.store({ type: "plugin", name: "i18n" });
|
7
7
|
};
|
@@ -174,6 +174,352 @@ const getI18nLocaleArgPlugin = ({ nexus, typeRegistry }) => {
|
|
174
174
|
}
|
175
175
|
});
|
176
176
|
};
|
177
|
+
const register = ({ strapi: strapi2 }) => {
|
178
|
+
extendContentTypes(strapi2);
|
179
|
+
addContentManagerLocaleMiddleware(strapi2);
|
180
|
+
};
|
181
|
+
const addContentManagerLocaleMiddleware = (strapi2) => {
|
182
|
+
strapi2.server.router.use("/content-manager/collection-types/:model", (ctx, next) => {
|
183
|
+
if (ctx.method === "POST" || ctx.method === "PUT") {
|
184
|
+
return validateLocaleCreation(ctx, next);
|
185
|
+
}
|
186
|
+
return next();
|
187
|
+
});
|
188
|
+
strapi2.server.router.use("/content-manager/single-types/:model", (ctx, next) => {
|
189
|
+
if (ctx.method === "POST" || ctx.method === "PUT") {
|
190
|
+
return validateLocaleCreation(ctx, next);
|
191
|
+
}
|
192
|
+
return next();
|
193
|
+
});
|
194
|
+
};
|
195
|
+
const extendContentTypes = (strapi2) => {
|
196
|
+
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
197
|
+
Object.values(strapi2.contentTypes).forEach((contentType) => {
|
198
|
+
const { attributes: attributes2 } = contentType;
|
199
|
+
const isLocalized = isLocalizedContentType2(contentType);
|
200
|
+
_.set(attributes2, "locale", {
|
201
|
+
writable: true,
|
202
|
+
private: !isLocalized,
|
203
|
+
configurable: false,
|
204
|
+
visible: false,
|
205
|
+
type: "string"
|
206
|
+
});
|
207
|
+
_.set(attributes2, "localizations", {
|
208
|
+
type: "relation",
|
209
|
+
relation: "oneToMany",
|
210
|
+
target: contentType.uid,
|
211
|
+
writable: false,
|
212
|
+
private: !isLocalized,
|
213
|
+
configurable: false,
|
214
|
+
visible: false,
|
215
|
+
unstable_virtual: true,
|
216
|
+
joinColumn: {
|
217
|
+
name: "document_id",
|
218
|
+
referencedColumn: "document_id",
|
219
|
+
referencedTable: strapi2.db.metadata.identifiers.getTableName(contentType.collectionName),
|
220
|
+
// ensure the population will not include the results we already loaded
|
221
|
+
on({ results }) {
|
222
|
+
return {
|
223
|
+
id: {
|
224
|
+
$notIn: results.map((r) => r.id)
|
225
|
+
}
|
226
|
+
};
|
227
|
+
}
|
228
|
+
}
|
229
|
+
});
|
230
|
+
});
|
231
|
+
if (strapi2.plugin("graphql")) {
|
232
|
+
graphqlProvider({ strapi: strapi2 }).register();
|
233
|
+
}
|
234
|
+
};
|
235
|
+
const info = {
|
236
|
+
singularName: "locale",
|
237
|
+
pluralName: "locales",
|
238
|
+
collectionName: "locales",
|
239
|
+
displayName: "Locale",
|
240
|
+
description: ""
|
241
|
+
};
|
242
|
+
const options = {};
|
243
|
+
const pluginOptions = {
|
244
|
+
"content-manager": {
|
245
|
+
visible: false
|
246
|
+
},
|
247
|
+
"content-type-builder": {
|
248
|
+
visible: false
|
249
|
+
}
|
250
|
+
};
|
251
|
+
const attributes = {
|
252
|
+
name: {
|
253
|
+
type: "string",
|
254
|
+
min: 1,
|
255
|
+
max: 50,
|
256
|
+
configurable: false
|
257
|
+
},
|
258
|
+
code: {
|
259
|
+
type: "string",
|
260
|
+
unique: true,
|
261
|
+
configurable: false
|
262
|
+
}
|
263
|
+
};
|
264
|
+
const schema = {
|
265
|
+
info,
|
266
|
+
options,
|
267
|
+
pluginOptions,
|
268
|
+
attributes
|
269
|
+
};
|
270
|
+
const locale = {
|
271
|
+
schema
|
272
|
+
};
|
273
|
+
const contentTypes$1 = {
|
274
|
+
locale
|
275
|
+
};
|
276
|
+
const actions = [
|
277
|
+
{
|
278
|
+
section: "settings",
|
279
|
+
category: "Internationalization",
|
280
|
+
subCategory: "Locales",
|
281
|
+
pluginName: "i18n",
|
282
|
+
displayName: "Create",
|
283
|
+
uid: "locale.create"
|
284
|
+
},
|
285
|
+
{
|
286
|
+
section: "settings",
|
287
|
+
category: "Internationalization",
|
288
|
+
subCategory: "Locales",
|
289
|
+
pluginName: "i18n",
|
290
|
+
displayName: "Read",
|
291
|
+
uid: "locale.read",
|
292
|
+
aliases: [
|
293
|
+
{ actionId: "plugin::content-manager.explorer.read", subjects: ["plugin::i18n.locale"] }
|
294
|
+
]
|
295
|
+
},
|
296
|
+
{
|
297
|
+
section: "settings",
|
298
|
+
category: "Internationalization",
|
299
|
+
subCategory: "Locales",
|
300
|
+
pluginName: "i18n",
|
301
|
+
displayName: "Update",
|
302
|
+
uid: "locale.update"
|
303
|
+
},
|
304
|
+
{
|
305
|
+
section: "settings",
|
306
|
+
category: "Internationalization",
|
307
|
+
subCategory: "Locales",
|
308
|
+
pluginName: "i18n",
|
309
|
+
displayName: "Delete",
|
310
|
+
uid: "locale.delete"
|
311
|
+
}
|
312
|
+
];
|
313
|
+
const addLocalesPropertyIfNeeded = ({ value: action }) => {
|
314
|
+
const {
|
315
|
+
section,
|
316
|
+
options: { applyToProperties }
|
317
|
+
} = action;
|
318
|
+
if (section !== "contentTypes") {
|
319
|
+
return;
|
320
|
+
}
|
321
|
+
if (isArray(applyToProperties) && applyToProperties.includes("locales")) {
|
322
|
+
return;
|
323
|
+
}
|
324
|
+
action.options.applyToProperties = isArray(applyToProperties) ? applyToProperties.concat("locales") : ["locales"];
|
325
|
+
};
|
326
|
+
const shouldApplyLocalesPropertyToSubject = ({ property, subject }) => {
|
327
|
+
if (property === "locales") {
|
328
|
+
const model = strapi.getModel(subject);
|
329
|
+
return getService("content-types").isLocalizedContentType(model);
|
330
|
+
}
|
331
|
+
return true;
|
332
|
+
};
|
333
|
+
const addAllLocalesToPermissions = async (permissions2) => {
|
334
|
+
const { actionProvider } = strapi.service("admin::permission");
|
335
|
+
const { find: findAllLocales } = getService("locales");
|
336
|
+
const allLocales = await findAllLocales();
|
337
|
+
const allLocalesCode = allLocales.map(prop("code"));
|
338
|
+
return Promise.all(
|
339
|
+
permissions2.map(async (permission) => {
|
340
|
+
const { action, subject } = permission;
|
341
|
+
const appliesToLocalesProperty = await actionProvider.appliesToProperty(
|
342
|
+
"locales",
|
343
|
+
action,
|
344
|
+
subject
|
345
|
+
);
|
346
|
+
if (!appliesToLocalesProperty) {
|
347
|
+
return permission;
|
348
|
+
}
|
349
|
+
const oldPermissionProperties = getOr({}, "properties", permission);
|
350
|
+
return { ...permission, properties: { ...oldPermissionProperties, locales: allLocalesCode } };
|
351
|
+
})
|
352
|
+
);
|
353
|
+
};
|
354
|
+
const syncSuperAdminPermissionsWithLocales = async () => {
|
355
|
+
const roleService = strapi.service("admin::role");
|
356
|
+
const permissionService = strapi.service("admin::permission");
|
357
|
+
const superAdminRole = await roleService.getSuperAdmin();
|
358
|
+
if (!superAdminRole) {
|
359
|
+
return;
|
360
|
+
}
|
361
|
+
const superAdminPermissions = await permissionService.findMany({
|
362
|
+
where: {
|
363
|
+
role: {
|
364
|
+
id: superAdminRole.id
|
365
|
+
}
|
366
|
+
}
|
367
|
+
});
|
368
|
+
const newSuperAdminPermissions = await addAllLocalesToPermissions(superAdminPermissions);
|
369
|
+
await roleService.assignPermissions(superAdminRole.id, newSuperAdminPermissions);
|
370
|
+
};
|
371
|
+
const registerI18nActions = async () => {
|
372
|
+
const { actionProvider } = strapi.service("admin::permission");
|
373
|
+
await actionProvider.registerMany(actions);
|
374
|
+
};
|
375
|
+
const registerI18nActionsHooks = () => {
|
376
|
+
const { actionProvider } = strapi.service("admin::permission");
|
377
|
+
const { hooks } = strapi.service("admin::role");
|
378
|
+
actionProvider.hooks.appliesPropertyToSubject.register(shouldApplyLocalesPropertyToSubject);
|
379
|
+
hooks.willResetSuperAdminPermissions.register(addAllLocalesToPermissions);
|
380
|
+
};
|
381
|
+
const updateActionsProperties = () => {
|
382
|
+
const { actionProvider } = strapi.service("admin::permission");
|
383
|
+
actionProvider.hooks.willRegister.register(addLocalesPropertyIfNeeded);
|
384
|
+
actionProvider.values().forEach((action) => addLocalesPropertyIfNeeded({ value: action }));
|
385
|
+
};
|
386
|
+
const i18nActionsService = {
|
387
|
+
actions,
|
388
|
+
registerI18nActions,
|
389
|
+
registerI18nActionsHooks,
|
390
|
+
updateActionsProperties,
|
391
|
+
syncSuperAdminPermissionsWithLocales
|
392
|
+
};
|
393
|
+
const localesPropertyHandler = async ({ action, section }) => {
|
394
|
+
const { actionProvider } = strapi.service("admin::permission");
|
395
|
+
const locales2 = await getService("locales").find();
|
396
|
+
if (isEmpty(locales2)) {
|
397
|
+
return;
|
398
|
+
}
|
399
|
+
for (const subject of section.subjects) {
|
400
|
+
const applies = await actionProvider.appliesToProperty("locales", action.actionId, subject.uid);
|
401
|
+
const hasLocalesProperty = subject.properties.find(
|
402
|
+
(property) => property.value === "locales"
|
403
|
+
);
|
404
|
+
if (applies && !hasLocalesProperty) {
|
405
|
+
subject.properties.push({
|
406
|
+
label: "Locales",
|
407
|
+
value: "locales",
|
408
|
+
children: locales2.map(({ name, code }) => ({ label: name || code, value: code }))
|
409
|
+
});
|
410
|
+
}
|
411
|
+
}
|
412
|
+
};
|
413
|
+
const registerLocalesPropertyHandler = () => {
|
414
|
+
const { sectionsBuilder } = strapi.service("admin::permission");
|
415
|
+
sectionsBuilder.addHandler("singleTypes", localesPropertyHandler);
|
416
|
+
sectionsBuilder.addHandler("collectionTypes", localesPropertyHandler);
|
417
|
+
};
|
418
|
+
const sectionsBuilderService = {
|
419
|
+
localesPropertyHandler,
|
420
|
+
registerLocalesPropertyHandler
|
421
|
+
};
|
422
|
+
const willRegisterPermission = (context) => {
|
423
|
+
const { permission, condition, user } = context;
|
424
|
+
const { subject, properties } = permission;
|
425
|
+
const isSuperAdmin = strapi.service("admin::role").hasSuperAdminRole(user);
|
426
|
+
if (isSuperAdmin) {
|
427
|
+
return;
|
428
|
+
}
|
429
|
+
const { locales: locales2 } = properties || {};
|
430
|
+
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
431
|
+
if (!subject) {
|
432
|
+
return;
|
433
|
+
}
|
434
|
+
const ct = strapi.contentTypes[subject];
|
435
|
+
if (!isLocalizedContentType2(ct)) {
|
436
|
+
return;
|
437
|
+
}
|
438
|
+
if (locales2 === null) {
|
439
|
+
return;
|
440
|
+
}
|
441
|
+
condition.and({
|
442
|
+
locale: {
|
443
|
+
$in: locales2 || []
|
444
|
+
}
|
445
|
+
});
|
446
|
+
};
|
447
|
+
const registerI18nPermissionsHandlers = () => {
|
448
|
+
const { engine } = strapi.service("admin::permission");
|
449
|
+
engine.hooks["before-register.permission"].register(willRegisterPermission);
|
450
|
+
};
|
451
|
+
const engineService = {
|
452
|
+
willRegisterPermission,
|
453
|
+
registerI18nPermissionsHandlers
|
454
|
+
};
|
455
|
+
const permissions = () => ({
|
456
|
+
actions: i18nActionsService,
|
457
|
+
sectionsBuilder: sectionsBuilderService,
|
458
|
+
engine: engineService
|
459
|
+
});
|
460
|
+
const sendDidInitializeEvent = async () => {
|
461
|
+
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
462
|
+
const numberOfContentTypes = reduce(
|
463
|
+
(sum, contentType) => isLocalizedContentType2(contentType) ? sum + 1 : sum,
|
464
|
+
0
|
465
|
+
)(strapi.contentTypes);
|
466
|
+
await strapi.telemetry.send("didInitializeI18n", { groupProperties: { numberOfContentTypes } });
|
467
|
+
};
|
468
|
+
const sendDidUpdateI18nLocalesEvent = async () => {
|
469
|
+
const numberOfLocales = await getService("locales").count();
|
470
|
+
await strapi.telemetry.send("didUpdateI18nLocales", {
|
471
|
+
groupProperties: { numberOfLocales }
|
472
|
+
});
|
473
|
+
};
|
474
|
+
const metrics = () => ({
|
475
|
+
sendDidInitializeEvent,
|
476
|
+
sendDidUpdateI18nLocalesEvent
|
477
|
+
});
|
478
|
+
const syncNonLocalizedAttributes = async (sourceEntry, model) => {
|
479
|
+
const { copyNonLocalizedAttributes: copyNonLocalizedAttributes2 } = getService("content-types");
|
480
|
+
const nonLocalizedAttributes = copyNonLocalizedAttributes2(model, sourceEntry);
|
481
|
+
if (isEmpty(nonLocalizedAttributes)) {
|
482
|
+
return;
|
483
|
+
}
|
484
|
+
const uid = model.uid;
|
485
|
+
const documentId = sourceEntry.documentId;
|
486
|
+
const locale2 = sourceEntry.locale;
|
487
|
+
const status = sourceEntry?.publishedAt ? "published" : "draft";
|
488
|
+
const localeEntriesToUpdate = await strapi.db.query(uid).findMany({
|
489
|
+
where: {
|
490
|
+
documentId,
|
491
|
+
publishedAt: status === "published" ? { $ne: null } : null,
|
492
|
+
locale: { $ne: locale2 }
|
493
|
+
},
|
494
|
+
select: ["locale", "id"]
|
495
|
+
});
|
496
|
+
const entryData = await strapi.documents(uid).omitComponentData(nonLocalizedAttributes);
|
497
|
+
await async.map(localeEntriesToUpdate, async (entry) => {
|
498
|
+
const transformedData = await strapi.documents.utils.transformData(
|
499
|
+
cloneDeep(nonLocalizedAttributes),
|
500
|
+
{
|
501
|
+
uid,
|
502
|
+
status,
|
503
|
+
locale: entry.locale,
|
504
|
+
allowMissingId: true
|
505
|
+
}
|
506
|
+
);
|
507
|
+
const componentData = await strapi.documents(uid).updateComponents(entry, transformedData);
|
508
|
+
await strapi.db.query(uid).update({
|
509
|
+
where: {
|
510
|
+
documentId,
|
511
|
+
publishedAt: status === "published" ? { $ne: null } : null,
|
512
|
+
locale: { $eq: entry.locale }
|
513
|
+
},
|
514
|
+
// The data we send to the update function is the entry data merged with
|
515
|
+
// the updated component data
|
516
|
+
data: Object.assign(cloneDeep(entryData), componentData)
|
517
|
+
});
|
518
|
+
});
|
519
|
+
};
|
520
|
+
const localizations = () => ({
|
521
|
+
syncNonLocalizedAttributes
|
522
|
+
});
|
177
523
|
const isoLocales = [
|
178
524
|
{
|
179
525
|
code: "af",
|
@@ -2551,6 +2897,14 @@ const isoLocales = [
|
|
2551
2897
|
code: "cy-GB",
|
2552
2898
|
name: "Welsh (United Kingdom) (cy-GB)"
|
2553
2899
|
},
|
2900
|
+
{
|
2901
|
+
code: "wo",
|
2902
|
+
name: "Wolof (wo)"
|
2903
|
+
},
|
2904
|
+
{
|
2905
|
+
code: "xh",
|
2906
|
+
name: "Xhosa (xh)"
|
2907
|
+
},
|
2554
2908
|
{
|
2555
2909
|
code: "yav",
|
2556
2910
|
name: "Yangben (yav)"
|
@@ -2567,437 +2921,40 @@ const isoLocales = [
|
|
2567
2921
|
code: "yo-NG",
|
2568
2922
|
name: "Yoruba (Nigeria) (yo-NG)"
|
2569
2923
|
},
|
2570
|
-
{
|
2571
|
-
code: "dje",
|
2572
|
-
name: "Zarma (dje)"
|
2573
|
-
},
|
2574
|
-
{
|
2575
|
-
code: "dje-NE",
|
2576
|
-
name: "Zarma (Niger) (dje-NE)"
|
2577
|
-
},
|
2578
|
-
{
|
2579
|
-
code: "zu",
|
2580
|
-
name: "Zulu (zu)"
|
2581
|
-
},
|
2582
|
-
{
|
2583
|
-
code: "zu-ZA",
|
2584
|
-
name: "Zulu (South Africa) (zu-ZA)"
|
2585
|
-
}
|
2586
|
-
];
|
2587
|
-
const getInitLocale = () => {
|
2588
|
-
const envLocaleCode = process.env.STRAPI_PLUGIN_I18N_INIT_LOCALE_CODE;
|
2589
|
-
if (envLocaleCode) {
|
2590
|
-
const matchingLocale = isoLocales.find(({ code }) => code === envLocaleCode);
|
2591
|
-
if (!matchingLocale) {
|
2592
|
-
throw new Error(
|
2593
|
-
"Unknown locale code provided in the environment variable STRAPI_PLUGIN_I18N_INIT_LOCALE_CODE"
|
2594
|
-
);
|
2595
|
-
}
|
2596
|
-
return { ...matchingLocale };
|
2597
|
-
}
|
2598
|
-
return {
|
2599
|
-
code: "en",
|
2600
|
-
name: "English (en)"
|
2601
|
-
};
|
2602
|
-
};
|
2603
|
-
const DEFAULT_LOCALE = getInitLocale();
|
2604
|
-
const enableContentType = async ({ oldContentTypes, contentTypes: contentTypes2 }) => {
|
2605
|
-
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
2606
|
-
const { getDefaultLocale: getDefaultLocale2 } = getService("locales");
|
2607
|
-
if (!oldContentTypes) {
|
2608
|
-
return;
|
2609
|
-
}
|
2610
|
-
for (const uid in contentTypes2) {
|
2611
|
-
if (!oldContentTypes[uid]) {
|
2612
|
-
continue;
|
2613
|
-
}
|
2614
|
-
const oldContentType = oldContentTypes[uid];
|
2615
|
-
const contentType = contentTypes2[uid];
|
2616
|
-
if (!isLocalizedContentType2(oldContentType) && isLocalizedContentType2(contentType)) {
|
2617
|
-
const defaultLocale = await getDefaultLocale2() || DEFAULT_LOCALE.code;
|
2618
|
-
await strapi.db.query(uid).updateMany({
|
2619
|
-
where: { locale: null },
|
2620
|
-
data: { locale: defaultLocale }
|
2621
|
-
});
|
2622
|
-
}
|
2623
|
-
}
|
2624
|
-
};
|
2625
|
-
const disableContentType = async ({ oldContentTypes, contentTypes: contentTypes2 }) => {
|
2626
|
-
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
2627
|
-
const { getDefaultLocale: getDefaultLocale2 } = getService("locales");
|
2628
|
-
if (!oldContentTypes) {
|
2629
|
-
return;
|
2630
|
-
}
|
2631
|
-
for (const uid in contentTypes2) {
|
2632
|
-
if (!oldContentTypes[uid]) {
|
2633
|
-
continue;
|
2634
|
-
}
|
2635
|
-
const oldContentType = oldContentTypes[uid];
|
2636
|
-
const contentType = contentTypes2[uid];
|
2637
|
-
if (isLocalizedContentType2(oldContentType) && !isLocalizedContentType2(contentType)) {
|
2638
|
-
const defaultLocale = await getDefaultLocale2() || DEFAULT_LOCALE.code;
|
2639
|
-
await Promise.all([
|
2640
|
-
// Delete all entities that are not in the default locale
|
2641
|
-
strapi.db.query(uid).deleteMany({
|
2642
|
-
where: { locale: { $ne: defaultLocale } }
|
2643
|
-
}),
|
2644
|
-
// Set locale to null for the rest
|
2645
|
-
strapi.db.query(uid).updateMany({
|
2646
|
-
where: { locale: { $eq: defaultLocale } },
|
2647
|
-
data: { locale: null }
|
2648
|
-
})
|
2649
|
-
]);
|
2650
|
-
}
|
2651
|
-
}
|
2652
|
-
};
|
2653
|
-
const register = ({ strapi: strapi2 }) => {
|
2654
|
-
extendContentTypes(strapi2);
|
2655
|
-
addContentManagerLocaleMiddleware(strapi2);
|
2656
|
-
addContentTypeSyncHooks(strapi2);
|
2657
|
-
};
|
2658
|
-
const addContentManagerLocaleMiddleware = (strapi2) => {
|
2659
|
-
strapi2.server.router.use("/content-manager/collection-types/:model", (ctx, next) => {
|
2660
|
-
if (ctx.method === "POST" || ctx.method === "PUT") {
|
2661
|
-
return validateLocaleCreation(ctx, next);
|
2662
|
-
}
|
2663
|
-
return next();
|
2664
|
-
});
|
2665
|
-
strapi2.server.router.use("/content-manager/single-types/:model", (ctx, next) => {
|
2666
|
-
if (ctx.method === "POST" || ctx.method === "PUT") {
|
2667
|
-
return validateLocaleCreation(ctx, next);
|
2668
|
-
}
|
2669
|
-
return next();
|
2670
|
-
});
|
2671
|
-
};
|
2672
|
-
const addContentTypeSyncHooks = (strapi2) => {
|
2673
|
-
strapi2.hook("strapi::content-types.beforeSync").register(disableContentType);
|
2674
|
-
strapi2.hook("strapi::content-types.afterSync").register(enableContentType);
|
2675
|
-
};
|
2676
|
-
const extendContentTypes = (strapi2) => {
|
2677
|
-
Object.values(strapi2.contentTypes).forEach((contentType) => {
|
2678
|
-
const { attributes: attributes2 } = contentType;
|
2679
|
-
_.set(attributes2, "locale", {
|
2680
|
-
writable: true,
|
2681
|
-
private: false,
|
2682
|
-
configurable: false,
|
2683
|
-
visible: false,
|
2684
|
-
type: "string"
|
2685
|
-
});
|
2686
|
-
_.set(attributes2, "localizations", {
|
2687
|
-
type: "relation",
|
2688
|
-
relation: "oneToMany",
|
2689
|
-
target: contentType.uid,
|
2690
|
-
writable: false,
|
2691
|
-
private: false,
|
2692
|
-
configurable: false,
|
2693
|
-
visible: false,
|
2694
|
-
joinColumn: {
|
2695
|
-
name: "document_id",
|
2696
|
-
referencedColumn: "document_id",
|
2697
|
-
referencedTable: strapi2.db.metadata.identifiers.getTableName(contentType.collectionName),
|
2698
|
-
// ensure the population will not include the results we already loaded
|
2699
|
-
on({ results }) {
|
2700
|
-
return {
|
2701
|
-
id: {
|
2702
|
-
$notIn: results.map((r) => r.id)
|
2703
|
-
}
|
2704
|
-
};
|
2705
|
-
}
|
2706
|
-
}
|
2707
|
-
});
|
2708
|
-
});
|
2709
|
-
if (strapi2.plugin("graphql")) {
|
2710
|
-
graphqlProvider({ strapi: strapi2 }).register();
|
2711
|
-
}
|
2712
|
-
};
|
2713
|
-
const info = {
|
2714
|
-
singularName: "locale",
|
2715
|
-
pluralName: "locales",
|
2716
|
-
collectionName: "locales",
|
2717
|
-
displayName: "Locale",
|
2718
|
-
description: ""
|
2719
|
-
};
|
2720
|
-
const options = {};
|
2721
|
-
const pluginOptions = {
|
2722
|
-
"content-manager": {
|
2723
|
-
visible: false
|
2724
|
-
},
|
2725
|
-
"content-type-builder": {
|
2726
|
-
visible: false
|
2727
|
-
}
|
2728
|
-
};
|
2729
|
-
const attributes = {
|
2730
|
-
name: {
|
2731
|
-
type: "string",
|
2732
|
-
min: 1,
|
2733
|
-
max: 50,
|
2734
|
-
configurable: false
|
2735
|
-
},
|
2736
|
-
code: {
|
2737
|
-
type: "string",
|
2738
|
-
unique: true,
|
2739
|
-
configurable: false
|
2740
|
-
}
|
2741
|
-
};
|
2742
|
-
const schema = {
|
2743
|
-
info,
|
2744
|
-
options,
|
2745
|
-
pluginOptions,
|
2746
|
-
attributes
|
2747
|
-
};
|
2748
|
-
const locale = {
|
2749
|
-
schema
|
2750
|
-
};
|
2751
|
-
const contentTypes$1 = {
|
2752
|
-
locale
|
2753
|
-
};
|
2754
|
-
const actions = [
|
2755
|
-
{
|
2756
|
-
section: "settings",
|
2757
|
-
category: "Internationalization",
|
2758
|
-
subCategory: "Locales",
|
2759
|
-
pluginName: "i18n",
|
2760
|
-
displayName: "Create",
|
2761
|
-
uid: "locale.create"
|
2924
|
+
{
|
2925
|
+
code: "dje",
|
2926
|
+
name: "Zarma (dje)"
|
2762
2927
|
},
|
2763
2928
|
{
|
2764
|
-
|
2765
|
-
|
2766
|
-
subCategory: "Locales",
|
2767
|
-
pluginName: "i18n",
|
2768
|
-
displayName: "Read",
|
2769
|
-
uid: "locale.read",
|
2770
|
-
aliases: [
|
2771
|
-
{ actionId: "plugin::content-manager.explorer.read", subjects: ["plugin::i18n.locale"] }
|
2772
|
-
]
|
2929
|
+
code: "dje-NE",
|
2930
|
+
name: "Zarma (Niger) (dje-NE)"
|
2773
2931
|
},
|
2774
2932
|
{
|
2775
|
-
|
2776
|
-
|
2777
|
-
subCategory: "Locales",
|
2778
|
-
pluginName: "i18n",
|
2779
|
-
displayName: "Update",
|
2780
|
-
uid: "locale.update"
|
2933
|
+
code: "zu",
|
2934
|
+
name: "Zulu (zu)"
|
2781
2935
|
},
|
2782
2936
|
{
|
2783
|
-
|
2784
|
-
|
2785
|
-
subCategory: "Locales",
|
2786
|
-
pluginName: "i18n",
|
2787
|
-
displayName: "Delete",
|
2788
|
-
uid: "locale.delete"
|
2937
|
+
code: "zu-ZA",
|
2938
|
+
name: "Zulu (South Africa) (zu-ZA)"
|
2789
2939
|
}
|
2790
2940
|
];
|
2791
|
-
const
|
2792
|
-
const
|
2793
|
-
|
2794
|
-
|
2795
|
-
|
2796
|
-
|
2797
|
-
|
2798
|
-
}
|
2799
|
-
if (isArray(applyToProperties) && applyToProperties.includes("locales")) {
|
2800
|
-
return;
|
2801
|
-
}
|
2802
|
-
action.options.applyToProperties = isArray(applyToProperties) ? applyToProperties.concat("locales") : ["locales"];
|
2803
|
-
};
|
2804
|
-
const shouldApplyLocalesPropertyToSubject = ({ property, subject }) => {
|
2805
|
-
if (property === "locales") {
|
2806
|
-
const model = strapi.getModel(subject);
|
2807
|
-
return getService("content-types").isLocalizedContentType(model);
|
2808
|
-
}
|
2809
|
-
return true;
|
2810
|
-
};
|
2811
|
-
const addAllLocalesToPermissions = async (permissions2) => {
|
2812
|
-
const { actionProvider } = strapi.service("admin::permission");
|
2813
|
-
const { find: findAllLocales } = getService("locales");
|
2814
|
-
const allLocales = await findAllLocales();
|
2815
|
-
const allLocalesCode = allLocales.map(prop("code"));
|
2816
|
-
return Promise.all(
|
2817
|
-
permissions2.map(async (permission) => {
|
2818
|
-
const { action, subject } = permission;
|
2819
|
-
const appliesToLocalesProperty = await actionProvider.appliesToProperty(
|
2820
|
-
"locales",
|
2821
|
-
action,
|
2822
|
-
subject
|
2941
|
+
const getInitLocale = () => {
|
2942
|
+
const envLocaleCode = process.env.STRAPI_PLUGIN_I18N_INIT_LOCALE_CODE;
|
2943
|
+
if (envLocaleCode) {
|
2944
|
+
const matchingLocale = isoLocales.find(({ code }) => code === envLocaleCode);
|
2945
|
+
if (!matchingLocale) {
|
2946
|
+
throw new Error(
|
2947
|
+
"Unknown locale code provided in the environment variable STRAPI_PLUGIN_I18N_INIT_LOCALE_CODE"
|
2823
2948
|
);
|
2824
|
-
if (!appliesToLocalesProperty) {
|
2825
|
-
return permission;
|
2826
|
-
}
|
2827
|
-
const oldPermissionProperties = getOr({}, "properties", permission);
|
2828
|
-
return { ...permission, properties: { ...oldPermissionProperties, locales: allLocalesCode } };
|
2829
|
-
})
|
2830
|
-
);
|
2831
|
-
};
|
2832
|
-
const syncSuperAdminPermissionsWithLocales = async () => {
|
2833
|
-
const roleService = strapi.service("admin::role");
|
2834
|
-
const permissionService = strapi.service("admin::permission");
|
2835
|
-
const superAdminRole = await roleService.getSuperAdmin();
|
2836
|
-
if (!superAdminRole) {
|
2837
|
-
return;
|
2838
|
-
}
|
2839
|
-
const superAdminPermissions = await permissionService.findMany({
|
2840
|
-
where: {
|
2841
|
-
role: {
|
2842
|
-
id: superAdminRole.id
|
2843
|
-
}
|
2844
|
-
}
|
2845
|
-
});
|
2846
|
-
const newSuperAdminPermissions = await addAllLocalesToPermissions(superAdminPermissions);
|
2847
|
-
await roleService.assignPermissions(superAdminRole.id, newSuperAdminPermissions);
|
2848
|
-
};
|
2849
|
-
const registerI18nActions = async () => {
|
2850
|
-
const { actionProvider } = strapi.service("admin::permission");
|
2851
|
-
await actionProvider.registerMany(actions);
|
2852
|
-
};
|
2853
|
-
const registerI18nActionsHooks = () => {
|
2854
|
-
const { actionProvider } = strapi.service("admin::permission");
|
2855
|
-
const { hooks } = strapi.service("admin::role");
|
2856
|
-
actionProvider.hooks.appliesPropertyToSubject.register(shouldApplyLocalesPropertyToSubject);
|
2857
|
-
hooks.willResetSuperAdminPermissions.register(addAllLocalesToPermissions);
|
2858
|
-
};
|
2859
|
-
const updateActionsProperties = () => {
|
2860
|
-
const { actionProvider } = strapi.service("admin::permission");
|
2861
|
-
actionProvider.hooks.willRegister.register(addLocalesPropertyIfNeeded);
|
2862
|
-
actionProvider.values().forEach((action) => addLocalesPropertyIfNeeded({ value: action }));
|
2863
|
-
};
|
2864
|
-
const i18nActionsService = {
|
2865
|
-
actions,
|
2866
|
-
registerI18nActions,
|
2867
|
-
registerI18nActionsHooks,
|
2868
|
-
updateActionsProperties,
|
2869
|
-
syncSuperAdminPermissionsWithLocales
|
2870
|
-
};
|
2871
|
-
const localesPropertyHandler = async ({ action, section }) => {
|
2872
|
-
const { actionProvider } = strapi.service("admin::permission");
|
2873
|
-
const locales2 = await getService("locales").find();
|
2874
|
-
if (isEmpty(locales2)) {
|
2875
|
-
return;
|
2876
|
-
}
|
2877
|
-
for (const subject of section.subjects) {
|
2878
|
-
const applies = await actionProvider.appliesToProperty("locales", action.actionId, subject.uid);
|
2879
|
-
const hasLocalesProperty = subject.properties.find(
|
2880
|
-
(property) => property.value === "locales"
|
2881
|
-
);
|
2882
|
-
if (applies && !hasLocalesProperty) {
|
2883
|
-
subject.properties.push({
|
2884
|
-
label: "Locales",
|
2885
|
-
value: "locales",
|
2886
|
-
children: locales2.map(({ name, code }) => ({ label: name || code, value: code }))
|
2887
|
-
});
|
2888
|
-
}
|
2889
|
-
}
|
2890
|
-
};
|
2891
|
-
const registerLocalesPropertyHandler = () => {
|
2892
|
-
const { sectionsBuilder } = strapi.service("admin::permission");
|
2893
|
-
sectionsBuilder.addHandler("singleTypes", localesPropertyHandler);
|
2894
|
-
sectionsBuilder.addHandler("collectionTypes", localesPropertyHandler);
|
2895
|
-
};
|
2896
|
-
const sectionsBuilderService = {
|
2897
|
-
localesPropertyHandler,
|
2898
|
-
registerLocalesPropertyHandler
|
2899
|
-
};
|
2900
|
-
const willRegisterPermission = (context) => {
|
2901
|
-
const { permission, condition, user } = context;
|
2902
|
-
const { subject, properties } = permission;
|
2903
|
-
const isSuperAdmin = strapi.service("admin::role").hasSuperAdminRole(user);
|
2904
|
-
if (isSuperAdmin) {
|
2905
|
-
return;
|
2906
|
-
}
|
2907
|
-
const { locales: locales2 } = properties || {};
|
2908
|
-
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
2909
|
-
if (!subject) {
|
2910
|
-
return;
|
2911
|
-
}
|
2912
|
-
const ct = strapi.contentTypes[subject];
|
2913
|
-
if (!isLocalizedContentType2(ct)) {
|
2914
|
-
return;
|
2915
|
-
}
|
2916
|
-
if (locales2 === null) {
|
2917
|
-
return;
|
2918
|
-
}
|
2919
|
-
condition.and({
|
2920
|
-
locale: {
|
2921
|
-
$in: locales2 || []
|
2922
2949
|
}
|
2923
|
-
|
2924
|
-
};
|
2925
|
-
const registerI18nPermissionsHandlers = () => {
|
2926
|
-
const { engine } = strapi.service("admin::permission");
|
2927
|
-
engine.hooks["before-register.permission"].register(willRegisterPermission);
|
2928
|
-
};
|
2929
|
-
const engineService = {
|
2930
|
-
willRegisterPermission,
|
2931
|
-
registerI18nPermissionsHandlers
|
2932
|
-
};
|
2933
|
-
const permissions = () => ({
|
2934
|
-
actions: i18nActionsService,
|
2935
|
-
sectionsBuilder: sectionsBuilderService,
|
2936
|
-
engine: engineService
|
2937
|
-
});
|
2938
|
-
const sendDidInitializeEvent = async () => {
|
2939
|
-
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
2940
|
-
const numberOfContentTypes = reduce(
|
2941
|
-
(sum, contentType) => isLocalizedContentType2(contentType) ? sum + 1 : sum,
|
2942
|
-
0
|
2943
|
-
)(strapi.contentTypes);
|
2944
|
-
await strapi.telemetry.send("didInitializeI18n", { groupProperties: { numberOfContentTypes } });
|
2945
|
-
};
|
2946
|
-
const sendDidUpdateI18nLocalesEvent = async () => {
|
2947
|
-
const numberOfLocales = await getService("locales").count();
|
2948
|
-
await strapi.telemetry.send("didUpdateI18nLocales", {
|
2949
|
-
groupProperties: { numberOfLocales }
|
2950
|
-
});
|
2951
|
-
};
|
2952
|
-
const metrics = () => ({
|
2953
|
-
sendDidInitializeEvent,
|
2954
|
-
sendDidUpdateI18nLocalesEvent
|
2955
|
-
});
|
2956
|
-
const syncNonLocalizedAttributes = async (sourceEntry, model) => {
|
2957
|
-
const { copyNonLocalizedAttributes: copyNonLocalizedAttributes2 } = getService("content-types");
|
2958
|
-
const nonLocalizedAttributes = copyNonLocalizedAttributes2(model, sourceEntry);
|
2959
|
-
if (isEmpty(nonLocalizedAttributes)) {
|
2960
|
-
return;
|
2950
|
+
return { ...matchingLocale };
|
2961
2951
|
}
|
2962
|
-
|
2963
|
-
|
2964
|
-
|
2965
|
-
|
2966
|
-
const localeEntriesToUpdate = await strapi.db.query(uid).findMany({
|
2967
|
-
where: {
|
2968
|
-
documentId,
|
2969
|
-
publishedAt: status === "published" ? { $ne: null } : null,
|
2970
|
-
locale: { $ne: locale2 }
|
2971
|
-
},
|
2972
|
-
select: ["locale", "id"]
|
2973
|
-
});
|
2974
|
-
const entryData = await strapi.documents(uid).omitComponentData(nonLocalizedAttributes);
|
2975
|
-
await async.map(localeEntriesToUpdate, async (entry) => {
|
2976
|
-
const transformedData = await strapi.documents.utils.transformData(
|
2977
|
-
cloneDeep(nonLocalizedAttributes),
|
2978
|
-
{
|
2979
|
-
uid,
|
2980
|
-
status,
|
2981
|
-
locale: entry.locale,
|
2982
|
-
allowMissingId: true
|
2983
|
-
}
|
2984
|
-
);
|
2985
|
-
const componentData = await strapi.documents(uid).updateComponents(entry, transformedData);
|
2986
|
-
await strapi.db.query(uid).update({
|
2987
|
-
where: {
|
2988
|
-
documentId,
|
2989
|
-
publishedAt: status === "published" ? { $ne: null } : null,
|
2990
|
-
locale: { $eq: entry.locale }
|
2991
|
-
},
|
2992
|
-
// The data we send to the update function is the entry data merged with
|
2993
|
-
// the updated component data
|
2994
|
-
data: Object.assign(cloneDeep(entryData), componentData)
|
2995
|
-
});
|
2996
|
-
});
|
2952
|
+
return {
|
2953
|
+
code: "en",
|
2954
|
+
name: "English (en)"
|
2955
|
+
};
|
2997
2956
|
};
|
2998
|
-
const
|
2999
|
-
syncNonLocalizedAttributes
|
3000
|
-
});
|
2957
|
+
const DEFAULT_LOCALE = getInitLocale();
|
3001
2958
|
const find = (params = {}) => strapi.db.query("plugin::i18n.locale").findMany({ where: params });
|
3002
2959
|
const findById = (id) => strapi.db.query("plugin::i18n.locale").findOne({ where: { id } });
|
3003
2960
|
const findByCode = (code) => strapi.db.query("plugin::i18n.locale").findOne({ where: { code } });
|
@@ -3189,11 +3146,32 @@ const contentTypes = () => ({
|
|
3189
3146
|
fillNonLocalizedAttributes,
|
3190
3147
|
getNestedPopulateOfNonLocalizedAttributes
|
3191
3148
|
});
|
3149
|
+
const LOCALIZATION_FIELDS = ["locale", "localizations"];
|
3150
|
+
const sanitize = ({ strapi: strapi2 }) => {
|
3151
|
+
const { isLocalizedContentType: isLocalizedContentType2 } = getService("content-types");
|
3152
|
+
const sanitizeLocalizationFields = curry(
|
3153
|
+
(schema2, entity) => traverseEntity(
|
3154
|
+
({ key, schema: schema22 }, { remove }) => {
|
3155
|
+
const isLocalized = isLocalizedContentType2(schema22);
|
3156
|
+
const isLocalizationField = LOCALIZATION_FIELDS.includes(key);
|
3157
|
+
if (!isLocalized && isLocalizationField) {
|
3158
|
+
remove(key);
|
3159
|
+
}
|
3160
|
+
},
|
3161
|
+
{ schema: schema2, getModel: strapi2.getModel.bind(strapi2) },
|
3162
|
+
entity
|
3163
|
+
)
|
3164
|
+
);
|
3165
|
+
return {
|
3166
|
+
sanitizeLocalizationFields
|
3167
|
+
};
|
3168
|
+
};
|
3192
3169
|
const services = {
|
3193
3170
|
permissions,
|
3194
3171
|
metrics,
|
3195
3172
|
localizations,
|
3196
3173
|
locales,
|
3174
|
+
sanitize,
|
3197
3175
|
"iso-locales": isoLocalesService,
|
3198
3176
|
"content-types": contentTypes
|
3199
3177
|
};
|