fix: remove unused data from post/topic/user hashes

This commit is contained in:
Barış Soner Uşaklı
2019-10-07 23:13:43 -04:00
parent 4bc77d064b
commit 75bcb0f484
8 changed files with 156 additions and 39 deletions

View File

@@ -237,8 +237,6 @@ function continueLogin(req, res, next) {
return helpers.noScriptErrors(req, res, info, 403); return helpers.noScriptErrors(req, res, info, 403);
} }
var passwordExpiry = userData.passwordExpiry !== undefined ? parseInt(userData.passwordExpiry, 10) : null;
// Alter user cookie depending on passed-in option // Alter user cookie depending on passed-in option
if (req.body.remember === 'on') { if (req.body.remember === 'on') {
var duration = 1000 * 60 * 60 * 24 * meta.config.loginDays; var duration = 1000 * 60 * 60 * 24 * meta.config.loginDays;
@@ -249,7 +247,7 @@ function continueLogin(req, res, next) {
req.session.cookie.expires = false; req.session.cookie.expires = false;
} }
if (passwordExpiry && passwordExpiry < Date.now()) { if (userData.passwordExpiry && userData.passwordExpiry < Date.now()) {
winston.verbose('[auth] Triggering password reset for uid ' + userData.uid + ' due to password policy'); winston.verbose('[auth] Triggering password reset for uid ' + userData.uid + ' due to password policy');
req.session.passwordExpired = true; req.session.passwordExpired = true;

View File

@@ -35,7 +35,6 @@ module.exports = function (Posts) {
tid: tid, tid: tid,
content: content, content: content,
timestamp: timestamp, timestamp: timestamp,
deleted: 0,
}; };
if (data.toPid) { if (data.toPid) {

View File

@@ -0,0 +1,98 @@
'use strict';
const db = require('../../database');
const batch = require('../../batch');
const posts = require('../../posts');
const topics = require('../../topics');
module.exports = {
name: 'Clean up post hash data',
timestamp: Date.UTC(2019, 9, 7),
method: async function (callback) {
const progress = this.progress;
await cleanPost(progress);
await cleanTopic(progress);
callback();
},
};
async function cleanPost(progress) {
await batch.processSortedSet('posts:pid', async function (pids) {
progress.incr(pids.length);
const postData = await posts.getPostsData(pids);
await Promise.all(postData.map(async function (post) {
if (!post) {
return;
}
const fields = [];
if (post.editor === '') {
fields.push('editor');
}
if (post.deleted === 0) {
fields.push('deleted');
}
if (post.edited === 0) {
fields.push('edited');
}
// cleanup legacy fields, these are not used anymore
const legacyFields = [
'show_banned', 'fav_star_class', 'relativeEditTime',
'post_rep', 'relativeTime', 'fav_button_class',
'edited-class',
];
legacyFields.forEach((field) => {
if (post.hasOwnProperty(field)) {
fields.push(field);
}
});
if (fields.length) {
await db.deleteObjectFields('post:' + post.pid, fields);
}
}));
}, {
batch: 500,
progress: progress,
});
}
async function cleanTopic(progress) {
await batch.processSortedSet('topics:tid', async function (tids) {
progress.incr(tids.length);
const topicData = await topics.getTopicsData(tids);
await Promise.all(topicData.map(async function (topic) {
if (!topic) {
return;
}
const fields = [];
if (topic.deleted === 0) {
fields.push('deleted');
}
if (topic.pinned === 0) {
fields.push('pinned');
}
if (topic.locked === 0) {
fields.push('locked');
}
// cleanup legacy fields, these are not used anymore
const legacyFields = [
'category_name', 'category_slug',
];
legacyFields.forEach((field) => {
if (topic.hasOwnProperty(field)) {
fields.push(field);
}
});
if (fields.length) {
await db.deleteObjectFields('topic:' + topic.tid, fields);
}
}));
}, {
batch: 500,
progress: progress,
});
}

View File

@@ -2,9 +2,10 @@
const db = require('../../database'); const db = require('../../database');
const batch = require('../../batch'); const batch = require('../../batch');
const user = require('../../user');
module.exports = { module.exports = {
name: 'Clean up old notifications', name: 'Clean up old notifications and hash data',
timestamp: Date.UTC(2019, 9, 7), timestamp: Date.UTC(2019, 9, 7),
method: async function (callback) { method: async function (callback) {
const progress = this.progress; const progress = this.progress;
@@ -16,6 +17,32 @@ module.exports = {
db.sortedSetsRemoveRangeByScore(uids.map(uid => 'uid:' + uid + ':notifications:unread'), '-inf', cutoffTime), db.sortedSetsRemoveRangeByScore(uids.map(uid => 'uid:' + uid + ':notifications:unread'), '-inf', cutoffTime),
db.sortedSetsRemoveRangeByScore(uids.map(uid => 'uid:' + uid + ':notifications:read'), '-inf', cutoffTime), db.sortedSetsRemoveRangeByScore(uids.map(uid => 'uid:' + uid + ':notifications:read'), '-inf', cutoffTime),
]); ]);
const userData = await user.getUsersData(uids);
await Promise.all(userData.map(async function (user) {
if (!user) {
return;
}
const fields = [];
['picture', 'fullname', 'location', 'birthday', 'website', 'signature', 'uploadedpicture'].forEach((field) => {
if (user[field] === '') {
fields.push(field);
}
});
['profileviews', 'reputation', 'postcount', 'topiccount', 'lastposttime', 'banned'].forEach((field) => {
if (user[field] === 0) {
fields.push(field);
}
});
if (user['icon:text']) {
fields.push('icon:text');
}
if (user['icon:bgColor']) {
fields.push('icon:bgColor');
}
if (fields.length) {
await db.deleteObjectFields('user:' + user.uid, fields);
}
}));
}, { }, {
batch: 500, batch: 500,
progress: progress, progress: progress,

View File

@@ -24,23 +24,20 @@ module.exports = function (User) {
email: data.email || '', email: data.email || '',
joindate: timestamp, joindate: timestamp,
lastonline: timestamp, lastonline: timestamp,
picture: data.picture || '',
fullname: data.fullname || '',
location: data.location || '',
birthday: data.birthday || '',
website: '',
signature: '',
uploadedpicture: '',
profileviews: 0,
reputation: 0,
postcount: 0,
topiccount: 0,
lastposttime: 0,
banned: 0,
status: 'online', status: 'online',
gdpr_consent: data.gdpr_consent === true ? 1 : 0,
acceptTos: data.acceptTos === true ? 1 : 0,
}; };
['picture', 'fullname', 'location', 'birthday'].forEach((field) => {
if (data[field]) {
userData[field] = data[field];
}
});
if (data.gdpr_consent === true) {
userData.gdpr_consent = 1;
}
if (data.acceptTos === true) {
userData.acceptTos = 1;
}
const renamedUsername = await User.uniqueUsername(userData); const renamedUsername = await User.uniqueUsername(userData);
const userNameChanged = !!renamedUsername; const userNameChanged = !!renamedUsername;
if (userNameChanged) { if (userNameChanged) {

View File

@@ -12,7 +12,7 @@ const utils = require('../utils');
const intFields = [ const intFields = [
'uid', 'postcount', 'topiccount', 'reputation', 'profileviews', 'uid', 'postcount', 'topiccount', 'reputation', 'profileviews',
'banned', 'banned:expire', 'email:confirmed', 'joindate', 'lastonline', 'lastqueuetime', 'banned', 'banned:expire', 'email:confirmed', 'joindate', 'lastonline', 'lastqueuetime',
'lastposttime', 'followingCount', 'followerCount', 'lastposttime', 'followingCount', 'followerCount', 'passwordExpiry',
]; ];
module.exports = function (User) { module.exports = function (User) {

View File

@@ -86,7 +86,9 @@ UserReset.updateExpiry = async function (uid) {
const oneDay = 1000 * 60 * 60 * 24; const oneDay = 1000 * 60 * 60 * 24;
const expireDays = meta.config.passwordExpiryDays; const expireDays = meta.config.passwordExpiryDays;
const expiry = Date.now() + (oneDay * expireDays); const expiry = Date.now() + (oneDay * expireDays);
await user.setUserField(uid, 'passwordExpiry', expireDays > 0 ? expiry : 0); if (expireDays > 0) {
await user.setUserField(uid, 'passwordExpiry', expiry);
}
}; };
UserReset.clean = async function () { UserReset.clean = async function () {

View File

@@ -50,25 +50,21 @@ describe('User', function () {
describe('.create(), when created', function () { describe('.create(), when created', function () {
it('should be created properly', function (done) { it('should be created properly', async function () {
User.create({ username: userData.username, password: userData.password, email: userData.email }, function (error, userId) { testUid = await User.create({ username: userData.username, password: userData.password, email: userData.email });
assert.equal(error, null, 'was created with error'); assert.ok(testUid);
assert.ok(userId);
testUid = userId;
done();
});
}); });
it('should be created properly', function (done) { it('should be created properly', async function () {
User.create({ username: 'weirdemail', email: '<h1>test</h1>@gmail.com' }, function (err, uid) { const uid = await User.create({ username: 'weirdemail', email: '<h1>test</h1>@gmail.com' });
assert.ifError(err); const data = await User.getUserData(uid);
User.getUserData(uid, function (err, data) { assert.equal(data.email, '&lt;h1&gt;test&lt;&#x2F;h1&gt;@gmail.com');
assert.ifError(err); assert.strictEqual(data.profileviews, 0);
assert.equal(data.email, '&lt;h1&gt;test&lt;&#x2F;h1&gt;@gmail.com'); assert.strictEqual(data.reputation, 0);
done(); assert.strictEqual(data.postcount, 0);
}); assert.strictEqual(data.topiccount, 0);
}); assert.strictEqual(data.lastposttime, 0);
assert.strictEqual(data.banned, 0);
}); });
it('should have a valid email, if using an email', function (done) { it('should have a valid email, if using an email', function (done) {