mirror of
https://github.com/NodeBB/NodeBB.git
synced 2025-11-02 03:55:55 +01:00
Load languages with other plugin data
- Added more tests - Should speed up builds - Allows for incremental builds in the future
This commit is contained in:
@@ -6,6 +6,7 @@ var async = require('async');
|
|||||||
var fs = require('fs');
|
var fs = require('fs');
|
||||||
var mkdirp = require('mkdirp');
|
var mkdirp = require('mkdirp');
|
||||||
var rimraf = require('rimraf');
|
var rimraf = require('rimraf');
|
||||||
|
var _ = require('underscore');
|
||||||
|
|
||||||
var file = require('../file');
|
var file = require('../file');
|
||||||
var Plugins = require('../plugins');
|
var Plugins = require('../plugins');
|
||||||
@@ -15,73 +16,34 @@ var coreLanguagesPath = path.join(__dirname, '../../public/language');
|
|||||||
|
|
||||||
function getTranslationTree(callback) {
|
function getTranslationTree(callback) {
|
||||||
async.waterfall([
|
async.waterfall([
|
||||||
// get plugin data
|
|
||||||
Plugins.data.getActive,
|
|
||||||
|
|
||||||
// generate list of languages and namespaces
|
// generate list of languages and namespaces
|
||||||
function (plugins, next) {
|
function (next) {
|
||||||
|
file.walk(coreLanguagesPath, next);
|
||||||
|
},
|
||||||
|
function (paths, next) {
|
||||||
var languages = [];
|
var languages = [];
|
||||||
var namespaces = [];
|
var namespaces = [];
|
||||||
|
|
||||||
// pull languages and namespaces from paths
|
paths.forEach(function (p) {
|
||||||
function extrude(languageDir, paths) {
|
if (!p.endsWith('.json')) {
|
||||||
paths.forEach(function (p) {
|
return;
|
||||||
var rel = p.split(languageDir)[1].split(/[/\\]/).slice(1);
|
|
||||||
var language = rel.shift().replace('_', '-').replace('@', '-x-');
|
|
||||||
var namespace = rel.join('/').replace(/\.json$/, '');
|
|
||||||
|
|
||||||
if (!language || !namespace) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (languages.indexOf(language) === -1) {
|
|
||||||
languages.push(language);
|
|
||||||
}
|
|
||||||
if (namespaces.indexOf(namespace) === -1) {
|
|
||||||
namespaces.push(namespace);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
plugins = plugins.filter(function (pluginData) {
|
|
||||||
return (typeof pluginData.languages === 'string');
|
|
||||||
});
|
|
||||||
async.parallel([
|
|
||||||
// get core languages and namespaces
|
|
||||||
function (nxt) {
|
|
||||||
file.walk(coreLanguagesPath, function (err, paths) {
|
|
||||||
if (err) {
|
|
||||||
return nxt(err);
|
|
||||||
}
|
|
||||||
|
|
||||||
extrude(coreLanguagesPath, paths);
|
|
||||||
nxt();
|
|
||||||
});
|
|
||||||
},
|
|
||||||
// get plugin languages and namespaces
|
|
||||||
function (nxt) {
|
|
||||||
async.each(plugins, function (pluginData, cb) {
|
|
||||||
var pathToFolder = path.join(__dirname, '../../node_modules/', pluginData.id, pluginData.languages);
|
|
||||||
file.walk(pathToFolder, function (err, paths) {
|
|
||||||
if (err) {
|
|
||||||
return cb(err);
|
|
||||||
}
|
|
||||||
|
|
||||||
extrude(pathToFolder, paths);
|
|
||||||
cb();
|
|
||||||
});
|
|
||||||
}, nxt);
|
|
||||||
},
|
|
||||||
], function (err) {
|
|
||||||
if (err) {
|
|
||||||
return next(err);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
next(null, {
|
var rel = path.relative(coreLanguagesPath, p).split(/[/\\]/);
|
||||||
languages: languages,
|
var language = rel.shift().replace('_', '-').replace('@', '-x-');
|
||||||
namespaces: namespaces,
|
var namespace = rel.join('/').replace(/\.json$/, '');
|
||||||
plugins: plugins,
|
|
||||||
});
|
if (!language || !namespace) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
languages.push(language);
|
||||||
|
namespaces.push(namespace);
|
||||||
|
});
|
||||||
|
|
||||||
|
next(null, {
|
||||||
|
languages: _.union(languages, Plugins.languageData.languages).sort().filter(Boolean),
|
||||||
|
namespaces: _.union(namespaces, Plugins.languageData.namespaces).sort().filter(Boolean),
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
@@ -94,8 +56,8 @@ function getTranslationTree(callback) {
|
|||||||
},
|
},
|
||||||
function (x, next) {
|
function (x, next) {
|
||||||
fs.writeFile(path.join(buildLanguagesPath, 'metadata.json'), JSON.stringify({
|
fs.writeFile(path.join(buildLanguagesPath, 'metadata.json'), JSON.stringify({
|
||||||
languages: ref.languages.sort(),
|
languages: ref.languages,
|
||||||
namespaces: ref.namespaces.sort(),
|
namespaces: ref.namespaces,
|
||||||
}), next);
|
}), next);
|
||||||
},
|
},
|
||||||
function (next) {
|
function (next) {
|
||||||
@@ -110,40 +72,42 @@ function getTranslationTree(callback) {
|
|||||||
function (ref, next) {
|
function (ref, next) {
|
||||||
var languages = ref.languages;
|
var languages = ref.languages;
|
||||||
var namespaces = ref.namespaces;
|
var namespaces = ref.namespaces;
|
||||||
var plugins = ref.plugins;
|
var plugins = _.values(Plugins.pluginsData).filter(function (plugin) {
|
||||||
|
return typeof plugin.languages === 'string';
|
||||||
|
});
|
||||||
|
|
||||||
var tree = {};
|
var tree = {};
|
||||||
|
|
||||||
async.eachLimit(languages, 10, function (lang, nxt) {
|
async.eachLimit(languages, 10, function (lang, next) {
|
||||||
async.eachLimit(namespaces, 10, function (ns, cb) {
|
async.eachLimit(namespaces, 10, function (namespace, next) {
|
||||||
var translations = {};
|
var translations = {};
|
||||||
|
|
||||||
async.series([
|
async.series([
|
||||||
// core first
|
// core first
|
||||||
function (n) {
|
function (cb) {
|
||||||
fs.readFile(path.join(coreLanguagesPath, lang, ns + '.json'), function (err, buffer) {
|
fs.readFile(path.join(coreLanguagesPath, lang, namespace + '.json'), function (err, buffer) {
|
||||||
if (err) {
|
if (err) {
|
||||||
if (err.code === 'ENOENT') {
|
if (err.code === 'ENOENT') {
|
||||||
return n();
|
return cb();
|
||||||
}
|
}
|
||||||
return n(err);
|
return cb(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Object.assign(translations, JSON.parse(buffer.toString()));
|
Object.assign(translations, JSON.parse(buffer.toString()));
|
||||||
n();
|
cb();
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
n(err);
|
cb(err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
function (n) {
|
function (cb) {
|
||||||
// for each plugin, fallback in this order:
|
// for each plugin, fallback in this order:
|
||||||
// 1. correct language string (en-GB)
|
// 1. correct language string (en-GB)
|
||||||
// 2. old language string (en_GB)
|
// 2. old language string (en_GB)
|
||||||
// 3. corrected plugin defaultLang (en-US)
|
// 3. corrected plugin defaultLang (en-US)
|
||||||
// 4. old plugin defaultLang (en_US)
|
// 4. old plugin defaultLang (en_US)
|
||||||
async.eachLimit(plugins, 10, function (pluginData, call) {
|
async.eachLimit(plugins, 10, function (pluginData, done) {
|
||||||
var pluginLanguages = path.join(__dirname, '../../node_modules/', pluginData.id, pluginData.languages);
|
var pluginLanguages = path.join(__dirname, '../../node_modules/', pluginData.id, pluginData.languages);
|
||||||
var defaultLang = pluginData.defaultLang || 'en-GB';
|
var defaultLang = pluginData.defaultLang || 'en-GB';
|
||||||
|
|
||||||
@@ -153,7 +117,7 @@ function getTranslationTree(callback) {
|
|||||||
defaultLang.replace('_', '-').replace('@', '-x-'),
|
defaultLang.replace('_', '-').replace('@', '-x-'),
|
||||||
defaultLang.replace('-', '_').replace('-x-', '@'),
|
defaultLang.replace('-', '_').replace('-x-', '@'),
|
||||||
], function (language, next) {
|
], function (language, next) {
|
||||||
fs.readFile(path.join(pluginLanguages, language, ns + '.json'), function (err, buffer) {
|
fs.readFile(path.join(pluginLanguages, language, namespace + '.json'), function (err, buffer) {
|
||||||
if (err) {
|
if (err) {
|
||||||
if (err.code === 'ENOENT') {
|
if (err.code === 'ENOENT') {
|
||||||
return next(null, false);
|
return next(null, false);
|
||||||
@@ -168,21 +132,21 @@ function getTranslationTree(callback) {
|
|||||||
next(err);
|
next(err);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}, call);
|
}, done);
|
||||||
}, function (err) {
|
}, function (err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return n(err);
|
return cb(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Object.keys(translations).length) {
|
if (Object.keys(translations).length) {
|
||||||
tree[lang] = tree[lang] || {};
|
tree[lang] = tree[lang] || {};
|
||||||
tree[lang][ns] = translations;
|
tree[lang][namespace] = translations;
|
||||||
}
|
}
|
||||||
n();
|
cb();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
], cb);
|
], next);
|
||||||
}, nxt);
|
}, next);
|
||||||
}, function (err) {
|
}, function (err) {
|
||||||
next(err, tree);
|
next(err, tree);
|
||||||
});
|
});
|
||||||
@@ -193,9 +157,9 @@ function getTranslationTree(callback) {
|
|||||||
// write translation hashes from the generated tree to language files
|
// write translation hashes from the generated tree to language files
|
||||||
function writeLanguageFiles(tree, callback) {
|
function writeLanguageFiles(tree, callback) {
|
||||||
// iterate over languages and namespaces
|
// iterate over languages and namespaces
|
||||||
async.eachLimit(Object.keys(tree), 10, function (language, cb) {
|
async.eachLimit(Object.keys(tree), 100, function (language, cb) {
|
||||||
var namespaces = tree[language];
|
var namespaces = tree[language];
|
||||||
async.eachLimit(Object.keys(namespaces), 100, function (namespace, next) {
|
async.eachLimit(Object.keys(namespaces), 10, function (namespace, next) {
|
||||||
var translations = namespaces[namespace];
|
var translations = namespaces[namespace];
|
||||||
|
|
||||||
var filePath = path.join(buildLanguagesPath, language, namespace + '.json');
|
var filePath = path.join(buildLanguagesPath, language, namespace + '.json');
|
||||||
|
|||||||
@@ -23,6 +23,7 @@ var middleware;
|
|||||||
Plugins.getPluginPaths = Plugins.data.getPluginPaths;
|
Plugins.getPluginPaths = Plugins.data.getPluginPaths;
|
||||||
Plugins.loadPluginInfo = Plugins.data.loadPluginInfo;
|
Plugins.loadPluginInfo = Plugins.data.loadPluginInfo;
|
||||||
|
|
||||||
|
Plugins.pluginsData = {};
|
||||||
Plugins.libraries = {};
|
Plugins.libraries = {};
|
||||||
Plugins.loadedHooks = {};
|
Plugins.loadedHooks = {};
|
||||||
Plugins.staticDirs = {};
|
Plugins.staticDirs = {};
|
||||||
@@ -33,6 +34,7 @@ var middleware;
|
|||||||
Plugins.libraryPaths = [];
|
Plugins.libraryPaths = [];
|
||||||
Plugins.versionWarning = [];
|
Plugins.versionWarning = [];
|
||||||
Plugins.soundpacks = [];
|
Plugins.soundpacks = [];
|
||||||
|
Plugins.languageData = {};
|
||||||
|
|
||||||
Plugins.initialized = false;
|
Plugins.initialized = false;
|
||||||
|
|
||||||
|
|||||||
@@ -299,3 +299,38 @@ function getSoundpack(pluginData, callback) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
Data.getSoundpack = getSoundpack;
|
Data.getSoundpack = getSoundpack;
|
||||||
|
|
||||||
|
function getLanguageData(pluginData, callback) {
|
||||||
|
if (typeof pluginData.languages !== 'string') {
|
||||||
|
return callback();
|
||||||
|
}
|
||||||
|
|
||||||
|
var pathToFolder = path.join(__dirname, '../../node_modules/', pluginData.id, pluginData.languages);
|
||||||
|
file.walk(pathToFolder, function (err, paths) {
|
||||||
|
if (err) {
|
||||||
|
return callback(err);
|
||||||
|
}
|
||||||
|
|
||||||
|
var namespaces = [];
|
||||||
|
var languages = [];
|
||||||
|
|
||||||
|
paths.forEach(function (p) {
|
||||||
|
var rel = path.relative(pathToFolder, p).split(/[/\\]/);
|
||||||
|
var language = rel.shift().replace('_', '-').replace('@', '-x-');
|
||||||
|
var namespace = rel.join('/').replace(/\.json$/, '');
|
||||||
|
|
||||||
|
if (!language || !namespace) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
languages.push(language);
|
||||||
|
namespaces.push(namespace);
|
||||||
|
});
|
||||||
|
|
||||||
|
callback(null, {
|
||||||
|
languages: languages,
|
||||||
|
namespaces: namespaces,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
Data.getLanguageData = getLanguageData;
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ var semver = require('semver');
|
|||||||
var async = require('async');
|
var async = require('async');
|
||||||
var winston = require('winston');
|
var winston = require('winston');
|
||||||
var nconf = require('nconf');
|
var nconf = require('nconf');
|
||||||
|
var _ = require('underscore');
|
||||||
|
|
||||||
var meta = require('../meta');
|
var meta = require('../meta');
|
||||||
|
|
||||||
@@ -36,6 +37,9 @@ module.exports = function (Plugins) {
|
|||||||
soundpack: function (next) {
|
soundpack: function (next) {
|
||||||
Plugins.data.getSoundpack(pluginData, next);
|
Plugins.data.getSoundpack(pluginData, next);
|
||||||
},
|
},
|
||||||
|
languageData: function (next) {
|
||||||
|
Plugins.data.getLanguageData(pluginData, next);
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
var methods;
|
var methods;
|
||||||
@@ -62,6 +66,11 @@ module.exports = function (Plugins) {
|
|||||||
if (results.soundpack) {
|
if (results.soundpack) {
|
||||||
Plugins.soundpacks.push(results.soundpack);
|
Plugins.soundpacks.push(results.soundpack);
|
||||||
}
|
}
|
||||||
|
if (results.languageData) {
|
||||||
|
Plugins.languageData.languages = _.union(Plugins.languageData.languages, results.languageData.languages);
|
||||||
|
Plugins.languageData.namespaces = _.union(Plugins.languageData.namespaces, results.languageData.namespaces);
|
||||||
|
}
|
||||||
|
Plugins.pluginsData[pluginData.id] = pluginData;
|
||||||
|
|
||||||
callback();
|
callback();
|
||||||
});
|
});
|
||||||
@@ -73,6 +82,8 @@ module.exports = function (Plugins) {
|
|||||||
Plugins.clientScripts.length = 0;
|
Plugins.clientScripts.length = 0;
|
||||||
Plugins.acpScripts.length = 0;
|
Plugins.acpScripts.length = 0;
|
||||||
Plugins.soundpacks.length = 0;
|
Plugins.soundpacks.length = 0;
|
||||||
|
Plugins.languageData.languages = [];
|
||||||
|
Plugins.languageData.namespaces = [];
|
||||||
|
|
||||||
var map = {
|
var map = {
|
||||||
'plugin static dirs': ['staticDirs'],
|
'plugin static dirs': ['staticDirs'],
|
||||||
@@ -82,6 +93,7 @@ module.exports = function (Plugins) {
|
|||||||
'client side styles': ['cssFiles', 'lessFiles'],
|
'client side styles': ['cssFiles', 'lessFiles'],
|
||||||
'admin control panel styles': ['cssFiles', 'lessFiles'],
|
'admin control panel styles': ['cssFiles', 'lessFiles'],
|
||||||
sounds: ['soundpack'],
|
sounds: ['soundpack'],
|
||||||
|
languages: ['languageData'],
|
||||||
};
|
};
|
||||||
|
|
||||||
var fields = targets.reduce(function (prev, target) {
|
var fields = targets.reduce(function (prev, target) {
|
||||||
|
|||||||
@@ -107,6 +107,7 @@ describe('Build', function (done) {
|
|||||||
async.parallel([
|
async.parallel([
|
||||||
async.apply(rimraf, path.join(__dirname, '../build/public')),
|
async.apply(rimraf, path.join(__dirname, '../build/public')),
|
||||||
db.setupMockDefaults,
|
db.setupMockDefaults,
|
||||||
|
async.apply(db.activatePlugin, 'nodebb-plugin-markdown'),
|
||||||
], done);
|
], done);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -181,10 +182,17 @@ describe('Build', function (done) {
|
|||||||
it('should build languages', function (done) {
|
it('should build languages', function (done) {
|
||||||
build.build(['languages'], function (err) {
|
build.build(['languages'], function (err) {
|
||||||
assert.ifError(err);
|
assert.ifError(err);
|
||||||
var filename = path.join(__dirname, '../build/public/language/en-GB/global.json');
|
|
||||||
assert(file.existsSync(filename));
|
var globalFile = path.join(__dirname, '../build/public/language/en-GB/global.json');
|
||||||
var global = fs.readFileSync(filename).toString();
|
assert(file.existsSync(globalFile));
|
||||||
|
var global = fs.readFileSync(globalFile).toString();
|
||||||
assert.strictEqual(JSON.parse(global).home, 'Home');
|
assert.strictEqual(JSON.parse(global).home, 'Home');
|
||||||
|
|
||||||
|
var mdFile = path.join(__dirname, '../build/public/language/en-GB/markdown.json');
|
||||||
|
assert(file.existsSync(mdFile));
|
||||||
|
var md = fs.readFileSync(mdFile).toString();
|
||||||
|
assert.strictEqual(JSON.parse(md).bold, 'bolded text');
|
||||||
|
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -192,8 +200,15 @@ describe('Build', function (done) {
|
|||||||
it('should build sounds', function (done) {
|
it('should build sounds', function (done) {
|
||||||
build.build(['sounds'], function (err) {
|
build.build(['sounds'], function (err) {
|
||||||
assert.ifError(err);
|
assert.ifError(err);
|
||||||
var filename = path.join(__dirname, '../build/public/sounds/fileMap.json');
|
|
||||||
assert(file.existsSync(filename));
|
var mapFile = path.join(__dirname, '../build/public/sounds/fileMap.json');
|
||||||
|
assert(file.existsSync(mapFile));
|
||||||
|
var fileMap = JSON.parse(fs.readFileSync(mapFile));
|
||||||
|
assert.strictEqual(fileMap['Default | Deedle-dum'], 'nodebb-plugin-soundpack-default/notification.mp3');
|
||||||
|
|
||||||
|
var deebleDumFile = path.join(__dirname, '../build/public/sounds/nodebb-plugin-soundpack-default/notification.mp3');
|
||||||
|
assert(file.existsSync(deebleDumFile));
|
||||||
|
|
||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -178,9 +178,14 @@ function enableDefaultPlugins(callback) {
|
|||||||
|
|
||||||
var defaultEnabled = [
|
var defaultEnabled = [
|
||||||
'nodebb-plugin-dbsearch',
|
'nodebb-plugin-dbsearch',
|
||||||
|
'nodebb-plugin-soundpack-default',
|
||||||
];
|
];
|
||||||
|
|
||||||
winston.info('[install/enableDefaultPlugins] activating default plugins', defaultEnabled);
|
winston.info('[install/enableDefaultPlugins] activating default plugins', defaultEnabled);
|
||||||
|
|
||||||
db.sortedSetAdd('plugins:active', [0], defaultEnabled, callback);
|
db.sortedSetAdd('plugins:active', Object.keys(defaultEnabled), defaultEnabled, callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
db.activatePlugin = function (id, callback) {
|
||||||
|
db.sortedSetAdd('plugins:active', Date.now(), id, callback);
|
||||||
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user