You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
nodebb/src/meta.js

542 lines
14 KiB
JavaScript

"use strict";
11 years ago
var fs = require('fs'),
path = require('path'),
11 years ago
async = require('async'),
winston = require('winston'),
11 years ago
nconf = require('nconf'),
_ = require('underscore'),
less = require('less'),
11 years ago
fork = require('child_process').fork,
rimraf = require('rimraf'),
mkdirp = require('mkdirp'),
11 years ago
utils = require('./../public/src/utils'),
translator = require('./../public/src/translator'),
11 years ago
db = require('./database'),
plugins = require('./plugins'),
user = require('./user'),
groups = require('./groups');
11 years ago
(function (Meta) {
Meta.restartRequired = false;
Meta.config = {};
Meta.configs = {
init: function (callback) {
delete Meta.config;
Meta.configs.list(function (err, config) {
11 years ago
if(err) {
winston.error(err);
11 years ago
return callback(err);
}
11 years ago
Meta.config = config;
callback();
});
},
list: function (callback) {
11 years ago
db.getObject('config', function (err, config) {
if(err) {
11 years ago
return callback(err);
}
11 years ago
config = config || {};
config.status = 'ok';
callback(err, config);
});
},
get: function (field, callback) {
11 years ago
db.getObjectField('config', field, callback);
},
getFields: function (fields, callback) {
11 years ago
db.getObjectFields('config', fields, callback);
},
set: function (field, value, callback) {
if(!field) {
return callback(new Error('invalid config field'));
}
11 years ago
db.setObjectField('config', field, value, function(err, res) {
if (callback) {
if(!err && Meta.config) {
11 years ago
Meta.config[field] = value;
}
callback(err, res);
}
});
},
setOnEmpty: function (field, value, callback) {
11 years ago
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) {
11 years ago
db.deleteObjectField('config', field);
}
};
Meta.themes = {
get: function (callback) {
var themePath = nconf.get('themes_path');
11 years ago
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 = {
11 years ago
'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 : '';
11 years ago
themeData['theme:src'] = '';
db.setObject('config', themeData, next);
}
], callback);
11 years ago
Meta.restartRequired = true;
break;
case 'bootswatch':
11 years ago
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) {
11 years ago
title = Meta.config.browserTitle || 'NodeBB';
} else {
11 years ago
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];
11 years ago
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',
11 years ago
'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) {
11 years ago
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;
11 years ago
} else {
winston.warn('[meta.scripts.get] Could not resolve mapped path: ' + jsPath + '. Are you sure it is defined by a plugin?');
11 years ago
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();
});
11 years ago
},
minify: function(minify) {
// Prepare js for minification/concatenation
11 years ago
var minifier = this.minifierProc = fork('minifier.js');
11 years ago
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
});
});
11 years ago
},
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<numLESS;x++) {
source += '\n@import ".' + path.sep + plugins.lessFiles[x] + '";';
}
// ... and for each CSS file
for(x=0,numCSS=plugins.cssFiles.length;x<numCSS;x++) {
source += '\n@import (inline) ".' + path.sep + plugins.cssFiles[x] + '";';
}
source += '\n@import (inline) "..' + path.sep + '..' + path.sep + 'public/vendor/jquery/css/smoothness/jquery-ui-1.10.4.custom.min.css";';
source += '\n@import (inline) "..' + path.sep + '..' + path.sep + 'public/vendor/jquery/bootstrap-tagsinput/bootstrap-tagsinput.css";';
var parser = new (less.Parser)({
paths: paths
});
parser.parse(source, function(err, tree) {
if (err) {
winston.error('[meta/css] Could not minify LESS/CSS: ' + err.message);
process.exit();
}
var css = tree.toCSS({
cleancss: true
});
Meta.css.cache = css;
var re = /.brand-([\S]*?)[ ]*?{[\s\S]*?color:([\S\s]*?)}/gi,
match;
while (match = re.exec(css)) {
Meta.css.branding[match[1]] = match[2];
}
Meta.css.defaultBranding = Meta.css.branding;
Meta.css.updateBranding();
winston.info('[meta/css] Done.');
});
});
};
Meta.css.updateBranding = function() {
var Settings = require('./settings');
var branding = new Settings('branding', '0', {}, function() {
branding = branding.cfg._;
for (var b in branding) {
if (branding.hasOwnProperty(b)) {
Meta.css.cache = Meta.css.cache.replace(new RegExp(Meta.css.branding[b], 'g'), branding[b]);
}
}
Meta.css.branding = branding;
});
};
/* Sounds */
Meta.sounds = {};
Meta.sounds.init = function() {
var soundsPath = path.join(__dirname, '../public/sounds');
plugins.fireHook('filter:sounds.get', [], function(err, filePaths) {
if (err) {
winston.error('Could not initialise sound files:' + err.message);
}
// Clear the sounds directory
async.series([
function(next) {
rimraf(soundsPath, next);
},
function(next) {
mkdirp(soundsPath, next);
}
], function(err) {
if (err) {
winston.error('Could not initialise sound files:' + err.message);
}
// Link paths
async.each(filePaths, function(filePath, next) {
fs[process.platform !== 'win32' ? 'symlink' : 'link'](filePath, path.join(soundsPath, path.basename(filePath)), 'file', next);
}, function(err) {
if (!err) {
winston.info('[sounds] Sounds OK');
} else {
winston.error('[sounds] Could not initialise sounds: ' + err.message);
}
});
});
});
};
Meta.sounds.getFiles = function(callback) {
// todo: Possibly move these into a bundled module?
fs.readdir(path.join(__dirname, '../public/sounds'), function(err, files) {
var localList = {};
if (err) {
winston.error('Could not get local sound files:' + err.message);
console.log(err.stack);
return callback(null, []);
}
// Return proper paths
files.forEach(function(filename) {
11 years ago
localList[filename] = nconf.get('relative_path') + '/sounds/' + filename;
});
callback(null, localList);
});
};
Meta.sounds.getMapping = function(callback) {
db.getObject('settings:sounds', function(err, sounds) {
if (err || !sounds) {
// Send default sounds
var defaults = {
'notification': 'notification.mp3',
'chat-incoming': 'waterdrop-high.mp3',
'chat-outgoing': 'waterdrop-low.mp3'
};
return callback(null, defaults);
}
callback.apply(null, arguments);
});
};
/* Settings */
Meta.settings = {};
Meta.settings.get = function(hash, callback) {
hash = 'settings:' + hash;
db.getObject(hash, function(err, settings) {
if (err) {
callback(err);
} else {
callback(null, settings || {});
}
});
};
Meta.settings.getOne = function(hash, field, callback) {
hash = 'settings:' + hash;
db.getObjectField(hash, field, callback);
};
Meta.settings.set = function(hash, values, callback) {
hash = 'settings:' + hash;
plugins.fireHook('action:settings.set', hash, values);
db.setObject(hash, values, callback);
};
Meta.settings.setOne = function(hash, field, value, callback) {
hash = 'settings:' + hash;
db.setObjectField(hash, field, value, callback);
};
Meta.settings.setOnEmpty = function (hash, field, value, callback) {
Meta.settings.getOne(hash, field, function (err, curValue) {
if (err) {
return callback(err);
}
if (!curValue) {
Meta.settings.setOne(hash, field, value, callback);
} else {
callback();
}
});
};
/* Assorted */
Meta.userOrGroupExists = function(slug, callback) {
async.parallel([
async.apply(user.exists, slug),
async.apply(groups.exists, slug)
], function(err, results) {
callback(err, results.some(function(result) { return result }));
});
};
Meta.restart = function() {
if (process.send) {
process.send({
action: 'restart'
});
} else {
winston.error('[meta.restart] Could not restart, are you sure NodeBB was started with `./nodebb start`?');
}
};
}(exports));