mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-26 16:46:12 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			189 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			189 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| 'use strict';
 | |
| 
 | |
| const winston = require('winston');
 | |
| 
 | |
| const db = require('../database');
 | |
| const topics = require('../topics');
 | |
| const posts = require('../posts');
 | |
| const utils = require('../utils');
 | |
| const pubsub = require('../pubsub');
 | |
| const slugify = require('../slugify');
 | |
| 
 | |
| const activitypub = module.parent.exports;
 | |
| const Notes = module.exports;
 | |
| 
 | |
| Notes.resolveId = async (uid, id) => {
 | |
| 	({ id } = await activitypub.get('uid', uid, id));
 | |
| 	return id;
 | |
| };
 | |
| 
 | |
| // todo: when asserted, notes aren't added to a global sorted set
 | |
| // also, db.exists call is probably expensive
 | |
| Notes.assert = async (uid, input, options = {}) => {
 | |
| 	// Ensures that each note has been saved to the database
 | |
| 	await Promise.all(input.map(async (item) => {
 | |
| 		const id = activitypub.helpers.isUri(item) ? item : item.pid;
 | |
| 		const key = `post:${id}`;
 | |
| 		const exists = await db.exists(key);
 | |
| 		winston.verbose(`[activitypub/notes.assert] Asserting note id ${id}`);
 | |
| 
 | |
| 		if (!exists || options.update === true) {
 | |
| 			let postData;
 | |
| 			winston.verbose(`[activitypub/notes.assert] Not found, retrieving note for persistence...`);
 | |
| 			if (activitypub.helpers.isUri(item)) {
 | |
| 				// get failure throws for now but should save intermediate object
 | |
| 				const object = await activitypub.get('uid', uid, item);
 | |
| 				postData = await activitypub.mocks.post(object);
 | |
| 			} else {
 | |
| 				postData = item;
 | |
| 			}
 | |
| 
 | |
| 			await db.setObject(key, postData);
 | |
| 			winston.verbose(`[activitypub/notes.assert] Note ${id} saved.`);
 | |
| 		}
 | |
| 
 | |
| 		if (options.update === true) {
 | |
| 			require('../posts/cache').del(String(id));
 | |
| 			pubsub.publish('post:edit', String(id));
 | |
| 		}
 | |
| 	}));
 | |
| };
 | |
| 
 | |
