mirror of
				https://github.com/NodeBB/NodeBB.git
				synced 2025-10-26 08:36:12 +01:00 
			
		
		
		
	feat: Announce(Note) and Undo(Announce)
This commit is contained in:
		| @@ -5,6 +5,7 @@ const winston = require('winston'); | ||||
| const db = require('../database'); | ||||
| const user = require('../user'); | ||||
| const posts = require('../posts'); | ||||
| const topics = require('../topics'); | ||||
| const categories = require('../categories'); | ||||
| const activitypub = require('.'); | ||||
|  | ||||
| @@ -65,6 +66,46 @@ inbox.like = async (req) => { | ||||
| 	await posts.upvote(id, actor); | ||||
| }; | ||||
|  | ||||
| inbox.announce = async (req) => { | ||||
| 	const { actor, object, published } = req.body; | ||||
| 	let timestamp = Date.now(); | ||||
| 	try { | ||||
| 		timestamp = new Date(published).getTime(); | ||||
| 	} catch (e) { | ||||
| 		// ok to fail | ||||
| 	} | ||||
|  | ||||
| 	const { type, id } = await activitypub.helpers.resolveLocalId(object); | ||||
| 	if (type !== 'post' || !(await posts.exists(id))) { | ||||
| 		throw new Error('[[error:activitypub.invalid-id]]'); | ||||
| 	} | ||||
|  | ||||
| 	const assertion = await activitypub.actors.assert(actor); | ||||
| 	if (!assertion) { | ||||
| 		throw new Error('[[error:activitypub.invalid-id]]'); | ||||
| 	} | ||||
|  | ||||
| 	const tid = await posts.getPostField(id, 'tid'); | ||||
|  | ||||
| 	// No double-announce allowed | ||||
| 	const existing = await topics.events.find(tid, { | ||||
| 		type: 'announce', | ||||
| 		uid: actor, | ||||
| 		pid: id, | ||||
| 	}); | ||||
| 	if (existing.length) { | ||||
| 		await topics.events.purge(tid, existing); | ||||
| 	} | ||||
|  | ||||
| 	await topics.events.log(tid, { | ||||
| 		type: 'announce', | ||||
| 		uid: actor, | ||||
| 		href: `/post/${id}`, | ||||
| 		pid: id, | ||||
| 		timestamp, | ||||
| 	}); | ||||
| }; | ||||
|  | ||||
| inbox.follow = async (req) => { | ||||
| 	// Sanity checks | ||||
| 	const { type, id } = await helpers.resolveLocalId(req.body.object); | ||||
| @@ -157,6 +198,10 @@ inbox.undo = async (req) => { | ||||
| 	const { actor, object } = req.body; | ||||
| 	const { type } = object; | ||||
|  | ||||
| 	if (actor !== object.actor) { | ||||
| 		throw new Error('[[error:activitypub.actor-mismatch]]'); | ||||
| 	} | ||||
|  | ||||
| 	const assertion = await activitypub.actors.assert(actor); | ||||
| 	if (!assertion) { | ||||
| 		throw new Error('[[error:activitypub.invalid-id]]'); | ||||
| @@ -202,5 +247,23 @@ inbox.undo = async (req) => { | ||||
| 			await posts.unvote(id, actor); | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		case 'Announce': { | ||||
| 			const exists = await posts.exists(id); | ||||
| 			if (localType !== 'post' || !exists) { | ||||
| 				throw new Error('[[error:invalid-pid]]'); | ||||
| 			} | ||||
|  | ||||
| 			const tid = await posts.getPostField(id, 'tid'); | ||||
| 			const existing = await topics.events.find(tid, { | ||||
| 				type: 'announce', | ||||
| 				uid: actor, | ||||
| 				pid: id, | ||||
| 			}); | ||||
|  | ||||
| 			if (existing.length) { | ||||
| 				await topics.events.purge(tid, existing); | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| }; | ||||
|   | ||||
| @@ -237,3 +237,15 @@ ActivityPub.send = async (type, id, targets, payload) => { | ||||
| 		} | ||||
| 	})); | ||||
| }; | ||||
|  | ||||
| setTimeout(async () => { | ||||
| 	await ActivityPub.send('uid', 1, 'https://localhost/uid/1', { | ||||
| 		// type: 'Undo', | ||||
| 		// object: { | ||||
| 		type: 'Announce', | ||||
| 		actor: `https://localhost/uid/1`, | ||||
| 		object: 'https://localhost/post/1', | ||||
| 		published: new Date().toISOString(), | ||||
| 		// }, | ||||
| 	}); | ||||
| }, 2000); | ||||
|   | ||||
| @@ -112,42 +112,11 @@ Controller.getInbox = async (req, res) => { | ||||
|  | ||||
| Controller.postInbox = async (req, res) => { | ||||
| 	// Note: underlying methods are internal use only, hence no exposure via src/api | ||||
| 	switch (req.body.type) { | ||||
| 		case 'Create': { | ||||
| 			await activitypub.inbox.create(req); | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		case 'Update': { | ||||
| 			await activitypub.inbox.update(req); | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		case 'Like': { | ||||
| 			await activitypub.inbox.like(req); | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		case 'Follow': { | ||||
| 			await activitypub.inbox.follow(req); | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		case 'Accept': { | ||||
| 			await activitypub.inbox.accept(req); | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		case 'Undo': { | ||||
| 			await activitypub.inbox.undo(req); | ||||
| 			break; | ||||
| 		} | ||||
|  | ||||
| 		default: { | ||||
| 			res.sendStatus(501); | ||||
| 			break; | ||||
| 		} | ||||
| 	const method = String(req.body.type).toLowerCase(); | ||||
| 	if (req.body.hasOwnProperty(method)) { | ||||
| 		return res.sendStatus(501); | ||||
| 	} | ||||
|  | ||||
| 	await activitypub.inbox[method](req); | ||||
| 	res.sendStatus(200); | ||||
| }; | ||||
|   | ||||
| @@ -68,6 +68,10 @@ Events._types = { | ||||
| 		icon: 'fa-code-fork', | ||||
| 		translation: async (event, language) => translateEventArgs(event, language, 'topic:user-forked-topic', renderUser(event), `${relative_path}${event.href}`, renderTimeago(event)), | ||||
| 	}, | ||||
| 	announce: { | ||||
| 		icon: 'fa-share-alt', | ||||
| 		translation: async (event, language) => translateEventArgs(event, language, 'activitypub:topic-event-announce', renderUser(event), `${relative_path}${event.href}`, renderTimeago(event)), | ||||
| 	}, | ||||
| }; | ||||
|  | ||||
| Events.init = async () => { | ||||
| @@ -199,8 +203,9 @@ async function modifyEvent({ tid, uid, eventIds, timestamps, events }) { | ||||
| 		event.id = parseInt(eventIds[idx], 10); | ||||
| 		event.timestamp = timestamps[idx]; | ||||
| 		event.timestampISO = new Date(timestamps[idx]).toISOString(); | ||||
| 		event.uid = utils.isNumber(event.uid) ? parseInt(event.uid, 10) : event.uid; | ||||
| 		if (event.hasOwnProperty('uid')) { | ||||
| 			event.user = users.get(event.uid === 'system' ? 'system' : parseInt(event.uid, 10)); | ||||
| 			event.user = users.get(event.uid === 'system' ? 'system' : event.uid); | ||||
| 		} | ||||
| 		if (event.hasOwnProperty('fromCid')) { | ||||
| 			event.fromCategory = fromCategories[event.fromCid]; | ||||
|   | ||||
		Reference in New Issue
	
	Block a user