mirror of
https://github.com/NodeBB/NodeBB.git
synced 2025-10-27 09:06:15 +01:00
feat: Create(Note) on new topic or reply
This is a naive WIP implementation that federates everything out publicly. It does not take category privileges into account!
This commit is contained in:
@@ -7,6 +7,7 @@ const { createHash, createSign, createVerify } = require('crypto');
|
|||||||
const request = require('../request');
|
const request = require('../request');
|
||||||
const db = require('../database');
|
const db = require('../database');
|
||||||
const user = require('../user');
|
const user = require('../user');
|
||||||
|
const posts = require('../posts');
|
||||||
const utils = require('../utils');
|
const utils = require('../utils');
|
||||||
const ttl = require('../cache/ttl');
|
const ttl = require('../cache/ttl');
|
||||||
|
|
||||||
@@ -14,6 +15,10 @@ const requestCache = ttl({ ttl: 1000 * 60 * 5 }); // 5 minutes
|
|||||||
const actorCache = ttl({ ttl: 1000 * 60 * 60 * 24 }); // 24 hours
|
const actorCache = ttl({ ttl: 1000 * 60 * 60 * 24 }); // 24 hours
|
||||||
const ActivityPub = module.exports;
|
const ActivityPub = module.exports;
|
||||||
|
|
||||||
|
ActivityPub._constants = Object.freeze({
|
||||||
|
publicAddress: 'https://www.w3.org/ns/activitystreams#Public',
|
||||||
|
});
|
||||||
|
|
||||||
ActivityPub.helpers = require('./helpers');
|
ActivityPub.helpers = require('./helpers');
|
||||||
ActivityPub.inbox = require('./inbox');
|
ActivityPub.inbox = require('./inbox');
|
||||||
ActivityPub.mocks = require('./mocks');
|
ActivityPub.mocks = require('./mocks');
|
||||||
@@ -59,10 +64,19 @@ ActivityPub.getActor = async (uid, input) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
ActivityPub.resolveInboxes = async (uid, ids) => await Promise.all(ids.map(async (id) => {
|
ActivityPub.resolveInboxes = async (uid, ids) => {
|
||||||
|
const inboxes = new Set();
|
||||||
|
|
||||||
|
await Promise.all(ids.map(async (id) => {
|
||||||
const actor = await ActivityPub.getActor(uid, id);
|
const actor = await ActivityPub.getActor(uid, id);
|
||||||
return actor.inbox;
|
const inbox = actor.sharedInbox || actor.inbox;
|
||||||
}));
|
if (inbox) {
|
||||||
|
inboxes.add(inbox);
|
||||||
|
}
|
||||||
|
}));
|
||||||
|
|
||||||
|
return Array.from(inboxes);
|
||||||
|
};
|
||||||
|
|
||||||
ActivityPub.getPublicKey = async (uid) => {
|
ActivityPub.getPublicKey = async (uid) => {
|
||||||
let publicKey;
|
let publicKey;
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ const nconf = require('nconf');
|
|||||||
const db = require('../database');
|
const db = require('../database');
|
||||||
const activitypub = require('../activitypub');
|
const activitypub = require('../activitypub');
|
||||||
const user = require('../user');
|
const user = require('../user');
|
||||||
|
const posts = require('../posts');
|
||||||
|
|
||||||
const activitypubApi = module.exports;
|
const activitypubApi = module.exports;
|
||||||
|
|
||||||
@@ -49,3 +50,57 @@ activitypubApi.unfollow = async (caller, { uid: actorId }) => {
|
|||||||
db.decrObjectField(`user:${caller.uid}`, 'followingRemoteCount'),
|
db.decrObjectField(`user:${caller.uid}`, 'followingRemoteCount'),
|
||||||
]);
|
]);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
activitypubApi.create = {};
|
||||||
|
|
||||||
|
activitypubApi.create.post = async (caller, { post }) => {
|
||||||
|
const id = `${nconf.get('url')}/post/${post.pid}`;
|
||||||
|
const published = new Date(post.timestamp).toISOString();
|
||||||
|
const [userslug, raw, followers] = await Promise.all([
|
||||||
|
user.getUserField(caller.uid, 'userslug'),
|
||||||
|
posts.getPostField(post.pid, 'content'),
|
||||||
|
db.getSortedSetMembers(`followersRemote:${caller.uid}`),
|
||||||
|
]);
|
||||||
|
|
||||||
|
// todo: post visibility, category privileges integration
|
||||||
|
const recipients = {
|
||||||
|
to: [activitypub._constants.publicAddress],
|
||||||
|
cc: [`${nconf.get('url')}/user/${userslug}/followers`],
|
||||||
|
};
|
||||||
|
const targets = new Set(followers);
|
||||||
|
|
||||||
|
let inReplyTo = null;
|
||||||
|
if (post.toPid) {
|
||||||
|
inReplyTo = activitypub.helpers.isUri(post.toPid) ? post.toPid : id;
|
||||||
|
const parentId = await posts.getPostField(post.toPid, 'uid');
|
||||||
|
if (activitypub.helpers.isUri(parentId)) {
|
||||||
|
recipients.to.unshift(parentId);
|
||||||
|
targets.add(parentId);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const object = {
|
||||||
|
id,
|
||||||
|
type: 'Note',
|
||||||
|
...recipients,
|
||||||
|
inReplyTo,
|
||||||
|
published,
|
||||||
|
url: id,
|
||||||
|
attributedTo: `${nconf.get('url')}/user/${post.user.userslug}`,
|
||||||
|
sensitive: false, // todo
|
||||||
|
content: post.content,
|
||||||
|
source: {
|
||||||
|
content: raw,
|
||||||
|
mediaType: 'text/markdown',
|
||||||
|
},
|
||||||
|
// replies: {} todo...
|
||||||
|
};
|
||||||
|
|
||||||
|
const payload = {
|
||||||
|
type: 'Create',
|
||||||
|
...recipients,
|
||||||
|
object,
|
||||||
|
};
|
||||||
|
|
||||||
|
await activitypub.send(caller.uid, Array.from(targets), payload);
|
||||||
|
};
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ const posts = require('../posts');
|
|||||||
const meta = require('../meta');
|
const meta = require('../meta');
|
||||||
const privileges = require('../privileges');
|
const privileges = require('../privileges');
|
||||||
|
|
||||||
|
const activitypubApi = require('./activitypub');
|
||||||
const apiHelpers = require('./helpers');
|
const apiHelpers = require('./helpers');
|
||||||
|
|
||||||
const { doTopicAction } = apiHelpers;
|
const { doTopicAction } = apiHelpers;
|
||||||
@@ -79,6 +80,7 @@ topicsAPI.create = async function (caller, data) {
|
|||||||
socketHelpers.emitToUids('event:new_post', { posts: [result.postData] }, [caller.uid]);
|
socketHelpers.emitToUids('event:new_post', { posts: [result.postData] }, [caller.uid]);
|
||||||
socketHelpers.emitToUids('event:new_topic', result.topicData, [caller.uid]);
|
socketHelpers.emitToUids('event:new_topic', result.topicData, [caller.uid]);
|
||||||
socketHelpers.notifyNew(caller.uid, 'newTopic', { posts: [result.postData], topic: result.topicData });
|
socketHelpers.notifyNew(caller.uid, 'newTopic', { posts: [result.postData], topic: result.topicData });
|
||||||
|
activitypubApi.create.post(caller, { post: result.postData });
|
||||||
|
|
||||||
return result.topicData;
|
return result.topicData;
|
||||||
};
|
};
|
||||||
@@ -113,6 +115,7 @@ topicsAPI.reply = async function (caller, data) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
socketHelpers.notifyNew(caller.uid, 'newPost', result);
|
socketHelpers.notifyNew(caller.uid, 'newPost', result);
|
||||||
|
activitypubApi.create.post(caller, { post: postData });
|
||||||
|
|
||||||
return postObj[0];
|
return postObj[0];
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user