Skip to content

Commit 8cbf94e

Browse files
[TECH] Ajouter un script pour actualiser les fichiers de traductions
#11255
2 parents 7d9bc6f + 2f3387b commit 8cbf94e

File tree

5 files changed

+402
-2
lines changed

5 files changed

+402
-2
lines changed

api/scripts/update-audit-api-csv-file.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -60,7 +60,7 @@ async function main() {
6060
}
6161

6262
// fetch current route from prod
63-
// eslint-disable-next-line n/no-unsupported-features/node-builtins
63+
6464
const response = await fetch(swaggerUrl);
6565
const swaggerJson = await response.json();
6666
const currentPixRoutes = extractRoutes(swaggerJson);

api/scripts/update-translations.js

+128
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,128 @@
1+
import fs from 'node:fs';
2+
import path from 'node:path';
3+
4+
import { Script } from '../src/shared/application/scripts/script.js';
5+
import { ScriptRunner } from '../src/shared/application/scripts/script-runner.js';
6+
7+
export class UpdateTranslations extends Script {
8+
constructor() {
9+
super({
10+
description: 'This script is used to update the translations in the application',
11+
permanent: true,
12+
options: {
13+
source: {
14+
type: 'string',
15+
describe: 'The source file',
16+
demandOption: true,
17+
},
18+
targets: {
19+
type: 'array',
20+
describe: 'The list of target files.',
21+
demandOption: true,
22+
},
23+
dryRun: {
24+
type: 'boolean',
25+
describe: 'If true: Write in files',
26+
default: false,
27+
},
28+
},
29+
});
30+
this.informations = {
31+
new: 0,
32+
old: 0,
33+
write: 0,
34+
delete: 0,
35+
};
36+
}
37+
38+
async handle({ options, logger }) {
39+
const { source, targets, dryRun } = options;
40+
this.checkExtension(source, targets);
41+
const sourceFile = await this.readAndConvertFile(source);
42+
for (const target of targets) {
43+
logger.info(`Runs on ${target}`);
44+
const targetFile = await this.readAndConvertFile(target);
45+
const updater = this.checkAndUpdate({ baseLanguage: sourceFile, targetLanguage: targetFile, logger, dryRun });
46+
const withoutOld = this.clearOldValues({ baseLanguage: sourceFile, targetLanguage: updater, logger, dryRun });
47+
if (!dryRun) {
48+
await this.writeFile(target, withoutOld);
49+
}
50+
logger.info(
51+
`Done with ${this.informations.new} new key found, ${this.informations.write} written, ${this.informations.old} old translation and ${this.informations.delete} deleted.`,
52+
);
53+
this.informations = {
54+
new: 0,
55+
old: 0,
56+
write: 0,
57+
delete: 0,
58+
};
59+
}
60+
}
61+
62+
checkExtension(source, targets) {
63+
const sourceExtension = path.extname(source);
64+
if (sourceExtension !== '.json') {
65+
throw new Error('The source file must be a JSON file');
66+
}
67+
if (targets.map((target) => path.extname(target)).some((ext) => ext !== '.json')) {
68+
throw new Error('All target files must be JSON files');
69+
}
70+
return true;
71+
}
72+
73+
async readAndConvertFile(filePath) {
74+
const file = await fs.promises.readFile(filePath, 'utf-8');
75+
return JSON.parse(file);
76+
}
77+
78+
async writeFile(filePath, content) {
79+
await fs.promises.writeFile(filePath, JSON.stringify(content, null, 2));
80+
}
81+
82+
checkAndUpdate({ baseLanguage, targetLanguage, dryRun, logger }) {
83+
for (const key in baseLanguage) {
84+
if (typeof baseLanguage[key] === 'object') {
85+
targetLanguage[key] = this.checkAndUpdate({
86+
baseLanguage: baseLanguage[key],
87+
targetLanguage: targetLanguage[key] || {},
88+
logger,
89+
dryRun,
90+
});
91+
} else {
92+
if (!targetLanguage[key]) {
93+
logger.info(`New key found: ${key}`);
94+
if (!dryRun) {
95+
targetLanguage[key] = `*${baseLanguage[key]}`;
96+
this.informations.write++;
97+
}
98+
this.informations.new++;
99+
}
100+
}
101+
}
102+
return targetLanguage;
103+
}
104+
105+
clearOldValues({ baseLanguage, targetLanguage, logger, dryRun }) {
106+
for (const key in targetLanguage) {
107+
if (typeof targetLanguage[key] === 'object') {
108+
targetLanguage[key] = this.clearOldValues({
109+
baseLanguage: baseLanguage[key] || {},
110+
targetLanguage: targetLanguage[key],
111+
logger,
112+
dryRun,
113+
});
114+
}
115+
if (!baseLanguage[key]) {
116+
logger.info(`Old key found: ${key}`);
117+
if (!dryRun) {
118+
delete targetLanguage[key];
119+
this.informations.delete++;
120+
}
121+
this.informations.old++;
122+
}
123+
}
124+
return targetLanguage;
125+
}
126+
}
127+
128+
await ScriptRunner.execute(import.meta.url, UpdateTranslations);

api/scripts/wait-for-api-deployment.js

