docusaurus/lib/write-translations.js
Laxman cfabaedc99 Fix: conflicting strings issue in translations (#917)
* Fix conflicting strings issue in translations

* Preserve structure of `customTranslations`

* Use `deepmerge` to merge whole of `localized-strings`

* Simplify and make deep property access on an object safe

* Fix deep property accessor and rename it to idx
2018-08-29 00:04:02 +08:00

211 lines
5.9 KiB
JavaScript
Executable file

#!/usr/bin/env node
/**
* Copyright (c) 2017-present, Facebook, Inc.
*
* This source code is licensed under the MIT license found in the
* LICENSE file in the root directory of this source tree.
*/
/* generate the i18n/en.json file */
require('babel-register')({
babelrc: false,
only: [__dirname, `${process.cwd()}/core`],
plugins: [
require('./server/translate-plugin.js'),
'transform-class-properties',
'transform-object-rest-spread',
],
presets: ['react', 'env'],
});
const traverse = require('babel-traverse').default;
const babylon = require('babylon');
const fs = require('fs-extra');
const glob = require('glob');
const mkdirp = require('mkdirp');
const nodePath = require('path');
const deepmerge = require('deepmerge');
const readMetadata = require('./server/readMetadata.js');
const CWD = process.cwd();
const siteConfig = require(`${CWD}/siteConfig.js`);
const sidebars = require(`${CWD}/sidebars.json`);
let customTranslations = {
'localized-strings': {
docs: {},
links: {},
categories: {},
},
'pages-strings': {},
};
if (fs.existsSync(`${CWD}/data/custom-translation-strings.json`)) {
customTranslations = deepmerge(
JSON.parse(
fs.readFileSync(`${CWD}/data/custom-translation-strings.json`, 'utf8')
),
customTranslations
);
}
function writeFileAndCreateFolder(file, content) {
mkdirp.sync(file.replace(new RegExp('/[^/]*$'), ''));
fs.writeFileSync(file, content);
}
function execute() {
const translations = {
'localized-strings': {
next: 'Next',
previous: 'Previous',
tagline: siteConfig.tagline,
docs: {},
links: {},
categories: {},
},
'pages-strings': {},
};
// look through markdown headers of docs for titles and categories to translate
const docsDir = nodePath.join(CWD, '../', readMetadata.getDocsPath());
const versionedDocsDir = nodePath.join(CWD, 'versioned_docs');
let files = [
...glob.sync(`${docsDir}/**`),
...glob.sync(`${versionedDocsDir}/**`),
];
files.forEach(file => {
const extension = nodePath.extname(file);
if (extension === '.md' || extension === '.markdown') {
let res;
try {
res = readMetadata.processMetadata(file, docsDir);
} catch (e) {
console.error(e);
process.exit(1);
}
if (!res) {
return;
}
const metadata = res.metadata;
const id = metadata.localized_id;
translations['localized-strings'].docs[id] = {};
translations['localized-strings'].docs[id].title = metadata.title;
if (metadata.sidebar_label) {
translations['localized-strings'].docs[id].sidebar_label =
metadata.sidebar_label;
}
}
});
// look through header links for text to translate
siteConfig.headerLinks.forEach(link => {
if (link.label) {
translations['localized-strings'].links[link.label] = link.label;
}
});
// find sidebar category titles to translate
Object.keys(sidebars).forEach(sb => {
const categories = sidebars[sb];
Object.keys(categories).forEach(category => {
translations['localized-strings'].categories[category] = category;
});
});
files = glob.sync(`${CWD}/versioned_sidebars/*`);
files.forEach(file => {
if (!file.endsWith('-sidebars.json')) {
if (file.endsWith('-sidebar.json')) {
console.warn(
`Skipping ${file}. Make sure your sidebar filenames follow this format: 'version-VERSION-sidebars.json'.`
);
}
return;
}
let sidebarContent;
try {
sidebarContent = JSON.parse(fs.readFileSync(file, 'utf8'));
} catch (e) {
console.error(`Could not parse ${file} into json. ${e}`);
process.exit(1);
}
Object.keys(sidebarContent).forEach(sb => {
const categories = sidebarContent[sb];
Object.keys(categories).forEach(category => {
translations['localized-strings'].categories[category] = category;
});
});
});
// go through pages to look for text inside translate tags
files = glob.sync(`${CWD}/pages/en/**`);
files.forEach(file => {
const extension = nodePath.extname(file);
if (extension === '.js') {
const ast = babylon.parse(fs.readFileSync(file, 'utf8'), {
plugins: ['jsx'],
});
traverse(ast, {
enter(path) {
if (
path.node.type === 'JSXElement' &&
path.node.openingElement.name.name === 'translate'
) {
const text = path.node.children[0].value
.trim()
.replace(/\s+/g, ' ');
let description = 'no description given';
const attributes = path.node.openingElement.attributes;
for (let i = 0; i < attributes.length; i++) {
if (attributes[i].name.name === 'desc') {
description = attributes[i].value.value;
}
}
translations['pages-strings'][`${text}|${description}`] = text;
}
},
});
}
});
// Manually add 'Help Translate' to en.json
translations['pages-strings'][
'Help Translate|recruit community translators for your project'
] = 'Help Translate';
translations['pages-strings'][
'Edit this Doc|recruitment message asking to edit the doc source'
] = 'Edit';
translations['pages-strings'][
'Translate this Doc|recruitment message asking to translate the docs'
] = 'Translate';
translations['pages-strings'] = Object.assign(
translations['pages-strings'],
customTranslations['pages-strings']
);
translations['localized-strings'] = deepmerge(
translations['localized-strings'],
customTranslations['localized-strings']
);
writeFileAndCreateFolder(
`${CWD}/i18n/en.json`,
`${JSON.stringify(
Object.assign(
{
_comment: 'This file is auto-generated by write-translations.js',
},
translations
),
null,
2
)}\n`
);
}
execute();
module.exports = execute;