Files
NodeBB/src/controllers/api.js

228 lines
8.4 KiB
JavaScript
Raw Normal View History

2017-02-18 01:56:23 -07:00
'use strict';
2016-03-08 11:24:32 +02:00
var async = require('async');
var validator = require('validator');
var nconf = require('nconf');
var meta = require('../meta');
var user = require('../user');
var posts = require('../posts');
var topics = require('../topics');
var categories = require('../categories');
var privileges = require('../privileges');
var plugins = require('../plugins');
var translator = require('../translator');
var languages = require('../languages');
2017-03-02 16:11:11 +03:00
var apiController = module.exports;
apiController.loadConfig = function (req, callback) {
var config = {};
2014-11-29 23:38:36 -05:00
config.relative_path = nconf.get('relative_path');
config.upload_url = nconf.get('upload_url');
config.siteTitle = validator.escape(String(meta.config.title || meta.config.browserTitle || 'NodeBB'));
config.browserTitle = validator.escape(String(meta.config.browserTitle || meta.config.title || 'NodeBB'));
2015-09-23 01:59:13 -04:00
config.titleLayout = (meta.config.titleLayout || '{pageTitle} | {browserTitle}').replace(/{/g, '{').replace(/}/g, '}');
config.showSiteTitle = meta.config.showSiteTitle === 1;
config.minimumTitleLength = meta.config.minimumTitleLength;
config.maximumTitleLength = meta.config.maximumTitleLength;
config.minimumPostLength = meta.config.minimumPostLength;
2015-02-03 19:55:14 -05:00
config.maximumPostLength = meta.config.maximumPostLength;
config.minimumTagsPerTopic = meta.config.minimumTagsPerTopic || 0;
config.maximumTagsPerTopic = meta.config.maximumTagsPerTopic || 5;
2016-02-10 12:53:39 +02:00
config.minimumTagLength = meta.config.minimumTagLength || 3;
config.maximumTagLength = meta.config.maximumTagLength || 15;
config.useOutgoingLinksPage = meta.config.useOutgoingLinksPage === 1;
config.allowGuestHandles = meta.config.allowGuestHandles === 1;
config.allowFileUploads = meta.config.allowFileUploads === 1;
config.allowTopicsThumbnail = meta.config.allowTopicsThumbnail === 1;
config.usePagination = meta.config.usePagination === 1;
config.disableChat = meta.config.disableChat === 1;
config.disableChatMessageEditing = meta.config.disableChatMessageEditing === 1;
config.maximumChatMessageLength = meta.config.maximumChatMessageLength || 1000;
2016-02-10 12:53:39 +02:00
config.socketioTransports = nconf.get('socket.io:transports') || ['polling', 'websocket'];
config.socketioOrigins = nconf.get('socket.io:origins');
2016-02-10 12:53:39 +02:00
config.websocketAddress = nconf.get('socket.io:address') || '';
2014-04-03 17:36:27 -04:00
config.maxReconnectionAttempts = meta.config.maxReconnectionAttempts || 5;
config.reconnectionDelay = meta.config.reconnectionDelay || 1500;
config.topicsPerPage = meta.config.topicsPerPage || 20;
config.postsPerPage = meta.config.postsPerPage || 20;
config.maximumFileSize = meta.config.maximumFileSize;
config['theme:id'] = meta.config['theme:id'];
config['theme:src'] = meta.config['theme:src'];
config.defaultLang = meta.config.defaultLang || 'en-GB';
config.userLang = req.query.lang ? validator.escape(String(req.query.lang)) : config.defaultLang;
2014-07-24 21:11:46 -04:00
config.loggedIn = !!req.user;
2018-05-08 13:33:25 -04:00
config.uid = req.uid;
config['cache-buster'] = meta.config['cache-buster'] || '';
config.requireEmailConfirmation = meta.config.requireEmailConfirmation === 1;
config.topicPostSort = meta.config.topicPostSort || 'oldest_to_newest';
2015-01-08 13:47:15 -05:00
config.categoryTopicSort = meta.config.categoryTopicSort || 'newest_to_oldest';
config.csrf_token = req.uid >= 0 && req.csrfToken && req.csrfToken();
config.searchEnabled = plugins.hasListeners('filter:search.query');
config.bootswatchSkin = meta.config.bootswatchSkin || '';
config.enablePostHistory = (meta.config.enablePostHistory || 1) === 1;
config.notificationAlertTimeout = meta.config.notificationAlertTimeout || 5000;
config.timeagoCodes = languages.timeagoCodes;
2016-11-30 16:28:25 +03:00
2017-03-10 14:03:07 -05:00
if (config.useOutgoingLinksPage) {
config.outgoingLinksWhitelist = meta.config['outgoingLinks:whitelist'];
}
2016-11-30 16:28:25 +03:00
var timeagoCutoff = meta.config.timeagoCutoff === undefined ? 30 : meta.config.timeagoCutoff;
config.timeagoCutoff = timeagoCutoff !== '' ? Math.max(0, parseInt(timeagoCutoff, 10)) : timeagoCutoff;
2016-11-01 12:59:51 -04:00
config.cookies = {
enabled: meta.config.cookieConsentEnabled === 1,
2017-03-24 14:46:51 -04:00
message: translator.escape(validator.escape(meta.config.cookieConsentMessage || '[[global:cookies.message]]')).replace(/\\/g, '\\\\'),
dismiss: translator.escape(validator.escape(meta.config.cookieConsentDismiss || '[[global:cookies.accept]]')).replace(/\\/g, '\\\\'),
link: translator.escape(validator.escape(meta.config.cookieConsentLink || '[[global:cookies.learn_more]]')).replace(/\\/g, '\\\\'),
link_url: translator.escape(validator.escape(meta.config.cookieConsentLinkUrl || 'https://www.cookiesandyou.com')).replace(/\\/g, '\\\\'),
2016-11-01 12:59:51 -04:00
};
2016-03-08 11:24:32 +02:00
async.waterfall([
function (next) {
2018-01-31 15:20:17 -05:00
if (!req.loggedIn) {
2016-03-08 11:24:32 +02:00
return next(null, config);
}
2016-08-29 12:43:32 +03:00
user.getSettings(req.uid, next);
2016-03-08 11:24:32 +02:00
},
2016-08-29 12:43:32 +03:00
function (settings, next) {
// Handle old skin configs
const oldSkins = ['noskin', 'default'];
settings.bootswatchSkin = oldSkins.includes(settings.bootswatchSkin) ? '' : settings.bootswatchSkin;
2016-08-29 12:43:32 +03:00
config.usePagination = settings.usePagination;
config.topicsPerPage = settings.topicsPerPage;
config.postsPerPage = settings.postsPerPage;
config.userLang = (req.query.lang ? validator.escape(String(req.query.lang)) : null) || settings.userLang || config.defaultLang;
2018-02-12 17:12:18 -05:00
config.acpLang = (req.query.lang ? validator.escape(String(req.query.lang)) : null) || settings.acpLang;
2016-08-29 12:43:32 +03:00
config.openOutgoingLinksInNewTab = settings.openOutgoingLinksInNewTab;
config.topicPostSort = settings.topicPostSort || config.topicPostSort;
config.categoryTopicSort = settings.categoryTopicSort || config.categoryTopicSort;
config.topicSearchEnabled = settings.topicSearchEnabled || false;
config.bootswatchSkin = (meta.config.disableCustomUserSkins !== 1 && settings.bootswatchSkin && settings.bootswatchSkin !== '') ? settings.bootswatchSkin : '';
2016-03-08 11:24:32 +02:00
plugins.fireHook('filter:config.get', config, next);
2017-02-17 19:31:21 -07:00
},
function (config, next) {
req.res.locals.config = config;
process.nextTick(next, null, config);
},
], callback);
};
apiController.getConfig = function (req, res, next) {
async.waterfall([
function (next) {
apiController.loadConfig(req, next);
2015-09-16 00:46:11 -07:00
},
function (config, next) {
if (res.locals.isAPI) {
res.json(config);
} else {
next(null, config);
2017-02-18 02:38:03 -07:00
}
},
], next);
2014-07-09 15:41:03 -04:00
};
apiController.getPostData = function (pid, uid, callback) {
2016-07-01 13:01:09 +03:00
async.parallel({
privileges: function (next) {
2016-07-01 13:01:09 +03:00
privileges.posts.get([pid], uid, next);
},
post: function (next) {
2016-07-01 13:01:09 +03:00
posts.getPostData(pid, next);
2017-02-17 19:31:21 -07:00
},
voted: async.apply(posts.hasVoted, pid, uid),
}, function (err, results) {
2016-07-01 13:01:09 +03:00
if (err || !results.post) {
return callback(err);
2016-03-08 11:24:32 +02:00
}
2016-07-01 13:01:09 +03:00
var post = results.post;
Object.assign(post, results.voted);
2016-07-01 13:01:09 +03:00
var privileges = results.privileges[0];
2016-07-01 13:01:09 +03:00
if (!privileges.read || !privileges['topics:read']) {
return callback();
}
post.ip = privileges.isAdminOrMod ? post.ip : undefined;
var selfPost = uid && uid === parseInt(post.uid, 10);
if (post.deleted && !(privileges.isAdminOrMod || selfPost)) {
post.content = '[[topic:post_is_deleted]]';
}
callback(null, post);
2016-03-08 11:24:32 +02:00
});
};
apiController.getTopicData = function (tid, uid, callback) {
2016-07-01 13:01:09 +03:00
async.parallel({
privileges: function (next) {
2016-07-01 13:01:09 +03:00
privileges.topics.get(tid, uid, next);
2015-09-21 12:59:41 -04:00
},
topic: function (next) {
2016-07-01 13:01:09 +03:00
topics.getTopicData(tid, next);
2017-02-17 19:31:21 -07:00
},
}, function (err, results) {
2016-07-01 13:01:09 +03:00
if (err || !results.topic) {
return callback(err);
2015-09-21 12:59:41 -04:00
}
2018-10-21 19:33:46 -04:00
if (!results.privileges.read || !results.privileges['topics:read'] || (results.topic.deleted && !results.privileges.view_deleted)) {
2016-07-01 13:01:09 +03:00
return callback();
}
callback(null, results.topic);
});
};
2015-09-21 12:59:41 -04:00
apiController.getCategoryData = function (cid, uid, callback) {
2016-07-01 13:01:09 +03:00
async.parallel({
privileges: function (next) {
2016-07-01 13:01:09 +03:00
privileges.categories.get(cid, uid, next);
2016-03-08 11:24:32 +02:00
},
category: function (next) {
2016-07-01 13:01:09 +03:00
categories.getCategoryData(cid, next);
2017-02-17 19:31:21 -07:00
},
}, function (err, results) {
2016-07-01 13:01:09 +03:00
if (err || !results.category) {
return callback(err);
}
if (!results.privileges.read) {
return callback();
2015-09-21 12:59:41 -04:00
}
2016-07-01 13:01:09 +03:00
callback(null, results.category);
});
};
apiController.getObject = function (req, res, next) {
2016-07-01 13:01:09 +03:00
var methods = {
post: apiController.getPostData,
topic: apiController.getTopicData,
2017-02-17 19:31:21 -07:00
category: apiController.getCategoryData,
2016-07-01 13:01:09 +03:00
};
var method = methods[req.params.type];
if (!method) {
return next();
}
method(req.params.id, req.uid, function (err, result) {
2016-07-01 13:01:09 +03:00
if (err || !result) {
return next(err);
}
res.json(result);
});
2015-09-21 12:59:41 -04:00
};
apiController.getModerators = function (req, res, next) {
categories.getModerators(req.params.cid, function (err, moderators) {
2015-09-25 18:29:02 -04:00
if (err) {
return next(err);
}
2017-02-18 12:30:49 -07:00
res.json({ moderators: moderators });
});
};