| Notes.getParentChain = async (uid, input) => {
 | |
| 	// Traverse upwards via `inReplyTo` until you find the root-level Note
 | |
| 	const id = activitypub.helpers.isUri(input) ? input : input.id;
 | |
| 
 | |
| 	const chain = new Set();
 | |
| 	const traverse = async (uid, id) => {
 | |
| 		// Handle remote reference to local post
 | |
| 		const { type, id: localId } = await activitypub.helpers.resolveLocalId(id);
 | |
| 		if (type === 'post' && localId) {
 | |
| 			return traverse(uid, localId);
 | |
| 		}
 | |
| 
 | |
| 		const exists = await db.exists(`post:${id}`);
 | |
| 		if (exists) {
 | |
| 			const postData = await posts.getPostData(id);
 | |
| 			chain.add(postData);
 | |
| 			if (postData.toPid) {
 | |
| 				await traverse(uid, postData.toPid);
 | |
| 			}
 | |
| 		} else {
 | |
| 			let object;
 | |
| 			try {
 | |
| 				object = await activitypub.get('uid', uid, id);
 | |
| 			} catch (e) {
 | |
| 				winston.warn(`[activitypub/notes/getParentChain] Cannot retrieve ${id}, terminating here.`);
 | |
| 			}
 | |
| 
 | |
| 			// Handle incorrect id passed in
 | |
| 			if (id !== object.id) {
 | |
| 				return await traverse(uid, object.id);
 | |
| 			}
 | |
| 
 | |
| 			object = await activitypub.mocks.post(object);
 | |
| 			if (object) {
 | |
| 				chain.add(object);
 | |
| 				if (object.toPid) {
 | |
| 					await traverse(uid, object.toPid);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	};
 | |
| 
 | |
| 	await traverse(uid, id);
 | |
| 	return chain;
 | |
| };
 | |
| 
 | |
| Notes.assertParentChain = async (chain, tid) => {
 | |
| 	const data = [];
 | |
| 	chain.reduce((child, parent) => {
 | |
| 		data.push([`pid:${parent.pid}:replies`, child.timestamp, child.pid]);
 | |
| 		return parent;
 | |
| 	});
 | |
| 
 | |
| 	await Promise.all([
 | |
| 		db.sortedSetAddBulk(data),
 | |
| 		db.setObjectBulk(chain.map(post => [`post:${post.pid}`, { tid }])),
 | |
| 	]);
 | |
| };
 | |
| 
 | |
| Notes.assertTopic = async (uid, id) => {
 | |
| 	/**
 | |
| 	 * Given the id of any post, traverses up to cache the entire threaded context
 | |
| 	 *
 | |
| 	 * Unfortunately, due to limitations and fragmentation of the existing ActivityPub landscape,
 | |
| 	 * retrieving the entire reply tree is not possible at this time.
 | |
| 	 */
 | |
| 
 | |
| 	const chain = Array.from(await Notes.getParentChain(uid, id));
 | |
| 	let { pid: mainPid, tid, uid: authorId, timestamp, name, content } = chain[chain.length - 1];
 | |
| 	const members = await db.isSortedSetMembers(`tid:${tid}:posts`, chain.map(p => p.pid));
 | |
| 	if (tid && members.every(Boolean)) {
 | |
| 		// All cached, return early.
 | |
| 		winston.info('[notes/assertTopic] No new notes to process.');
 | |
| 		return tid;
 | |
| 	}
 | |
| 
 | |
| 	tid = tid || utils.generateUUID();
 | |
| 	const cid = await topics.getTopicField(tid, 'cid');
 | |
| 
 | |
| 	let title = name || utils.decodeHTMLEntities(utils.stripHTMLTags(content));
 | |
| 	if (title.length > 64) {
 | |
| 		title = `${title.slice(0, 64)}...`;
 | |
| 	}
 | |
| 
 | |
| 	const unprocessed = chain.filter((p, idx) => !members[idx]);
 | |
| 	winston.info(`[notes/assertTopic] ${unprocessed.length} new note(s) found.`);
 | |
| 
 | |
| 	const [ids, timestamps] = [
 | |
| 		unprocessed.map(n => n.pid),
 | |
| 		unprocessed.map(n => n.timestamp),
 | |
| 	];
 | |
| 
 | |
| 	// mainPid doesn't belong in posts zset
 | |
| 	ids.pop();
 | |
| 	timestamps.pop();
 | |
| 
 | |
| 	await Promise.all([
 | |
| 		db.setObject(`topic:${tid}`, {
 | |
| 			tid,
 | |
| 			uid: authorId,
 | |
| 			cid: cid || -1,
 | |
| 			mainPid,
 | |
| 			title,
 | |
| 			slug: `${tid}/${slugify(title)}`,
 | |
| 			timestamp,
 | |
| 		}),
 | |
| 		db.sortedSetAdd(`tid:${tid}:posts`, timestamps, ids),
 | |
| 		Notes.assert(uid, unprocessed),
 | |
| 	]);
 | |
| 	await Promise.all([ // must be done after .assert()
 | |
| 		Notes.assertParentChain(chain, tid),
 | |
| 		Notes.updateTopicCounts(tid),
 | |
| 		topics.updateLastPostTimeFromLastPid(tid),
 | |
| 		topics.updateTeaser(tid),
 | |
| 	]);
 | |
| 
 | |
| 	return tid;
 | |
| };
 | |
| 
 | |
| Notes.updateTopicCounts = async function (tid) {
 | |
| 	const pids = await db.getSortedSetMembers(`tid:${tid}:posts`);
 | |
| 	let uids = await db.getObjectsFields(pids.map(p => `post:${p}`), ['uid']);
 | |
| 	uids = uids.reduce((set, { uid }) => {
 | |
| 		set.add(uid);
 | |
| 		return set;
 | |
| 	}, new Set());
 | |
| 
 | |
| 	db.setObject(`topic:${tid}`, {
 | |
| 		postercount: uids.size,
 | |
| 		postcount: pids.length,
 | |
| 	});
 | |
| };
 | |
| 
 | |
| Notes.getTopicPosts = async (tid, uid, start, stop) => {
 | |
| 	const pids = await db.getSortedSetRange(`tid:${tid}:posts`, start, stop);
 | |
| 	return await posts.getPostsByPids(pids, uid);
 | |
| };
 |