Files
NodeBB/src/posts/parse.js

175 lines
4.8 KiB
JavaScript
Raw Normal View History

'use strict';
2021-02-04 00:06:15 -07:00
const nconf = require('nconf');
const url = require('url');
const winston = require('winston');
const sanitize = require('sanitize-html');
const _ = require('lodash');
2016-05-16 08:22:23 -04:00
2021-02-04 00:06:15 -07:00
const meta = require('../meta');
const plugins = require('../plugins');
const translator = require('../translator');
const utils = require('../utils');
let sanitizeConfig = {
allowedTags: sanitize.defaults.allowedTags.concat([
// Some safe-to-use tags to add
'sup', 'ins', 'del', 'img', 'button',
'video', 'audio', 'iframe', 'embed',
// 'sup' still necessary until https://github.com/apostrophecms/sanitize-html/pull/422 merged
]),
allowedAttributes: {
...sanitize.defaults.allowedAttributes,
a: ['href', 'name', 'hreflang', 'media', 'rel', 'target', 'type'],
img: ['alt', 'height', 'ismap', 'src', 'usemap', 'width', 'srcset'],
iframe: ['height', 'name', 'src', 'width'],
video: ['autoplay', 'controls', 'height', 'loop', 'muted', 'poster', 'preload', 'src', 'width'],
audio: ['autoplay', 'controls', 'loop', 'muted', 'preload', 'src'],
embed: ['height', 'src', 'type', 'width'],
},
globalAttributes: ['accesskey', 'class', 'contenteditable', 'dir',
'draggable', 'dropzone', 'hidden', 'id', 'lang', 'spellcheck', 'style',
'tabindex', 'title', 'translate', 'aria-expanded', 'data-*',
],
allowedClasses: {
...sanitize.defaults.allowedClasses,
},
};
module.exports = function (Posts) {
2017-05-28 01:10:16 -04:00
Posts.urlRegex = {
regex: /href="([^"]+)"/g,
length: 6,
};
2017-05-28 01:26:56 -04:00
2017-05-28 01:10:16 -04:00
Posts.imgRegex = {
2017-05-28 01:26:56 -04:00
regex: /src="([^"]+)"/g,
2017-05-28 01:10:16 -04:00
length: 5,
};
2019-07-17 19:05:55 -04:00
Posts.parsePost = async function (postData) {
2018-10-21 19:33:46 -04:00
if (!postData) {
2019-07-17 19:05:55 -04:00
return postData;
2018-10-21 19:33:46 -04:00
}
2017-03-09 19:52:48 +03:00
postData.content = String(postData.content || '');
2019-07-17 19:05:55 -04:00
const cache = require('./cache');
const pid = String(postData.pid);
const cachedContent = cache.get(pid);
if (postData.pid && cachedContent !== undefined) {
postData.content = cachedContent;
return postData;
}
const data = await plugins.hooks.fire('filter:parse.post', { postData: postData });
2019-07-17 19:05:55 -04:00
data.postData.content = translator.escape(data.postData.content);
if (data.postData.pid) {
2019-07-17 19:05:55 -04:00
cache.set(pid, data.postData.content);
}
return data.postData;
};
2019-07-17 19:05:55 -04:00
Posts.parseSignature = async function (userData, uid) {
2016-11-21 11:07:20 -05:00
userData.signature = sanitizeSignature(userData.signature || '');
return await plugins.hooks.fire('filter:parse.signature', { userData: userData, uid: uid });
};
2016-05-16 08:22:23 -04:00
2017-05-28 01:10:16 -04:00
Posts.relativeToAbsolute = function (content, regex) {
// Turns relative links in content to absolute urls
if (!content) {
return content;
}
2021-02-04 00:06:15 -07:00
let parsed;
let current = regex.regex.exec(content);
let absolute;
while (current !== null) {
2016-05-16 08:22:23 -04:00
if (current[1]) {
2016-06-03 11:20:53 +03:00
try {
parsed = url.parse(current[1]);
if (!parsed.protocol) {
if (current[1].startsWith('/')) {
// Internal link
2017-05-28 01:10:16 -04:00
absolute = nconf.get('base_url') + current[1];
2016-06-03 11:20:53 +03:00
} else {
// External link
2021-02-03 23:59:08 -07:00
absolute = `//${current[1]}`;
2016-06-03 11:20:53 +03:00
}
2021-02-04 02:07:29 -07:00
content = content.slice(0, current.index + regex.length) +
absolute +
content.slice(current.index + regex.length + current[1].length);
2016-05-16 08:22:23 -04:00
}
} catch (err) {
winston.verbose(err.messsage);
}
2016-05-16 08:22:23 -04:00
}
2017-05-28 01:10:16 -04:00
current = regex.regex.exec(content);
2016-05-16 08:22:23 -04:00
}
return content;
};
2016-11-21 11:07:20 -05:00
Posts.sanitize = function (content) {
return sanitize(content, {
allowedTags: sanitizeConfig.allowedTags,
allowedAttributes: sanitizeConfig.allowedAttributes,
allowedClasses: sanitizeConfig.allowedClasses,
});
};
Posts.configureSanitize = async () => {
// Each allowed tags should have some common global attributes...
sanitizeConfig.allowedTags.forEach((tag) => {
2021-02-04 02:07:29 -07:00
sanitizeConfig.allowedAttributes[tag] = _.union(
sanitizeConfig.allowedAttributes[tag],
sanitizeConfig.globalAttributes
);
});
2019-09-04 11:44:04 -04:00
// Some plugins might need to adjust or whitelist their own tags...
sanitizeConfig = await plugins.hooks.fire('filter:sanitize.config', sanitizeConfig);
};
Posts.registerHooks = () => {
plugins.hooks.register('core', {
hook: 'filter:parse.post',
method: async (data) => {
data.postData.content = Posts.sanitize(data.postData.content);
return data;
},
});
plugins.hooks.register('core', {
hook: 'filter:parse.raw',
method: async content => Posts.sanitize(content),
});
plugins.hooks.register('core', {
hook: 'filter:parse.aboutme',
method: async content => Posts.sanitize(content),
});
plugins.hooks.register('core', {
hook: 'filter:parse.signature',
method: async (data) => {
data.userData.signature = Posts.sanitize(data.userData.signature);
return data;
},
});
};
2016-11-21 11:07:20 -05:00
function sanitizeSignature(signature) {
signature = translator.escape(signature);
2021-02-04 00:06:15 -07:00
const tagsToStrip = [];
2016-11-21 11:07:20 -05:00
if (meta.config['signatures:disableLinks']) {
2016-11-21 11:07:20 -05:00
tagsToStrip.push('a');
}
if (meta.config['signatures:disableImages']) {
2016-11-21 11:07:20 -05:00
tagsToStrip.push('img');
}
2017-10-13 21:02:41 -06:00
return utils.stripHTMLTags(signature, tagsToStrip);
2016-11-21 11:07:20 -05:00
}
2015-11-26 23:34:55 -05:00
};