-
Notifications
You must be signed in to change notification settings - Fork 984
/
Copy pathwebpack.plugin.localize.js
174 lines (155 loc) · 6.37 KB
/
webpack.plugin.localize.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
/* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
/* This is a webpack plugin.
* This plugin generates one javascript bundle per locale.
*
* It replaces the javascript translation function arguments with the locale-specific data.
* The javascript functions are in `warehouse/static/js/warehouse/utils/messages-access.js`.
*
* Run 'make translations' before webpack to extract the translatable text to gettext format files.
*/
// ref: https://webpack.js.org/contribute/writing-a-plugin/
// ref: https://github.com/zainulbr/i18n-webpack-plugin/blob/v2.0.3/src/index.js
// ref: https://github.com/webpack/webpack/discussions/14956
// ref: https://github.com/webpack/webpack/issues/9992
/* global module, __dirname */
const ConstDependency = require("webpack/lib/dependencies/ConstDependency");
const fs = require("node:fs");
const {resolve} = require("node:path");
const path = require("path");
const gettextParser = require("gettext-parser");
// generate and then load the locale translation data
const baseDir = __dirname;
const localeDir = path.resolve(baseDir, "warehouse/locale");
// This list should match `warehouse.i18n.KNOWN_LOCALES`
const KNOWN_LOCALES = [
"en", // English
"es", // Spanish
"fr", // French
"ja", // Japanese
"pt_BR", // Brazilian Portuguese
"uk", // Ukrainian
"el", // Greek
"de", // German
"zh_Hans", // Simplified Chinese
"zh_Hant", // Traditional Chinese
"ru", // Russian
"he", // Hebrew
"eo", // Esperanto
"ko", // Korean
];
// A custom regular expression to do some basic checking of the plural form,
// to try to ensure the plural form expression contains only expected characters.
// - the plural form expression MUST NOT have any type of quotes and
// the only whitespace allowed is space (not tab or form feed)
// - MUST NOT allow brackets other than parentheses (()),
// as allowing braces ({}) might allow ending the function early
// - MUST allow space, number variable (n), numbers, groups (()),
// comparisons (<>!=), ternary expressions (?:), and/or (&|),
// remainder (%)
const pluralFormPattern = new RegExp("^ *nplurals *= *[0-9]+ *; *plural *=[ n0-9()<>!=?:&|%]+;?$");
const allLocaleData = KNOWN_LOCALES
.filter(langCode => langCode !== "en")
.map((langCode) => resolve(localeDir, langCode, "LC_MESSAGES/messages.po"))
.filter((file) => fs.statSync(file).isFile())
.map((file) => ({path: path.relative(baseDir, file), data: fs.readFileSync(file, "utf8")}))
.map((data) => {
try {
const lines = data.data
.split("\n")
// gettext-parser does not support obsolete previous translations,
// so filter out those lines
// see: https://github.com/smhg/gettext-parser/issues/79
.filter(line => !line.startsWith("#~|"))
.join("\n");
const parsed = gettextParser.po.parse(lines);
const language = parsed.headers["Language"];
const pluralForms = parsed.headers["Plural-Forms"];
const result = {
"": {
"language": language,
"plural-forms": pluralForms,
},
};
if (!pluralFormPattern.test(pluralForms)) {
throw new Error(`Invalid plural forms for '${language}': "${pluralForms}"`);
}
const translations = parsed.translations[""];
for (const key in translations) {
if (key === "") {
continue;
}
const value = translations[key];
const refs = value.comments.reference.split("\n");
if (refs.every(refLine => !refLine.includes(".js:"))) {
continue;
}
result[value.msgid] = value.msgstr.map(function(str) {
return str
.replace(/&/g, "&")
.replace(/</g, "<")
.replace(/>/g, ">")
.replace(/"/g, """)
.replace(/'/g, "'");
});
}
return result;
} catch (e) {
throw new Error(`Could not parse file ${data.path}: ${e.message}\n${e}`);
}
});
const pluginName = "WebpackLocalisationPlugin";
class WebpackLocalisationPlugin {
constructor(localeData) {
this.localeData = localeData || {};
}
apply(compiler) {
const self = this;
// create a handler for each factory.hooks.parser
const handler = function (parser) {
parser.hooks.statement.tap(pluginName, (statement) => {
if (statement.type === "VariableDeclaration" &&
statement.declarations.length === 1 &&
statement.declarations[0].id.name === "messagesAccessLocaleData") {
const initData = statement.declarations[0].init;
const dep = new ConstDependency(JSON.stringify(self.localeData), initData.range);
dep.loc = initData.loc;
parser.state.current.addDependency(dep);
return true;
} else if (statement.type === "VariableDeclaration" &&
statement.declarations.length === 1 &&
statement.declarations[0].id.name === "messagesAccessPluralFormFunction") {
const initData = statement.declarations[0].init;
const pluralForms = self.localeData[""]["plural-forms"];
const newValue = `function (n) {
let nplurals, plural;
${pluralForms}
return {total: nplurals, index: ((nplurals > 1 && plural === true) ? 1 : (plural ? plural : 0))};
}`;
const dep = new ConstDependency(newValue, initData.range);
dep.loc = initData.loc;
parser.state.current.addDependency(dep);
return true;
}
});
};
// place the handler into the hooks for the webpack compiler module factories
compiler.hooks.normalModuleFactory.tap(pluginName, factory => {
factory.hooks.parser.for("javascript/auto").tap(pluginName, handler);
factory.hooks.parser.for("javascript/dynamic").tap(pluginName, handler);
factory.hooks.parser.for("javascript/esm").tap(pluginName, handler);
});
}
}
module.exports.WebpackLocalisationPlugin = WebpackLocalisationPlugin;
module.exports.allLocaleData = allLocaleData;