"use strict"; var fs = require('fs'), path = require('path'), async = require('async'), winston = require('winston'), nconf = require('nconf'), _ = require('underscore'), less = require('less'), fork = require('child_process').fork, rimraf = require('rimraf'), mkdirp = require('mkdirp'), utils = require('./../public/src/utils'), translator = require('./../public/src/translator'), db = require('./database'), plugins = require('./plugins'), user = require('./user'), groups = require('./groups'); (function (Meta) { Meta.restartRequired = false; Meta.config = {}; Meta.configs = { init: function (callback) { delete Meta.config; Meta.configs.list(function (err, config) { if(err) { winston.error(err); return callback(err); } Meta.config = config; callback(); }); }, list: function (callback) { db.getObject('config', function (err, config) { if(err) { return callback(err); } config = config || {}; config.status = 'ok'; callback(err, config); }); }, get: function (field, callback) { db.getObjectField('config', field, callback); }, getFields: function (fields, callback) { db.getObjectFields('config', fields, callback); }, set: function (field, value, callback) { if(!field) { return callback(new Error('invalid config field')); } db.setObjectField('config', field, value, function(err, res) { if (callback) { if(!err && Meta.config) { Meta.config[field] = value; } callback(err, res); } }); }, setOnEmpty: function (field, value, callback) { Meta.configs.get(field, function (err, curValue) { if(err) { return callback(err); } if (!curValue) { Meta.configs.set(field, value, callback); } else { callback(); } }); }, remove: function (field) { db.deleteObjectField('config', field); } }; Meta.themes = { get: function (callback) { var themePath = nconf.get('themes_path'); if (typeof themePath !== 'string') { return callback(null, []); } fs.readdir(themePath, function (err, files) { async.filter(files, function (file, next) { fs.stat(path.join(themePath, file), function (err, fileStat) { if (err) { return next(false); } next((fileStat.isDirectory() && file.slice(0, 13) === 'nodebb-theme-')); }); }, function (themes) { async.map(themes, function (theme, next) { var config = path.join(themePath, theme, 'theme.json'); if (fs.existsSync(config)) { fs.readFile(config, function (err, file) { if (err) { return next(); } else { var configObj = JSON.parse(file.toString()); next(err, configObj); } }); } else { next(); } }, function (err, themes) { themes = themes.filter(function (theme) { return (theme !== undefined); }); callback(null, themes); }); }); }); }, set: function(data, callback) { var themeData = { 'theme:type': data.type, 'theme:id': data.id, 'theme:staticDir': '', 'theme:templates': '', 'theme:src': '' }; switch(data.type) { case 'local': async.waterfall([ function(next) { fs.readFile(path.join(nconf.get('themes_path'), data.id, 'theme.json'), function(err, config) { if (!err) { config = JSON.parse(config.toString()); next(null, config); } else { next(err); } }); }, function(config, next) { themeData['theme:staticDir'] = config.staticDir ? config.staticDir : ''; themeData['theme:templates'] = config.templates ? config.templates : ''; themeData['theme:src'] = ''; db.setObject('config', themeData, next); } ], callback); Meta.restartRequired = true; break; case 'bootswatch': Meta.configs.set('theme:src', data.src, callback); break; } } }; Meta.title = { tests: { isCategory: /^category\/\d+\/?/, isTopic: /^topic\/\d+\/?/, isUserPage: /^user\/[^\/]+(\/[\w]+)?/ }, build: function (urlFragment, language, callback) { Meta.title.parseFragment(decodeURIComponent(urlFragment), language, function(err, title) { if (err) { title = Meta.config.browserTitle || 'NodeBB'; } else { title = (title ? title + ' | ' : '') + (Meta.config.browserTitle || 'NodeBB'); } callback(null, title); }); }, parseFragment: function (urlFragment, language, callback) { var translated = ['', 'recent', 'unread', 'users', 'notifications']; if (translated.indexOf(urlFragment) !== -1) { if (!urlFragment.length) { urlFragment = 'home'; } translator.translate('[[pages:' + urlFragment + ']]', language, function(translated) { callback(null, translated); }); } else if (this.tests.isCategory.test(urlFragment)) { var cid = urlFragment.match(/category\/(\d+)/)[1]; require('./categories').getCategoryField(cid, 'name', function (err, name) { callback(null, name); }); } else if (this.tests.isTopic.test(urlFragment)) { var tid = urlFragment.match(/topic\/(\d+)/)[1]; require('./topics').getTopicField(tid, 'title', function (err, title) { callback(null, title); }); } else if (this.tests.isUserPage.test(urlFragment)) { var matches = urlFragment.match(/user\/([^\/]+)\/?([\w]+)?/), userslug = matches[1], subpage = matches[2]; user.getUsernameByUserslug(userslug, function(err, username) { if (subpage) { translator.translate('[[pages:user.' + subpage + ', ' + username + ']]', language, function(translated) { callback(null, translated); }); } else { callback(null, username); } }); } else { callback(null); } } }; Meta.js = { cache: undefined, prepared: false, scripts: [ 'vendor/jquery/js/jquery.js', 'vendor/jquery/js/jquery-ui-1.10.4.custom.js', 'vendor/jquery/timeago/jquery.timeago.min.js', 'vendor/jquery/js/jquery.form.min.js', 'vendor/jquery/serializeObject/jquery.ba-serializeobject.min.js', 'vendor/bootstrap/js/bootstrap.min.js', 'vendor/jquery/bootstrap-tagsinput/bootstrap-tagsinput.min.js', 'vendor/requirejs/require.js', 'vendor/bootbox/bootbox.min.js', 'vendor/tinycon/tinycon.js', 'vendor/xregexp/xregexp.js', 'vendor/xregexp/unicode/unicode-base.js', 'src/utils.js', 'src/app.js', 'src/templates.js', 'src/ajaxify.js', 'src/variables.js', 'src/widgets.js', 'src/translator.js', 'src/helpers.js', 'src/overrides.js' ], minFile: 'nodebb.min.js', prepare: function (callback) { plugins.fireHook('filter:scripts.get', this.scripts, function(err, scripts) { var ctime, jsPaths = scripts.map(function (jsPath) { jsPath = path.normalize(jsPath); if (jsPath.substring(0, 7) === 'plugins') { var matches = _.map(plugins.staticDirs, function(realPath, mappedPath) { if (jsPath.match(mappedPath)) { return mappedPath; } else { return null; } }).filter(function(a) { return a; }); if (matches.length) { var relPath = jsPath.slice(('plugins/' + matches[0]).length), pluginId = matches[0].split(path.sep)[0]; return plugins.staticDirs[matches[0]] + relPath; } else { winston.warn('[meta.scripts.get] Could not resolve mapped path: ' + jsPath + '. Are you sure it is defined by a plugin?'); return null; } } else { return path.join(__dirname, '..', '/public', jsPath); } }); Meta.js.scripts = jsPaths.filter(function(path) { return path !== null; }); // Add plugin scripts Meta.js.scripts = Meta.js.scripts.concat(plugins.clientScripts); Meta.js.prepared = true; callback(); }); }, minify: function(minify) { // Prepare js for minification/concatenation var minifier = this.minifierProc = fork('minifier.js'); minifier.on('message', function(payload) { if (payload.action !== 'error') { winston.info('[meta/js] Compilation complete'); Meta.js.cache = payload.data; minifier.kill(); } else { winston.error('[meta/js] Could not compile client-side scripts!'); winston.error('[meta/js] ' + payload.error.message); minifier.kill(); process.exit(); } }); this.prepare(function() { minifier.send({ action: minify ? 'js.minify' : 'js.concatenate', scripts: Meta.js.scripts }); }); }, killMinifier: function(callback) { if (this.minifierProc) { this.minifierProc.kill('SIGTERM'); } } }; /* Themes */ Meta.css = {}; Meta.css.cache = undefined; Meta.css.branding = {}; Meta.css.defaultBranding = {}; Meta.css.minify = function() { winston.info('[meta/css] Minifying LESS/CSS'); db.getObjectFields('config', ['theme:type', 'theme:id'], function(err, themeData) { var themeId = (themeData['theme:id'] || 'nodebb-theme-vanilla'), baseThemePath = path.join(nconf.get('themes_path'), (themeData['theme:type'] && themeData['theme:type'] === 'local' ? themeId : 'nodebb-theme-vanilla')), paths = [ baseThemePath, path.join(__dirname, '../node_modules'), path.join(__dirname, '../public/vendor/fontawesome/less') ], source = '@import "./theme";\n@import "font-awesome";', x, numLESS, numCSS; // Add the imports for each LESS file for(x=0,numLESS=plugins.lessFiles.length;x