mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-31 02:55:58 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			145 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			145 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 'use strict';
 | |
| 
 | |
| var nconf = require('nconf');
 | |
| var url = require('url');
 | |
| var winston = require('winston');
 | |
| const sanitize = require('sanitize-html');
 | |
| const _ = require('lodash');
 | |
| 
 | |
| var meta = require('../meta');
 | |
| var plugins = require('../plugins');
 | |
| var translator = require('../translator');
 | |
| var utils = require('../utils');
 | |
| 
 | |
| let sanitizeConfig = {
 | |
| 	allowedTags: sanitize.defaults.allowedTags.concat([
 | |
| 		// Some safe-to-use tags to add
 | |
| 		'span', 'a', 'pre', 'small',
 | |
| 		'sup', 'sub', 'u', 'del',
 | |
| 		'video', 'audio', 'iframe', 'embed',
 | |
| 		'img', 'tfoot', 'h1', 'h2',
 | |
| 		's', 'button', 'i',
 | |
| 	]),
 | |
| 	allowedAttributes: {
 | |
| 		...sanitize.defaults.allowedAttributes,
 | |
| 		a: ['href', '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) {
 | |
| 	Posts.urlRegex = {
 | |
| 		regex: /href="([^"]+)"/g,
 | |
| 		length: 6,
 | |
| 	};
 | |
| 
 | |
| 	Posts.imgRegex = {
 | |
| 		regex: /src="([^"]+)"/g,
 | |
| 		length: 5,
 | |
| 	};
 | |
| 
 | |
| 	Posts.parsePost = async function (postData) {
 | |
| 		if (!postData) {
 | |
| 			return postData;
 | |
| 		}
 | |
| 		postData.content = String(postData.content || '');
 | |
| 		const cache = require('./cache');
 | |
| 		const pid = String(postData.pid);
 | |
| 		const cachedContent = cache.get(pid);
 | |
| 		if (postData.pid && cachedContent !== undefined) {
 | |
| 			postData.content = cachedContent;
 | |
| 			cache.hits += 1;
 | |
| 			return postData;
 | |
| 		}
 | |
| 		cache.misses += 1;
 | |
| 		const data = await plugins.fireHook('filter:parse.post', { postData: postData });
 | |
| 		data.postData.content = translator.escape(data.postData.content);
 | |
| 		if (global.env === 'production' && data.postData.pid) {
 | |
| 			cache.set(pid, data.postData.content);
 | |
| 		}
 | |
| 		return data.postData;
 | |
| 	};
 | |
| 
 | |
| 	Posts.parseSignature = async function (userData, uid) {
 | |
| 		userData.signature = sanitizeSignature(userData.signature || '');
 | |
| 		return await plugins.fireHook('filter:parse.signature', { userData: userData, uid: uid });
 | |
| 	};
 | |
| 
 | |
| 	Posts.relativeToAbsolute = function (content, regex) {
 | |
| 		// Turns relative links in content to absolute urls
 | |
| 		if (!content) {
 | |
| 			return content;
 | |
| 		}
 | |
| 		var parsed;
 | |
| 		var current = regex.regex.exec(content);
 | |
| 		var absolute;
 | |
| 		while (current !== null) {
 | |
| 			if (current[1]) {
 | |
| 				try {
 | |
| 					parsed = url.parse(current[1]);
 | |
| 					if (!parsed.protocol) {
 | |
| 						if (current[1].startsWith('/')) {
 | |
| 							// Internal link
 | |
| 							absolute = nconf.get('base_url') + current[1];
 | |
| 						} else {
 | |
| 							// External link
 | |
| 							absolute = '//' + current[1];
 | |
| 						}
 | |
| 
 | |
| 						content = content.slice(0, current.index + regex.length) + absolute + content.slice(current.index + regex.length + current[1].length);
 | |
| 					}
 | |
| 				} catch (err) {
 | |
| 					winston.verbose(err.messsage);
 | |
| 				}
 | |
| 			}
 | |
| 			current = regex.regex.exec(content);
 | |
| 		}
 | |
| 
 | |
| 		return content;
 | |
| 	};
 | |
| 
 | |
| 	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) => {
 | |
| 			sanitizeConfig.allowedAttributes[tag] = _.union(sanitizeConfig.allowedAttributes[tag], sanitizeConfig.globalAttributes);
 | |
| 		});
 | |
| 
 | |
| 		// Some plugins might need to adjust or whitelist their own tags...
 | |
| 		sanitizeConfig = await plugins.fireHook('filter:sanitize.config', sanitizeConfig);
 | |
| 	};
 | |
| 
 | |
| 	function sanitizeSignature(signature) {
 | |
| 		signature = translator.escape(signature);
 | |
| 		var tagsToStrip = [];
 | |
| 
 | |
| 		if (meta.config['signatures:disableLinks']) {
 | |
| 			tagsToStrip.push('a');
 | |
| 		}
 | |
| 
 | |
| 		if (meta.config['signatures:disableImages']) {
 | |
| 			tagsToStrip.push('img');
 | |
| 		}
 | |
| 
 | |
| 		return utils.stripHTMLTags(signature, tagsToStrip);
 | |
| 	}
 | |
| };
 |