Implement like

This commit is contained in:
syuilo 2018-04-07 17:05:14 +09:00
parent 321f61f1cb
commit c5f23bce78
7 changed files with 131 additions and 137 deletions

View file

@ -1,9 +1,10 @@
import { Object } from '../type';
import { IRemoteUser } from '../../../models/user';
import create from './create'; import create from './create';
import performDeleteActivity from './delete'; import performDeleteActivity from './delete';
import follow from './follow'; import follow from './follow';
import undo from './undo'; import undo from './undo';
import { Object } from '../type'; import like from './like';
import { IRemoteUser } from '../../../models/user';
const self = async (actor: IRemoteUser, activity: Object): Promise<void> => { const self = async (actor: IRemoteUser, activity: Object): Promise<void> => {
switch (activity.type) { switch (activity.type) {
@ -23,6 +24,10 @@ const self = async (actor: IRemoteUser, activity: Object): Promise<void> => {
// noop // noop
break; break;
case 'Like':
await like(actor, activity);
break;
case 'Undo': case 'Undo':
await undo(actor, activity); await undo(actor, activity);
break; break;
@ -33,7 +38,7 @@ const self = async (actor: IRemoteUser, activity: Object): Promise<void> => {
break; break;
default: default:
console.warn(`unknown activity type: ${activity.type}`); console.warn(`unknown activity type: ${(activity as any).type}`);
return null; return null;
} }
}; };

View file

@ -1,10 +1,10 @@
import { MongoError } from 'mongodb';
import Reaction, { IPostReaction } from '../../../models/post-reaction';
import Post from '../../../models/post'; import Post from '../../../models/post';
import queue from '../../../queue'; import { IRemoteUser } from '../../../models/user';
import { ILike } from '../type';
import create from '../../../services/post/reaction/create';
export default async (resolver, actor, activity, distribute) => { export default async (actor: IRemoteUser, activity: ILike) => {
const id = activity.object.id || activity.object; const id = typeof activity.object == 'string' ? activity.object : activity.object.id;
// Transform: // Transform:
// https://misskey.ex/@syuilo/xxxx to // https://misskey.ex/@syuilo/xxxx to
@ -16,48 +16,5 @@ export default async (resolver, actor, activity, distribute) => {
throw new Error(); throw new Error();
} }
if (!distribute) { await create(actor, post, 'pudding');
const { _id } = await Reaction.findOne({
userId: actor._id,
postId: post._id
});
return {
resolver,
object: { $ref: 'postPeactions', $id: _id }
};
}
const promisedReaction = Reaction.insert({
createdAt: new Date(),
userId: actor._id,
postId: post._id,
reaction: 'pudding'
}).then(reaction => new Promise<IPostReaction>((resolve, reject) => {
queue.create('http', {
type: 'reaction',
reactionId: reaction._id
}).save(error => {
if (error) {
reject(error);
} else {
resolve(reaction);
}
});
}), async error => {
// duplicate key error
if (error instanceof MongoError && error.code === 11000) {
return Reaction.findOne({
userId: actor._id,
postId: post._id
});
}
throw error;
});
return promisedReaction.then(({ _id }) => ({
resolver,
object: { $ref: 'postPeactions', $id: _id }
}));
}; };

View file

@ -0,0 +1,9 @@
import config from '../../../config';
export default (user, post) => {
return {
type: 'Like',
actor: `${config.url}/@${user.username}`,
object: post.uri ? post.uri : `${config.url}/posts/${post._id}`
};
};

View file

@ -23,7 +23,7 @@ export default async (user: IUser, post: IPost) => {
}); });
if (inReplyToUser !== null) { if (inReplyToUser !== null) {
inReplyTo = inReplyToPost.uri || `${config.url}/@${inReplyToUser.username}/${inReplyToPost._id}`; inReplyTo = inReplyToPost.uri || `${config.url}/posts/${inReplyToPost._id}`;
} }
} }
} else { } else {
@ -33,7 +33,7 @@ export default async (user: IUser, post: IPost) => {
const attributedTo = `${config.url}/@${user.username}`; const attributedTo = `${config.url}/@${user.username}`;
return { return {
id: `${attributedTo}/${post._id}`, id: `${config.url}/posts/${post._id}`,
type: 'Note', type: 'Note',
attributedTo, attributedTo,
content: post.textHtml, content: post.textHtml,

View file

@ -55,6 +55,10 @@ export interface IAccept extends IActivity {
type: 'Accept'; type: 'Accept';
} }
export interface ILike extends IActivity {
type: 'Like';
}
export type Object = export type Object =
ICollection | ICollection |
IOrderedCollection | IOrderedCollection |
@ -62,4 +66,5 @@ export type Object =
IDelete | IDelete |
IUndo | IUndo |
IFollow | IFollow |
IAccept; IAccept |
ILike;

View file

@ -3,20 +3,11 @@
*/ */
import $ from 'cafy'; import $ from 'cafy';
import Reaction from '../../../../../models/post-reaction'; import Reaction from '../../../../../models/post-reaction';
import Post, { pack as packPost } from '../../../../../models/post'; import Post from '../../../../../models/post';
import { pack as packUser } from '../../../../../models/user'; import create from '../../../../../services/post/reaction/create';
import Watching from '../../../../../models/post-watching';
import watch from '../../../../../post/watch';
import { publishPostStream } from '../../../../../publishers/stream';
import notify from '../../../../../publishers/notify';
import pushSw from '../../../../../publishers/push-sw';
/** /**
* React to a post * React to a post
*
* @param {any} params
* @param {any} user
* @return {Promise<any>}
*/ */
module.exports = (params, user) => new Promise(async (res, rej) => { module.exports = (params, user) => new Promise(async (res, rej) => {
// Get 'postId' parameter // Get 'postId' parameter
@ -46,78 +37,11 @@ module.exports = (params, user) => new Promise(async (res, rej) => {
return rej('post not found'); return rej('post not found');
} }
// Myself try {
if (post.userId.equals(user._id)) { await create(user, post, reaction);
return rej('cannot react to my post'); } catch (e) {
rej(e);
} }
// if already reacted
const exist = await Reaction.findOne({
postId: post._id,
userId: user._id,
deletedAt: { $exists: false }
});
if (exist !== null) {
return rej('already reacted');
}
// Create reaction
await Reaction.insert({
createdAt: new Date(),
postId: post._id,
userId: user._id,
reaction: reaction
});
// Send response
res(); res();
const inc = {};
inc[`reactionCounts.${reaction}`] = 1;
// Increment reactions count
await Post.update({ _id: post._id }, {
$inc: inc
});
publishPostStream(post._id, 'reacted');
// Notify
notify(post.userId, user._id, 'reaction', {
postId: post._id,
reaction: reaction
});
pushSw(post.userId, 'reaction', {
user: await packUser(user, post.userId),
post: await packPost(post, post.userId),
reaction: reaction
});
// Fetch watchers
Watching
.find({
postId: post._id,
userId: { $ne: user._id },
// 削除されたドキュメントは除く
deletedAt: { $exists: false }
}, {
fields: {
userId: true
}
})
.then(watchers => {
watchers.forEach(watcher => {
notify(watcher.userId, user._id, 'reaction', {
postId: post._id,
reaction: reaction
});
});
});
// この投稿をWatchする
if (user.account.settings.autoWatch !== false) {
watch(user._id, post);
}
}); });

View file

@ -0,0 +1,94 @@
import { IUser, pack as packUser, isLocalUser, isRemoteUser } from '../../../models/user';
import Post, { IPost, pack as packPost } from '../../../models/post';
import PostReaction from '../../../models/post-reaction';
import { publishPostStream } from '../../../publishers/stream';
import notify from '../../../publishers/notify';
import pushSw from '../../../publishers/push-sw';
import PostWatching from '../../../models/post-watching';
import watch from '../watch';
import renderLike from '../../../remote/activitypub/renderer/like';
import { deliver } from '../../../queue';
import context from '../../../remote/activitypub/renderer/context';
export default async (user: IUser, post: IPost, reaction: string) => new Promise(async (res, rej) => {
// Myself
if (post.userId.equals(user._id)) {
return rej('cannot react to my post');
}
// if already reacted
const exist = await PostReaction.findOne({
postId: post._id,
userId: user._id
});
if (exist !== null) {
return rej('already reacted');
}
// Create reaction
await PostReaction.insert({
createdAt: new Date(),
postId: post._id,
userId: user._id,
reaction
});
res();
const inc = {};
inc[`reactionCounts.${reaction}`] = 1;
// Increment reactions count
await Post.update({ _id: post._id }, {
$inc: inc
});
publishPostStream(post._id, 'reacted');
// Notify
notify(post.userId, user._id, 'reaction', {
postId: post._id,
reaction: reaction
});
pushSw(post.userId, 'reaction', {
user: await packUser(user, post.userId),
post: await packPost(post, post.userId),
reaction: reaction
});
// Fetch watchers
PostWatching
.find({
postId: post._id,
userId: { $ne: user._id }
}, {
fields: {
userId: true
}
})
.then(watchers => {
watchers.forEach(watcher => {
notify(watcher.userId, user._id, 'reaction', {
postId: post._id,
reaction: reaction
});
});
});
// ユーザーがローカルユーザーかつ自動ウォッチ設定がオンならばこの投稿をWatchする
if (isLocalUser(user) && user.account.settings.autoWatch !== false) {
watch(user._id, post);
}
//#region 配信
const content = renderLike(user, post);
content['@context'] = context;
// リアクターがローカルユーザーかつリアクション対象がリモートユーザーの投稿なら配送
if (isLocalUser(user) && isRemoteUser(post._user)) {
deliver(user, content, post._user.account.inbox).save();
}
//#endregion
});