-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,6 @@ while (true) {
2121
async function fetchApiInfoURL() {
2222
console.info('Fetching', apiInfoURL.href);
2323

24-
// eslint-disable-next-line n/no-unsupported-features/node-builtins
2524
const res = await fetch(apiInfoURL);
2625
if (!res.ok) {
2726
if (res.status === 404) {
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,123 @@
1+
import fs from 'node:fs';
2+
3+
import { UpdateTranslations } from '../../../scripts/update-translations.js';
4+
import { expect, sinon } from '../../test-helper.js';
5+
6+
describe('integration | scripts | update-translations', function () {
7+
let script;
8+
let logger;
9+
10+
beforeEach(function () {
11+
logger = { info: sinon.stub() };
12+
script = new UpdateTranslations();
13+
sinon.stub(fs.promises, 'writeFile');
14+
sinon.stub(fs.promises, 'readFile');
15+
});
16+
17+
describe('#handle', function () {
18+
context('when a file is not a json', function () {
19+
it('should return an error if source is not a json', function () {
20+
// given
21+
const source = 'source';
22+
const target = ['target.json'];
23+
24+
// when
25+
const result = script.handle(source, target, logger);
26+
27+
// then
28+
expect(result).to.be.rejectedWith('The source file must be a JSON file');
29+
});
30+
it('should return an error if one target file is not a json', function () {
31+
// given
32+
const source = 'source.json';
33+
const target = ['target1'];
34+
35+
// when
36+
const result = script.handle(source, target, logger);
37+
38+
// then
39+
expect(result).to.be.rejectedWith('The target file must be a JSON file');
40+
});
41+
});
42+
43+
describe('dryRun mode', function () {
44+
it('should display translations informations', async function () {
45+
// given
46+
const source = 'source.json';
47+
const targets = ['target.json'];
48+
fs.promises.readFile.withArgs(source).resolves('{"key": "value", "key2": "value2"}');
49+
fs.promises.readFile.withArgs(targets[0]).resolves('{"key": "value"}');
50+
51+
// when
52+
await script.handle({ options: { source, targets, dryRun: true }, logger });
53+
54+
// then
55+
expect(logger.info).to.have.been.calledWith('New key found: key2');
56+
expect(logger.info).to.have.been.calledWith(
57+
'Done with 1 new key found, 0 written, 0 old translation and 0 deleted.',
58+
);
59+
});
60+
});
61+
62+
describe('without dryRun', function () {
63+
it('should actualize translations', async function () {
64+
// given
65+
const source = 'source.json';
66+
const targets = ['target1.json', 'target2.json'];
67+
const sourceTranslation = {
68+
key: 'value',
69+
key2: 'value2',
70+
object: { subkey1: 'subvalue1', subkey2: 'subvalue2' },
71+
};
72+
const targetTranslation1 = {
73+
key: 'value',
74+
object: { subkey1: 'subvalue1', subkey2: 'subvalue2' },
75+
};
76+
const expectedTranslation1 = {
77+
key: 'value',
78+
object: { subkey1: 'subvalue1', subkey2: 'subvalue2' },
79+
key2: '*value2',
80+
};
81+
const targetTranslation2 = { key: 'value', key3: 'value3', object: { subkey1: 'subvalue1' } };
82+
const expectedTranslation2 = {
83+
key: 'value',
84+
object: { subkey1: 'subvalue1', subkey2: '*subvalue2' },
85+
key2: '*value2',
86+
};
87+
fs.promises.readFile.withArgs(source).resolves(JSON.stringify(sourceTranslation));
88+
fs.promises.readFile.withArgs(targets[0]).resolves(JSON.stringify(targetTranslation1));
89+
fs.promises.readFile.withArgs(targets[1]).resolves(JSON.stringify(targetTranslation2));
90+
fs.promises.writeFile.withArgs(targets[0], expectedTranslation1).resolves();
91+
fs.promises.writeFile.withArgs(targets[1], expectedTranslation2).resolves();
92+
93+
// when
94+
await script.handle({ options: { source, targets, dryRun: false }, logger });
95+
96+
// then
97+
const expectedLogs = [
98+
'Runs on target1.json',
99+
'New key found: key2',
100+
'Done with 1 new key found, 1 written, 0 old translation and 0 deleted.',
101+
'Runs on target2.json',
102+
'New key found: key2',
103+
'New key found: subkey2',
104+
'Old key found: key3',
105+
'Done with 2 new key found, 2 written, 1 old translation and 1 deleted.',
106+
];
107+
expect(logger.info).to.have.callCount(expectedLogs.length);
108+
expectedLogs.forEach((log, index) => {
109+
expect(logger.info.getCall(index)).to.have.been.calledWith(log);
110+
});
111+
expect(fs.promises.writeFile).to.have.callCount(2);
112+
expect(fs.promises.writeFile).to.have.been.calledWith(
113+
targets[0],
114+
JSON.stringify(expectedTranslation1, null, 2),
115+
);
116+
expect(fs.promises.writeFile).to.have.been.calledWith(
117+
targets[1],
118+
JSON.stringify(expectedTranslation2, null, 2),
119+
);
120+
});
121+
});
122+
});
123+
});

0 commit comments

Comments
 (0)