From 753f1576ceceaaaf03bcb5b2fc62593cf084fc62 Mon Sep 17 00:00:00 2001 From: Baris Usakli <barisusakli@gmail.com> Date: Wed, 16 Aug 2017 16:47:52 -0400 Subject: [PATCH 01/13] processSortedSet --- src/user/profile.js | 4 +++- src/user/reset.js | 9 ++------- 2 files changed, 5 insertions(+), 8 deletions(-) diff --git a/src/user/profile.js b/src/user/profile.js index 54ed58267b..dbdcbcba85 100644 --- a/src/user/profile.js +++ b/src/user/profile.js @@ -185,7 +185,9 @@ module.exports = function (User) { function (next) { db.sortedSetAdd('users:notvalidated', Date.now(), uid, next); }, - async.apply(User.reset.cleanByUid, uid), + function (next) { + User.reset.cleanByUid(uid, next); + }, ], function (err) { next(err); }); diff --git a/src/user/reset.js b/src/user/reset.js index 618ba680bc..db52ed033e 100644 --- a/src/user/reset.js +++ b/src/user/reset.js @@ -171,17 +171,12 @@ UserReset.clean = function (callback) { }; UserReset.cleanByUid = function (uid, callback) { - if (typeof callback !== 'function') { - callback = function () {}; - } - var toClean = []; uid = parseInt(uid, 10); async.waterfall([ - async.apply(db.getSortedSetRange.bind(db), 'reset:issueDate', 0, -1), - function (tokens, next) { - batch.processArray(tokens, function (tokens, next) { + function (next) { + batch.processSortedSet('reset:issueDate', function (tokens, next) { db.getObjectFields('reset:uid', tokens, function (err, results) { for (var code in results) { if (results.hasOwnProperty(code) && parseInt(results[code], 10) === uid) { From 4e98c4b39fb430da2e7175f1de3a6c44c6937956 Mon Sep 17 00:00:00 2001 From: "Misty (Bot)" <deploy@nodebb.org> Date: Thu, 17 Aug 2017 09:23:28 +0000 Subject: [PATCH 02/13] Latest translations and fallbacks --- public/language/zh-CN/admin/advanced/database.json | 4 ++-- public/language/zh-CN/admin/advanced/errors.json | 2 +- public/language/zh-CN/admin/advanced/events.json | 2 +- public/language/zh-CN/admin/appearance/skins.json | 2 +- public/language/zh-CN/admin/development/logger.json | 2 +- public/language/zh-CN/admin/general/dashboard.json | 2 +- public/language/zh-CN/admin/manage/categories.json | 2 +- public/language/zh-CN/admin/manage/registration.json | 2 +- public/language/zh-CN/category.json | 4 ++-- 9 files changed, 11 insertions(+), 11 deletions(-) diff --git a/public/language/zh-CN/admin/advanced/database.json b/public/language/zh-CN/admin/advanced/database.json index 825469bb2d..f8487c7fcb 100644 --- a/public/language/zh-CN/admin/advanced/database.json +++ b/public/language/zh-CN/admin/advanced/database.json @@ -17,14 +17,14 @@ "mongo.file-size": "文件大小", "mongo.resident-memory": "驻留内存", "mongo.virtual-memory": "虚拟内存", - "mongo.mapped-memory": "映射", + "mongo.mapped-memory": "已映射内存", "mongo.raw-info": "MongoDB 原始信息", "redis": "Redis", "redis.version": "Redis 版本", "redis.connected-clients": "已连接客户端", "redis.connected-slaves": "已连接从", - "redis.blocked-clients": "阻止的客户端", + "redis.blocked-clients": "受阻的客户端", "redis.used-memory": "已使用内存", "redis.memory-frag-ratio": "内存碎片比率", "redis.total-connections-recieved": "已接收的连接总数", diff --git a/public/language/zh-CN/admin/advanced/errors.json b/public/language/zh-CN/admin/advanced/errors.json index 257f88661c..6fd8b532b6 100644 --- a/public/language/zh-CN/admin/advanced/errors.json +++ b/public/language/zh-CN/admin/advanced/errors.json @@ -4,7 +4,7 @@ "error.404": "404 页面不存在", "error.503": "503 服务不可用", "manage-error-log": "管理错误日志", - "export-error-log": "提取错误日志(.csv)", + "export-error-log": "导出错误日志 (.csv)", "clear-error-log": "清空错误日志", "route": "路由", "count": "计数", diff --git a/public/language/zh-CN/admin/advanced/events.json b/public/language/zh-CN/admin/advanced/events.json index 85d741ff27..b9f44389e7 100644 --- a/public/language/zh-CN/admin/advanced/events.json +++ b/public/language/zh-CN/admin/advanced/events.json @@ -1,6 +1,6 @@ { "events": "事件", - "no-events": "暂无事件。", + "no-events": "暂无事件", "control-panel": "事件控制面板", "delete-events": "清除事件" } \ No newline at end of file diff --git a/public/language/zh-CN/admin/appearance/skins.json b/public/language/zh-CN/admin/appearance/skins.json index 730c79c2f0..1a075fe9f9 100644 --- a/public/language/zh-CN/admin/appearance/skins.json +++ b/public/language/zh-CN/admin/appearance/skins.json @@ -5,5 +5,5 @@ "current-skin": "当前皮肤", "skin-updated": "皮肤已更新", "applied-success": "%1 皮肤已成功应用", - "revert-success": "皮肤恢复到基础颜色" + "revert-success": "皮肤已恢复到基础颜色" } \ No newline at end of file diff --git a/public/language/zh-CN/admin/development/logger.json b/public/language/zh-CN/admin/development/logger.json index b7e6b36f17..769fd15b11 100644 --- a/public/language/zh-CN/admin/development/logger.json +++ b/public/language/zh-CN/admin/development/logger.json @@ -5,7 +5,7 @@ "enable-http": "启用 HTTP 日志", "enable-socket": "启用 socket.io 事件日志", "file-path": "日志文件路径", - "file-path-placeholder": "如/path/to/log/file.log ::: 如想在终端中显示日志请留空此项", + "file-path-placeholder": "如 /path/to/log/file.log ::: 如想在终端中显示日志请留空此项", "control-panel": "日志记录器控制面板", "update-settings": "更新日志记录器设置" diff --git a/public/language/zh-CN/admin/general/dashboard.json b/public/language/zh-CN/admin/general/dashboard.json index f89eecfc59..f263627aa2 100644 --- a/public/language/zh-CN/admin/general/dashboard.json +++ b/public/language/zh-CN/admin/general/dashboard.json @@ -24,7 +24,7 @@ "keep-updated": "请确保您已及时更新 NodeBB 以获得最新的安全补丁与 Bug 修复。", "up-to-date": "<p>正在使用 <strong>最新版本</strong> <i class=\"fa fa-check\"></i></p>", "upgrade-available": "<p> 新版本 (v%1) 已经发布! 建议 <a href=\"https://docs.nodebb.org/configuring/upgrade/\">更新你的 NodeBB</a>。</p>", - "prerelease-upgrade-available": "<p>正在使用NodeBB过期的实验版本。新的版本 (v%1) 已经发布。 请考虑<a href=\"https://docs.nodebb.org/configuring/upgrade/\">更新你的 NodeBB</a>。", + "prerelease-upgrade-available": "<p>正在使用过时的测试版 NodeBB。新的版本 (v%1) 已经发布。 请考虑<a href=\"https://docs.nodebb.org/configuring/upgrade/\">更新你的 NodeBB</a>。", "prerelease-warning": "<p>正在使用<strong>测试版</strong> NodeBB。可能会出现意外的 Bug。<i class=\"fa fa-exclamation-triangle\"></i></p>", "running-in-development": "<span>论坛正处于开发模式,这可能使其暴露于潜在的危险之中;请联系您的系统管理员。</span>", diff --git a/public/language/zh-CN/admin/manage/categories.json b/public/language/zh-CN/admin/manage/categories.json index babc97bb9f..63f94bdebe 100644 --- a/public/language/zh-CN/admin/manage/categories.json +++ b/public/language/zh-CN/admin/manage/categories.json @@ -60,7 +60,7 @@ "alert.copy-success": "设置已复制!", "alert.set-parent-category": "设置父版块", "alert.updated": "版块已更新", - "alert.updated-success": "版块ID %1 成功更新。", + "alert.updated-success": "版块 ID %1 成功更新。", "alert.upload-image": "上传版块图片", "alert.find-user": "查找用户", "alert.user-search": "在这里查找用户…", diff --git a/public/language/zh-CN/admin/manage/registration.json b/public/language/zh-CN/admin/manage/registration.json index aa46fcdb49..fa2e26c5be 100644 --- a/public/language/zh-CN/admin/manage/registration.json +++ b/public/language/zh-CN/admin/manage/registration.json @@ -11,7 +11,7 @@ "list.ip-spam": "频率:%1 显示: %2", "invitations": "邀请", - "invitations.description": "下面是一份完整的邀请请求列表。请使用Ctrl-F键以及电子邮件或者用户名以便搜索这个列表。<br><br>那些已经接受他们邀请的用户的用户名将显示在电子邮箱右边。", + "invitations.description": "下面列出了所有已发送的邀请。您可以使用 Ctrl+F 快捷键搜索列表中的邮箱地址或用户名。<br><br>如果用户接受了邀请,他的用户名将会被显示在邮箱右边。", "invitations.inviter-username": "邀请人用户名", "invitations.invitee-email": "受邀请的电子邮箱", "invitations.invitee-username": "受邀请的用户名(如果已经注册)", diff --git a/public/language/zh-CN/category.json b/public/language/zh-CN/category.json index e9ce41a2e3..4c9902d3d8 100644 --- a/public/language/zh-CN/category.json +++ b/public/language/zh-CN/category.json @@ -2,7 +2,7 @@ "category": "版块", "subcategories": "子版块", "new_topic_button": "新主题", - "guest-login-post": "登录后发表", + "guest-login-post": "登录以发表", "no_topics": "<strong>此版块还没有任何内容。</strong><br />赶紧来发帖吧!", "browsing": "正在浏览", "no_replies": "尚无回复", @@ -10,7 +10,7 @@ "share_this_category": "分享此版块", "watch": "关注", "ignore": "忽略", - "watching": "正在关注", + "watching": "已关注", "ignoring": "已忽略", "watching.description": "显示未读主题", "ignoring.description": "不显示未读主题", From f14c2f44ffba5972e65fd8f37f40cfae6b238c4f Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Thu, 17 Aug 2017 11:53:01 -0400 Subject: [PATCH 03/13] up themes for rtl fixes --- package.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index c12be8d8db..392a6986b2 100644 --- a/package.json +++ b/package.json @@ -65,9 +65,9 @@ "nodebb-plugin-spam-be-gone": "0.5.1", "nodebb-rewards-essentials": "0.0.9", "nodebb-theme-lavender": "4.0.5", - "nodebb-theme-persona": "5.0.22", + "nodebb-theme-persona": "5.0.23", "nodebb-theme-slick": "1.1.0", - "nodebb-theme-vanilla": "6.0.17", + "nodebb-theme-vanilla": "6.0.18", "nodebb-widget-essentials": "3.0.1", "nodemailer": "2.6.4", "nodemailer-sendmail-transport": "1.0.0", From 89b2d94b6151c4802b359cf38113184626d46828 Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Thu, 17 Aug 2017 12:54:51 -0400 Subject: [PATCH 04/13] up themes again --- package.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index 392a6986b2..ecd8a34141 100644 --- a/package.json +++ b/package.json @@ -65,9 +65,9 @@ "nodebb-plugin-spam-be-gone": "0.5.1", "nodebb-rewards-essentials": "0.0.9", "nodebb-theme-lavender": "4.0.5", - "nodebb-theme-persona": "5.0.23", + "nodebb-theme-persona": "5.0.24", "nodebb-theme-slick": "1.1.0", - "nodebb-theme-vanilla": "6.0.18", + "nodebb-theme-vanilla": "6.0.19", "nodebb-widget-essentials": "3.0.1", "nodemailer": "2.6.4", "nodemailer-sendmail-transport": "1.0.0", From 4dae008e16a9389fd1c47cb87f962e846462adee Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Thu, 17 Aug 2017 13:17:24 -0400 Subject: [PATCH 05/13] up themes again :shipit: --- package.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index ecd8a34141..0c85ba1c53 100644 --- a/package.json +++ b/package.json @@ -65,9 +65,9 @@ "nodebb-plugin-spam-be-gone": "0.5.1", "nodebb-rewards-essentials": "0.0.9", "nodebb-theme-lavender": "4.0.5", - "nodebb-theme-persona": "5.0.24", + "nodebb-theme-persona": "5.0.25", "nodebb-theme-slick": "1.1.0", - "nodebb-theme-vanilla": "6.0.19", + "nodebb-theme-vanilla": "6.0.20", "nodebb-widget-essentials": "3.0.1", "nodemailer": "2.6.4", "nodemailer-sendmail-transport": "1.0.0", From db13aac106df3b0d4d4078d8e4e13276d79a5bf0 Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Thu, 17 Aug 2017 13:35:40 -0400 Subject: [PATCH 06/13] up themes, for the last time today. --- package.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index 0c85ba1c53..455e5b8957 100644 --- a/package.json +++ b/package.json @@ -65,9 +65,9 @@ "nodebb-plugin-spam-be-gone": "0.5.1", "nodebb-rewards-essentials": "0.0.9", "nodebb-theme-lavender": "4.0.5", - "nodebb-theme-persona": "5.0.25", + "nodebb-theme-persona": "5.0.27", "nodebb-theme-slick": "1.1.0", - "nodebb-theme-vanilla": "6.0.20", + "nodebb-theme-vanilla": "6.0.21", "nodebb-widget-essentials": "3.0.1", "nodemailer": "2.6.4", "nodemailer-sendmail-transport": "1.0.0", From dc9b21021ab08cba0478baa25e75017059ade05d Mon Sep 17 00:00:00 2001 From: Baris Usakli <barisusakli@gmail.com> Date: Fri, 18 Aug 2017 11:23:15 -0400 Subject: [PATCH 07/13] escape moderation notes --- src/controllers/accounts/info.js | 33 +++++++++++++++----------------- src/user/info.js | 1 + test/user.js | 5 +++-- 3 files changed, 19 insertions(+), 20 deletions(-) diff --git a/src/controllers/accounts/info.js b/src/controllers/accounts/info.js index 7f9edb2462..cb3b6f1abf 100644 --- a/src/controllers/accounts/info.js +++ b/src/controllers/accounts/info.js @@ -47,24 +47,21 @@ infoController.get = function (req, res, callback) { }, }, next); }, - ], function (err, data) { - if (err) { - return callback(err); - } + function (data) { + userData.history = data.history; + userData.sessions = data.sessions; + userData.usernames = data.usernames; + userData.emails = data.emails; - userData.history = data.history; - userData.sessions = data.sessions; - userData.usernames = data.usernames; - userData.emails = data.emails; - - if (userData.isAdminOrGlobalModeratorOrModerator) { - userData.moderationNotes = data.notes.notes; - var pageCount = Math.ceil(data.notes.count / itemsPerPage); - userData.pagination = pagination.create(page, pageCount, req.query); - } - userData.title = '[[pages:account/info]]'; - userData.breadcrumbs = helpers.buildBreadcrumbs([{ text: userData.username, url: '/user/' + userData.userslug }, { text: '[[user:account_info]]' }]); + if (userData.isAdminOrGlobalModeratorOrModerator) { + userData.moderationNotes = data.notes.notes; + var pageCount = Math.ceil(data.notes.count / itemsPerPage); + userData.pagination = pagination.create(page, pageCount, req.query); + } + userData.title = '[[pages:account/info]]'; + userData.breadcrumbs = helpers.buildBreadcrumbs([{ text: userData.username, url: '/user/' + userData.userslug }, { text: '[[user:account_info]]' }]); - res.render('account/info', userData); - }); + res.render('account/info', userData); + }, + ], callback); }; diff --git a/src/user/info.js b/src/user/info.js index e8642989a1..5e91c6cf08 100644 --- a/src/user/info.js +++ b/src/user/info.js @@ -166,6 +166,7 @@ module.exports = function (User) { var data = JSON.parse(note); uids.push(data.uid); data.timestampISO = utils.toISOString(data.timestamp); + data.note = validator.escape(String(data.note)); return data; } catch (err) { return next(err); diff --git a/test/user.js b/test/user.js index a5f92a1744..77144ea9c7 100644 --- a/test/user.js +++ b/test/user.js @@ -1236,15 +1236,16 @@ describe('User', function () { setTimeout(next, 50); }, function (next) { - socketUser.setModerationNote({ uid: adminUid }, { uid: testUid, note: 'second moderation note' }, next); + socketUser.setModerationNote({ uid: adminUid }, { uid: testUid, note: '<svg/onload=alert(document.location);//' }, next); }, function (next) { User.getModerationNotes(testUid, 0, -1, next); }, ], function (err, notes) { assert.ifError(err); - assert.equal(notes[0].note, 'second moderation note'); + assert.equal(notes[0].note, '<svg/onload=alert(document.location);//'); assert.equal(notes[0].uid, adminUid); + assert.equal(notes[1].note, 'this is a test user'); assert(notes[0].timestamp); done(); }); From b4a4fd8679c2106f7ad1ffdb66532099d2a60fa2 Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Fri, 18 Aug 2017 12:25:41 -0400 Subject: [PATCH 08/13] up composer for #5886 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 455e5b8957..6fa914da46 100644 --- a/package.json +++ b/package.json @@ -55,7 +55,7 @@ "morgan": "^1.3.2", "mousetrap": "^1.5.3", "nconf": "~0.8.2", - "nodebb-plugin-composer-default": "5.0.5", + "nodebb-plugin-composer-default": "5.0.6", "nodebb-plugin-dbsearch": "2.0.6", "nodebb-plugin-emoji-extended": "1.1.1", "nodebb-plugin-emoji-one": "1.2.1", From 1159abf9ec8f33328d0bb61a99b6f61bcab74aa2 Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Fri, 18 Aug 2017 15:05:49 -0400 Subject: [PATCH 09/13] fixes #5879 --- package.json | 1 + src/meta/blacklist.js | 20 +++++++++++++++----- 2 files changed, 16 insertions(+), 5 deletions(-) diff --git a/package.json b/package.json index 6fa914da46..148c7c4ac2 100644 --- a/package.json +++ b/package.json @@ -41,6 +41,7 @@ "express-useragent": "1.0.7", "html-to-text": "3.3.0", "ip": "1.1.5", + "ip-range-check": "^0.0.2", "jimp": "0.2.28", "jquery": "^3.1.0", "json-2-csv": "^2.0.22", diff --git a/src/meta/blacklist.js b/src/meta/blacklist.js index 1ef6cfe0a9..80f3ca1190 100644 --- a/src/meta/blacklist.js +++ b/src/meta/blacklist.js @@ -1,6 +1,7 @@ 'use strict'; var ip = require('ip'); +var ipRangeCheck = require('ip-range-check'); var winston = require('winston'); var async = require('async'); @@ -27,6 +28,7 @@ Blacklist.load = function (callback) { ipv4: rules.ipv4, ipv6: rules.ipv6, cidr: rules.cidr, + cidr6: rules.cidr6, }; next(); }, @@ -53,11 +55,12 @@ Blacklist.get = function (callback) { Blacklist.test = function (clientIp, callback) { if ( - Blacklist._rules.ipv4.indexOf(clientIp) === -1 &&// not explicitly specified in ipv4 list - Blacklist._rules.ipv6.indexOf(clientIp) === -1 &&// not explicitly specified in ipv6 list + Blacklist._rules.ipv4.indexOf(clientIp) === -1 && // not explicitly specified in ipv4 list + Blacklist._rules.ipv6.indexOf(clientIp) === -1 && // not explicitly specified in ipv6 list !Blacklist._rules.cidr.some(function (subnet) { return ip.cidrSubnet(subnet).contains(clientIp); - }) // not in a blacklisted cidr range + }) && // not in a blacklisted IPv4 cidr range + !ipRangeCheck(clientIp, Blacklist._rules.cidr6) // not in a blacklisted IPv6 cidr range ) { if (typeof callback === 'function') { setImmediate(callback); @@ -81,9 +84,11 @@ Blacklist.validate = function (rules, callback) { var ipv4 = []; var ipv6 = []; var cidr = []; + var cidr6 = []; var invalid = []; - var isCidrSubnet = /^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])(\/([0-9]|[1-2][0-9]|3[0-2]))$/; + var isIPv4CidrSubnet = /^(([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])\.){3}([0-9]|[1-9][0-9]|1[0-9]{2}|2[0-4][0-9]|25[0-5])(\/([0-9]|[1-2][0-9]|3[0-2]))$/; + var isIPv6CidrSubnet = /^s*((([0-9A-Fa-f]{1,4}:){7}([0-9A-Fa-f]{1,4}|:))|(([0-9A-Fa-f]{1,4}:){6}(:[0-9A-Fa-f]{1,4}|((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3})|:))|(([0-9A-Fa-f]{1,4}:){5}(((:[0-9A-Fa-f]{1,4}){1,2})|:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3})|:))|(([0-9A-Fa-f]{1,4}:){4}(((:[0-9A-Fa-f]{1,4}){1,3})|((:[0-9A-Fa-f]{1,4})?:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){3}(((:[0-9A-Fa-f]{1,4}){1,4})|((:[0-9A-Fa-f]{1,4}){0,2}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){2}(((:[0-9A-Fa-f]{1,4}){1,5})|((:[0-9A-Fa-f]{1,4}){0,3}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(([0-9A-Fa-f]{1,4}:){1}(((:[0-9A-Fa-f]{1,4}){1,6})|((:[0-9A-Fa-f]{1,4}){0,4}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:))|(:(((:[0-9A-Fa-f]{1,4}){1,7})|((:[0-9A-Fa-f]{1,4}){0,5}:((25[0-5]|2[0-4]d|1dd|[1-9]?d)(.(25[0-5]|2[0-4]d|1dd|[1-9]?d)){3}))|:)))(%.+)?s*(\/([0-9]|[1-9][0-9]|1[0-1][0-9]|12[0-8]))?$/; var inlineCommentMatch = /#.*$/; var whitelist = ['127.0.0.1', '::1', '::ffff:0:127.0.0.1']; @@ -109,7 +114,11 @@ Blacklist.validate = function (rules, callback) { ipv6.push(rule); return true; } - if (isCidrSubnet.test(rule)) { + if (isIPv4CidrSubnet.test(rule)) { + cidr.push(rule); + return true; + } + if (isIPv6CidrSubnet.test(rule)) { cidr.push(rule); return true; } @@ -123,6 +132,7 @@ Blacklist.validate = function (rules, callback) { ipv4: ipv4, ipv6: ipv6, cidr: cidr, + cidr6: cidr6, valid: rules, invalid: invalid, }); From c1d7b06ded71a6ddf8fb4ca7bbae5211db99cca7 Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Fri, 18 Aug 2017 16:30:04 -0400 Subject: [PATCH 10/13] Fixes #5873 - Notifications.getMultiple now takes an optional uid parameter - If a notification link in dropdown points to a topic and you're in said topic, you will be scrolled to the post instead of ajaxified to it. --- public/src/modules/notifications.js | 15 +++++++++-- src/notifications.js | 39 ++++++++++++++++++++++++----- src/user/notifications.js | 2 +- 3 files changed, 47 insertions(+), 9 deletions(-) diff --git a/public/src/modules/notifications.js b/public/src/modules/notifications.js index 928884b69a..ea78918c1b 100644 --- a/public/src/modules/notifications.js +++ b/public/src/modules/notifications.js @@ -1,7 +1,7 @@ 'use strict'; -define('notifications', ['sounds', 'translator', 'components'], function (sounds, translator, components) { +define('notifications', ['sounds', 'translator', 'components', 'navigator'], function (sounds, translator, components, navigator) { var Notifications = {}; var unreadNotifs = {}; @@ -20,7 +20,18 @@ define('notifications', ['sounds', 'translator', 'components'], function (sounds Notifications.loadNotifications(notifList); }); - notifList.on('click', '[data-nid]', function () { + notifList.on('click', '[data-nid]', function (e) { + // Scroll to index if already in topic (gh#5873) + var index = $(this).attr('data-index'); + var tid = $(this).attr('data-tid'); + if (index && ajaxify.data.template.topic && parseInt(ajaxify.data.tid, 10) === parseInt(tid, 10)) { + e.stopPropagation(); + e.preventDefault(); + + navigator.scrollToIndex(index, true); + notifTrigger.dropdown('toggle'); + } + var unread = $(this).hasClass('unread'); if (!unread) { return; diff --git a/src/notifications.js b/src/notifications.js index 58ccf2e8fa..715f4cf264 100644 --- a/src/notifications.js +++ b/src/notifications.js @@ -13,6 +13,7 @@ var groups = require('./groups'); var meta = require('./meta'); var batch = require('./batch'); var plugins = require('./plugins'); +var posts = require('./posts'); var utils = require('./utils'); var Notifications = module.exports; @@ -22,13 +23,19 @@ Notifications.startJobs = function () { new cron('*/30 * * * *', Notifications.prune, null, true); }; -Notifications.get = function (nid, callback) { - Notifications.getMultiple([nid], function (err, notifications) { +Notifications.get = function (nid, uid, callback) { + Notifications.getMultiple([nid], uid, function (err, notifications) { callback(err, Array.isArray(notifications) && notifications.length ? notifications[0] : null); }); }; -Notifications.getMultiple = function (nids, callback) { +Notifications.getMultiple = function (nids, uid, callback) { + if (typeof uid === 'function' && !callback) { + // no uid passed in + callback = uid; + uid = undefined; + } + if (!Array.isArray(nids) || !nids.length) { return setImmediate(callback, null, []); } @@ -37,8 +44,19 @@ Notifications.getMultiple = function (nids, callback) { }); var notifications; + var userSettings; async.waterfall([ + function (next) { + if (!uid) { + return setImmediate(next); + } + + User.getSettings(uid, function (err, settings) { + userSettings = settings; + next(err); + }); + }, function (next) { db.getObjects(keys, next); }, @@ -51,7 +69,7 @@ Notifications.getMultiple = function (nids, callback) { User.getUsersFields(userKeys, ['username', 'userslug', 'picture'], next); }, function (usersData, next) { - notifications.forEach(function (notification, index) { + async.eachOf(notifications, function (notification, index, next) { if (notification) { notification.datetimeISO = utils.toISOString(notification.datetime); @@ -68,10 +86,19 @@ Notifications.getMultiple = function (nids, callback) { } else if (notification.image === 'brand:logo' || !notification.image) { notification.image = meta.config['brand:logo'] || nconf.get('relative_path') + '/logo.png'; } + + if (notification.path.startsWith('/post/')) { + posts.getPidIndex(notification.pid, notification.tid, userSettings.topicPostSort, function (err, index) { + notification.index = index; + next(err); + }); + } else { + next(); + } } + }, function (err) { + next(err, notifications); }); - - next(null, notifications); }, ], callback); }; diff --git a/src/user/notifications.js b/src/user/notifications.js index 4e2dcba7e8..cb3f208e0f 100644 --- a/src/user/notifications.js +++ b/src/user/notifications.js @@ -140,7 +140,7 @@ UserNotifications.getNotifications = function (nids, uid, callback) { function (next) { async.parallel({ notifications: function (next) { - notifications.getMultiple(nids, next); + notifications.getMultiple(nids, uid, next); }, hasRead: function (next) { db.isSortedSetMembers('uid:' + uid + ':notifications:read', nids, next); From 31ae8c7fd5eeb00a2fcc70214bae1c29d397816c Mon Sep 17 00:00:00 2001 From: Julian Lam <julian@nodebb.org> Date: Fri, 18 Aug 2017 16:32:07 -0400 Subject: [PATCH 11/13] up themes for #5873 --- package.json | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/package.json b/package.json index 148c7c4ac2..2a7575b3b5 100644 --- a/package.json +++ b/package.json @@ -66,9 +66,9 @@ "nodebb-plugin-spam-be-gone": "0.5.1", "nodebb-rewards-essentials": "0.0.9", "nodebb-theme-lavender": "4.0.5", - "nodebb-theme-persona": "5.0.27", + "nodebb-theme-persona": "5.0.28", "nodebb-theme-slick": "1.1.0", - "nodebb-theme-vanilla": "6.0.21", + "nodebb-theme-vanilla": "6.0.22", "nodebb-widget-essentials": "3.0.1", "nodemailer": "2.6.4", "nodemailer-sendmail-transport": "1.0.0", From 5dfb2fb83a12fc9185d8bacc72740196e3aec184 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= <baris@nodebb.org> Date: Fri, 18 Aug 2017 19:00:48 -0400 Subject: [PATCH 12/13] up themes, fix notif test --- package.json | 4 +-- public/src/modules/notifications.js | 31 ++++++++++++++--------- src/notifications.js | 38 ++++------------------------- src/user/notifications.js | 2 +- 4 files changed, 28 insertions(+), 47 deletions(-) diff --git a/package.json b/package.json index 2a7575b3b5..56a606d617 100644 --- a/package.json +++ b/package.json @@ -66,9 +66,9 @@ "nodebb-plugin-spam-be-gone": "0.5.1", "nodebb-rewards-essentials": "0.0.9", "nodebb-theme-lavender": "4.0.5", - "nodebb-theme-persona": "5.0.28", + "nodebb-theme-persona": "5.0.29", "nodebb-theme-slick": "1.1.0", - "nodebb-theme-vanilla": "6.0.22", + "nodebb-theme-vanilla": "6.0.23", "nodebb-widget-essentials": "3.0.1", "nodemailer": "2.6.4", "nodemailer-sendmail-transport": "1.0.0", diff --git a/public/src/modules/notifications.js b/public/src/modules/notifications.js index ea78918c1b..f99e1b8de4 100644 --- a/public/src/modules/notifications.js +++ b/public/src/modules/notifications.js @@ -20,23 +20,19 @@ define('notifications', ['sounds', 'translator', 'components', 'navigator'], fun Notifications.loadNotifications(notifList); }); - notifList.on('click', '[data-nid]', function (e) { - // Scroll to index if already in topic (gh#5873) - var index = $(this).attr('data-index'); - var tid = $(this).attr('data-tid'); - if (index && ajaxify.data.template.topic && parseInt(ajaxify.data.tid, 10) === parseInt(tid, 10)) { - e.stopPropagation(); - e.preventDefault(); - - navigator.scrollToIndex(index, true); + notifList.on('click', '[data-nid]', function (ev) { + var notifEl = $(this); + if (scrollToPostIndexIfOnPage(notifEl)) { + ev.stopPropagation(); + ev.preventDefault(); notifTrigger.dropdown('toggle'); } - var unread = $(this).hasClass('unread'); + var unread = notifEl.hasClass('unread'); if (!unread) { return; } - var nid = $(this).attr('data-nid'); + var nid = notifEl.attr('data-nid'); socket.emit('notifications.markRead', nid, function (err) { if (err) { return app.alertError(err.message); @@ -118,6 +114,19 @@ define('notifications', ['sounds', 'translator', 'components', 'navigator'], fun }); }; + function scrollToPostIndexIfOnPage(notifEl) { + // Scroll to index if already in topic (gh#5873) + var pid = notifEl.attr('data-pid'); + var tid = notifEl.attr('data-tid'); + var path = notifEl.attr('data-path'); + var postEl = components.get('post', 'pid', pid); + if (path.startsWith(config.relative_path + '/post/') && pid && postEl.length && ajaxify.data.template.topic && parseInt(ajaxify.data.tid, 10) === parseInt(tid, 10)) { + navigator.scrollToIndex(postEl.attr('data-index'), true); + return true; + } + return false; + } + Notifications.loadNotifications = function (notifList) { socket.emit('notifications.get', null, function (err, data) { if (err) { diff --git a/src/notifications.js b/src/notifications.js index 715f4cf264..098efe5d9f 100644 --- a/src/notifications.js +++ b/src/notifications.js @@ -13,7 +13,6 @@ var groups = require('./groups'); var meta = require('./meta'); var batch = require('./batch'); var plugins = require('./plugins'); -var posts = require('./posts'); var utils = require('./utils'); var Notifications = module.exports; @@ -23,19 +22,13 @@ Notifications.startJobs = function () { new cron('*/30 * * * *', Notifications.prune, null, true); }; -Notifications.get = function (nid, uid, callback) { - Notifications.getMultiple([nid], uid, function (err, notifications) { +Notifications.get = function (nid, callback) { + Notifications.getMultiple([nid], function (err, notifications) { callback(err, Array.isArray(notifications) && notifications.length ? notifications[0] : null); }); }; -Notifications.getMultiple = function (nids, uid, callback) { - if (typeof uid === 'function' && !callback) { - // no uid passed in - callback = uid; - uid = undefined; - } - +Notifications.getMultiple = function (nids, callback) { if (!Array.isArray(nids) || !nids.length) { return setImmediate(callback, null, []); } @@ -44,19 +37,8 @@ Notifications.getMultiple = function (nids, uid, callback) { }); var notifications; - var userSettings; async.waterfall([ - function (next) { - if (!uid) { - return setImmediate(next); - } - - User.getSettings(uid, function (err, settings) { - userSettings = settings; - next(err); - }); - }, function (next) { db.getObjects(keys, next); }, @@ -69,7 +51,7 @@ Notifications.getMultiple = function (nids, uid, callback) { User.getUsersFields(userKeys, ['username', 'userslug', 'picture'], next); }, function (usersData, next) { - async.eachOf(notifications, function (notification, index, next) { + notifications.forEach(function (notification, index) { if (notification) { notification.datetimeISO = utils.toISOString(notification.datetime); @@ -86,19 +68,9 @@ Notifications.getMultiple = function (nids, uid, callback) { } else if (notification.image === 'brand:logo' || !notification.image) { notification.image = meta.config['brand:logo'] || nconf.get('relative_path') + '/logo.png'; } - - if (notification.path.startsWith('/post/')) { - posts.getPidIndex(notification.pid, notification.tid, userSettings.topicPostSort, function (err, index) { - notification.index = index; - next(err); - }); - } else { - next(); - } } - }, function (err) { - next(err, notifications); }); + next(null, notifications); }, ], callback); }; diff --git a/src/user/notifications.js b/src/user/notifications.js index cb3f208e0f..4e2dcba7e8 100644 --- a/src/user/notifications.js +++ b/src/user/notifications.js @@ -140,7 +140,7 @@ UserNotifications.getNotifications = function (nids, uid, callback) { function (next) { async.parallel({ notifications: function (next) { - notifications.getMultiple(nids, uid, next); + notifications.getMultiple(nids, next); }, hasRead: function (next) { db.isSortedSetMembers('uid:' + uid + ':notifications:read', nids, next); From 5344edc2a762480d8f39712baa52f626233a7295 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Bar=C4=B1=C5=9F=20Soner=20U=C5=9Fakl=C4=B1?= <baris@nodebb.org> Date: Fri, 18 Aug 2017 20:08:19 -0400 Subject: [PATCH 13/13] closes #5885 --- package.json | 4 ++-- src/controllers/accounts/helpers.js | 3 +-- src/privileges/posts.js | 17 +++++++++++++++++ src/socket.io/posts/tools.js | 4 ++++ 4 files changed, 24 insertions(+), 4 deletions(-) diff --git a/package.json b/package.json index 56a606d617..b2b7e352c0 100644 --- a/package.json +++ b/package.json @@ -66,9 +66,9 @@ "nodebb-plugin-spam-be-gone": "0.5.1", "nodebb-rewards-essentials": "0.0.9", "nodebb-theme-lavender": "4.0.5", - "nodebb-theme-persona": "5.0.29", + "nodebb-theme-persona": "5.0.30", "nodebb-theme-slick": "1.1.0", - "nodebb-theme-vanilla": "6.0.23", + "nodebb-theme-vanilla": "6.0.24", "nodebb-widget-essentials": "3.0.1", "nodemailer": "2.6.4", "nodemailer-sendmail-transport": "1.0.0", diff --git a/src/controllers/accounts/helpers.js b/src/controllers/accounts/helpers.js index 5bd65a33af..e35ac514d9 100644 --- a/src/controllers/accounts/helpers.js +++ b/src/controllers/accounts/helpers.js @@ -182,9 +182,8 @@ function filterLinks(links, states) { admin: true, }, link.visibility); - // Iterate through states and permit if every test passes (or is not defined) var permit = Object.keys(states).some(function (state) { - return states[state] === link.visibility[state]; + return states[state] && link.visibility[state]; }); links[index].public = permit; diff --git a/src/privileges/posts.js b/src/privileges/posts.js index 89ef1e0f48..f2bfe38428 100644 --- a/src/privileges/posts.js +++ b/src/privileges/posts.js @@ -10,6 +10,7 @@ var topics = require('../topics'); var user = require('../user'); var helpers = require('./helpers'); var plugins = require('../plugins'); +var utils = require('../utils'); module.exports = function (privileges) { privileges.posts = {}; @@ -190,6 +191,22 @@ module.exports = function (privileges) { ], callback); }; + privileges.posts.canFlag = function (pid, uid, callback) { + async.waterfall([ + function (next) { + async.parallel({ + userReputation: async.apply(user.getUserField, uid, 'reputation'), + isAdminOrMod: async.apply(isAdminOrMod, pid, uid), + }, next); + }, + function (results, next) { + var minimumReputation = utils.isNumber(meta.config['privileges:flag']) ? parseInt(meta.config['privileges:flag'], 10) : 1; + var canFlag = results.isAdminOrMod || parseInt(results.userReputation, 10) >= minimumReputation; + next(null, { flag: canFlag }); + }, + ], callback); + }; + privileges.posts.canMove = function (pid, uid, callback) { async.waterfall([ function (next) { diff --git a/src/socket.io/posts/tools.js b/src/socket.io/posts/tools.js index c075a96a8e..7f80ce9805 100644 --- a/src/socket.io/posts/tools.js +++ b/src/socket.io/posts/tools.js @@ -31,6 +31,9 @@ module.exports = function (SocketPosts) { canDelete: function (next) { privileges.posts.canDelete(data.pid, socket.uid, next); }, + canFlag: function (next) { + privileges.posts.canFlag(data.pid, socket.uid, next); + }, bookmarked: function (next) { posts.hasBookmarked(data.pid, socket.uid, next); }, @@ -49,6 +52,7 @@ module.exports = function (SocketPosts) { results.posts.selfPost = socket.uid && socket.uid === parseInt(results.posts.uid, 10); results.posts.display_edit_tools = results.canEdit.flag; results.posts.display_delete_tools = results.canDelete.flag; + results.posts.display_flag_tools = socket.uid && !results.posts.selfPost && results.canFlag.flag; results.posts.display_moderator_tools = results.posts.display_edit_tools || results.posts.display_delete_tools; results.posts.display_move_tools = results.isAdminOrMod; next(null, results);