mirror of
https://github.com/NodeBB/NodeBB.git
synced 2026-02-28 09:31:17 +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 mkdirp = require('mkdirp');
|
||||
var rimraf = require('rimraf');
|
||||
var _ = require('underscore');
|
||||
|
||||
var file = require('../file');
|
||||
var Plugins = require('../plugins');
|
||||
@@ -15,73 +16,34 @@ var coreLanguagesPath = path.join(__dirname, '../../public/language');
|
||||
|
||||
function getTranslationTree(callback) {
|
||||
async.waterfall([
|
||||
// get plugin data
|
||||
Plugins.data.getActive,
|
||||
|
||||
// generate list of languages and namespaces
|
||||
function (plugins, next) {
|
||||
function (next) {
|
||||
file.walk(coreLanguagesPath, next);
|
||||
},
|
||||
function (paths, next) {
|
||||
var languages = [];
|
||||
var namespaces = [];
|
||||
|
||||
// pull languages and namespaces from paths
|
||||
function extrude(languageDir, paths) {
|
||||
paths.forEach(function (p) {
|
||||
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);
|
||||
paths.forEach(function (p) {
|
||||
if (!p.endsWith('.json')) {
|
||||
return;
|
||||
}
|
||||
|
||||
next(null, {
|
||||
languages: languages,
|
||||
namespaces: namespaces,
|
||||
plugins: plugins,
|
||||
});
|
||||
var rel = path.relative(coreLanguagesPath, 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);
|
||||
});
|
||||
|
||||
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) {
|
||||
fs.writeFile(path.join(buildLanguagesPath, 'metadata.json'), JSON.stringify({
|
||||
languages: ref.languages.sort(),
|
||||
namespaces: ref.namespaces.sort(),
|
||||
languages: ref.languages,
|
||||
namespaces: ref.namespaces,
|
||||
}), next);
|
||||
},
|
||||
function (next) {
|
||||
@@ -110,40 +72,42 @@ function getTranslationTree(callback) {
|
||||
function (ref, next) {
|
||||
var languages = ref.languages;
|
||||
var namespaces = ref.namespaces;
|
||||
var plugins = ref.plugins;
|
||||
var plugins = _.values(Plugins.pluginsData).filter(function (plugin) {
|
||||
return typeof plugin.languages === 'string';
|
||||
});
|
||||
|
||||
var tree = {};
|
||||
|
||||
async.eachLimit(languages, 10, function (lang, nxt) {
|
||||
async.eachLimit(namespaces, 10, function (ns, cb) {
|
||||
async.eachLimit(languages, 10, function (lang, next) {
|
||||
async.eachLimit(namespaces, 10, function (namespace, next) {
|
||||
var translations = {};
|
||||
|
||||
async.series([
|
||||
// core first
|
||||
function (n) {
|
||||
fs.readFile(path.join(coreLanguagesPath, lang, ns + '.json'), function (err, buffer) {
|
||||
function (cb) {
|
||||
fs.readFile(path.join(coreLanguagesPath, lang, namespace + '.json'), function (err, buffer) {
|
||||
if (err) {
|
||||
if (err.code === 'ENOENT') {
|
||||
return n();
|
||||
return cb();
|
||||
}
|
||||
return n(err);
|
||||
return cb(err);
|
||||
}
|
||||
|
||||
try {
|
||||
Object.assign(translations, JSON.parse(buffer.toString()));
|
||||
n();
|
||||
cb();
|
||||
} catch (err) {
|
||||
n(err);
|
||||
cb(err);
|
||||
}
|
||||
});
|
||||
},
|
||||
function (n) {
|
||||
function (cb) {
|
||||
// for each plugin, fallback in this order:
|
||||
// 1. correct language string (en-GB)
|
||||
// 2. old language string (en_GB)
|
||||
// 3. corrected 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 defaultLang = pluginData.defaultLang || 'en-GB';
|
||||
|
||||
@@ -153,7 +117,7 @@ function getTranslationTree(callback) {
|
||||
defaultLang.replace('_', '-').replace('@', '-x-'),
|
||||
defaultLang.replace('-', '_').replace('-x-', '@'),
|
||||
], 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.code === 'ENOENT') {
|
||||
return next(null, false);
|
||||
@@ -168,21 +132,21 @@ function getTranslationTree(callback) {
|
||||
next(err);
|
||||
}
|
||||
});
|
||||
}, call);
|
||||
}, done);
|
||||
}, function (err) {
|
||||
if (err) {
|
||||
return n(err);
|
||||
return cb(err);
|
||||
}
|
||||
|
||||
if (Object.keys(translations).length) {
|
||||
tree[lang] = tree[lang] || {};
|
||||
tree[lang][ns] = translations;
|
||||
tree[lang][namespace] = translations;
|
||||
}
|
||||
n();
|
||||
cb();
|
||||
});
|
||||
},
|
||||
], cb);
|
||||
}, nxt);
|
||||
], next);
|
||||
}, next);
|
||||
}, function (err) {
|
||||
next(err, tree);
|
||||
});
|
||||
@@ -193,9 +157,9 @@ function getTranslationTree(callback) {
|
||||
// write translation hashes from the generated tree to language files
|
||||
function writeLanguageFiles(tree, callback) {
|
||||
// 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];
|
||||
async.eachLimit(Object.keys(namespaces), 100, function (namespace, next) {
|
||||
async.eachLimit(Object.keys(namespaces), 10, function (namespace, next) {
|
||||
var translations = namespaces[namespace];
|
||||
|
||||
var filePath = path.join(buildLanguagesPath, language, namespace + '.json');
|
||||
|
||||
@@ -23,6 +23,7 @@ var middleware;
|
||||
Plugins.getPluginPaths = Plugins.data.getPluginPaths;
|
||||
Plugins.loadPluginInfo = Plugins.data.loadPluginInfo;
|
||||
|
||||
Plugins.pluginsData = {};
|
||||
Plugins.libraries = {};
|
||||
Plugins.loadedHooks = {};
|
||||
Plugins.staticDirs = {};
|
||||
@@ -33,6 +34,7 @@ var middleware;
|
||||
Plugins.libraryPaths = [];
|
||||
Plugins.versionWarning = [];
|
||||
Plugins.soundpacks = [];
|
||||
Plugins.languageData = {};
|
||||
|
||||
Plugins.initialized = false;
|
||||
|
||||
|
||||
@@ -299,3 +299,38 @@ function getSoundpack(pluginData, callback) {
|
||||
});
|
||||
}
|
||||
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 winston = require('winston');
|
||||
var nconf = require('nconf');
|
||||
var _ = require('underscore');
|
||||
|
||||
var meta = require('../meta');
|
||||
|
||||
@@ -36,6 +37,9 @@ module.exports = function (Plugins) {
|
||||
soundpack: function (next) {
|
||||
Plugins.data.getSoundpack(pluginData, next);
|
||||
},
|
||||
languageData: function (next) {
|
||||
Plugins.data.getLanguageData(pluginData, next);
|
||||
},
|
||||
};
|
||||
|
||||
var methods;
|
||||
@@ -62,6 +66,11 @@ module.exports = function (Plugins) {
|
||||
if (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();
|
||||
});
|
||||
@@ -73,6 +82,8 @@ module.exports = function (Plugins) {
|
||||
Plugins.clientScripts.length = 0;
|
||||
Plugins.acpScripts.length = 0;
|
||||
Plugins.soundpacks.length = 0;
|
||||
Plugins.languageData.languages = [];
|
||||
Plugins.languageData.namespaces = [];
|
||||
|
||||
var map = {
|
||||
'plugin static dirs': ['staticDirs'],
|
||||
@@ -82,6 +93,7 @@ module.exports = function (Plugins) {
|
||||
'client side styles': ['cssFiles', 'lessFiles'],
|
||||
'admin control panel styles': ['cssFiles', 'lessFiles'],
|
||||
sounds: ['soundpack'],
|
||||
languages: ['languageData'],
|
||||
};
|
||||
|
||||
var fields = targets.reduce(function (prev, target) {
|
||||
|
||||
@@ -107,6 +107,7 @@ describe('Build', function (done) {
|
||||
async.parallel([
|
||||
async.apply(rimraf, path.join(__dirname, '../build/public')),
|
||||
db.setupMockDefaults,
|
||||
async.apply(db.activatePlugin, 'nodebb-plugin-markdown'),
|
||||
], done);
|
||||
});
|
||||
|
||||
@@ -181,10 +182,17 @@ describe('Build', function (done) {
|
||||
it('should build languages', function (done) {
|
||||
build.build(['languages'], function (err) {
|
||||
assert.ifError(err);
|
||||
var filename = path.join(__dirname, '../build/public/language/en-GB/global.json');
|
||||
assert(file.existsSync(filename));
|
||||
var global = fs.readFileSync(filename).toString();
|
||||
|
||||
var globalFile = path.join(__dirname, '../build/public/language/en-GB/global.json');
|
||||
assert(file.existsSync(globalFile));
|
||||
var global = fs.readFileSync(globalFile).toString();
|
||||
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();
|
||||
});
|
||||
});
|
||||
@@ -192,8 +200,15 @@ describe('Build', function (done) {
|
||||
it('should build sounds', function (done) {
|
||||
build.build(['sounds'], function (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();
|
||||
});
|
||||
});
|
||||
|
||||
@@ -178,9 +178,14 @@ function enableDefaultPlugins(callback) {
|
||||
|
||||
var defaultEnabled = [
|
||||
'nodebb-plugin-dbsearch',
|
||||
'nodebb-plugin-soundpack-default',
|
||||
];
|
||||
|
||||
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