mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-31 11:05:54 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			561 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			561 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 
 | |
| 'use strict';
 | |
| 
 | |
| var nconf = require('nconf'),
 | |
| 	async = require('async'),
 | |
| 	winston = require('winston'),
 | |
| 
 | |
| 	topics = require('../topics'),
 | |
| 	categories = require('../categories'),
 | |
| 	privileges = require('../privileges'),
 | |
| 	plugins = require('../plugins'),
 | |
| 	notifications = require('../notifications'),
 | |
| 	threadTools = require('../threadTools'),
 | |
| 	websockets = require('./index'),
 | |
| 	user = require('../user'),
 | |
| 	db = require('../database'),
 | |
| 	meta = require('../meta'),
 | |
| 	events = require('../events'),
 | |
| 	utils = require('../../public/src/utils'),
 | |
| 	SocketPosts = require('./posts'),
 | |
| 
 | |
| 	SocketTopics = {};
 | |
| 
 | |
| SocketTopics.post = function(socket, data, callback) {
 | |
| 	if(!data) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	topics.post({
 | |
| 		uid: socket.uid,
 | |
| 		handle: data.handle,
 | |
| 		title: data.title,
 | |
| 		content: data.content,
 | |
| 		cid: data.category_id,
 | |
| 		thumb: data.topic_thumb,
 | |
| 		tags: data.tags,
 | |
| 		req: websockets.reqFromSocket(socket)
 | |
| 	}, function(err, result) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		callback(null, result.topicData);
 | |
| 		socket.emit('event:new_post', {posts: [result.postData]});
 | |
| 		socket.emit('event:new_topic', result.topicData);
 | |
| 
 | |
| 		async.waterfall([
 | |
| 			function(next) {
 | |
| 				user.getUidsFromSet('users:online', 0, -1, next);
 | |
| 			},
 | |
| 			function(uids, next) {
 | |
| 				privileges.categories.filterUids('read', result.topicData.cid, uids, next);
 | |
| 			},
 | |
| 			function(uids, next) {
 | |
| 				plugins.fireHook('filter:sockets.sendNewPostToUids', {uidsTo: uids, uidFrom: data.uid, type: 'newTopic'}, next);
 | |
| 			}
 | |
| 		], function(err, data) {
 | |
| 			if (err) {
 | |
| 				return winston.error(err.stack);
 | |
| 			}
 | |
| 
 | |
| 			var uids = data.uidsTo;
 | |
| 
 | |
| 			for(var i=0; i<uids.length; ++i) {
 | |
| 				if (parseInt(uids[i], 10) !== socket.uid) {
 | |
| 					websockets.in('uid_' + uids[i]).emit('event:new_post', {posts: [result.postData]});
 | |
| 					websockets.in('uid_' + uids[i]).emit('event:new_topic', result.topicData);
 | |
| 				}
 | |
| 			}
 | |
| 		});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.enter = function(socket, tid, callback) {
 | |
| 	if (!parseInt(tid, 10) || !socket.uid) {
 | |
| 		return;
 | |
| 	}
 | |
| 	async.parallel({
 | |
| 		markAsRead: function(next) {
 | |
| 			SocketTopics.markAsRead(socket, [tid], next);
 | |
| 		},
 | |
| 		users: function(next) {
 | |
| 			websockets.getUsersInRoom(socket.uid, 'topic_' + tid, next);
 | |
| 		}
 | |
| 	}, function(err, result) {
 | |
| 		callback(err, result ? result.users : null);
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.postcount = function(socket, tid, callback) {
 | |
| 	topics.getTopicField(tid, 'postcount', callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.markAsRead = function(socket, tids, callback) {
 | |
| 	if(!Array.isArray(tids) || !socket.uid) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	if (!tids.length) {
 | |
| 		return callback();
 | |
| 	}
 | |
| 	tids = tids.filter(function(tid) {
 | |
| 		return tid && utils.isNumber(tid);
 | |
| 	});
 | |
| 
 | |
| 	topics.markAsRead(tids, socket.uid, function(err) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		topics.pushUnreadCount(socket.uid);
 | |
| 
 | |
| 		for (var i=0; i<tids.length; ++i) {
 | |
| 			topics.markTopicNotificationsRead(tids[i], socket.uid);
 | |
| 		}
 | |
| 		callback();
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.markTopicNotificationsRead = function(socket, tid, callback) {
 | |
| 	if(!tid || !socket.uid) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 	topics.markTopicNotificationsRead(tid, socket.uid);
 | |
| };
 | |
| 
 | |
| SocketTopics.markAllRead = function(socket, data, callback) {
 | |
| 	topics.getLatestTidsFromSet('topics:recent', 0, -1, 'day', function(err, tids) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		SocketTopics.markAsRead(socket, tids, callback);
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.markCategoryTopicsRead = function(socket, cid, callback) {
 | |
| 	topics.getUnreadTids(socket.uid, 0, -1, function(err, tids) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		topics.getTopicsFields(tids, ['tid', 'cid'], function(err, topicData) {
 | |
| 			if (err) {
 | |
| 				return callback(err);
 | |
| 			}
 | |
| 
 | |
| 			tids = topicData.filter(function(topic) {
 | |
| 				return topic && parseInt(topic.cid, 10) === parseInt(cid, 10);
 | |
| 			}).map(function(topic) {
 | |
| 				return topic.tid;
 | |
| 			});
 | |
| 
 | |
| 			SocketTopics.markAsRead(socket, tids, callback);
 | |
| 		});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.markAsUnreadForAll = function(socket, tids, callback) {
 | |
| 	if (!Array.isArray(tids)) {
 | |
| 		return callback(new Error('[[error:invalid-tid]]'));
 | |
| 	}
 | |
| 
 | |
| 	if (!socket.uid) {
 | |
| 		return callback(new Error('[[error:no-privileges]]'));
 | |
| 	}
 | |
| 
 | |
| 	user.isAdministrator(socket.uid, function(err, isAdmin) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		async.each(tids, function(tid, next) {
 | |
| 			async.waterfall([
 | |
| 				function(next) {
 | |
| 					topics.exists(tid, next);
 | |
| 				},
 | |
| 				function(exists, next) {
 | |
| 					if (!exists) {
 | |
| 						return next(new Error('[[error:invalid-tid]]'));
 | |
| 					}
 | |
| 					topics.getTopicField(tid, 'cid', next);
 | |
| 				},
 | |
| 				function(cid, next) {
 | |
| 					user.isModerator(socket.uid, cid, next);
 | |
| 				},
 | |
| 				function(isMod, next) {
 | |
| 					if (!isAdmin && !isMod) {
 | |
| 						return next(new Error('[[error:no-privileges]]'));
 | |
| 					}
 | |
| 					topics.markAsUnreadForAll(tid, next);
 | |
| 				},
 | |
| 				function(next) {
 | |
| 					topics.updateRecent(tid, Date.now(), next);
 | |
| 				}
 | |
| 			], next);
 | |
| 		}, function(err) {
 | |
| 			if (err) {
 | |
| 				return callback(err);
 | |
| 			}
 | |
| 			topics.pushUnreadCount(socket.uid);
 | |
| 		});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.delete = function(socket, data, callback) {
 | |
| 	doTopicAction('delete', socket, data, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.restore = function(socket, data, callback) {
 | |
| 	doTopicAction('restore', socket, data, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.purge = function(socket, data, callback) {
 | |
| 	doTopicAction('purge', socket, data, function(err) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		websockets.in('category_' + data.cid).emit('event:topic_purged', data.tids);
 | |
| 		async.each(data.tids, function(tid, next) {
 | |
| 			websockets.in('topic_' + tid).emit('event:topic_purged', tid);
 | |
| 			next();
 | |
| 		}, callback);
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.lock = function(socket, data, callback) {
 | |
| 	doTopicAction('lock', socket, data, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.unlock = function(socket, data, callback) {
 | |
| 	doTopicAction('unlock', socket, data, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.pin = function(socket, data, callback) {
 | |
| 	doTopicAction('pin', socket, data, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.unpin = function(socket, data, callback) {
 | |
| 	doTopicAction('unpin', socket, data, callback);
 | |
| };
 | |
| 
 | |
| function doTopicAction(action, socket, data, callback) {
 | |
| 	if (!socket.uid) {
 | |
| 		return;
 | |
| 	}
 | |
| 	if(!data || !Array.isArray(data.tids) || !data.cid) {
 | |
| 		return callback(new Error('[[error:invalid-tid]]'));
 | |
| 	}
 | |
| 
 | |
| 	async.each(data.tids, function(tid, next) {
 | |
| 		privileges.topics.canEdit(tid, socket.uid, function(err, canEdit) {
 | |
| 			if(err) {
 | |
| 				return next(err);
 | |
| 			}
 | |
| 
 | |
| 			if(!canEdit) {
 | |
| 				return next(new Error('[[error:no-privileges]]'));
 | |
| 			}
 | |
| 
 | |
| 			if(typeof threadTools[action] === 'function') {
 | |
| 				threadTools[action](tid, socket.uid, function(err) {
 | |
| 					if (err) {
 | |
| 						return next(err);
 | |
| 					}
 | |
| 
 | |
| 					if (action === 'delete' || action === 'restore' || action === 'purge') {
 | |
| 						events.log({
 | |
| 							type: 'topic-' + action,
 | |
| 							uid: socket.uid,
 | |
| 							ip: socket.ip,
 | |
| 							tid: tid
 | |
| 						});
 | |
| 					}
 | |
| 					next();
 | |
| 				});
 | |
| 			}
 | |
| 		});
 | |
| 	}, callback);
 | |
| }
 | |
| 
 | |
| SocketTopics.createTopicFromPosts = function(socket, data, callback) {
 | |
| 	if(!socket.uid) {
 | |
| 		return callback(new Error('[[error:not-logged-in]]'));
 | |
| 	}
 | |
| 
 | |
| 	if(!data || !data.title || !data.pids || !Array.isArray(data.pids)) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	topics.createTopicFromPosts(socket.uid, data.title, data.pids, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.movePost = function(socket, data, callback) {
 | |
| 	if (!socket.uid) {
 | |
| 		return callback(new Error('[[error:not-logged-in]]'));
 | |
| 	}
 | |
| 
 | |
| 	if (!data || !data.pid || !data.tid) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	privileges.posts.canMove(data.pid, socket.uid, function(err, canMove) {
 | |
| 		if (err || !canMove) {
 | |
| 			return callback(err || new Error('[[error:no-privileges]]'));
 | |
| 		}
 | |
| 
 | |
| 		topics.movePostToTopic(data.pid, data.tid, function(err) {
 | |
| 			if (err) {
 | |
| 				return callback(err);
 | |
| 			}
 | |
| 
 | |
| 			SocketPosts.sendNotificationToPostOwner(data.pid, socket.uid, 'notifications:moved_your_post');
 | |
| 			callback();
 | |
| 		});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.move = function(socket, data, callback) {
 | |
| 	if(!data || !Array.isArray(data.tids) || !data.cid) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	async.eachLimit(data.tids, 10, function(tid, next) {
 | |
| 		var oldCid;
 | |
| 		async.waterfall([
 | |
| 			function(next) {
 | |
| 				privileges.topics.canMove(tid, socket.uid, next);
 | |
| 			},
 | |
| 			function(canMove, next) {
 | |
| 				if (!canMove) {
 | |
| 					return next(new Error('[[error:no-privileges]]'));
 | |
| 				}
 | |
| 				next();
 | |
| 			},
 | |
| 			function(next) {
 | |
| 				topics.getTopicField(tid, 'cid', next);
 | |
| 			},
 | |
| 			function(cid, next) {
 | |
| 				oldCid = cid;
 | |
| 				threadTools.move(tid, data.cid, socket.uid, next);
 | |
| 			}
 | |
| 		], function(err) {
 | |
| 			if(err) {
 | |
| 				return next(err);
 | |
| 			}
 | |
| 
 | |
| 			websockets.in('topic_' + tid).emit('event:topic_moved', {
 | |
| 				tid: tid
 | |
| 			});
 | |
| 
 | |
| 			websockets.in('category_' + oldCid).emit('event:topic_moved', {
 | |
| 				tid: tid
 | |
| 			});
 | |
| 
 | |
| 			SocketTopics.sendNotificationToTopicOwner(tid, socket.uid, 'notifications:moved_your_topic');
 | |
| 
 | |
| 			next();
 | |
| 		});
 | |
| 	}, callback);
 | |
| };
 | |
| 
 | |
| 
 | |
| SocketTopics.sendNotificationToTopicOwner = function(tid, fromuid, notification) {
 | |
| 	if(!tid || !fromuid) {
 | |
| 		return;
 | |
| 	}
 | |
| 
 | |
| 	async.parallel({
 | |
| 		username: async.apply(user.getUserField, fromuid, 'username'),
 | |
| 		topicData: async.apply(topics.getTopicFields, tid, ['uid', 'slug']),
 | |
| 	}, function(err, results) {
 | |
| 		if (err || fromuid === parseInt(results.topicData.uid, 10)) {
 | |
| 			return;
 | |
| 		}
 | |
| 
 | |
| 		notifications.create({
 | |
| 			bodyShort: '[[' + notification + ', ' + results.username + ']]',
 | |
| 			path: nconf.get('relative_path') + '/topic/' + results.topicData.slug,
 | |
| 			nid: 'tid:' + tid + ':uid:' + fromuid,
 | |
| 			from: fromuid
 | |
| 		}, function(err, notification) {
 | |
| 			if (!err && notification) {
 | |
| 				notifications.push(notification, [results.topicData.uid]);
 | |
| 			}
 | |
| 		});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| 
 | |
| SocketTopics.moveAll = function(socket, data, callback) {
 | |
| 	if(!data || !data.cid || !data.currentCid) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	privileges.categories.canMoveAllTopics(data.currentCid, data.cid, data.uid, function(err, canMove) {
 | |
| 		if (err || canMove) {
 | |
| 			return callback(err || new Error('[[error:no-privileges]]'));
 | |
| 		}
 | |
| 
 | |
| 		categories.getTopicIds('cid:' + data.currentCid + ':tids', true, 0, -1, function(err, tids) {
 | |
| 			if (err) {
 | |
| 				return callback(err);
 | |
| 			}
 | |
| 
 | |
| 			async.eachLimit(tids, 10, function(tid, next) {
 | |
| 				threadTools.move(tid, data.cid, socket.uid, next);
 | |
| 			}, callback);
 | |
| 		});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.toggleFollow = function(socket, tid, callback) {
 | |
| 	if(!socket.uid) {
 | |
| 		return callback(new Error('[[error:not-logged-in]]'));
 | |
| 	}
 | |
| 
 | |
| 	topics.toggleFollow(tid, socket.uid, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.follow = function(socket, tid, callback) {
 | |
| 	if(!socket.uid) {
 | |
| 		return callback(new Error('[[error:not-logged-in]]'));
 | |
| 	}
 | |
| 
 | |
| 	topics.follow(tid, socket.uid, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.loadMore = function(socket, data, callback) {
 | |
| 	if(!data || !data.tid || !utils.isNumber(data.after) || parseInt(data.after, 10) < 0)  {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	async.parallel({
 | |
| 		settings: function(next) {
 | |
| 			user.getSettings(socket.uid, next);
 | |
| 		},
 | |
| 		privileges: function(next) {
 | |
| 			privileges.topics.get(data.tid, socket.uid, next);
 | |
| 		},
 | |
| 		postCount: function(next) {
 | |
| 			topics.getPostCount(data.tid, next);
 | |
| 		}
 | |
| 	}, function(err, results) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		if (!results.privileges.read) {
 | |
| 			return callback(new Error('[[error:no-privileges]]'));
 | |
| 		}
 | |
| 
 | |
| 		var set = 'tid:' + data.tid + ':posts',
 | |
| 			reverse = false,
 | |
| 			start = Math.max(parseInt(data.after, 10) - 1, 0);
 | |
| 
 | |
| 		if (results.settings.topicPostSort === 'newest_to_oldest' || results.settings.topicPostSort === 'most_votes') {
 | |
| 			reverse = true;
 | |
| 			data.after = results.postCount - 1 - data.after;
 | |
| 			start = Math.max(parseInt(data.after, 10), 0);
 | |
| 			if (results.settings.topicPostSort === 'most_votes') {
 | |
| 				set = 'tid:' + data.tid + ':posts:votes';
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		var end = start + results.settings.postsPerPage - 1;
 | |
| 
 | |
| 		async.parallel({
 | |
| 			posts: function(next) {
 | |
| 				topics.getTopicPosts(data.tid, set, start, end, socket.uid, reverse, next);
 | |
| 			},
 | |
| 			privileges: function(next) {
 | |
| 				next(null, results.privileges);
 | |
| 			},
 | |
| 			'reputation:disabled': function(next) {
 | |
| 				next(null, parseInt(meta.config['reputation:disabled'], 10) === 1);
 | |
| 			},
 | |
| 			'downvote:disabled': function(next) {
 | |
| 				next(null, parseInt(meta.config['downvote:disabled'], 10) === 1);
 | |
| 			}
 | |
| 		}, callback);
 | |
| 	});
 | |
| };
 | |
| 
 | |
| SocketTopics.loadMoreUnreadTopics = function(socket, data, callback) {
 | |
| 	if(!data || !data.after) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	var start = parseInt(data.after, 10),
 | |
| 		end = start + 9;
 | |
| 
 | |
| 	topics.getUnreadTopics(socket.uid, start, end, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.loadMoreFromSet = function(socket, data, callback) {
 | |
| 	if(!data || !data.after || !data.set) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	var start = parseInt(data.after, 10),
 | |
| 		end = start + 9;
 | |
| 
 | |
| 	topics.getTopicsFromSet(data.set, socket.uid, start, end, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.loadTopics = function(socket, data, callback) {
 | |
| 	if(!data || !data.set || !utils.isNumber(data.start) || !utils.isNumber(data.end)) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	topics.getTopicsFromSet(data.set, socket.uid, data.start, data.end, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.getPageCount = function(socket, tid, callback) {
 | |
| 	topics.getPageCount(tid, socket.uid, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.getTidPage = function(socket, tid, callback) {
 | |
| 	topics.getTidPage(tid, socket.uid, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.getTidIndex = function(socket, tid, callback) {
 | |
| 	categories.getTopicIndex(tid, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.searchTags = function(socket, data, callback) {
 | |
| 	topics.searchTags(data, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.search = function(socket, data, callback) {
 | |
| 	topics.search(data.tid, data.term, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.searchAndLoadTags = function(socket, data, callback) {
 | |
| 	if (!data) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 	topics.searchAndLoadTags(data, callback);
 | |
| };
 | |
| 
 | |
| SocketTopics.loadMoreTags = function(socket, data, callback) {
 | |
| 	if(!data || !utils.isNumber(data.after)) {
 | |
| 		return callback(new Error('[[error:invalid-data]]'));
 | |
| 	}
 | |
| 
 | |
| 	var start = parseInt(data.after, 10),
 | |
| 		end = start + 99;
 | |
| 
 | |
| 	topics.getTags(start, end, function(err, tags) {
 | |
| 		if (err) {
 | |
| 			return callback(err);
 | |
| 		}
 | |
| 
 | |
| 		callback(null, {tags: tags, nextStart: end + 1});
 | |
| 	});
 | |
| };
 | |
| 
 | |
| module.exports = SocketTopics;
 |