Merge branch 'master' into bookmark2
commit
2a82b5db91
@ -0,0 +1,13 @@
|
|||||||
|
# These files are text and should be normalized (convert crlf => lf)
|
||||||
|
*.json text
|
||||||
|
*.css text
|
||||||
|
*.less text
|
||||||
|
*.tpl text
|
||||||
|
*.html text
|
||||||
|
*.js text
|
||||||
|
*.md text
|
||||||
|
|
||||||
|
# Images should be treated as binary
|
||||||
|
# (binary is a macro for -text -diff)
|
||||||
|
*.png binary
|
||||||
|
*.jpg binary
|
@ -1,75 +1,75 @@
|
|||||||
# <img alt="NodeBB" src="http://i.imgur.com/mYxPPtB.png" />
|
# <img alt="NodeBB" src="http://i.imgur.com/mYxPPtB.png" />
|
||||||
|
|
||||||
[](https://gitter.im/NodeBB/NodeBB?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
|
[](https://gitter.im/NodeBB/NodeBB?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
|
||||||
[](https://travis-ci.org/NodeBB/NodeBB)
|
[](https://travis-ci.org/NodeBB/NodeBB)
|
||||||
[](https://david-dm.org/nodebb/nodebb)
|
[](https://david-dm.org/nodebb/nodebb)
|
||||||
[](https://codeclimate.com/github/NodeBB/NodeBB)
|
[](https://codeclimate.com/github/NodeBB/NodeBB)
|
||||||
[](https://readthedocs.org/projects/nodebb/?badge=latest)
|
[](https://readthedocs.org/projects/nodebb/?badge=latest)
|
||||||
|
|
||||||
**NodeBB Forum Software** is powered by Node.js and built on either a Redis or MongoDB database. It utilizes web sockets for instant interactions and real-time notifications. NodeBB is compatible down to IE8 and has many modern features out of the box such as social network integration and streaming discussions.
|
**NodeBB Forum Software** is powered by Node.js and built on either a Redis or MongoDB database. It utilizes web sockets for instant interactions and real-time notifications. NodeBB is compatible down to IE8 and has many modern features out of the box such as social network integration and streaming discussions.
|
||||||
|
|
||||||
Additional functionality is enabled through the use of third-party plugins.
|
Additional functionality is enabled through the use of third-party plugins.
|
||||||
|
|
||||||
* [Get NodeBB](http://www.nodebb.org/ "NodeBB")
|
* [Get NodeBB](http://www.nodebb.org/ "NodeBB")
|
||||||
* [Demo & Meta Discussion](http://community.nodebb.org)
|
* [Demo & Meta Discussion](http://community.nodebb.org)
|
||||||
* [Documentation & Installation Instructions](http://docs.nodebb.org)
|
* [Documentation & Installation Instructions](http://docs.nodebb.org)
|
||||||
* [Help translate NodeBB](https://www.transifex.com/projects/p/nodebb/)
|
* [Help translate NodeBB](https://www.transifex.com/projects/p/nodebb/)
|
||||||
* [NodeBB Blog](http://blog.nodebb.org)
|
* [NodeBB Blog](http://blog.nodebb.org)
|
||||||
* [Join us on IRC](https://kiwiirc.com/client/irc.freenode.net/nodebb) - #nodebb on Freenode
|
* [Join us on IRC](https://kiwiirc.com/client/irc.freenode.net/nodebb) - #nodebb on Freenode
|
||||||
* [Follow us on Twitter](http://www.twitter.com/NodeBB/ "NodeBB Twitter")
|
* [Follow us on Twitter](http://www.twitter.com/NodeBB/ "NodeBB Twitter")
|
||||||
* [Like us on Facebook](http://www.facebook.com/NodeBB/ "NodeBB Facebook")
|
* [Like us on Facebook](http://www.facebook.com/NodeBB/ "NodeBB Facebook")
|
||||||
|
|
||||||
## Screenshots
|
## Screenshots
|
||||||
|
|
||||||
[](http://i.imgur.com/VCoOFyq.png)
|
[](http://i.imgur.com/VCoOFyq.png)
|
||||||
[](http://i.imgur.com/FLOUuIq.png)
|
[](http://i.imgur.com/FLOUuIq.png)
|
||||||
[](http://i.imgur.com/Ud1LrfI.png)
|
[](http://i.imgur.com/Ud1LrfI.png)
|
||||||
[](http://i.imgur.com/h6yZ66s.png)
|
[](http://i.imgur.com/h6yZ66s.png)
|
||||||
[](http://i.imgur.com/o90kVPi.png)
|
[](http://i.imgur.com/o90kVPi.png)
|
||||||
[](http://i.imgur.com/AaRRrU2.png)
|
[](http://i.imgur.com/AaRRrU2.png)
|
||||||
[](http://i.imgur.com/LmHtPho.png)
|
[](http://i.imgur.com/LmHtPho.png)
|
||||||
[](http://i.imgur.com/paiJPJk.jpg)
|
[](http://i.imgur.com/paiJPJk.jpg)
|
||||||
|
|
||||||
[](http://i.imgur.com/8OLssij.png)
|
[](http://i.imgur.com/8OLssij.png)
|
||||||
[](http://i.imgur.com/JKOc0LZ.png)
|
[](http://i.imgur.com/JKOc0LZ.png)
|
||||||
|
|
||||||
## How can I follow along/contribute?
|
## How can I follow along/contribute?
|
||||||
|
|
||||||
* Our feature roadmap is hosted on the project wiki's [Version History / Roadmap](https://github.com/NodeBB/NodeBB/wiki/Version-History-%26-Roadmap)
|
* Our feature roadmap is hosted on the project wiki's [Version History / Roadmap](https://github.com/NodeBB/NodeBB/wiki/Version-History-%26-Roadmap)
|
||||||
* If you are a developer, feel free to check out the source and submit pull requests. We also have a wide array of [plugins](http://community.nodebb.org/category/7/nodebb-plugins) which would be a great starting point for learning the codebase.
|
* If you are a developer, feel free to check out the source and submit pull requests. We also have a wide array of [plugins](http://community.nodebb.org/category/7/nodebb-plugins) which would be a great starting point for learning the codebase.
|
||||||
* If you are a designer, [NodeBB needs themes](http://community.nodebb.org/category/10/nodebb-themes)! NodeBB's theming system allows extention of the base templates as well as styling via LESS or CSS. NodeBB's base theme utilizes [Bootstrap 3](http://getbootstrap.com/) but themes can choose to use a different framework altogether.
|
* If you are a designer, [NodeBB needs themes](http://community.nodebb.org/category/10/nodebb-themes)! NodeBB's theming system allows extention of the base templates as well as styling via LESS or CSS. NodeBB's base theme utilizes [Bootstrap 3](http://getbootstrap.com/) but themes can choose to use a different framework altogether.
|
||||||
* If you know languages other than English you can help us translate NodeBB. We use [Transifex](https://www.transifex.com/projects/p/nodebb/) for internationalization.
|
* If you know languages other than English you can help us translate NodeBB. We use [Transifex](https://www.transifex.com/projects/p/nodebb/) for internationalization.
|
||||||
* Please don't forget to **like**, **follow**, and **star our repo**! Join our growing [community](http://community.nodebb.org) to keep up to date with the latest NodeBB development.
|
* Please don't forget to **like**, **follow**, and **star our repo**! Join our growing [community](http://community.nodebb.org) to keep up to date with the latest NodeBB development.
|
||||||
|
|
||||||
## Requirements
|
## Requirements
|
||||||
|
|
||||||
NodeBB requires the following software to be installed:
|
NodeBB requires the following software to be installed:
|
||||||
|
|
||||||
* A version of Node.js at least 0.10 or greater
|
* A version of Node.js at least 0.10 or greater
|
||||||
* Redis, version 2.8.9 or greater **or** MongoDB, version 2.6 or greater
|
* Redis, version 2.8.9 or greater **or** MongoDB, version 2.6 or greater
|
||||||
* nginx, version 1.3.13 or greater (**only if** intending to use nginx to proxy requests to a NodeBB)
|
* nginx, version 1.3.13 or greater (**only if** intending to use nginx to proxy requests to a NodeBB)
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
[Please refer to platform-specific installation documentation](http://docs.nodebb.org/en/latest/installing/os.html)
|
[Please refer to platform-specific installation documentation](http://docs.nodebb.org/en/latest/installing/os.html)
|
||||||
|
|
||||||
## Securing NodeBB
|
## Securing NodeBB
|
||||||
|
|
||||||
It is important to ensure that your NodeBB and database servers are secured. Bear these points in mind:
|
It is important to ensure that your NodeBB and database servers are secured. Bear these points in mind:
|
||||||
|
|
||||||
1. While some distributions set up Redis with a more restrictive configuration, Redis by default listens to all interfaces, which is especially dangerous when a server is open to the public. Some suggestions:
|
1. While some distributions set up Redis with a more restrictive configuration, Redis by default listens to all interfaces, which is especially dangerous when a server is open to the public. Some suggestions:
|
||||||
* Set `bind_address` to `127.0.0.1` so as to restrict access to the local machine only
|
* Set `bind_address` to `127.0.0.1` so as to restrict access to the local machine only
|
||||||
* Use `requirepass` to secure Redis behind a password (preferably a long one)
|
* Use `requirepass` to secure Redis behind a password (preferably a long one)
|
||||||
* Familiarise yourself with [Redis Security](http://redis.io/topics/security)
|
* Familiarise yourself with [Redis Security](http://redis.io/topics/security)
|
||||||
2. Use `iptables` to secure your server from unintended open ports. In Ubuntu, `ufw` provides a friendlier interface to working with `iptables`.
|
2. Use `iptables` to secure your server from unintended open ports. In Ubuntu, `ufw` provides a friendlier interface to working with `iptables`.
|
||||||
* e.g. If your NodeBB is proxied, no ports should be open except 80 (and possibly 22, for SSH access)
|
* e.g. If your NodeBB is proxied, no ports should be open except 80 (and possibly 22, for SSH access)
|
||||||
|
|
||||||
## Upgrading NodeBB
|
## Upgrading NodeBB
|
||||||
|
|
||||||
Detailed upgrade instructions are listed in [Upgrading NodeBB](https://docs.nodebb.org/en/latest/upgrading/index.html)
|
Detailed upgrade instructions are listed in [Upgrading NodeBB](https://docs.nodebb.org/en/latest/upgrading/index.html)
|
||||||
|
|
||||||
## License
|
## License
|
||||||
|
|
||||||
NodeBB is licensed under the **GNU General Public License v3 (GPL-3)** (http://www.gnu.org/copyleft/gpl.html).
|
NodeBB is licensed under the **GNU General Public License v3 (GPL-3)** (http://www.gnu.org/copyleft/gpl.html).
|
||||||
|
|
||||||
Interested in a sublicense agreement for use of NodeBB in a non-free/restrictive environment? Contact us at sales@nodebb.org.
|
Interested in a sublicense agreement for use of NodeBB in a non-free/restrictive environment? Contact us at sales@nodebb.org.
|
@ -1,416 +1,416 @@
|
|||||||
/*
|
/*
|
||||||
NodeBB - A better forum platform for the modern web
|
NodeBB - A better forum platform for the modern web
|
||||||
https://github.com/NodeBB/NodeBB/
|
https://github.com/NodeBB/NodeBB/
|
||||||
Copyright (C) 2013-2014 NodeBB Inc.
|
Copyright (C) 2013-2014 NodeBB Inc.
|
||||||
|
|
||||||
This program is free software: you can redistribute it and/or modify
|
This program is free software: you can redistribute it and/or modify
|
||||||
it under the terms of the GNU General Public License as published by
|
it under the terms of the GNU General Public License as published by
|
||||||
the Free Software Foundation, either version 3 of the License, or
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
(at your option) any later version.
|
(at your option) any later version.
|
||||||
|
|
||||||
This program is distributed in the hope that it will be useful,
|
This program is distributed in the hope that it will be useful,
|
||||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
GNU General Public License for more details.
|
GNU General Public License for more details.
|
||||||
|
|
||||||
You should have received a copy of the GNU General Public License
|
You should have received a copy of the GNU General Public License
|
||||||
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
"use strict";
|
"use strict";
|
||||||
/*global require, global, process*/
|
/*global require, global, process*/
|
||||||
|
|
||||||
var nconf = require('nconf');
|
var nconf = require('nconf');
|
||||||
nconf.argv().env('__');
|
nconf.argv().env('__');
|
||||||
|
|
||||||
var fs = require('fs'),
|
var fs = require('fs'),
|
||||||
os = require('os'),
|
os = require('os'),
|
||||||
url = require('url'),
|
url = require('url'),
|
||||||
async = require('async'),
|
async = require('async'),
|
||||||
semver = require('semver'),
|
semver = require('semver'),
|
||||||
winston = require('winston'),
|
winston = require('winston'),
|
||||||
colors = require('colors'),
|
colors = require('colors'),
|
||||||
path = require('path'),
|
path = require('path'),
|
||||||
pkg = require('./package.json'),
|
pkg = require('./package.json'),
|
||||||
utils = require('./public/src/utils.js');
|
utils = require('./public/src/utils.js');
|
||||||
|
|
||||||
global.env = process.env.NODE_ENV || 'production';
|
global.env = process.env.NODE_ENV || 'production';
|
||||||
|
|
||||||
winston.remove(winston.transports.Console);
|
winston.remove(winston.transports.Console);
|
||||||
winston.add(winston.transports.Console, {
|
winston.add(winston.transports.Console, {
|
||||||
colorize: true,
|
colorize: true,
|
||||||
timestamp: function() {
|
timestamp: function() {
|
||||||
var date = new Date();
|
var date = new Date();
|
||||||
return date.getDate() + '/' + (date.getMonth() + 1) + ' ' + date.toTimeString().substr(0,5) + ' [' + global.process.pid + ']';
|
return date.getDate() + '/' + (date.getMonth() + 1) + ' ' + date.toTimeString().substr(0,5) + ' [' + global.process.pid + ']';
|
||||||
},
|
},
|
||||||
level: nconf.get('log-level') || (global.env === 'production' ? 'info' : 'verbose')
|
level: nconf.get('log-level') || (global.env === 'production' ? 'info' : 'verbose')
|
||||||
});
|
});
|
||||||
|
|
||||||
if(os.platform() === 'linux') {
|
if(os.platform() === 'linux') {
|
||||||
require('child_process').exec('/usr/bin/which convert', function(err, stdout, stderr) {
|
require('child_process').exec('/usr/bin/which convert', function(err, stdout, stderr) {
|
||||||
if(err || !stdout) {
|
if(err || !stdout) {
|
||||||
winston.warn('Couldn\'t find convert. Did you install imagemagick?');
|
winston.warn('Couldn\'t find convert. Did you install imagemagick?');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Alternate configuration file support
|
// Alternate configuration file support
|
||||||
var configFile = path.join(__dirname, '/config.json'),
|
var configFile = path.join(__dirname, '/config.json'),
|
||||||
configExists;
|
configExists;
|
||||||
|
|
||||||
if (nconf.get('config')) {
|
if (nconf.get('config')) {
|
||||||
configFile = path.resolve(__dirname, nconf.get('config'));
|
configFile = path.resolve(__dirname, nconf.get('config'));
|
||||||
}
|
}
|
||||||
configExists = fs.existsSync(configFile);
|
configExists = fs.existsSync(configFile);
|
||||||
|
|
||||||
if (!nconf.get('setup') && !nconf.get('install') && !nconf.get('upgrade') && !nconf.get('reset') && configExists) {
|
if (!nconf.get('setup') && !nconf.get('install') && !nconf.get('upgrade') && !nconf.get('reset') && configExists) {
|
||||||
start();
|
start();
|
||||||
} else if (nconf.get('setup') || nconf.get('install')) {
|
} else if (nconf.get('setup') || nconf.get('install')) {
|
||||||
setup();
|
setup();
|
||||||
} else if (!configExists) {
|
} else if (!configExists) {
|
||||||
require('./install/web').install(nconf.get('port'));
|
require('./install/web').install(nconf.get('port'));
|
||||||
} else if (nconf.get('upgrade')) {
|
} else if (nconf.get('upgrade')) {
|
||||||
upgrade();
|
upgrade();
|
||||||
} else if (nconf.get('reset')) {
|
} else if (nconf.get('reset')) {
|
||||||
reset();
|
reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
function loadConfig() {
|
function loadConfig() {
|
||||||
nconf.file({
|
nconf.file({
|
||||||
file: configFile
|
file: configFile
|
||||||
});
|
});
|
||||||
|
|
||||||
nconf.defaults({
|
nconf.defaults({
|
||||||
base_dir: __dirname,
|
base_dir: __dirname,
|
||||||
themes_path: path.join(__dirname, 'node_modules'),
|
themes_path: path.join(__dirname, 'node_modules'),
|
||||||
views_dir: path.join(__dirname, 'public/templates'),
|
views_dir: path.join(__dirname, 'public/templates'),
|
||||||
version: pkg.version
|
version: pkg.version
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!nconf.get('isCluster')) {
|
if (!nconf.get('isCluster')) {
|
||||||
nconf.set('isPrimary', 'true');
|
nconf.set('isPrimary', 'true');
|
||||||
nconf.set('isCluster', 'false');
|
nconf.set('isCluster', 'false');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure themes_path is a full filepath
|
// Ensure themes_path is a full filepath
|
||||||
nconf.set('themes_path', path.resolve(__dirname, nconf.get('themes_path')));
|
nconf.set('themes_path', path.resolve(__dirname, nconf.get('themes_path')));
|
||||||
nconf.set('core_templates_path', path.join(__dirname, 'src/views'));
|
nconf.set('core_templates_path', path.join(__dirname, 'src/views'));
|
||||||
nconf.set('base_templates_path', path.join(nconf.get('themes_path'), 'nodebb-theme-vanilla/templates'));
|
nconf.set('base_templates_path', path.join(nconf.get('themes_path'), 'nodebb-theme-vanilla/templates'));
|
||||||
|
|
||||||
if (!process.send) {
|
if (!process.send) {
|
||||||
// If run using `node app`, log GNU copyright info along with server info
|
// If run using `node app`, log GNU copyright info along with server info
|
||||||
winston.info('NodeBB v' + nconf.get('version') + ' Copyright (C) 2013-2014 NodeBB Inc.');
|
winston.info('NodeBB v' + nconf.get('version') + ' Copyright (C) 2013-2014 NodeBB Inc.');
|
||||||
winston.info('This program comes with ABSOLUTELY NO WARRANTY.');
|
winston.info('This program comes with ABSOLUTELY NO WARRANTY.');
|
||||||
winston.info('This is free software, and you are welcome to redistribute it under certain conditions.');
|
winston.info('This is free software, and you are welcome to redistribute it under certain conditions.');
|
||||||
winston.info('');
|
winston.info('');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function start() {
|
function start() {
|
||||||
loadConfig();
|
loadConfig();
|
||||||
var db = require('./src/database');
|
var db = require('./src/database');
|
||||||
|
|
||||||
// nconf defaults, if not set in config
|
// nconf defaults, if not set in config
|
||||||
if (!nconf.get('upload_path')) {
|
if (!nconf.get('upload_path')) {
|
||||||
nconf.set('upload_path', '/public/uploads');
|
nconf.set('upload_path', '/public/uploads');
|
||||||
}
|
}
|
||||||
// Parse out the relative_url and other goodies from the configured URL
|
// Parse out the relative_url and other goodies from the configured URL
|
||||||
var urlObject = url.parse(nconf.get('url'));
|
var urlObject = url.parse(nconf.get('url'));
|
||||||
var relativePath = urlObject.pathname !== '/' ? urlObject.pathname : '';
|
var relativePath = urlObject.pathname !== '/' ? urlObject.pathname : '';
|
||||||
nconf.set('base_url', urlObject.protocol + '//' + urlObject.host);
|
nconf.set('base_url', urlObject.protocol + '//' + urlObject.host);
|
||||||
nconf.set('use_port', !!urlObject.port);
|
nconf.set('use_port', !!urlObject.port);
|
||||||
nconf.set('relative_path', relativePath);
|
nconf.set('relative_path', relativePath);
|
||||||
nconf.set('port', urlObject.port || nconf.get('port') || nconf.get('PORT') || 4567);
|
nconf.set('port', urlObject.port || nconf.get('port') || nconf.get('PORT') || 4567);
|
||||||
nconf.set('upload_url', '/uploads/');
|
nconf.set('upload_url', '/uploads/');
|
||||||
|
|
||||||
if (nconf.get('isPrimary') === 'true') {
|
if (nconf.get('isPrimary') === 'true') {
|
||||||
winston.info('Time: %s', (new Date()).toString());
|
winston.info('Time: %s', (new Date()).toString());
|
||||||
winston.info('Initializing NodeBB v%s', nconf.get('version'));
|
winston.info('Initializing NodeBB v%s', nconf.get('version'));
|
||||||
winston.verbose('* using configuration stored in: %s', configFile);
|
winston.verbose('* using configuration stored in: %s', configFile);
|
||||||
|
|
||||||
var host = nconf.get(nconf.get('database') + ':host'),
|
var host = nconf.get(nconf.get('database') + ':host'),
|
||||||
storeLocation = host ? 'at ' + host + (host.indexOf('/') === -1 ? ':' + nconf.get(nconf.get('database') + ':port') : '') : '';
|
storeLocation = host ? 'at ' + host + (host.indexOf('/') === -1 ? ':' + nconf.get(nconf.get('database') + ':port') : '') : '';
|
||||||
|
|
||||||
winston.verbose('* using %s store %s', nconf.get('database'), storeLocation);
|
winston.verbose('* using %s store %s', nconf.get('database'), storeLocation);
|
||||||
winston.verbose('* using themes stored in: %s', nconf.get('themes_path'));
|
winston.verbose('* using themes stored in: %s', nconf.get('themes_path'));
|
||||||
}
|
}
|
||||||
|
|
||||||
process.on('SIGTERM', shutdown);
|
process.on('SIGTERM', shutdown);
|
||||||
process.on('SIGINT', shutdown);
|
process.on('SIGINT', shutdown);
|
||||||
process.on('SIGHUP', restart);
|
process.on('SIGHUP', restart);
|
||||||
process.on('message', function(message) {
|
process.on('message', function(message) {
|
||||||
if (typeof message !== 'object') {
|
if (typeof message !== 'object') {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
var meta = require('./src/meta');
|
var meta = require('./src/meta');
|
||||||
var emitter = require('./src/emitter');
|
var emitter = require('./src/emitter');
|
||||||
switch (message.action) {
|
switch (message.action) {
|
||||||
case 'reload':
|
case 'reload':
|
||||||
meta.reload();
|
meta.reload();
|
||||||
break;
|
break;
|
||||||
case 'js-propagate':
|
case 'js-propagate':
|
||||||
meta.js.cache = message.cache;
|
meta.js.cache = message.cache;
|
||||||
meta.js.map = message.map;
|
meta.js.map = message.map;
|
||||||
meta.js.hash = message.hash;
|
meta.js.hash = message.hash;
|
||||||
emitter.emit('meta:js.compiled');
|
emitter.emit('meta:js.compiled');
|
||||||
winston.verbose('[cluster] Client-side javascript and mapping propagated to worker %s', process.pid);
|
winston.verbose('[cluster] Client-side javascript and mapping propagated to worker %s', process.pid);
|
||||||
break;
|
break;
|
||||||
case 'css-propagate':
|
case 'css-propagate':
|
||||||
meta.css.cache = message.cache;
|
meta.css.cache = message.cache;
|
||||||
meta.css.acpCache = message.acpCache;
|
meta.css.acpCache = message.acpCache;
|
||||||
meta.css.hash = message.hash;
|
meta.css.hash = message.hash;
|
||||||
emitter.emit('meta:css.compiled');
|
emitter.emit('meta:css.compiled');
|
||||||
winston.verbose('[cluster] Stylesheets propagated to worker %s', process.pid);
|
winston.verbose('[cluster] Stylesheets propagated to worker %s', process.pid);
|
||||||
break;
|
break;
|
||||||
case 'templates:compiled':
|
case 'templates:compiled':
|
||||||
emitter.emit('templates:compiled');
|
emitter.emit('templates:compiled');
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
process.on('uncaughtException', function(err) {
|
process.on('uncaughtException', function(err) {
|
||||||
winston.error(err.stack);
|
winston.error(err.stack);
|
||||||
console.log(err.stack);
|
console.log(err.stack);
|
||||||
|
|
||||||
require('./src/meta').js.killMinifier();
|
require('./src/meta').js.killMinifier();
|
||||||
shutdown(1);
|
shutdown(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
async.waterfall([
|
async.waterfall([
|
||||||
async.apply(db.init),
|
async.apply(db.init),
|
||||||
async.apply(db.checkCompatibility),
|
async.apply(db.checkCompatibility),
|
||||||
function(next) {
|
function(next) {
|
||||||
require('./src/meta').configs.init(next);
|
require('./src/meta').configs.init(next);
|
||||||
},
|
},
|
||||||
function(next) {
|
function(next) {
|
||||||
require('./src/meta').dependencies.check(next);
|
require('./src/meta').dependencies.check(next);
|
||||||
},
|
},
|
||||||
function(next) {
|
function(next) {
|
||||||
require('./src/upgrade').check(next);
|
require('./src/upgrade').check(next);
|
||||||
},
|
},
|
||||||
function(next) {
|
function(next) {
|
||||||
var webserver = require('./src/webserver');
|
var webserver = require('./src/webserver');
|
||||||
require('./src/socket.io').init(webserver.server);
|
require('./src/socket.io').init(webserver.server);
|
||||||
|
|
||||||
if (nconf.get('isPrimary') === 'true' && !nconf.get('jobsDisabled')) {
|
if (nconf.get('isPrimary') === 'true' && !nconf.get('jobsDisabled')) {
|
||||||
require('./src/notifications').init();
|
require('./src/notifications').init();
|
||||||
require('./src/user').startJobs();
|
require('./src/user').startJobs();
|
||||||
}
|
}
|
||||||
|
|
||||||
webserver.listen();
|
webserver.listen();
|
||||||
}
|
}
|
||||||
], function(err) {
|
], function(err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
switch(err.message) {
|
switch(err.message) {
|
||||||
case 'schema-out-of-date':
|
case 'schema-out-of-date':
|
||||||
winston.warn('Your NodeBB schema is out-of-date. Please run the following command to bring your dataset up to spec:');
|
winston.warn('Your NodeBB schema is out-of-date. Please run the following command to bring your dataset up to spec:');
|
||||||
winston.warn(' ./nodebb upgrade');
|
winston.warn(' ./nodebb upgrade');
|
||||||
break;
|
break;
|
||||||
case 'dependencies-out-of-date':
|
case 'dependencies-out-of-date':
|
||||||
winston.warn('One or more of NodeBB\'s dependent packages are out-of-date. Please run the following command to update them:');
|
winston.warn('One or more of NodeBB\'s dependent packages are out-of-date. Please run the following command to update them:');
|
||||||
winston.warn(' ./nodebb upgrade');
|
winston.warn(' ./nodebb upgrade');
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
if (err.stacktrace !== false) {
|
if (err.stacktrace !== false) {
|
||||||
winston.error(err.stack);
|
winston.error(err.stack);
|
||||||
} else {
|
} else {
|
||||||
winston.error(err.message);
|
winston.error(err.message);
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Either way, bad stuff happened. Abort start.
|
// Either way, bad stuff happened. Abort start.
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function setup() {
|
function setup() {
|
||||||
loadConfig();
|
loadConfig();
|
||||||
|
|
||||||
winston.info('NodeBB Setup Triggered via Command Line');
|
winston.info('NodeBB Setup Triggered via Command Line');
|
||||||
|
|
||||||
var install = require('./src/install');
|
var install = require('./src/install');
|
||||||
|
|
||||||
process.stdout.write('\nWelcome to NodeBB!\n');
|
process.stdout.write('\nWelcome to NodeBB!\n');
|
||||||
process.stdout.write('\nThis looks like a new installation, so you\'ll have to answer a few questions about your environment before we can proceed.\n');
|
process.stdout.write('\nThis looks like a new installation, so you\'ll have to answer a few questions about your environment before we can proceed.\n');
|
||||||
process.stdout.write('Press enter to accept the default setting (shown in brackets).\n');
|
process.stdout.write('Press enter to accept the default setting (shown in brackets).\n');
|
||||||
|
|
||||||
install.setup(function (err, data) {
|
install.setup(function (err, data) {
|
||||||
var separator = ' ';
|
var separator = ' ';
|
||||||
if (process.stdout.columns > 10) {
|
if (process.stdout.columns > 10) {
|
||||||
for(var x=0,cols=process.stdout.columns-10;x<cols;x++) {
|
for(var x=0,cols=process.stdout.columns-10;x<cols;x++) {
|
||||||
separator += '=';
|
separator += '=';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
process.stdout.write('\n' + separator + '\n\n');
|
process.stdout.write('\n' + separator + '\n\n');
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
winston.error('There was a problem completing NodeBB setup: ', err.message);
|
winston.error('There was a problem completing NodeBB setup: ', err.message);
|
||||||
} else {
|
} else {
|
||||||
if (data.hasOwnProperty('password')) {
|
if (data.hasOwnProperty('password')) {
|
||||||
process.stdout.write('An administrative user was automatically created for you:\n');
|
process.stdout.write('An administrative user was automatically created for you:\n');
|
||||||
process.stdout.write(' Username: ' + data.username + '\n');
|
process.stdout.write(' Username: ' + data.username + '\n');
|
||||||
process.stdout.write(' Password: ' + data.password + '\n');
|
process.stdout.write(' Password: ' + data.password + '\n');
|
||||||
process.stdout.write('\n');
|
process.stdout.write('\n');
|
||||||
}
|
}
|
||||||
process.stdout.write('NodeBB Setup Completed. Run \'./nodebb start\' to manually start your NodeBB server.\n');
|
process.stdout.write('NodeBB Setup Completed. Run \'./nodebb start\' to manually start your NodeBB server.\n');
|
||||||
|
|
||||||
// If I am a child process, notify the parent of the returned data before exiting (useful for notifying
|
// If I am a child process, notify the parent of the returned data before exiting (useful for notifying
|
||||||
// hosts of auto-generated username/password during headless setups)
|
// hosts of auto-generated username/password during headless setups)
|
||||||
if (process.send) {
|
if (process.send) {
|
||||||
process.send(data);
|
process.send(data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
process.exit();
|
process.exit();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function upgrade() {
|
function upgrade() {
|
||||||
loadConfig();
|
loadConfig();
|
||||||
|
|
||||||
require('./src/database').init(function(err) {
|
require('./src/database').init(function(err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
winston.error(err.stack);
|
winston.error(err.stack);
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
require('./src/meta').configs.init(function () {
|
require('./src/meta').configs.init(function () {
|
||||||
require('./src/upgrade').upgrade();
|
require('./src/upgrade').upgrade();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function reset() {
|
function reset() {
|
||||||
loadConfig();
|
loadConfig();
|
||||||
|
|
||||||
require('./src/database').init(function(err) {
|
require('./src/database').init(function(err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
winston.error(err.message);
|
winston.error(err.message);
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (nconf.get('t')) {
|
if (nconf.get('t')) {
|
||||||
resetThemes();
|
resetThemes();
|
||||||
} else if (nconf.get('p')) {
|
} else if (nconf.get('p')) {
|
||||||
if (nconf.get('p') === true) {
|
if (nconf.get('p') === true) {
|
||||||
resetPlugins();
|
resetPlugins();
|
||||||
} else {
|
} else {
|
||||||
resetPlugin(nconf.get('p'));
|
resetPlugin(nconf.get('p'));
|
||||||
}
|
}
|
||||||
} else if (nconf.get('w')) {
|
} else if (nconf.get('w')) {
|
||||||
resetWidgets();
|
resetWidgets();
|
||||||
} else if (nconf.get('s')) {
|
} else if (nconf.get('s')) {
|
||||||
resetSettings();
|
resetSettings();
|
||||||
} else if (nconf.get('a')) {
|
} else if (nconf.get('a')) {
|
||||||
require('async').series([resetWidgets, resetThemes, resetPlugins, resetSettings], function(err) {
|
require('async').series([resetWidgets, resetThemes, resetPlugins, resetSettings], function(err) {
|
||||||
if (!err) {
|
if (!err) {
|
||||||
winston.info('[reset] Reset complete.');
|
winston.info('[reset] Reset complete.');
|
||||||
} else {
|
} else {
|
||||||
winston.error('[reset] Errors were encountered while resetting your forum settings: %s', err.message);
|
winston.error('[reset] Errors were encountered while resetting your forum settings: %s', err.message);
|
||||||
}
|
}
|
||||||
process.exit();
|
process.exit();
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
process.stdout.write('\nNodeBB Reset\n'.bold);
|
process.stdout.write('\nNodeBB Reset\n'.bold);
|
||||||
process.stdout.write('No arguments passed in, so nothing was reset.\n\n'.yellow);
|
process.stdout.write('No arguments passed in, so nothing was reset.\n\n'.yellow);
|
||||||
process.stdout.write('Use ./nodebb reset ' + '{-t|-p|-w|-s|-a}\n'.red);
|
process.stdout.write('Use ./nodebb reset ' + '{-t|-p|-w|-s|-a}\n'.red);
|
||||||
process.stdout.write(' -t\tthemes\n');
|
process.stdout.write(' -t\tthemes\n');
|
||||||
process.stdout.write(' -p\tplugins\n');
|
process.stdout.write(' -p\tplugins\n');
|
||||||
process.stdout.write(' -w\twidgets\n');
|
process.stdout.write(' -w\twidgets\n');
|
||||||
process.stdout.write(' -s\tsettings\n');
|
process.stdout.write(' -s\tsettings\n');
|
||||||
process.stdout.write(' -a\tall of the above\n');
|
process.stdout.write(' -a\tall of the above\n');
|
||||||
|
|
||||||
process.stdout.write('\nPlugin reset flag (-p) can take a single argument\n');
|
process.stdout.write('\nPlugin reset flag (-p) can take a single argument\n');
|
||||||
process.stdout.write(' e.g. ./nodebb reset -p nodebb-plugin-mentions\n');
|
process.stdout.write(' e.g. ./nodebb reset -p nodebb-plugin-mentions\n');
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function resetSettings(callback) {
|
function resetSettings(callback) {
|
||||||
var meta = require('./src/meta');
|
var meta = require('./src/meta');
|
||||||
meta.configs.set('allowLocalLogin', 1, function(err) {
|
meta.configs.set('allowLocalLogin', 1, function(err) {
|
||||||
winston.info('[reset] Settings reset to default');
|
winston.info('[reset] Settings reset to default');
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback(err);
|
callback(err);
|
||||||
} else {
|
} else {
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function resetThemes(callback) {
|
function resetThemes(callback) {
|
||||||
var meta = require('./src/meta');
|
var meta = require('./src/meta');
|
||||||
|
|
||||||
meta.themes.set({
|
meta.themes.set({
|
||||||
type: 'local',
|
type: 'local',
|
||||||
id: 'nodebb-theme-vanilla'
|
id: 'nodebb-theme-vanilla'
|
||||||
}, function(err) {
|
}, function(err) {
|
||||||
winston.info('[reset] Theme reset to Vanilla');
|
winston.info('[reset] Theme reset to Vanilla');
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback(err);
|
callback(err);
|
||||||
} else {
|
} else {
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function resetPlugin(pluginId) {
|
function resetPlugin(pluginId) {
|
||||||
var db = require('./src/database');
|
var db = require('./src/database');
|
||||||
db.sortedSetRemove('plugins:active', pluginId, function(err) {
|
db.sortedSetRemove('plugins:active', pluginId, function(err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
winston.error('[reset] Could not disable plugin: %s encountered error %s', pluginId, err.message);
|
winston.error('[reset] Could not disable plugin: %s encountered error %s', pluginId, err.message);
|
||||||
} else {
|
} else {
|
||||||
winston.info('[reset] Plugin `%s` disabled', pluginId);
|
winston.info('[reset] Plugin `%s` disabled', pluginId);
|
||||||
}
|
}
|
||||||
|
|
||||||
process.exit();
|
process.exit();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function resetPlugins(callback) {
|
function resetPlugins(callback) {
|
||||||
var db = require('./src/database');
|
var db = require('./src/database');
|
||||||
db.delete('plugins:active', function(err) {
|
db.delete('plugins:active', function(err) {
|
||||||
winston.info('[reset] All Plugins De-activated');
|
winston.info('[reset] All Plugins De-activated');
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback(err);
|
callback(err);
|
||||||
} else {
|
} else {
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function resetWidgets(callback) {
|
function resetWidgets(callback) {
|
||||||
require('./src/widgets').reset(function(err) {
|
require('./src/widgets').reset(function(err) {
|
||||||
winston.info('[reset] All Widgets moved to Draft Zone');
|
winston.info('[reset] All Widgets moved to Draft Zone');
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback(err);
|
callback(err);
|
||||||
} else {
|
} else {
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function shutdown(code) {
|
function shutdown(code) {
|
||||||
winston.info('[app] Shutdown (SIGTERM/SIGINT) Initialised.');
|
winston.info('[app] Shutdown (SIGTERM/SIGINT) Initialised.');
|
||||||
require('./src/database').close();
|
require('./src/database').close();
|
||||||
winston.info('[app] Database connection closed.');
|
winston.info('[app] Database connection closed.');
|
||||||
require('./src/webserver').server.close();
|
require('./src/webserver').server.close();
|
||||||
winston.info('[app] Web server closed to connections.');
|
winston.info('[app] Web server closed to connections.');
|
||||||
|
|
||||||
winston.info('[app] Shutdown complete.');
|
winston.info('[app] Shutdown complete.');
|
||||||
process.exit(code || 0);
|
process.exit(code || 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
function restart() {
|
function restart() {
|
||||||
if (process.send) {
|
if (process.send) {
|
||||||
winston.info('[app] Restarting...');
|
winston.info('[app] Restarting...');
|
||||||
process.send({
|
process.send({
|
||||||
action: 'restart'
|
action: 'restart'
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
winston.error('[app] Could not restart server. Shutting down.');
|
winston.error('[app] Could not restart server. Shutting down.');
|
||||||
shutdown(1);
|
shutdown(1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,34 +1,34 @@
|
|||||||
[
|
[
|
||||||
{
|
{
|
||||||
"name": "Announcements",
|
"name": "Announcements",
|
||||||
"description": "Announcements regarding our community",
|
"description": "Announcements regarding our community",
|
||||||
"bgColor": "#fda34b",
|
"bgColor": "#fda34b",
|
||||||
"color": "#fff",
|
"color": "#fff",
|
||||||
"icon" : "fa-bullhorn",
|
"icon" : "fa-bullhorn",
|
||||||
"order": 1
|
"order": 1
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "General Discussion",
|
"name": "General Discussion",
|
||||||
"description": "A place to talk about whatever you want",
|
"description": "A place to talk about whatever you want",
|
||||||
"bgColor": "#59b3d0",
|
"bgColor": "#59b3d0",
|
||||||
"color": "#fff",
|
"color": "#fff",
|
||||||
"icon" : "fa-comments-o",
|
"icon" : "fa-comments-o",
|
||||||
"order": 2
|
"order": 2
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Blogs",
|
"name": "Blogs",
|
||||||
"description": "Blog posts from individual members",
|
"description": "Blog posts from individual members",
|
||||||
"bgColor": "#86ba4b",
|
"bgColor": "#86ba4b",
|
||||||
"color": "#fff",
|
"color": "#fff",
|
||||||
"icon" : "fa-newspaper-o",
|
"icon" : "fa-newspaper-o",
|
||||||
"order": 4
|
"order": 4
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"name": "Comments & Feedback",
|
"name": "Comments & Feedback",
|
||||||
"description": "Got a question? Ask away!",
|
"description": "Got a question? Ask away!",
|
||||||
"bgColor": "#e95c5a",
|
"bgColor": "#e95c5a",
|
||||||
"color": "#fff",
|
"color": "#fff",
|
||||||
"icon" : "fa-question",
|
"icon" : "fa-question",
|
||||||
"order": 3
|
"order": 3
|
||||||
}
|
}
|
||||||
]
|
]
|
@ -1,312 +1,312 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
var ajaxify = ajaxify || {};
|
var ajaxify = ajaxify || {};
|
||||||
|
|
||||||
$(document).ready(function() {
|
$(document).ready(function() {
|
||||||
|
|
||||||
/*global app, templates, utils, socket, config, RELATIVE_PATH*/
|
/*global app, templates, utils, socket, config, RELATIVE_PATH*/
|
||||||
|
|
||||||
var location = document.location || window.location,
|
var location = document.location || window.location,
|
||||||
rootUrl = location.protocol + '//' + (location.hostname || location.host) + (location.port ? ':' + location.port : ''),
|
rootUrl = location.protocol + '//' + (location.hostname || location.host) + (location.port ? ':' + location.port : ''),
|
||||||
apiXHR = null,
|
apiXHR = null,
|
||||||
|
|
||||||
translator;
|
translator;
|
||||||
|
|
||||||
// Dumb hack to fool ajaxify into thinking translator is still a global
|
// Dumb hack to fool ajaxify into thinking translator is still a global
|
||||||
// When ajaxify is migrated to a require.js module, then this can be merged into the "define" call
|
// When ajaxify is migrated to a require.js module, then this can be merged into the "define" call
|
||||||
require(['translator'], function(_translator) {
|
require(['translator'], function(_translator) {
|
||||||
translator = _translator;
|
translator = _translator;
|
||||||
});
|
});
|
||||||
|
|
||||||
$(window).on('popstate', function (ev) {
|
$(window).on('popstate', function (ev) {
|
||||||
ev = ev.originalEvent;
|
ev = ev.originalEvent;
|
||||||
|
|
||||||
if (ev !== null && ev.state && ev.state.url !== undefined) {
|
if (ev !== null && ev.state && ev.state.url !== undefined) {
|
||||||
ajaxify.go(ev.state.url, function() {
|
ajaxify.go(ev.state.url, function() {
|
||||||
$(window).trigger('action:popstate', {url: ev.state.url});
|
$(window).trigger('action:popstate', {url: ev.state.url});
|
||||||
}, true);
|
}, true);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
ajaxify.currentPage = null;
|
ajaxify.currentPage = null;
|
||||||
|
|
||||||
ajaxify.go = function (url, callback, quiet, search) {
|
ajaxify.go = function (url, callback, quiet, search) {
|
||||||
if (!socket.connected) {
|
if (!socket.connected) {
|
||||||
if (ajaxify.reconnectAction) {
|
if (ajaxify.reconnectAction) {
|
||||||
$(window).off('action:reconnected', ajaxify.reconnectAction);
|
$(window).off('action:reconnected', ajaxify.reconnectAction);
|
||||||
}
|
}
|
||||||
ajaxify.reconnectAction = function(e) {
|
ajaxify.reconnectAction = function(e) {
|
||||||
ajaxify.go(url, callback, quiet, search);
|
ajaxify.go(url, callback, quiet, search);
|
||||||
$(window).off(e);
|
$(window).off(e);
|
||||||
}
|
}
|
||||||
$(window).on('action:reconnected', ajaxify.reconnectAction);
|
$(window).on('action:reconnected', ajaxify.reconnectAction);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ajaxify.handleRedirects(url)) {
|
if (ajaxify.handleRedirects(url)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
app.enterRoom('');
|
app.enterRoom('');
|
||||||
|
|
||||||
$(window).off('scroll');
|
$(window).off('scroll');
|
||||||
|
|
||||||
if ($('#content').hasClass('ajaxifying') && apiXHR) {
|
if ($('#content').hasClass('ajaxifying') && apiXHR) {
|
||||||
apiXHR.abort();
|
apiXHR.abort();
|
||||||
}
|
}
|
||||||
|
|
||||||
url = ajaxify.start(url, quiet, search);
|
url = ajaxify.start(url, quiet, search);
|
||||||
|
|
||||||
$('#footer, #content').removeClass('hide').addClass('ajaxifying');
|
$('#footer, #content').removeClass('hide').addClass('ajaxifying');
|
||||||
|
|
||||||
ajaxify.variables.flush();
|
ajaxify.variables.flush();
|
||||||
ajaxify.loadData(url, function(err, data) {
|
ajaxify.loadData(url, function(err, data) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return onAjaxError(err, url, callback, quiet);
|
return onAjaxError(err, url, callback, quiet);
|
||||||
}
|
}
|
||||||
|
|
||||||
app.template = data.template.name;
|
app.template = data.template.name;
|
||||||
|
|
||||||
require(['translator'], function(translator) {
|
require(['translator'], function(translator) {
|
||||||
translator.load(config.defaultLang, data.template.name);
|
translator.load(config.defaultLang, data.template.name);
|
||||||
renderTemplate(url, data.template.name, data, callback);
|
renderTemplate(url, data.template.name, data, callback);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
};
|
};
|
||||||
|
|
||||||
ajaxify.handleRedirects = function(url) {
|
ajaxify.handleRedirects = function(url) {
|
||||||
url = ajaxify.removeRelativePath(url.replace(/\/$/, '')).toLowerCase();
|
url = ajaxify.removeRelativePath(url.replace(/\/$/, '')).toLowerCase();
|
||||||
var isAdminRoute = url.startsWith('admin') && window.location.pathname.indexOf(RELATIVE_PATH + '/admin') !== 0;
|
var isAdminRoute = url.startsWith('admin') && window.location.pathname.indexOf(RELATIVE_PATH + '/admin') !== 0;
|
||||||
var uploadsOrApi = url.startsWith('uploads') || url.startsWith('api');
|
var uploadsOrApi = url.startsWith('uploads') || url.startsWith('api');
|
||||||
if (isAdminRoute || uploadsOrApi) {
|
if (isAdminRoute || uploadsOrApi) {
|
||||||
window.open(RELATIVE_PATH + '/' + url, '_top');
|
window.open(RELATIVE_PATH + '/' + url, '_top');
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
ajaxify.start = function(url, quiet, search) {
|
ajaxify.start = function(url, quiet, search) {
|
||||||
url = ajaxify.removeRelativePath(url.replace(/^\/|\/$/g, ''));
|
url = ajaxify.removeRelativePath(url.replace(/^\/|\/$/g, ''));
|
||||||
var hash = window.location.hash;
|
var hash = window.location.hash;
|
||||||
search = search || '';
|
search = search || '';
|
||||||
|
|
||||||
$(window).trigger('action:ajaxify.start', {url: url});
|
$(window).trigger('action:ajaxify.start', {url: url});
|
||||||
|
|
||||||
if (!window.location.pathname.match(/\/(403|404)$/g)) {
|
if (!window.location.pathname.match(/\/(403|404)$/g)) {
|
||||||
app.previousUrl = window.location.href;
|
app.previousUrl = window.location.href;
|
||||||
}
|
}
|
||||||
|
|
||||||
ajaxify.currentPage = url;
|
ajaxify.currentPage = url;
|
||||||
|
|
||||||
if (window.history && window.history.pushState) {
|
if (window.history && window.history.pushState) {
|
||||||
window.history[!quiet ? 'pushState' : 'replaceState']({
|
window.history[!quiet ? 'pushState' : 'replaceState']({
|
||||||
url: url + search + hash
|
url: url + search + hash
|
||||||
}, url, RELATIVE_PATH + '/' + url + search + hash);
|
}, url, RELATIVE_PATH + '/' + url + search + hash);
|
||||||
}
|
}
|
||||||
return url;
|
return url;
|
||||||
};
|
};
|
||||||
|
|
||||||
function onAjaxError(err, url, callback, quiet) {
|
function onAjaxError(err, url, callback, quiet) {
|
||||||
var data = err.data,
|
var data = err.data,
|
||||||
textStatus = err.textStatus;
|
textStatus = err.textStatus;
|
||||||
|
|
||||||
if (data) {
|
if (data) {
|
||||||
var status = parseInt(data.status, 10);
|
var status = parseInt(data.status, 10);
|
||||||
if (status === 403 || status === 404 || status === 500 || status === 502) {
|
if (status === 403 || status === 404 || status === 500 || status === 502) {
|
||||||
if (status === 502) {
|
if (status === 502) {
|
||||||
status = 500;
|
status = 500;
|
||||||
}
|
}
|
||||||
|
|
||||||
$('#footer, #content').removeClass('hide').addClass('ajaxifying');
|
$('#footer, #content').removeClass('hide').addClass('ajaxifying');
|
||||||
return renderTemplate(url, status.toString(), data.responseJSON, (new Date()).getTime(), callback);
|
return renderTemplate(url, status.toString(), data.responseJSON, (new Date()).getTime(), callback);
|
||||||
} else if (status === 401) {
|
} else if (status === 401) {
|
||||||
app.alertError('[[global:please_log_in]]');
|
app.alertError('[[global:please_log_in]]');
|
||||||
app.previousUrl = url;
|
app.previousUrl = url;
|
||||||
return ajaxify.go('login');
|
return ajaxify.go('login');
|
||||||
} else if (status === 302) {
|
} else if (status === 302) {
|
||||||
if (data.responseJSON.external) {
|
if (data.responseJSON.external) {
|
||||||
window.location.href = data.responseJSON.external;
|
window.location.href = data.responseJSON.external;
|
||||||
} else if (typeof data.responseJSON === 'string') {
|
} else if (typeof data.responseJSON === 'string') {
|
||||||
ajaxify.go(data.responseJSON.slice(1), callback, quiet);
|
ajaxify.go(data.responseJSON.slice(1), callback, quiet);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (textStatus !== 'abort') {
|
} else if (textStatus !== 'abort') {
|
||||||
app.alertError(data.responseJSON.error);
|
app.alertError(data.responseJSON.error);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function renderTemplate(url, tpl_url, data, callback) {
|
function renderTemplate(url, tpl_url, data, callback) {
|
||||||
$(window).trigger('action:ajaxify.loadingTemplates', {});
|
$(window).trigger('action:ajaxify.loadingTemplates', {});
|
||||||
|
|
||||||
templates.parse(tpl_url, data, function(template) {
|
templates.parse(tpl_url, data, function(template) {
|
||||||
translator.translate(template, function(translatedTemplate) {
|
translator.translate(template, function(translatedTemplate) {
|
||||||
$('#content').html(translatedTemplate);
|
$('#content').html(translatedTemplate);
|
||||||
|
|
||||||
ajaxify.end(url, tpl_url);
|
ajaxify.end(url, tpl_url);
|
||||||
|
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
|
|
||||||
$('#content, #footer').removeClass('ajaxifying');
|
$('#content, #footer').removeClass('ajaxifying');
|
||||||
|
|
||||||
app.refreshTitle(url);
|
app.refreshTitle(url);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
ajaxify.end = function(url, tpl_url) {
|
ajaxify.end = function(url, tpl_url) {
|
||||||
function done() {
|
function done() {
|
||||||
if (--count === 0) {
|
if (--count === 0) {
|
||||||
$(window).trigger('action:ajaxify.end', {url: url});
|
$(window).trigger('action:ajaxify.end', {url: url});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
var count = 2;
|
var count = 2;
|
||||||
|
|
||||||
ajaxify.variables.parse();
|
ajaxify.variables.parse();
|
||||||
|
|
||||||
ajaxify.loadScript(tpl_url, done);
|
ajaxify.loadScript(tpl_url, done);
|
||||||
|
|
||||||
ajaxify.widgets.render(tpl_url, url, done);
|
ajaxify.widgets.render(tpl_url, url, done);
|
||||||
|
|
||||||
$(window).trigger('action:ajaxify.contentLoaded', {url: url, tpl: tpl_url});
|
$(window).trigger('action:ajaxify.contentLoaded', {url: url, tpl: tpl_url});
|
||||||
|
|
||||||
app.processPage();
|
app.processPage();
|
||||||
};
|
};
|
||||||
|
|
||||||
ajaxify.removeRelativePath = function(url) {
|
ajaxify.removeRelativePath = function(url) {
|
||||||
if (url.startsWith(RELATIVE_PATH.slice(1))) {
|
if (url.startsWith(RELATIVE_PATH.slice(1))) {
|
||||||
url = url.slice(RELATIVE_PATH.length);
|
url = url.slice(RELATIVE_PATH.length);
|
||||||
}
|
}
|
||||||
return url;
|
return url;
|
||||||
};
|
};
|
||||||
|
|
||||||
ajaxify.refresh = function(e) {
|
ajaxify.refresh = function(e) {
|
||||||
if (e && e instanceof jQuery.Event) {
|
if (e && e instanceof jQuery.Event) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
}
|
}
|
||||||
|
|
||||||
ajaxify.go(ajaxify.currentPage, null, true);
|
ajaxify.go(ajaxify.currentPage, null, true);
|
||||||
};
|
};
|
||||||
|
|
||||||
ajaxify.loadScript = function(tpl_url, callback) {
|
ajaxify.loadScript = function(tpl_url, callback) {
|
||||||
var location = !app.inAdmin ? 'forum/' : '';
|
var location = !app.inAdmin ? 'forum/' : '';
|
||||||
|
|
||||||
require([location + tpl_url], function(script) {
|
require([location + tpl_url], function(script) {
|
||||||
if (script && script.init) {
|
if (script && script.init) {
|
||||||
script.init();
|
script.init();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (callback) {
|
if (callback) {
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
ajaxify.loadData = function(url, callback) {
|
ajaxify.loadData = function(url, callback) {
|
||||||
url = ajaxify.removeRelativePath(url);
|
url = ajaxify.removeRelativePath(url);
|
||||||
|
|
||||||
$(window).trigger('action:ajaxify.loadingData', {url: url});
|
$(window).trigger('action:ajaxify.loadingData', {url: url});
|
||||||
|
|
||||||
apiXHR = $.ajax({
|
apiXHR = $.ajax({
|
||||||
url: RELATIVE_PATH + '/api/' + url,
|
url: RELATIVE_PATH + '/api/' + url,
|
||||||
cache: false,
|
cache: false,
|
||||||
success: function(data) {
|
success: function(data) {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
ajaxify.data = data;
|
ajaxify.data = data;
|
||||||
data.relative_path = RELATIVE_PATH;
|
data.relative_path = RELATIVE_PATH;
|
||||||
$(window).trigger('action:ajaxify.dataLoaded', {url: url, data: data});
|
$(window).trigger('action:ajaxify.dataLoaded', {url: url, data: data});
|
||||||
|
|
||||||
if (callback) {
|
if (callback) {
|
||||||
callback(null, data);
|
callback(null, data);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
error: function(data, textStatus) {
|
error: function(data, textStatus) {
|
||||||
if (data.status === 0 && textStatus === 'error') {
|
if (data.status === 0 && textStatus === 'error') {
|
||||||
data.status = 500;
|
data.status = 500;
|
||||||
}
|
}
|
||||||
callback({
|
callback({
|
||||||
data: data,
|
data: data,
|
||||||
textStatus: textStatus
|
textStatus: textStatus
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
ajaxify.loadTemplate = function(template, callback) {
|
ajaxify.loadTemplate = function(template, callback) {
|
||||||
if (templates.cache[template]) {
|
if (templates.cache[template]) {
|
||||||
callback(templates.cache[template]);
|
callback(templates.cache[template]);
|
||||||
} else {
|
} else {
|
||||||
$.ajax({
|
$.ajax({
|
||||||
url: RELATIVE_PATH + '/templates/' + template + '.tpl' + (config['cache-buster'] ? '?v=' + config['cache-buster'] : ''),
|
url: RELATIVE_PATH + '/templates/' + template + '.tpl' + (config['cache-buster'] ? '?v=' + config['cache-buster'] : ''),
|
||||||
type: 'GET',
|
type: 'GET',
|
||||||
success: function(data) {
|
success: function(data) {
|
||||||
callback(data.toString());
|
callback(data.toString());
|
||||||
},
|
},
|
||||||
error: function(error) {
|
error: function(error) {
|
||||||
throw new Error("Unable to load template: " + template + " (" + error.statusText + ")");
|
throw new Error("Unable to load template: " + template + " (" + error.statusText + ")");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
function ajaxifyAnchors() {
|
function ajaxifyAnchors() {
|
||||||
templates.registerLoader(ajaxify.loadTemplate);
|
templates.registerLoader(ajaxify.loadTemplate);
|
||||||
|
|
||||||
function hrefEmpty(href) {
|
function hrefEmpty(href) {
|
||||||
return href === undefined || href === '' || href === 'javascript:;';
|
return href === undefined || href === '' || href === 'javascript:;';
|
||||||
}
|
}
|
||||||
|
|
||||||
// Enhancing all anchors to ajaxify...
|
// Enhancing all anchors to ajaxify...
|
||||||
$(document.body).on('click', 'a', function (e) {
|
$(document.body).on('click', 'a', function (e) {
|
||||||
if (this.target !== '' || (this.protocol !== 'http:' && this.protocol !== 'https:')) {
|
if (this.target !== '' || (this.protocol !== 'http:' && this.protocol !== 'https:')) {
|
||||||
return;
|
return;
|
||||||
} else if (hrefEmpty(this.href) || this.protocol === 'javascript:' || $(this).attr('data-ajaxify') === 'false' || $(this).attr('href') === '#') {
|
} else if (hrefEmpty(this.href) || this.protocol === 'javascript:' || $(this).attr('data-ajaxify') === 'false' || $(this).attr('href') === '#') {
|
||||||
return e.preventDefault();
|
return e.preventDefault();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!e.ctrlKey && !e.shiftKey && !e.metaKey && e.which === 1) {
|
if (!e.ctrlKey && !e.shiftKey && !e.metaKey && e.which === 1) {
|
||||||
if (
|
if (
|
||||||
this.host === '' || // Relative paths are always internal links...
|
this.host === '' || // Relative paths are always internal links...
|
||||||
(this.host === window.location.host && this.protocol === window.location.protocol && // Otherwise need to check that protocol and host match
|
(this.host === window.location.host && this.protocol === window.location.protocol && // Otherwise need to check that protocol and host match
|
||||||
(RELATIVE_PATH.length > 0 ? this.pathname.indexOf(RELATIVE_PATH) === 0 : true)) // Subfolder installs need this additional check
|
(RELATIVE_PATH.length > 0 ? this.pathname.indexOf(RELATIVE_PATH) === 0 : true)) // Subfolder installs need this additional check
|
||||||
) {
|
) {
|
||||||
// Internal link
|
// Internal link
|
||||||
var url = this.pathname.replace(RELATIVE_PATH + '/', '');
|
var url = this.pathname.replace(RELATIVE_PATH + '/', '');
|
||||||
|
|
||||||
// Special handling for urls with hashes
|
// Special handling for urls with hashes
|
||||||
if (window.location.pathname === this.pathname && this.hash.length) {
|
if (window.location.pathname === this.pathname && this.hash.length) {
|
||||||
window.location.hash = this.hash;
|
window.location.hash = this.hash;
|
||||||
} else {
|
} else {
|
||||||
window.location.hash = '';
|
window.location.hash = '';
|
||||||
if (ajaxify.go(url)) {
|
if (ajaxify.go(url)) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else if (window.location.pathname !== '/outgoing') {
|
} else if (window.location.pathname !== '/outgoing') {
|
||||||
// External Link
|
// External Link
|
||||||
if (config.openOutgoingLinksInNewTab) {
|
if (config.openOutgoingLinksInNewTab) {
|
||||||
window.open(this.href, '_blank');
|
window.open(this.href, '_blank');
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
} else if (config.useOutgoingLinksPage) {
|
} else if (config.useOutgoingLinksPage) {
|
||||||
ajaxify.go('outgoing?url=' + encodeURIComponent(this.href));
|
ajaxify.go('outgoing?url=' + encodeURIComponent(this.href));
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (window.history && window.history.pushState) {
|
if (window.history && window.history.pushState) {
|
||||||
// Progressive Enhancement, ajaxify available only to modern browsers
|
// Progressive Enhancement, ajaxify available only to modern browsers
|
||||||
ajaxifyAnchors();
|
ajaxifyAnchors();
|
||||||
}
|
}
|
||||||
|
|
||||||
app.load();
|
app.load();
|
||||||
templates.cache['500'] = $('.tpl-500').html();
|
templates.cache['500'] = $('.tpl-500').html();
|
||||||
|
|
||||||
});
|
});
|
File diff suppressed because it is too large
Load Diff
@ -1,29 +1,29 @@
|
|||||||
*.pyc
|
*.pyc
|
||||||
*.egg-info
|
*.egg-info
|
||||||
*.db
|
*.db
|
||||||
*.db.old
|
*.db.old
|
||||||
*.swp
|
*.swp
|
||||||
*.db-journal
|
*.db-journal
|
||||||
|
|
||||||
.coverage
|
.coverage
|
||||||
.DS_Store
|
.DS_Store
|
||||||
.installed.cfg
|
.installed.cfg
|
||||||
|
|
||||||
.idea/*
|
.idea/*
|
||||||
.svn/*
|
.svn/*
|
||||||
src/website/static/*
|
src/website/static/*
|
||||||
src/website/media/*
|
src/website/media/*
|
||||||
|
|
||||||
bin
|
bin
|
||||||
build
|
build
|
||||||
cfcache
|
cfcache
|
||||||
develop-eggs
|
develop-eggs
|
||||||
dist
|
dist
|
||||||
downloads
|
downloads
|
||||||
eggs
|
eggs
|
||||||
parts
|
parts
|
||||||
tmp
|
tmp
|
||||||
.sass-cache
|
.sass-cache
|
||||||
|
|
||||||
src/website/settingslocal.py
|
src/website/settingslocal.py
|
||||||
stunnel.log
|
stunnel.log
|
@ -1,19 +1,19 @@
|
|||||||
//Uzbek
|
//Uzbek
|
||||||
jQuery.timeago.settings.strings = {
|
jQuery.timeago.settings.strings = {
|
||||||
prefixAgo: null,
|
prefixAgo: null,
|
||||||
prefixFromNow: "keyin",
|
prefixFromNow: "keyin",
|
||||||
suffixAgo: "avval",
|
suffixAgo: "avval",
|
||||||
suffixFromNow: null,
|
suffixFromNow: null,
|
||||||
seconds: "bir necha soniya",
|
seconds: "bir necha soniya",
|
||||||
minute: "1 daqiqa",
|
minute: "1 daqiqa",
|
||||||
minutes: function(value) { return "%d daqiqa" },
|
minutes: function(value) { return "%d daqiqa" },
|
||||||
hour: "1 soat",
|
hour: "1 soat",
|
||||||
hours: function(value) { return "%d soat" },
|
hours: function(value) { return "%d soat" },
|
||||||
day: "1 kun",
|
day: "1 kun",
|
||||||
days: function(value) { return "%d kun" },
|
days: function(value) { return "%d kun" },
|
||||||
month: "1 oy",
|
month: "1 oy",
|
||||||
months: function(value) { return "%d oy" },
|
months: function(value) { return "%d oy" },
|
||||||
year: "1 yil",
|
year: "1 yil",
|
||||||
years: function(value) { return "%d yil" },
|
years: function(value) { return "%d yil" },
|
||||||
wordSeparator: " "
|
wordSeparator: " "
|
||||||
};
|
};
|
||||||
|
@ -1,85 +1,85 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
var express = require('express');
|
var express = require('express');
|
||||||
|
|
||||||
|
|
||||||
function apiRoutes(router, middleware, controllers) {
|
function apiRoutes(router, middleware, controllers) {
|
||||||
router.get('/users/csv', middleware.authenticate, controllers.admin.users.getCSV);
|
router.get('/users/csv', middleware.authenticate, controllers.admin.users.getCSV);
|
||||||
|
|
||||||
var multipart = require('connect-multiparty');
|
var multipart = require('connect-multiparty');
|
||||||
var multipartMiddleware = multipart();
|
var multipartMiddleware = multipart();
|
||||||
|
|
||||||
var middlewares = [multipartMiddleware, middleware.validateFiles, middleware.applyCSRF, middleware.authenticate];
|
var middlewares = [multipartMiddleware, middleware.validateFiles, middleware.applyCSRF, middleware.authenticate];
|
||||||
|
|
||||||
router.post('/category/uploadpicture', middlewares, controllers.admin.uploads.uploadCategoryPicture);
|
router.post('/category/uploadpicture', middlewares, controllers.admin.uploads.uploadCategoryPicture);
|
||||||
router.post('/uploadfavicon', middlewares, controllers.admin.uploads.uploadFavicon);
|
router.post('/uploadfavicon', middlewares, controllers.admin.uploads.uploadFavicon);
|
||||||
router.post('/uploadlogo', middlewares, controllers.admin.uploads.uploadLogo);
|
router.post('/uploadlogo', middlewares, controllers.admin.uploads.uploadLogo);
|
||||||
router.post('/uploadgravatardefault', middlewares, controllers.admin.uploads.uploadGravatarDefault);
|
router.post('/uploadgravatardefault', middlewares, controllers.admin.uploads.uploadGravatarDefault);
|
||||||
}
|
}
|
||||||
|
|
||||||
function adminRouter(middleware, controllers) {
|
function adminRouter(middleware, controllers) {
|
||||||
var router = express.Router();
|
var router = express.Router();
|
||||||
|
|
||||||
router.use(middleware.admin.buildHeader);
|
router.use(middleware.admin.buildHeader);
|
||||||
|
|
||||||
addRoutes(router, middleware, controllers);
|
addRoutes(router, middleware, controllers);
|
||||||
|
|
||||||
return router;
|
return router;
|
||||||
}
|
}
|
||||||
|
|
||||||
function apiRouter(middleware, controllers) {
|
function apiRouter(middleware, controllers) {
|
||||||
var router = express.Router();
|
var router = express.Router();
|
||||||
|
|
||||||
addRoutes(router, middleware, controllers);
|
addRoutes(router, middleware, controllers);
|
||||||
|
|
||||||
apiRoutes(router, middleware, controllers);
|
apiRoutes(router, middleware, controllers);
|
||||||
|
|
||||||
return router;
|
return router;
|
||||||
}
|
}
|
||||||
|
|
||||||
function addRoutes(router, middleware, controllers) {
|
function addRoutes(router, middleware, controllers) {
|
||||||
router.get('/', controllers.admin.home);
|
router.get('/', controllers.admin.home);
|
||||||
router.get('/general/dashboard', controllers.admin.home);
|
router.get('/general/dashboard', controllers.admin.home);
|
||||||
router.get('/general/languages', controllers.admin.languages.get);
|
router.get('/general/languages', controllers.admin.languages.get);
|
||||||
router.get('/general/sounds', controllers.admin.sounds.get);
|
router.get('/general/sounds', controllers.admin.sounds.get);
|
||||||
router.get('/general/navigation', controllers.admin.navigation.get);
|
router.get('/general/navigation', controllers.admin.navigation.get);
|
||||||
router.get('/general/homepage', controllers.admin.homepage.get);
|
router.get('/general/homepage', controllers.admin.homepage.get);
|
||||||
|
|
||||||
router.get('/manage/categories', controllers.admin.categories.getAll);
|
router.get('/manage/categories', controllers.admin.categories.getAll);
|
||||||
router.get('/manage/categories/:category_id', controllers.admin.categories.get);
|
router.get('/manage/categories/:category_id', controllers.admin.categories.get);
|
||||||
|
|
||||||
router.get('/manage/tags', controllers.admin.tags.get);
|
router.get('/manage/tags', controllers.admin.tags.get);
|
||||||
|
|
||||||
router.get('/manage/flags', controllers.admin.flags.get);
|
router.get('/manage/flags', controllers.admin.flags.get);
|
||||||
|
|
||||||
router.get('/manage/users', controllers.admin.users.sortByJoinDate);
|
router.get('/manage/users', controllers.admin.users.sortByJoinDate);
|
||||||
router.get('/manage/users/search', controllers.admin.users.search);
|
router.get('/manage/users/search', controllers.admin.users.search);
|
||||||
router.get('/manage/users/latest', controllers.admin.users.sortByJoinDate);
|
router.get('/manage/users/latest', controllers.admin.users.sortByJoinDate);
|
||||||
router.get('/manage/users/sort-posts', controllers.admin.users.sortByPosts);
|
router.get('/manage/users/sort-posts', controllers.admin.users.sortByPosts);
|
||||||
router.get('/manage/users/sort-reputation', controllers.admin.users.sortByReputation);
|
router.get('/manage/users/sort-reputation', controllers.admin.users.sortByReputation);
|
||||||
router.get('/manage/users/banned', controllers.admin.users.banned);
|
router.get('/manage/users/banned', controllers.admin.users.banned);
|
||||||
router.get('/manage/users/registration', controllers.admin.users.registrationQueue);
|
router.get('/manage/users/registration', controllers.admin.users.registrationQueue);
|
||||||
|
|
||||||
router.get('/manage/groups', controllers.admin.groups.list);
|
router.get('/manage/groups', controllers.admin.groups.list);
|
||||||
router.get('/manage/groups/:name', controllers.admin.groups.get);
|
router.get('/manage/groups/:name', controllers.admin.groups.get);
|
||||||
|
|
||||||
router.get('/settings/:term?', controllers.admin.settings.get);
|
router.get('/settings/:term?', controllers.admin.settings.get);
|
||||||
|
|
||||||
router.get('/appearance/:term?', controllers.admin.appearance.get);
|
router.get('/appearance/:term?', controllers.admin.appearance.get);
|
||||||
|
|
||||||
router.get('/extend/plugins', controllers.admin.plugins.get);
|
router.get('/extend/plugins', controllers.admin.plugins.get);
|
||||||
router.get('/extend/widgets', controllers.admin.extend.widgets);
|
router.get('/extend/widgets', controllers.admin.extend.widgets);
|
||||||
router.get('/extend/rewards', controllers.admin.extend.rewards);
|
router.get('/extend/rewards', controllers.admin.extend.rewards);
|
||||||
|
|
||||||
router.get('/advanced/database', controllers.admin.database.get);
|
router.get('/advanced/database', controllers.admin.database.get);
|
||||||
router.get('/advanced/events', controllers.admin.events.get);
|
router.get('/advanced/events', controllers.admin.events.get);
|
||||||
router.get('/advanced/logs', controllers.admin.logs.get);
|
router.get('/advanced/logs', controllers.admin.logs.get);
|
||||||
router.get('/advanced/post-cache', controllers.admin.postCache.get);
|
router.get('/advanced/post-cache', controllers.admin.postCache.get);
|
||||||
|
|
||||||
router.get('/development/logger', controllers.admin.logger.get);
|
router.get('/development/logger', controllers.admin.logger.get);
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports = function(app, middleware, controllers) {
|
module.exports = function(app, middleware, controllers) {
|
||||||
app.use('/admin/', adminRouter(middleware, controllers));
|
app.use('/admin/', adminRouter(middleware, controllers));
|
||||||
app.use('/api/admin/', apiRouter(middleware, controllers));
|
app.use('/api/admin/', apiRouter(middleware, controllers));
|
||||||
};
|
};
|
||||||
|
@ -1,86 +1,86 @@
|
|||||||
(function(Auth) {
|
(function(Auth) {
|
||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
var passport = require('passport'),
|
var passport = require('passport'),
|
||||||
passportLocal = require('passport-local').Strategy,
|
passportLocal = require('passport-local').Strategy,
|
||||||
nconf = require('nconf'),
|
nconf = require('nconf'),
|
||||||
winston = require('winston'),
|
winston = require('winston'),
|
||||||
express = require('express'),
|
express = require('express'),
|
||||||
|
|
||||||
controllers = require('../controllers'),
|
controllers = require('../controllers'),
|
||||||
plugins = require('../plugins'),
|
plugins = require('../plugins'),
|
||||||
hotswap = require('../hotswap'),
|
hotswap = require('../hotswap'),
|
||||||
|
|
||||||
loginStrategies = [];
|
loginStrategies = [];
|
||||||
|
|
||||||
Auth.initialize = function(app, middleware) {
|
Auth.initialize = function(app, middleware) {
|
||||||
app.use(passport.initialize());
|
app.use(passport.initialize());
|
||||||
app.use(passport.session());
|
app.use(passport.session());
|
||||||
|
|
||||||
app.use(function(req, res, next) {
|
app.use(function(req, res, next) {
|
||||||
req.uid = req.user ? parseInt(req.user.uid, 10) : 0;
|
req.uid = req.user ? parseInt(req.user.uid, 10) : 0;
|
||||||
next();
|
next();
|
||||||
});
|
});
|
||||||
|
|
||||||
Auth.app = app;
|
Auth.app = app;
|
||||||
Auth.middleware = middleware;
|
Auth.middleware = middleware;
|
||||||
};
|
};
|
||||||
|
|
||||||
Auth.getLoginStrategies = function() {
|
Auth.getLoginStrategies = function() {
|
||||||
return loginStrategies;
|
return loginStrategies;
|
||||||
};
|
};
|
||||||
|
|
||||||
Auth.reloadRoutes = function(callback) {
|
Auth.reloadRoutes = function(callback) {
|
||||||
var router = express.Router();
|
var router = express.Router();
|
||||||
router.hotswapId = 'auth';
|
router.hotswapId = 'auth';
|
||||||
|
|
||||||
loginStrategies.length = 0;
|
loginStrategies.length = 0;
|
||||||
|
|
||||||
if (plugins.hasListeners('action:auth.overrideLogin')) {
|
if (plugins.hasListeners('action:auth.overrideLogin')) {
|
||||||
winston.warn('[authentication] Login override detected, skipping local login strategy.');
|
winston.warn('[authentication] Login override detected, skipping local login strategy.');
|
||||||
plugins.fireHook('action:auth.overrideLogin');
|
plugins.fireHook('action:auth.overrideLogin');
|
||||||
} else {
|
} else {
|
||||||
passport.use(new passportLocal({passReqToCallback: true}, controllers.authentication.localLogin));
|
passport.use(new passportLocal({passReqToCallback: true}, controllers.authentication.localLogin));
|
||||||
}
|
}
|
||||||
|
|
||||||
plugins.fireHook('filter:auth.init', loginStrategies, function(err) {
|
plugins.fireHook('filter:auth.init', loginStrategies, function(err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
winston.error('filter:auth.init - plugin failure');
|
winston.error('filter:auth.init - plugin failure');
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
loginStrategies.forEach(function(strategy) {
|
loginStrategies.forEach(function(strategy) {
|
||||||
if (strategy.url) {
|
if (strategy.url) {
|
||||||
router.get(strategy.url, passport.authenticate(strategy.name, {
|
router.get(strategy.url, passport.authenticate(strategy.name, {
|
||||||
scope: strategy.scope
|
scope: strategy.scope
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|
||||||
router.get(strategy.callbackURL, passport.authenticate(strategy.name, {
|
router.get(strategy.callbackURL, passport.authenticate(strategy.name, {
|
||||||
successReturnToOrRedirect: nconf.get('relative_path') + (strategy.successUrl !== undefined ? strategy.successUrl : '/'),
|
successReturnToOrRedirect: nconf.get('relative_path') + (strategy.successUrl !== undefined ? strategy.successUrl : '/'),
|
||||||
failureRedirect: nconf.get('relative_path') + (strategy.failureUrl !== undefined ? strategy.failureUrl : '/login')
|
failureRedirect: nconf.get('relative_path') + (strategy.failureUrl !== undefined ? strategy.failureUrl : '/login')
|
||||||
}));
|
}));
|
||||||
});
|
});
|
||||||
|
|
||||||
router.post('/register', Auth.middleware.applyCSRF, controllers.authentication.register);
|
router.post('/register', Auth.middleware.applyCSRF, controllers.authentication.register);
|
||||||
router.post('/login', Auth.middleware.applyCSRF, controllers.authentication.login);
|
router.post('/login', Auth.middleware.applyCSRF, controllers.authentication.login);
|
||||||
router.post('/logout', Auth.middleware.applyCSRF, controllers.authentication.logout);
|
router.post('/logout', Auth.middleware.applyCSRF, controllers.authentication.logout);
|
||||||
|
|
||||||
hotswap.replace('auth', router);
|
hotswap.replace('auth', router);
|
||||||
if (typeof callback === 'function') {
|
if (typeof callback === 'function') {
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
passport.serializeUser(function(user, done) {
|
passport.serializeUser(function(user, done) {
|
||||||
done(null, user.uid);
|
done(null, user.uid);
|
||||||
});
|
});
|
||||||
|
|
||||||
passport.deserializeUser(function(uid, done) {
|
passport.deserializeUser(function(uid, done) {
|
||||||
done(null, {
|
done(null, {
|
||||||
uid: uid
|
uid: uid
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
}(exports));
|
}(exports));
|
||||||
|
@ -1,379 +1,379 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
|
|
||||||
var async = require('async'),
|
var async = require('async'),
|
||||||
validator = require('validator'),
|
validator = require('validator'),
|
||||||
|
|
||||||
_ = require('underscore'),
|
_ = require('underscore'),
|
||||||
db = require('./database'),
|
db = require('./database'),
|
||||||
posts = require('./posts'),
|
posts = require('./posts'),
|
||||||
utils = require('../public/src/utils'),
|
utils = require('../public/src/utils'),
|
||||||
plugins = require('./plugins'),
|
plugins = require('./plugins'),
|
||||||
user = require('./user'),
|
user = require('./user'),
|
||||||
categories = require('./categories'),
|
categories = require('./categories'),
|
||||||
privileges = require('./privileges');
|
privileges = require('./privileges');
|
||||||
|
|
||||||
(function(Topics) {
|
(function(Topics) {
|
||||||
|
|
||||||
require('./topics/create')(Topics);
|
require('./topics/create')(Topics);
|
||||||
require('./topics/delete')(Topics);
|
require('./topics/delete')(Topics);
|
||||||
require('./topics/unread')(Topics);
|
require('./topics/unread')(Topics);
|
||||||
require('./topics/recent')(Topics);
|
require('./topics/recent')(Topics);
|
||||||
require('./topics/popular')(Topics);
|
require('./topics/popular')(Topics);
|
||||||
require('./topics/user')(Topics);
|
require('./topics/user')(Topics);
|
||||||
require('./topics/fork')(Topics);
|
require('./topics/fork')(Topics);
|
||||||
require('./topics/posts')(Topics);
|
require('./topics/posts')(Topics);
|
||||||
require('./topics/follow')(Topics);
|
require('./topics/follow')(Topics);
|
||||||
require('./topics/tags')(Topics);
|
require('./topics/tags')(Topics);
|
||||||
require('./topics/teaser')(Topics);
|
require('./topics/teaser')(Topics);
|
||||||
require('./topics/suggested')(Topics);
|
require('./topics/suggested')(Topics);
|
||||||
|
|
||||||
Topics.exists = function(tid, callback) {
|
Topics.exists = function(tid, callback) {
|
||||||
db.isSortedSetMember('topics:tid', tid, callback);
|
db.isSortedSetMember('topics:tid', tid, callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopicData = function(tid, callback) {
|
Topics.getTopicData = function(tid, callback) {
|
||||||
db.getObject('topic:' + tid, function(err, topic) {
|
db.getObject('topic:' + tid, function(err, topic) {
|
||||||
if (err || !topic) {
|
if (err || !topic) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
modifyTopic(topic, callback);
|
modifyTopic(topic, callback);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopicsData = function(tids, callback) {
|
Topics.getTopicsData = function(tids, callback) {
|
||||||
var keys = [];
|
var keys = [];
|
||||||
|
|
||||||
for (var i=0; i<tids.length; ++i) {
|
for (var i=0; i<tids.length; ++i) {
|
||||||
keys.push('topic:' + tids[i]);
|
keys.push('topic:' + tids[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
db.getObjects(keys, function(err, topics) {
|
db.getObjects(keys, function(err, topics) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
async.map(topics, modifyTopic, callback);
|
async.map(topics, modifyTopic, callback);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
function modifyTopic(topic, callback) {
|
function modifyTopic(topic, callback) {
|
||||||
if (!topic) {
|
if (!topic) {
|
||||||
return callback(null, topic);
|
return callback(null, topic);
|
||||||
}
|
}
|
||||||
topic.title = validator.escape(topic.title);
|
topic.title = validator.escape(topic.title);
|
||||||
topic.relativeTime = utils.toISOString(topic.timestamp);
|
topic.relativeTime = utils.toISOString(topic.timestamp);
|
||||||
topic.lastposttimeISO = utils.toISOString(topic.lastposttime);
|
topic.lastposttimeISO = utils.toISOString(topic.lastposttime);
|
||||||
callback(null, topic);
|
callback(null, topic);
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.getPageCount = function(tid, uid, callback) {
|
Topics.getPageCount = function(tid, uid, callback) {
|
||||||
Topics.getTopicField(tid, 'postcount', function(err, postCount) {
|
Topics.getTopicField(tid, 'postcount', function(err, postCount) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
if (!parseInt(postCount, 10)) {
|
if (!parseInt(postCount, 10)) {
|
||||||
return callback(null, 1);
|
return callback(null, 1);
|
||||||
}
|
}
|
||||||
user.getSettings(uid, function(err, settings) {
|
user.getSettings(uid, function(err, settings) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
callback(null, Math.ceil((parseInt(postCount, 10) - 1) / settings.postsPerPage));
|
callback(null, Math.ceil((parseInt(postCount, 10) - 1) / settings.postsPerPage));
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTidPage = function(tid, uid, callback) {
|
Topics.getTidPage = function(tid, uid, callback) {
|
||||||
if(!tid) {
|
if(!tid) {
|
||||||
return callback(new Error('[[error:invalid-tid]]'));
|
return callback(new Error('[[error:invalid-tid]]'));
|
||||||
}
|
}
|
||||||
|
|
||||||
async.parallel({
|
async.parallel({
|
||||||
index: function(next) {
|
index: function(next) {
|
||||||
categories.getTopicIndex(tid, next);
|
categories.getTopicIndex(tid, next);
|
||||||
},
|
},
|
||||||
settings: function(next) {
|
settings: function(next) {
|
||||||
user.getSettings(uid, next);
|
user.getSettings(uid, next);
|
||||||
}
|
}
|
||||||
}, function(err, results) {
|
}, function(err, results) {
|
||||||
if(err) {
|
if(err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
callback(null, Math.ceil((results.index + 1) / results.settings.topicsPerPage));
|
callback(null, Math.ceil((results.index + 1) / results.settings.topicsPerPage));
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getCategoryData = function(tid, callback) {
|
Topics.getCategoryData = function(tid, callback) {
|
||||||
Topics.getTopicField(tid, 'cid', function(err, cid) {
|
Topics.getTopicField(tid, 'cid', function(err, cid) {
|
||||||
if (err) {
|
if (err) {
|
||||||
callback(err);
|
callback(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
categories.getCategoryData(cid, callback);
|
categories.getCategoryData(cid, callback);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopicsFromSet = function(set, uid, start, stop, callback) {
|
Topics.getTopicsFromSet = function(set, uid, start, stop, callback) {
|
||||||
async.waterfall([
|
async.waterfall([
|
||||||
function(next) {
|
function(next) {
|
||||||
db.getSortedSetRevRange(set, start, stop, next);
|
db.getSortedSetRevRange(set, start, stop, next);
|
||||||
},
|
},
|
||||||
function(tids, next) {
|
function(tids, next) {
|
||||||
Topics.getTopics(tids, uid, next);
|
Topics.getTopics(tids, uid, next);
|
||||||
},
|
},
|
||||||
function(topics, next) {
|
function(topics, next) {
|
||||||
next(null, {topics: topics, nextStart: stop + 1});
|
next(null, {topics: topics, nextStart: stop + 1});
|
||||||
}
|
}
|
||||||
], callback);
|
], callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopics = function(tids, uid, callback) {
|
Topics.getTopics = function(tids, uid, callback) {
|
||||||
async.waterfall([
|
async.waterfall([
|
||||||
function(next) {
|
function(next) {
|
||||||
privileges.topics.filterTids('read', tids, uid, next);
|
privileges.topics.filterTids('read', tids, uid, next);
|
||||||
},
|
},
|
||||||
function(tids, next) {
|
function(tids, next) {
|
||||||
Topics.getTopicsByTids(tids, uid, next);
|
Topics.getTopicsByTids(tids, uid, next);
|
||||||
}
|
}
|
||||||
], callback);
|
], callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopicsByTids = function(tids, uid, callback) {
|
Topics.getTopicsByTids = function(tids, uid, callback) {
|
||||||
if (!Array.isArray(tids) || !tids.length) {
|
if (!Array.isArray(tids) || !tids.length) {
|
||||||
return callback(null, []);
|
return callback(null, []);
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.getTopicsData(tids, function(err, topics) {
|
Topics.getTopicsData(tids, function(err, topics) {
|
||||||
function mapFilter(array, field) {
|
function mapFilter(array, field) {
|
||||||
return array.map(function(topic) {
|
return array.map(function(topic) {
|
||||||
return topic && topic[field] && topic[field].toString();
|
return topic && topic[field] && topic[field].toString();
|
||||||
}).filter(function(value, index, array) {
|
}).filter(function(value, index, array) {
|
||||||
return utils.isNumber(value) && array.indexOf(value) === index;
|
return utils.isNumber(value) && array.indexOf(value) === index;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
var uids = mapFilter(topics, 'uid');
|
var uids = mapFilter(topics, 'uid');
|
||||||
var cids = mapFilter(topics, 'cid');
|
var cids = mapFilter(topics, 'cid');
|
||||||
|
|
||||||
async.parallel({
|
async.parallel({
|
||||||
teasers: function(next) {
|
teasers: function(next) {
|
||||||
Topics.getTeasers(topics, next);
|
Topics.getTeasers(topics, next);
|
||||||
},
|
},
|
||||||
users: function(next) {
|
users: function(next) {
|
||||||
user.getMultipleUserFields(uids, ['uid', 'username', 'userslug', 'picture'], next);
|
user.getMultipleUserFields(uids, ['uid', 'username', 'userslug', 'picture'], next);
|
||||||
},
|
},
|
||||||
categories: function(next) {
|
categories: function(next) {
|
||||||
categories.getMultipleCategoryFields(cids, ['cid', 'name', 'slug', 'icon', 'bgColor', 'color', 'disabled'], next);
|
categories.getMultipleCategoryFields(cids, ['cid', 'name', 'slug', 'icon', 'bgColor', 'color', 'disabled'], next);
|
||||||
},
|
},
|
||||||
hasRead: function(next) {
|
hasRead: function(next) {
|
||||||
Topics.hasReadTopics(tids, uid, next);
|
Topics.hasReadTopics(tids, uid, next);
|
||||||
},
|
},
|
||||||
tags: function(next) {
|
tags: function(next) {
|
||||||
Topics.getTopicsTagsObjects(tids, next);
|
Topics.getTopicsTagsObjects(tids, next);
|
||||||
}
|
}
|
||||||
}, function(err, results) {
|
}, function(err, results) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
var users = _.object(uids, results.users);
|
var users = _.object(uids, results.users);
|
||||||
var categories = _.object(cids, results.categories);
|
var categories = _.object(cids, results.categories);
|
||||||
|
|
||||||
for (var i=0; i<topics.length; ++i) {
|
for (var i=0; i<topics.length; ++i) {
|
||||||
if (topics[i]) {
|
if (topics[i]) {
|
||||||
topics[i].category = categories[topics[i].cid];
|
topics[i].category = categories[topics[i].cid];
|
||||||
topics[i].user = users[topics[i].uid];
|
topics[i].user = users[topics[i].uid];
|
||||||
topics[i].teaser = results.teasers[i];
|
topics[i].teaser = results.teasers[i];
|
||||||
topics[i].tags = results.tags[i];
|
topics[i].tags = results.tags[i];
|
||||||
|
|
||||||
topics[i].isOwner = parseInt(topics[i].uid, 10) === parseInt(uid, 10);
|
topics[i].isOwner = parseInt(topics[i].uid, 10) === parseInt(uid, 10);
|
||||||
topics[i].pinned = parseInt(topics[i].pinned, 10) === 1;
|
topics[i].pinned = parseInt(topics[i].pinned, 10) === 1;
|
||||||
topics[i].locked = parseInt(topics[i].locked, 10) === 1;
|
topics[i].locked = parseInt(topics[i].locked, 10) === 1;
|
||||||
topics[i].deleted = parseInt(topics[i].deleted, 10) === 1;
|
topics[i].deleted = parseInt(topics[i].deleted, 10) === 1;
|
||||||
topics[i].unread = !results.hasRead[i];
|
topics[i].unread = !results.hasRead[i];
|
||||||
topics[i].unreplied = parseInt(topics[i].postcount, 10) <= 1;
|
topics[i].unreplied = parseInt(topics[i].postcount, 10) <= 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
topics = topics.filter(function(topic) {
|
topics = topics.filter(function(topic) {
|
||||||
return topic && topic.category && !topic.category.disabled;
|
return topic && topic.category && !topic.category.disabled;
|
||||||
});
|
});
|
||||||
|
|
||||||
plugins.fireHook('filter:topics.get', {topics: topics, uid: uid}, function(err, topicData) {
|
plugins.fireHook('filter:topics.get', {topics: topics, uid: uid}, function(err, topicData) {
|
||||||
callback(err, topicData.topics);
|
callback(err, topicData.topics);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopicWithPosts = function(tid, set, uid, start, stop, reverse, callback) {
|
Topics.getTopicWithPosts = function(tid, set, uid, start, stop, reverse, callback) {
|
||||||
Topics.getTopicData(tid, function(err, topicData) {
|
Topics.getTopicData(tid, function(err, topicData) {
|
||||||
if (err || !topicData) {
|
if (err || !topicData) {
|
||||||
return callback(err || new Error('[[error:no-topic]]'));
|
return callback(err || new Error('[[error:no-topic]]'));
|
||||||
}
|
}
|
||||||
|
|
||||||
async.parallel({
|
async.parallel({
|
||||||
posts: async.apply(getMainPostAndReplies, topicData, set, uid, start, stop, reverse),
|
posts: async.apply(getMainPostAndReplies, topicData, set, uid, start, stop, reverse),
|
||||||
category: async.apply(Topics.getCategoryData, tid),
|
category: async.apply(Topics.getCategoryData, tid),
|
||||||
threadTools: async.apply(plugins.fireHook, 'filter:topic.thread_tools', {topic: topicData, uid: uid, tools: []}),
|
threadTools: async.apply(plugins.fireHook, 'filter:topic.thread_tools', {topic: topicData, uid: uid, tools: []}),
|
||||||
tags: async.apply(Topics.getTopicTagsObjects, tid),
|
tags: async.apply(Topics.getTopicTagsObjects, tid),
|
||||||
isFollowing: async.apply(Topics.isFollowing, [tid], uid),
|
isFollowing: async.apply(Topics.isFollowing, [tid], uid),
|
||||||
bookmark: async.apply(Topics.getUserBookmark, tid, uid)
|
bookmark: async.apply(Topics.getUserBookmark, tid, uid)
|
||||||
}, function(err, results) {
|
}, function(err, results) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
topicData.posts = results.posts;
|
topicData.posts = results.posts;
|
||||||
topicData.category = results.category;
|
topicData.category = results.category;
|
||||||
topicData.thread_tools = results.threadTools.tools;
|
topicData.thread_tools = results.threadTools.tools;
|
||||||
topicData.tags = results.tags;
|
topicData.tags = results.tags;
|
||||||
topicData.isFollowing = results.isFollowing[0];
|
topicData.isFollowing = results.isFollowing[0];
|
||||||
topicData.bookmark = results.bookmark;
|
topicData.bookmark = results.bookmark;
|
||||||
|
|
||||||
topicData.unreplied = parseInt(topicData.postcount, 10) === 1;
|
topicData.unreplied = parseInt(topicData.postcount, 10) === 1;
|
||||||
topicData.deleted = parseInt(topicData.deleted, 10) === 1;
|
topicData.deleted = parseInt(topicData.deleted, 10) === 1;
|
||||||
topicData.locked = parseInt(topicData.locked, 10) === 1;
|
topicData.locked = parseInt(topicData.locked, 10) === 1;
|
||||||
topicData.pinned = parseInt(topicData.pinned, 10) === 1;
|
topicData.pinned = parseInt(topicData.pinned, 10) === 1;
|
||||||
|
|
||||||
plugins.fireHook('filter:topic.get', {topic: topicData, uid: uid}, function(err, data) {
|
plugins.fireHook('filter:topic.get', {topic: topicData, uid: uid}, function(err, data) {
|
||||||
callback(err, data ? data.topic : null);
|
callback(err, data ? data.topic : null);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
function getMainPostAndReplies(topic, set, uid, start, stop, reverse, callback) {
|
function getMainPostAndReplies(topic, set, uid, start, stop, reverse, callback) {
|
||||||
async.waterfall([
|
async.waterfall([
|
||||||
function(next) {
|
function(next) {
|
||||||
posts.getPidsFromSet(set, start, stop, reverse, next);
|
posts.getPidsFromSet(set, start, stop, reverse, next);
|
||||||
},
|
},
|
||||||
function(pids, next) {
|
function(pids, next) {
|
||||||
if ((!Array.isArray(pids) || !pids.length) && !topic.mainPid) {
|
if ((!Array.isArray(pids) || !pids.length) && !topic.mainPid) {
|
||||||
return callback(null, []);
|
return callback(null, []);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (topic.mainPid) {
|
if (topic.mainPid) {
|
||||||
pids.unshift(topic.mainPid);
|
pids.unshift(topic.mainPid);
|
||||||
}
|
}
|
||||||
posts.getPostsByPids(pids, uid, next);
|
posts.getPostsByPids(pids, uid, next);
|
||||||
},
|
},
|
||||||
function(posts, next) {
|
function(posts, next) {
|
||||||
if (!posts.length) {
|
if (!posts.length) {
|
||||||
return next(null, []);
|
return next(null, []);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (topic.mainPid) {
|
if (topic.mainPid) {
|
||||||
posts[0].index = 0;
|
posts[0].index = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
var indices = Topics.calculatePostIndices(start, stop, topic.postcount, reverse);
|
var indices = Topics.calculatePostIndices(start, stop, topic.postcount, reverse);
|
||||||
for (var i=1; i<posts.length; ++i) {
|
for (var i=1; i<posts.length; ++i) {
|
||||||
if (posts[i]) {
|
if (posts[i]) {
|
||||||
posts[i].index = indices[i - 1];
|
posts[i].index = indices[i - 1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.addPostData(posts, uid, callback);
|
Topics.addPostData(posts, uid, callback);
|
||||||
}
|
}
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.getMainPost = function(tid, uid, callback) {
|
Topics.getMainPost = function(tid, uid, callback) {
|
||||||
Topics.getMainPosts([tid], uid, function(err, mainPosts) {
|
Topics.getMainPosts([tid], uid, function(err, mainPosts) {
|
||||||
callback(err, Array.isArray(mainPosts) && mainPosts.length ? mainPosts[0] : null);
|
callback(err, Array.isArray(mainPosts) && mainPosts.length ? mainPosts[0] : null);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getMainPids = function(tids, callback) {
|
Topics.getMainPids = function(tids, callback) {
|
||||||
if (!Array.isArray(tids) || !tids.length) {
|
if (!Array.isArray(tids) || !tids.length) {
|
||||||
return callback(null, []);
|
return callback(null, []);
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.getTopicsFields(tids, ['mainPid'], function(err, topicData) {
|
Topics.getTopicsFields(tids, ['mainPid'], function(err, topicData) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
|
|
||||||
var mainPids = topicData.map(function(topic) {
|
var mainPids = topicData.map(function(topic) {
|
||||||
return topic && topic.mainPid;
|
return topic && topic.mainPid;
|
||||||
});
|
});
|
||||||
callback(null, mainPids);
|
callback(null, mainPids);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getMainPosts = function(tids, uid, callback) {
|
Topics.getMainPosts = function(tids, uid, callback) {
|
||||||
Topics.getMainPids(tids, function(err, mainPids) {
|
Topics.getMainPids(tids, function(err, mainPids) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
getMainPosts(mainPids, uid, callback);
|
getMainPosts(mainPids, uid, callback);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
function getMainPosts(mainPids, uid, callback) {
|
function getMainPosts(mainPids, uid, callback) {
|
||||||
posts.getPostsByPids(mainPids, uid, function(err, postData) {
|
posts.getPostsByPids(mainPids, uid, function(err, postData) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
postData.forEach(function(post) {
|
postData.forEach(function(post) {
|
||||||
if (post) {
|
if (post) {
|
||||||
post.index = 0;
|
post.index = 0;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
Topics.addPostData(postData, uid, callback);
|
Topics.addPostData(postData, uid, callback);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.getUserBookmark = function (tid, uid, callback) {
|
Topics.getUserBookmark = function (tid, uid, callback) {
|
||||||
Topics.getTopicField(tid + ':bookmarks', uid, callback);
|
Topics.getTopicField(tid + ':bookmarks', uid, callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.setUserBookmark = function(data, callback) {
|
Topics.setUserBookmark = function(data, callback) {
|
||||||
Topics.setTopicField(data.tid + ':bookmarks', data.uid, data.postIndex, callback);
|
Topics.setTopicField(data.tid + ':bookmarks', data.uid, data.postIndex, callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
Topics.getTopicField = function(tid, field, callback) {
|
Topics.getTopicField = function(tid, field, callback) {
|
||||||
db.getObjectField('topic:' + tid, field, callback);
|
db.getObjectField('topic:' + tid, field, callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopicFields = function(tid, fields, callback) {
|
Topics.getTopicFields = function(tid, fields, callback) {
|
||||||
db.getObjectFields('topic:' + tid, fields, callback);
|
db.getObjectFields('topic:' + tid, fields, callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.getTopicsFields = function(tids, fields, callback) {
|
Topics.getTopicsFields = function(tids, fields, callback) {
|
||||||
if (!Array.isArray(tids) || !tids.length) {
|
if (!Array.isArray(tids) || !tids.length) {
|
||||||
return callback(null, []);
|
return callback(null, []);
|
||||||
}
|
}
|
||||||
var keys = tids.map(function(tid) {
|
var keys = tids.map(function(tid) {
|
||||||
return 'topic:' + tid;
|
return 'topic:' + tid;
|
||||||
});
|
});
|
||||||
db.getObjectsFields(keys, fields, callback);
|
db.getObjectsFields(keys, fields, callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.setTopicField = function(tid, field, value, callback) {
|
Topics.setTopicField = function(tid, field, value, callback) {
|
||||||
db.setObjectField('topic:' + tid, field, value, callback);
|
db.setObjectField('topic:' + tid, field, value, callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.isLocked = function(tid, callback) {
|
Topics.isLocked = function(tid, callback) {
|
||||||
Topics.getTopicField(tid, 'locked', function(err, locked) {
|
Topics.getTopicField(tid, 'locked', function(err, locked) {
|
||||||
callback(err, parseInt(locked, 10) === 1);
|
callback(err, parseInt(locked, 10) === 1);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
Topics.search = function(tid, term, callback) {
|
Topics.search = function(tid, term, callback) {
|
||||||
if (plugins.hasListeners('filter:topic.search')) {
|
if (plugins.hasListeners('filter:topic.search')) {
|
||||||
plugins.fireHook('filter:topic.search', {
|
plugins.fireHook('filter:topic.search', {
|
||||||
tid: tid,
|
tid: tid,
|
||||||
term: term
|
term: term
|
||||||
}, callback);
|
}, callback);
|
||||||
} else {
|
} else {
|
||||||
callback(new Error('no-plugins-available'), []);
|
callback(new Error('no-plugins-available'), []);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
}(exports));
|
}(exports));
|
||||||
|
@ -1,230 +1,230 @@
|
|||||||
|
|
||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
var path = require('path'),
|
var path = require('path'),
|
||||||
fs = require('fs'),
|
fs = require('fs'),
|
||||||
nconf = require('nconf'),
|
nconf = require('nconf'),
|
||||||
express = require('express'),
|
express = require('express'),
|
||||||
app = express(),
|
app = express(),
|
||||||
server,
|
server,
|
||||||
winston = require('winston'),
|
winston = require('winston'),
|
||||||
async = require('async'),
|
async = require('async'),
|
||||||
|
|
||||||
emailer = require('./emailer'),
|
emailer = require('./emailer'),
|
||||||
meta = require('./meta'),
|
meta = require('./meta'),
|
||||||
logger = require('./logger'),
|
logger = require('./logger'),
|
||||||
plugins = require('./plugins'),
|
plugins = require('./plugins'),
|
||||||
middleware = require('./middleware'),
|
middleware = require('./middleware'),
|
||||||
routes = require('./routes'),
|
routes = require('./routes'),
|
||||||
emitter = require('./emitter'),
|
emitter = require('./emitter'),
|
||||||
|
|
||||||
helpers = require('../public/src/modules/helpers');
|
helpers = require('../public/src/modules/helpers');
|
||||||
|
|
||||||
if (nconf.get('ssl')) {
|
if (nconf.get('ssl')) {
|
||||||
server = require('https').createServer({
|
server = require('https').createServer({
|
||||||
key: fs.readFileSync(nconf.get('ssl').key),
|
key: fs.readFileSync(nconf.get('ssl').key),
|
||||||
cert: fs.readFileSync(nconf.get('ssl').cert)
|
cert: fs.readFileSync(nconf.get('ssl').cert)
|
||||||
}, app);
|
}, app);
|
||||||
} else {
|
} else {
|
||||||
server = require('http').createServer(app);
|
server = require('http').createServer(app);
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports.server = server;
|
module.exports.server = server;
|
||||||
|
|
||||||
server.on('error', function(err) {
|
server.on('error', function(err) {
|
||||||
winston.error(err);
|
winston.error(err);
|
||||||
if (err.code === 'EADDRINUSE') {
|
if (err.code === 'EADDRINUSE') {
|
||||||
winston.error('NodeBB address in use, exiting...');
|
winston.error('NodeBB address in use, exiting...');
|
||||||
process.exit(0);
|
process.exit(0);
|
||||||
} else {
|
} else {
|
||||||
throw err;
|
throw err;
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
module.exports.listen = function() {
|
module.exports.listen = function() {
|
||||||
emailer.registerApp(app);
|
emailer.registerApp(app);
|
||||||
|
|
||||||
middleware = middleware(app);
|
middleware = middleware(app);
|
||||||
|
|
||||||
helpers.register();
|
helpers.register();
|
||||||
|
|
||||||
logger.init(app);
|
logger.init(app);
|
||||||
|
|
||||||
emitter.all(['templates:compiled', 'meta:js.compiled', 'meta:css.compiled'], function() {
|
emitter.all(['templates:compiled', 'meta:js.compiled', 'meta:css.compiled'], function() {
|
||||||
winston.info('NodeBB Ready');
|
winston.info('NodeBB Ready');
|
||||||
emitter.emit('nodebb:ready');
|
emitter.emit('nodebb:ready');
|
||||||
listen();
|
listen();
|
||||||
});
|
});
|
||||||
|
|
||||||
initializeNodeBB(function(err) {
|
initializeNodeBB(function(err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
winston.error(err);
|
winston.error(err);
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
if (process.send) {
|
if (process.send) {
|
||||||
process.send({
|
process.send({
|
||||||
action: 'ready'
|
action: 'ready'
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
function initializeNodeBB(callback) {
|
function initializeNodeBB(callback) {
|
||||||
var skipJS, skipLess, fromFile = nconf.get('from-file') || '';
|
var skipJS, skipLess, fromFile = nconf.get('from-file') || '';
|
||||||
|
|
||||||
if (fromFile.match('js')) {
|
if (fromFile.match('js')) {
|
||||||
winston.info('[minifier] Minifying client-side JS skipped');
|
winston.info('[minifier] Minifying client-side JS skipped');
|
||||||
skipJS = true;
|
skipJS = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (fromFile.match('less')) {
|
if (fromFile.match('less')) {
|
||||||
winston.info('[minifier] Compiling LESS files skipped');
|
winston.info('[minifier] Compiling LESS files skipped');
|
||||||
skipLess = true;
|
skipLess = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
async.waterfall([
|
async.waterfall([
|
||||||
async.apply(cacheStaticFiles),
|
async.apply(cacheStaticFiles),
|
||||||
async.apply(meta.themes.setupPaths),
|
async.apply(meta.themes.setupPaths),
|
||||||
function(next) {
|
function(next) {
|
||||||
plugins.init(app, middleware, next);
|
plugins.init(app, middleware, next);
|
||||||
},
|
},
|
||||||
function(next) {
|
function(next) {
|
||||||
async.parallel([
|
async.parallel([
|
||||||
async.apply(meta.templates.compile),
|
async.apply(meta.templates.compile),
|
||||||
async.apply(!skipJS ? meta.js.minify : meta.js.getFromFile, app.enabled('minification')),
|
async.apply(!skipJS ? meta.js.minify : meta.js.getFromFile, app.enabled('minification')),
|
||||||
async.apply(!skipLess ? meta.css.minify : meta.css.getFromFile),
|
async.apply(!skipLess ? meta.css.minify : meta.css.getFromFile),
|
||||||
async.apply(meta.sounds.init)
|
async.apply(meta.sounds.init)
|
||||||
], next);
|
], next);
|
||||||
},
|
},
|
||||||
function(results, next) {
|
function(results, next) {
|
||||||
plugins.fireHook('static:app.preload', {
|
plugins.fireHook('static:app.preload', {
|
||||||
app: app,
|
app: app,
|
||||||
middleware: middleware
|
middleware: middleware
|
||||||
}, next);
|
}, next);
|
||||||
},
|
},
|
||||||
function(next) {
|
function(next) {
|
||||||
routes(app, middleware);
|
routes(app, middleware);
|
||||||
next();
|
next();
|
||||||
}
|
}
|
||||||
], callback);
|
], callback);
|
||||||
}
|
}
|
||||||
|
|
||||||
function cacheStaticFiles(callback) {
|
function cacheStaticFiles(callback) {
|
||||||
if (global.env === 'development') {
|
if (global.env === 'development') {
|
||||||
return callback();
|
return callback();
|
||||||
}
|
}
|
||||||
|
|
||||||
app.enable('cache');
|
app.enable('cache');
|
||||||
app.enable('minification');
|
app.enable('minification');
|
||||||
|
|
||||||
// Configure cache-buster timestamp
|
// Configure cache-buster timestamp
|
||||||
require('child_process').exec('git describe --tags', {
|
require('child_process').exec('git describe --tags', {
|
||||||
cwd: path.join(__dirname, '../')
|
cwd: path.join(__dirname, '../')
|
||||||
}, function(err, stdOut) {
|
}, function(err, stdOut) {
|
||||||
if (!err) {
|
if (!err) {
|
||||||
meta.config['cache-buster'] = stdOut.trim();
|
meta.config['cache-buster'] = stdOut.trim();
|
||||||
callback();
|
callback();
|
||||||
} else {
|
} else {
|
||||||
fs.stat(path.join(__dirname, '../package.json'), function(err, stats) {
|
fs.stat(path.join(__dirname, '../package.json'), function(err, stats) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return callback(err);
|
return callback(err);
|
||||||
}
|
}
|
||||||
meta.config['cache-buster'] = new Date(stats.mtime).getTime();
|
meta.config['cache-buster'] = new Date(stats.mtime).getTime();
|
||||||
callback();
|
callback();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function listen(callback) {
|
function listen(callback) {
|
||||||
var port = nconf.get('port');
|
var port = nconf.get('port');
|
||||||
|
|
||||||
if (Array.isArray(port)) {
|
if (Array.isArray(port)) {
|
||||||
if (!port.length) {
|
if (!port.length) {
|
||||||
winston.error('[startup] empty ports array in config.json');
|
winston.error('[startup] empty ports array in config.json');
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
winston.warn('[startup] If you want to start nodebb on multiple ports please use loader.js');
|
winston.warn('[startup] If you want to start nodebb on multiple ports please use loader.js');
|
||||||
winston.warn('[startup] Defaulting to first port in array, ' + port[0]);
|
winston.warn('[startup] Defaulting to first port in array, ' + port[0]);
|
||||||
port = port[0];
|
port = port[0];
|
||||||
if (!port) {
|
if (!port) {
|
||||||
winston.error('[startup] Invalid port, exiting');
|
winston.error('[startup] Invalid port, exiting');
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (port !== 80 && port !== 443 && nconf.get('use_port') === false) {
|
if (port !== 80 && port !== 443 && nconf.get('use_port') === false) {
|
||||||
winston.info('Enabling \'trust proxy\'');
|
winston.info('Enabling \'trust proxy\'');
|
||||||
app.enable('trust proxy');
|
app.enable('trust proxy');
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((port === 80 || port === 443) && process.env.NODE_ENV !== 'development') {
|
if ((port === 80 || port === 443) && process.env.NODE_ENV !== 'development') {
|
||||||
winston.info('Using ports 80 and 443 is not recommend; use a proxy instead. See README.md');
|
winston.info('Using ports 80 and 443 is not recommend; use a proxy instead. See README.md');
|
||||||
}
|
}
|
||||||
|
|
||||||
var isSocket = isNaN(port),
|
var isSocket = isNaN(port),
|
||||||
args = isSocket ? [port] : [port, nconf.get('bind_address')],
|
args = isSocket ? [port] : [port, nconf.get('bind_address')],
|
||||||
bind_address = ((nconf.get('bind_address') === "0.0.0.0" || !nconf.get('bind_address')) ? '0.0.0.0' : nconf.get('bind_address')) + ':' + port,
|
bind_address = ((nconf.get('bind_address') === "0.0.0.0" || !nconf.get('bind_address')) ? '0.0.0.0' : nconf.get('bind_address')) + ':' + port,
|
||||||
oldUmask;
|
oldUmask;
|
||||||
|
|
||||||
args.push(function(err) {
|
args.push(function(err) {
|
||||||
if (err) {
|
if (err) {
|
||||||
winston.info('[startup] NodeBB was unable to listen on: ' + bind_address);
|
winston.info('[startup] NodeBB was unable to listen on: ' + bind_address);
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
|
|
||||||
winston.info('NodeBB is now listening on: ' + (isSocket ? port : bind_address));
|
winston.info('NodeBB is now listening on: ' + (isSocket ? port : bind_address));
|
||||||
if (oldUmask) {
|
if (oldUmask) {
|
||||||
process.umask(oldUmask);
|
process.umask(oldUmask);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Alter umask if necessary
|
// Alter umask if necessary
|
||||||
if (isSocket) {
|
if (isSocket) {
|
||||||
oldUmask = process.umask('0000');
|
oldUmask = process.umask('0000');
|
||||||
module.exports.testSocket(port, function(err) {
|
module.exports.testSocket(port, function(err) {
|
||||||
if (!err) {
|
if (!err) {
|
||||||
server.listen.apply(server, args);
|
server.listen.apply(server, args);
|
||||||
} else {
|
} else {
|
||||||
winston.error('[startup] NodeBB was unable to secure domain socket access (' + port + ')');
|
winston.error('[startup] NodeBB was unable to secure domain socket access (' + port + ')');
|
||||||
winston.error('[startup] ' + err.message);
|
winston.error('[startup] ' + err.message);
|
||||||
process.exit();
|
process.exit();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
server.listen.apply(server, args);
|
server.listen.apply(server, args);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
module.exports.testSocket = function(socketPath, callback) {
|
module.exports.testSocket = function(socketPath, callback) {
|
||||||
if (typeof socketPath !== 'string') {
|
if (typeof socketPath !== 'string') {
|
||||||
return callback(new Error('invalid socket path : ' + socketPath));
|
return callback(new Error('invalid socket path : ' + socketPath));
|
||||||
}
|
}
|
||||||
var net = require('net');
|
var net = require('net');
|
||||||
async.series([
|
async.series([
|
||||||
function(next) {
|
function(next) {
|
||||||
fs.exists(socketPath, function(exists) {
|
fs.exists(socketPath, function(exists) {
|
||||||
if (exists) {
|
if (exists) {
|
||||||
next();
|
next();
|
||||||
} else {
|
} else {
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
function(next) {
|
function(next) {
|
||||||
var testSocket = new net.Socket();
|
var testSocket = new net.Socket();
|
||||||
testSocket.on('error', function(err) {
|
testSocket.on('error', function(err) {
|
||||||
next(err.code !== 'ECONNREFUSED' ? err : null);
|
next(err.code !== 'ECONNREFUSED' ? err : null);
|
||||||
});
|
});
|
||||||
testSocket.connect({ path: socketPath }, function() {
|
testSocket.connect({ path: socketPath }, function() {
|
||||||
// Something's listening here, abort
|
// Something's listening here, abort
|
||||||
callback(new Error('port-in-use'));
|
callback(new Error('port-in-use'));
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
async.apply(fs.unlink, socketPath), // The socket was stale, kick it out of the way
|
async.apply(fs.unlink, socketPath), // The socket was stale, kick it out of the way
|
||||||
], callback);
|
], callback);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue