mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-26 16:46:12 +01:00 
			
		
		
		
	feat: widgets/index.js
enable widget-essentials in tests fix widget test
This commit is contained in:
		| @@ -1,104 +1,84 @@ | ||||
| 'use strict'; | ||||
|  | ||||
| var async = require('async'); | ||||
| var winston = require('winston'); | ||||
| var _ = require('lodash'); | ||||
| var Benchpress = require('benchpressjs'); | ||||
| const async = require('async'); | ||||
| const winston = require('winston'); | ||||
| const _ = require('lodash'); | ||||
| const Benchpress = require('benchpressjs'); | ||||
| const util = require('util'); | ||||
|  | ||||
| var plugins = require('../plugins'); | ||||
| var groups = require('../groups'); | ||||
| var translator = require('../translator'); | ||||
| var db = require('../database'); | ||||
| var apiController = require('../controllers/api'); | ||||
| var meta = require('../meta'); | ||||
| const plugins = require('../plugins'); | ||||
| const groups = require('../groups'); | ||||
| const translator = require('../translator'); | ||||
| const db = require('../database'); | ||||
| const apiController = require('../controllers/api'); | ||||
| const loadConfigAsync = util.promisify(apiController.loadConfig); | ||||
| const meta = require('../meta'); | ||||
|  | ||||
| var widgets = module.exports; | ||||
| const widgets = module.exports; | ||||
|  | ||||
| widgets.render = function (uid, options, callback) { | ||||
| widgets.render = async function (uid, options) { | ||||
| 	if (!options.template) { | ||||
| 		return callback(new Error('[[error:invalid-data]]')); | ||||
| 		throw new Error('[[error:invalid-data]]'); | ||||
| 	} | ||||
|  | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			widgets.getWidgetDataForTemplates(['global', options.template], next); | ||||
| 		}, | ||||
| 		function (data, next) { | ||||
| 			var widgetsByLocation = {}; | ||||
|  | ||||
| 	const data = await widgets.getWidgetDataForTemplates(['global', options.template]); | ||||
| 	delete data.global.drafts; | ||||
|  | ||||
| 			var locations = _.uniq(Object.keys(data.global).concat(Object.keys(data[options.template]))); | ||||
| 	const locations = _.uniq(Object.keys(data.global).concat(Object.keys(data[options.template]))); | ||||
|  | ||||
| 			var returnData = {}; | ||||
| 	const widgetData = await Promise.all(locations.map(location => renderLocation(location, data, uid, options))); | ||||
|  | ||||
| 			async.each(locations, function (location, done) { | ||||
| 				widgetsByLocation[location] = (data[options.template][location] || []).concat(data.global[location] || []); | ||||
|  | ||||
| 				if (!widgetsByLocation[location].length) { | ||||
| 					return done(null, { location: location, widgets: [] }); | ||||
| 	const returnData = {}; | ||||
| 	locations.forEach(function (location, i) { | ||||
| 		if (Array.isArray(widgetData[i]) && widgetData[i].length) { | ||||
| 			returnData[location] = widgetData[i].filter(Boolean); | ||||
| 		} | ||||
|  | ||||
| 				async.map(widgetsByLocation[location], function (widget, next) { | ||||
| 					renderWidget(widget, uid, options, next); | ||||
| 				}, function (err, renderedWidgets) { | ||||
| 					if (err) { | ||||
| 						return done(err); | ||||
| 					} | ||||
| 					renderedWidgets = renderedWidgets.filter(Boolean); | ||||
| 					returnData[location] = renderedWidgets.length ? renderedWidgets : undefined; | ||||
|  | ||||
| 					done(); | ||||
| 	}); | ||||
| 			}, function (err) { | ||||
| 				next(err, returnData); | ||||
| 			}); | ||||
| 		}, | ||||
| 	], callback); | ||||
|  | ||||
| 	return returnData; | ||||
| }; | ||||
|  | ||||
| function renderWidget(widget, uid, options, callback) { | ||||
| 	var userLang; | ||||
| async function renderLocation(location, data, uid, options) { | ||||
| 	const widgetsAtLocation = (data[options.template][location] || []).concat(data.global[location] || []); | ||||
|  | ||||
| 	if (!widgetsAtLocation.length) { | ||||
| 		return []; | ||||
| 	} | ||||
|  | ||||
| 	const renderedWidgets = await Promise.all(widgetsAtLocation.map(widget => renderWidget(widget, uid, options))); | ||||
| 	return renderedWidgets; | ||||
| } | ||||
|  | ||||
| async function renderWidget(widget, uid, options) { | ||||
| 	if (!widget || !widget.data || (!!widget.data['hide-mobile'] && options.req.useragent.isMobile)) { | ||||
| 		return setImmediate(callback); | ||||
| 		return; | ||||
| 	} | ||||
|  | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			if (!widget.data.groups.length) { | ||||
| 				return next(null, true); | ||||
| 	let isVisible = true; | ||||
| 	if (widget.data.groups.length) { | ||||
| 		isVisible = await groups.isMemberOfAny(uid, widget.data.groups); | ||||
| 	} | ||||
| 			groups.isMemberOfAny(uid, widget.data.groups, next); | ||||
| 		}, | ||||
| 		function (isVisible, next) { | ||||
| 	if (!isVisible) { | ||||
| 				return callback(); | ||||
| 		return; | ||||
| 	} | ||||
| 	let config = options.res.locals.config || {}; | ||||
| 	if (options.res.locals.isAPI) { | ||||
| 		config = await loadConfigAsync(options.req); | ||||
| 	} | ||||
|  | ||||
| 			if (options.res.locals.isAPI) { | ||||
| 				apiController.loadConfig(options.req, next); | ||||
| 			} else { | ||||
| 				next(null, options.res.locals.config || {}); | ||||
| 			} | ||||
| 		}, | ||||
| 		function (config, next) { | ||||
| 			userLang = config.userLang || meta.config.defaultLang || 'en-GB'; | ||||
| 			var templateData = _.assign({ }, options.templateData, { config: config }); | ||||
| 			plugins.fireHook('filter:widget.render:' + widget.widget, { | ||||
| 	const userLang = config.userLang || meta.config.defaultLang || 'en-GB'; | ||||
| 	const templateData = _.assign({ }, options.templateData, { config: config }); | ||||
| 	const data = await plugins.fireHook('filter:widget.render:' + widget.widget, { | ||||
| 		uid: uid, | ||||
| 		area: options, | ||||
| 		templateData: templateData, | ||||
| 		data: widget.data, | ||||
| 		req: options.req, | ||||
| 		res: options.res, | ||||
| 			}, next); | ||||
| 		}, | ||||
| 		function (data, next) { | ||||
| 	}); | ||||
|  | ||||
| 	if (!data) { | ||||
| 				return callback(); | ||||
| 		return; | ||||
| 	} | ||||
| 			var html = data; | ||||
| 	let html = data; | ||||
| 	if (typeof html !== 'string') { | ||||
| 		html = data.html; | ||||
| 	} else { | ||||
| @@ -106,37 +86,31 @@ function renderWidget(widget, uid, options, callback) { | ||||
| 	} | ||||
|  | ||||
| 	if (widget.data.container && widget.data.container.match('{body}')) { | ||||
| 				Benchpress.compileParse(widget.data.container, { | ||||
| 		html = await Benchpress.compileRender(widget.data.container, { | ||||
| 			title: widget.data.title, | ||||
| 			body: html, | ||||
| 			template: data.templateData && data.templateData.template, | ||||
| 				}, next); | ||||
| 			} else { | ||||
| 				next(null, html); | ||||
| 			} | ||||
| 		}, | ||||
| 		function (html, next) { | ||||
| 			translator.translate(html, userLang, function (translatedHtml) { | ||||
| 				next(null, { html: translatedHtml }); | ||||
| 		}); | ||||
| 		}, | ||||
| 	], callback); | ||||
| 	} | ||||
|  | ||||
| 	if (html !== undefined) { | ||||
| 		html = await translator.translate(html, userLang); | ||||
| 	} | ||||
|  | ||||
| 	return { html: html }; | ||||
| } | ||||
|  | ||||
| widgets.getWidgetDataForTemplates = function (templates, callback) { | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| widgets.getWidgetDataForTemplates = async function (templates) { | ||||
| 	const keys = templates.map(tpl => 'widgets:' + tpl); | ||||
| 			db.getObjects(keys, next); | ||||
| 		}, | ||||
| 		function (data, next) { | ||||
| 			var returnData = {}; | ||||
| 	const data = await db.getObjects(keys); | ||||
|  | ||||
| 	const returnData = {}; | ||||
|  | ||||
| 	templates.forEach(function (template, index) { | ||||
| 		returnData[template] = returnData[template] || {}; | ||||
|  | ||||
| 				var templateWidgetData = data[index] || {}; | ||||
| 				var locations = Object.keys(templateWidgetData); | ||||
| 		const templateWidgetData = data[index] || {}; | ||||
| 		const locations = Object.keys(templateWidgetData); | ||||
|  | ||||
| 		locations.forEach(function (location) { | ||||
| 			if (templateWidgetData && templateWidgetData[location]) { | ||||
| @@ -152,29 +126,19 @@ widgets.getWidgetDataForTemplates = function (templates, callback) { | ||||
| 		}); | ||||
| 	}); | ||||
|  | ||||
| 			next(null, returnData); | ||||
| 		}, | ||||
| 	], callback); | ||||
| 	return returnData; | ||||
| }; | ||||
|  | ||||
| widgets.getArea = function (template, location, callback) { | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			db.getObjectField('widgets:' + template, location, next); | ||||
| 		}, | ||||
| 		function (result, next) { | ||||
| widgets.getArea = async function (template, location) { | ||||
| 	const result = await db.getObjectField('widgets:' + template, location); | ||||
| 	if (!result) { | ||||
| 				return callback(null, []); | ||||
| 		return []; | ||||
| 	} | ||||
| 	try { | ||||
| 				result = parseWidgetData(result); | ||||
| 		return parseWidgetData(result); | ||||
| 	} catch (err) { | ||||
| 				return callback(err); | ||||
| 		throw err; | ||||
| 	} | ||||
|  | ||||
| 			next(null, result); | ||||
| 		}, | ||||
| 	], callback); | ||||
| }; | ||||
|  | ||||
| function parseWidgetData(data) { | ||||
| @@ -190,84 +154,62 @@ function parseWidgetData(data) { | ||||
| 	return widgets; | ||||
| } | ||||
|  | ||||
| widgets.setArea = function (area, callback) { | ||||
| widgets.setArea = async function (area) { | ||||
| 	if (!area.location || !area.template) { | ||||
| 		return callback(new Error('Missing location and template data')); | ||||
| 		throw new Error('Missing location and template data'); | ||||
| 	} | ||||
|  | ||||
| 	db.setObjectField('widgets:' + area.template, area.location, JSON.stringify(area.widgets), callback); | ||||
| 	await db.setObjectField('widgets:' + area.template, area.location, JSON.stringify(area.widgets)); | ||||
| }; | ||||
|  | ||||
| widgets.reset = function (callback) { | ||||
| 	var defaultAreas = [ | ||||
| widgets.reset = async function () { | ||||
| 	const defaultAreas = [ | ||||
| 		{ name: 'Draft Zone', template: 'global', location: 'header' }, | ||||
| 		{ name: 'Draft Zone', template: 'global', location: 'footer' }, | ||||
| 		{ name: 'Draft Zone', template: 'global', location: 'sidebar' }, | ||||
| 	]; | ||||
| 	var drafts; | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			async.parallel({ | ||||
| 				areas: function (next) { | ||||
| 					plugins.fireHook('filter:widgets.getAreas', defaultAreas, next); | ||||
| 				}, | ||||
| 				drafts: function (next) { | ||||
| 					widgets.getArea('global', 'drafts', next); | ||||
| 				}, | ||||
| 			}, next); | ||||
| 		}, | ||||
| 		function (results, next) { | ||||
| 			drafts = results.drafts || []; | ||||
|  | ||||
| 			async.eachSeries(results.areas, function (area, next) { | ||||
| 				async.waterfall([ | ||||
| 					function (next) { | ||||
| 						widgets.getArea(area.template, area.location, next); | ||||
| 					}, | ||||
| 					function (areaData, next) { | ||||
| 						drafts = drafts.concat(areaData); | ||||
| 	const [areas, drafts] = await Promise.all([ | ||||
| 		plugins.fireHook('filter:widgets.getAreas', defaultAreas), | ||||
| 		widgets.getArea('global', 'drafts'), | ||||
| 	]); | ||||
|  | ||||
| 	let saveDrafts = drafts || []; | ||||
| 	for (const area of areas) { | ||||
| 		/* eslint-disable no-await-in-loop */ | ||||
| 		const areaData = await widgets.getArea(area.template, area.location); | ||||
| 		saveDrafts = saveDrafts.concat(areaData); | ||||
| 		area.widgets = []; | ||||
| 						widgets.setArea(area, next); | ||||
| 					}, | ||||
| 				], next); | ||||
| 			}, next); | ||||
| 		}, | ||||
| 		function (next) { | ||||
| 			widgets.setArea({ | ||||
| 		await widgets.setArea(area); | ||||
| 	} | ||||
|  | ||||
| 	await widgets.setArea({ | ||||
| 		template: 'global', | ||||
| 		location: 'drafts', | ||||
| 				widgets: drafts, | ||||
| 			}, next); | ||||
| 		}, | ||||
| 	], callback); | ||||
| 		widgets: saveDrafts, | ||||
| 	}); | ||||
| }; | ||||
|  | ||||
| widgets.resetTemplate = function (template, callback) { | ||||
| 	var toBeDrafted = []; | ||||
| 	async.waterfall([ | ||||
| 		function (next) { | ||||
| 			db.getObject('widgets:' + template + '.tpl', next); | ||||
| 		}, | ||||
| 		function (area, next) { | ||||
| widgets.resetTemplate = async function (template) { | ||||
| 	let toBeDrafted = []; | ||||
| 	const area = await db.getObject('widgets:' + template + '.tpl'); | ||||
| 	for (var location in area) { | ||||
| 		if (area.hasOwnProperty(location)) { | ||||
| 			toBeDrafted = toBeDrafted.concat(JSON.parse(area[location])); | ||||
| 		} | ||||
| 	} | ||||
| 			db.delete('widgets:' + template + '.tpl', next); | ||||
| 		}, | ||||
| 		function (next) { | ||||
| 			db.getObjectField('widgets:global', 'drafts', next); | ||||
| 		}, | ||||
| 		function (draftWidgets, next) { | ||||
| 	await db.delete('widgets:' + template + '.tpl'); | ||||
| 	let draftWidgets = await db.getObjectField('widgets:global', 'drafts'); | ||||
| 	draftWidgets = JSON.parse(draftWidgets).concat(toBeDrafted); | ||||
| 			db.setObjectField('widgets:global', 'drafts', JSON.stringify(draftWidgets), next); | ||||
| 		}, | ||||
| 	], callback); | ||||
| 	await db.setObjectField('widgets:global', 'drafts', JSON.stringify(draftWidgets)); | ||||
| }; | ||||
|  | ||||
| widgets.resetTemplates = function (templates, callback) { | ||||
| 	async.eachSeries(templates, widgets.resetTemplate, callback); | ||||
| widgets.resetTemplates = async function (templates) { | ||||
| 	async.eachSeries(templates, widgets.resetTemplate); | ||||
| 	for (const template of templates) { | ||||
| 		/* eslint-disable no-await-in-loop */ | ||||
| 		await widgets.resetTemplate(template); | ||||
| 	} | ||||
| }; | ||||
|  | ||||
| require('../promisify')(widgets); | ||||
|   | ||||
| @@ -885,15 +885,12 @@ describe('Controllers', function () { | ||||
| 						location: 'sidebar', | ||||
| 						widgets: [ | ||||
| 							{ | ||||
| 								widget: 'html', | ||||
| 								data: [{ | ||||
| 								widget: 'html', | ||||
| 								data: { | ||||
| 									html: 'test', | ||||
| 									title: '', | ||||
| 									container: '', | ||||
| 								}, | ||||
| 								}], | ||||
| 							}, | ||||
| 						], | ||||
| 					}; | ||||
| @@ -920,6 +917,7 @@ describe('Controllers', function () { | ||||
| 				assert.equal(res.statusCode, 200); | ||||
| 				assert(body.widgets); | ||||
| 				assert(body.widgets.sidebar); | ||||
| 				assert.equal(body.widgets.sidebar[0].html, 'test'); | ||||
| 				done(); | ||||
| 			}); | ||||
| 		}); | ||||
|   | ||||
| @@ -257,6 +257,7 @@ function enableDefaultPlugins(callback) { | ||||
| 	var defaultEnabled = [ | ||||
| 		'nodebb-plugin-dbsearch', | ||||
| 		'nodebb-plugin-soundpack-default', | ||||
| 		'nodebb-widget-essentials', | ||||
| 	]; | ||||
|  | ||||
| 	winston.info('[install/enableDefaultPlugins] activating default plugins', defaultEnabled); | ||||
|   | ||||
		Reference in New Issue
	
	Block a user