mirror of
				https://codeberg.org/yeentown/barkey.git
				synced 2025-11-04 15:34:13 +00:00 
			
		
		
		
	wip
This commit is contained in:
		
							parent
							
								
									b60121527d
								
							
						
					
					
						commit
						e8b42d7e16
					
				
					 20 changed files with 354 additions and 377 deletions
				
			
		| 
						 | 
					@ -1,8 +1,14 @@
 | 
				
			||||||
import parseAcct from '../acct/parse';
 | 
					import parseAcct from '../acct/parse';
 | 
				
			||||||
import Post from '../models/post';
 | 
					import Post, { pack } from '../models/post';
 | 
				
			||||||
import User from '../models/user';
 | 
					import User, { isLocalUser, isRemoteUser, IUser } from '../models/user';
 | 
				
			||||||
 | 
					import stream from '../publishers/stream';
 | 
				
			||||||
 | 
					import Following from '../models/following';
 | 
				
			||||||
 | 
					import { createHttp } from '../queue';
 | 
				
			||||||
 | 
					import renderNote from '../remote/activitypub/renderer/note';
 | 
				
			||||||
 | 
					import renderCreate from '../remote/activitypub/renderer/create';
 | 
				
			||||||
 | 
					import context from '../remote/activitypub/renderer/context';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default async (post, reply, repost, atMentions) => {
 | 
					export default async (user: IUser, post, reply, repost, atMentions) => {
 | 
				
			||||||
	post.mentions = [];
 | 
						post.mentions = [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	function addMention(mentionee) {
 | 
						function addMention(mentionee) {
 | 
				
			||||||
| 
						 | 
					@ -46,5 +52,98 @@ export default async (post, reply, repost, atMentions) => {
 | 
				
			||||||
		addMention(_id);
 | 
							addMention(_id);
 | 
				
			||||||
	}));
 | 
						}));
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return Post.insert(post);
 | 
						const inserted = await Post.insert(post);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						User.update({ _id: user._id }, {
 | 
				
			||||||
 | 
							// Increment my posts count
 | 
				
			||||||
 | 
							$inc: {
 | 
				
			||||||
 | 
								postsCount: 1
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							$set: {
 | 
				
			||||||
 | 
								latestPost: post._id
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						const postObj = await pack(inserted);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// タイムラインへの投稿
 | 
				
			||||||
 | 
						if (!post.channelId) {
 | 
				
			||||||
 | 
							// Publish event to myself's stream
 | 
				
			||||||
 | 
							stream(post.userId, 'post', postObj);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// Fetch all followers
 | 
				
			||||||
 | 
							const followers = await Following.aggregate([{
 | 
				
			||||||
 | 
								$lookup: {
 | 
				
			||||||
 | 
									from: 'users',
 | 
				
			||||||
 | 
									localField: 'followerId',
 | 
				
			||||||
 | 
									foreignField: '_id',
 | 
				
			||||||
 | 
									as: 'follower'
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}, {
 | 
				
			||||||
 | 
								$match: {
 | 
				
			||||||
 | 
									followeeId: post.userId
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}], {
 | 
				
			||||||
 | 
								_id: false
 | 
				
			||||||
 | 
							});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							const note = await renderNote(user, post);
 | 
				
			||||||
 | 
							const content = renderCreate(note);
 | 
				
			||||||
 | 
							content['@context'] = context;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							Promise.all(followers.map(({ follower }) => {
 | 
				
			||||||
 | 
								if (isLocalUser(follower)) {
 | 
				
			||||||
 | 
									// Publish event to followers stream
 | 
				
			||||||
 | 
									stream(follower._id, 'post', postObj);
 | 
				
			||||||
 | 
								} else {
 | 
				
			||||||
 | 
									// フォロワーがリモートユーザーかつ投稿者がローカルユーザーなら投稿を配信
 | 
				
			||||||
 | 
									if (isLocalUser(user)) {
 | 
				
			||||||
 | 
										createHttp({
 | 
				
			||||||
 | 
											type: 'deliver',
 | 
				
			||||||
 | 
											user,
 | 
				
			||||||
 | 
											content,
 | 
				
			||||||
 | 
											to: follower.account.inbox
 | 
				
			||||||
 | 
										}).save();
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}
 | 
				
			||||||
 | 
							}));
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// チャンネルへの投稿
 | 
				
			||||||
 | 
						/* TODO
 | 
				
			||||||
 | 
						if (post.channelId) {
 | 
				
			||||||
 | 
							promises.push(
 | 
				
			||||||
 | 
								// Increment channel index(posts count)
 | 
				
			||||||
 | 
								Channel.update({ _id: post.channelId }, {
 | 
				
			||||||
 | 
									$inc: {
 | 
				
			||||||
 | 
										index: 1
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								// Publish event to channel
 | 
				
			||||||
 | 
								promisedPostObj.then(postObj => {
 | 
				
			||||||
 | 
									publishChannelStream(post.channelId, 'post', postObj);
 | 
				
			||||||
 | 
								}),
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
								Promise.all([
 | 
				
			||||||
 | 
									promisedPostObj,
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
									// Get channel watchers
 | 
				
			||||||
 | 
									ChannelWatching.find({
 | 
				
			||||||
 | 
										channelId: post.channelId,
 | 
				
			||||||
 | 
										// 削除されたドキュメントは除く
 | 
				
			||||||
 | 
										deletedAt: { $exists: false }
 | 
				
			||||||
 | 
									})
 | 
				
			||||||
 | 
								]).then(([postObj, watches]) => {
 | 
				
			||||||
 | 
									// チャンネルの視聴者(のタイムライン)に配信
 | 
				
			||||||
 | 
									watches.forEach(w => {
 | 
				
			||||||
 | 
										stream(w.userId, 'post', postObj);
 | 
				
			||||||
 | 
									});
 | 
				
			||||||
 | 
								})
 | 
				
			||||||
 | 
							);
 | 
				
			||||||
 | 
						}*/
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return Promise.all(promises);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,93 +0,0 @@
 | 
				
			||||||
import Channel from '../../models/channel';
 | 
					 | 
				
			||||||
import Following from '../../models/following';
 | 
					 | 
				
			||||||
import ChannelWatching from '../../models/channel-watching';
 | 
					 | 
				
			||||||
import Post, { pack } from '../../models/post';
 | 
					 | 
				
			||||||
import User, { isLocalUser } from '../../models/user';
 | 
					 | 
				
			||||||
import stream, { publishChannelStream } from '../../publishers/stream';
 | 
					 | 
				
			||||||
import context from '../../remote/activitypub/renderer/context';
 | 
					 | 
				
			||||||
import renderCreate from '../../remote/activitypub/renderer/create';
 | 
					 | 
				
			||||||
import renderNote from '../../remote/activitypub/renderer/note';
 | 
					 | 
				
			||||||
import request from '../../remote/request';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default ({ data }) => Post.findOne({ _id: data.id }).then(post => {
 | 
					 | 
				
			||||||
	const promisedPostObj = pack(post);
 | 
					 | 
				
			||||||
	const promises = [];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// タイムラインへの投稿
 | 
					 | 
				
			||||||
	if (!post.channelId) {
 | 
					 | 
				
			||||||
		promises.push(
 | 
					 | 
				
			||||||
			// Publish event to myself's stream
 | 
					 | 
				
			||||||
			promisedPostObj.then(postObj => {
 | 
					 | 
				
			||||||
				stream(post.userId, 'post', postObj);
 | 
					 | 
				
			||||||
			}),
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			Promise.all([
 | 
					 | 
				
			||||||
				User.findOne({ _id: post.userId }),
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
				// Fetch all followers
 | 
					 | 
				
			||||||
				Following.aggregate([{
 | 
					 | 
				
			||||||
					$lookup: {
 | 
					 | 
				
			||||||
						from: 'users',
 | 
					 | 
				
			||||||
						localField: 'followerId',
 | 
					 | 
				
			||||||
						foreignField: '_id',
 | 
					 | 
				
			||||||
						as: 'follower'
 | 
					 | 
				
			||||||
					}
 | 
					 | 
				
			||||||
				}, {
 | 
					 | 
				
			||||||
					$match: {
 | 
					 | 
				
			||||||
						followeeId: post.userId
 | 
					 | 
				
			||||||
					}
 | 
					 | 
				
			||||||
				}], {
 | 
					 | 
				
			||||||
					_id: false
 | 
					 | 
				
			||||||
				})
 | 
					 | 
				
			||||||
			]).then(([user, followers]) => Promise.all(followers.map(following => {
 | 
					 | 
				
			||||||
				if (isLocalUser(following.follower)) {
 | 
					 | 
				
			||||||
					// Publish event to followers stream
 | 
					 | 
				
			||||||
					return promisedPostObj.then(postObj => {
 | 
					 | 
				
			||||||
						stream(following.followerId, 'post', postObj);
 | 
					 | 
				
			||||||
					});
 | 
					 | 
				
			||||||
				}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
				return renderNote(user, post).then(note => {
 | 
					 | 
				
			||||||
					const create = renderCreate(note);
 | 
					 | 
				
			||||||
					create['@context'] = context;
 | 
					 | 
				
			||||||
					return request(user, following.follower[0].account.inbox, create);
 | 
					 | 
				
			||||||
				});
 | 
					 | 
				
			||||||
			})))
 | 
					 | 
				
			||||||
		);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	// チャンネルへの投稿
 | 
					 | 
				
			||||||
	if (post.channelId) {
 | 
					 | 
				
			||||||
		promises.push(
 | 
					 | 
				
			||||||
			// Increment channel index(posts count)
 | 
					 | 
				
			||||||
			Channel.update({ _id: post.channelId }, {
 | 
					 | 
				
			||||||
				$inc: {
 | 
					 | 
				
			||||||
					index: 1
 | 
					 | 
				
			||||||
				}
 | 
					 | 
				
			||||||
			}),
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			// Publish event to channel
 | 
					 | 
				
			||||||
			promisedPostObj.then(postObj => {
 | 
					 | 
				
			||||||
				publishChannelStream(post.channelId, 'post', postObj);
 | 
					 | 
				
			||||||
			}),
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			Promise.all([
 | 
					 | 
				
			||||||
				promisedPostObj,
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
				// Get channel watchers
 | 
					 | 
				
			||||||
				ChannelWatching.find({
 | 
					 | 
				
			||||||
					channelId: post.channelId,
 | 
					 | 
				
			||||||
					// 削除されたドキュメントは除く
 | 
					 | 
				
			||||||
					deletedAt: { $exists: false }
 | 
					 | 
				
			||||||
				})
 | 
					 | 
				
			||||||
			]).then(([postObj, watches]) => {
 | 
					 | 
				
			||||||
				// チャンネルの視聴者(のタイムライン)に配信
 | 
					 | 
				
			||||||
				watches.forEach(w => {
 | 
					 | 
				
			||||||
					stream(w.userId, 'post', postObj);
 | 
					 | 
				
			||||||
				});
 | 
					 | 
				
			||||||
			})
 | 
					 | 
				
			||||||
		);
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	return Promise.all(promises);
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
| 
						 | 
					@ -1,39 +0,0 @@
 | 
				
			||||||
import { verifySignature } from 'http-signature';
 | 
					 | 
				
			||||||
import parseAcct from '../../acct/parse';
 | 
					 | 
				
			||||||
import User, { IRemoteUser } from '../../models/user';
 | 
					 | 
				
			||||||
import act from '../../remote/activitypub/act';
 | 
					 | 
				
			||||||
import resolvePerson from '../../remote/activitypub/resolve-person';
 | 
					 | 
				
			||||||
import Resolver from '../../remote/activitypub/resolver';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default async ({ data }): Promise<void> => {
 | 
					 | 
				
			||||||
	const keyIdLower = data.signature.keyId.toLowerCase();
 | 
					 | 
				
			||||||
	let user;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (keyIdLower.startsWith('acct:')) {
 | 
					 | 
				
			||||||
		const { username, host } = parseAcct(keyIdLower.slice('acct:'.length));
 | 
					 | 
				
			||||||
		if (host === null) {
 | 
					 | 
				
			||||||
			throw 'request was made by local user';
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		user = await User.findOne({ usernameLower: username, hostLower: host }) as IRemoteUser;
 | 
					 | 
				
			||||||
	} else {
 | 
					 | 
				
			||||||
		user = await User.findOne({
 | 
					 | 
				
			||||||
			host: { $ne: null },
 | 
					 | 
				
			||||||
			'account.publicKey.id': data.signature.keyId
 | 
					 | 
				
			||||||
		}) as IRemoteUser;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if (user === null) {
 | 
					 | 
				
			||||||
			user = await resolvePerson(data.signature.keyId);
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (user === null) {
 | 
					 | 
				
			||||||
		throw 'failed to resolve user';
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	if (!verifySignature(data.signature, user.account.publicKey.publicKeyPem)) {
 | 
					 | 
				
			||||||
		throw 'signature verification failed';
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	await Promise.all(await act(new Resolver(), user, data.inbox, true));
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
							
								
								
									
										10
									
								
								src/queue.ts
									
										
									
									
									
								
							
							
						
						
									
										10
									
								
								src/queue.ts
									
										
									
									
									
								
							| 
						 | 
					@ -1,10 +0,0 @@
 | 
				
			||||||
import { createQueue } from 'kue';
 | 
					 | 
				
			||||||
import config from './config';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default createQueue({
 | 
					 | 
				
			||||||
	redis: {
 | 
					 | 
				
			||||||
		port: config.redis.port,
 | 
					 | 
				
			||||||
		host: config.redis.host,
 | 
					 | 
				
			||||||
		auth: config.redis.pass
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
});
 | 
					 | 
				
			||||||
							
								
								
									
										37
									
								
								src/queue/index.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								src/queue/index.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,37 @@
 | 
				
			||||||
 | 
					import { createQueue } from 'kue';
 | 
				
			||||||
 | 
					import config from '../config';
 | 
				
			||||||
 | 
					import db from './processors/db';
 | 
				
			||||||
 | 
					import http from './processors/http';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					const queue = createQueue({
 | 
				
			||||||
 | 
						redis: {
 | 
				
			||||||
 | 
							port: config.redis.port,
 | 
				
			||||||
 | 
							host: config.redis.host,
 | 
				
			||||||
 | 
							auth: config.redis.pass
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export function createHttp(data) {
 | 
				
			||||||
 | 
						return queue
 | 
				
			||||||
 | 
							.create('http', data)
 | 
				
			||||||
 | 
							.attempts(16)
 | 
				
			||||||
 | 
							.backoff({ delay: 16384, type: 'exponential' });
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export function createDb(data) {
 | 
				
			||||||
 | 
						return queue.create('db', data);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export function process() {
 | 
				
			||||||
 | 
						queue.process('db', db);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						/*
 | 
				
			||||||
 | 
							256 is the default concurrency limit of Mozilla Firefox and Google
 | 
				
			||||||
 | 
							Chromium.
 | 
				
			||||||
 | 
							a8af215e691f3a2205a3758d2d96e9d328e100ff - chromium/src.git - Git at Google
 | 
				
			||||||
 | 
							https://chromium.googlesource.com/chromium/src.git/+/a8af215e691f3a2205a3758d2d96e9d328e100ff
 | 
				
			||||||
 | 
							Network.http.max-connections - MozillaZine Knowledge Base
 | 
				
			||||||
 | 
							http://kb.mozillazine.org/Network.http.max-connections
 | 
				
			||||||
 | 
						*/
 | 
				
			||||||
 | 
						queue.process('http', 256, http);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										17
									
								
								src/queue/processors/http/deliver.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										17
									
								
								src/queue/processors/http/deliver.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,17 @@
 | 
				
			||||||
 | 
					import * as kue from 'kue';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import Channel from '../../models/channel';
 | 
				
			||||||
 | 
					import Following from '../../models/following';
 | 
				
			||||||
 | 
					import ChannelWatching from '../../models/channel-watching';
 | 
				
			||||||
 | 
					import Post, { pack } from '../../models/post';
 | 
				
			||||||
 | 
					import User, { isLocalUser } from '../../models/user';
 | 
				
			||||||
 | 
					import stream, { publishChannelStream } from '../../publishers/stream';
 | 
				
			||||||
 | 
					import context from '../../remote/activitypub/renderer/context';
 | 
				
			||||||
 | 
					import renderCreate from '../../remote/activitypub/renderer/create';
 | 
				
			||||||
 | 
					import renderNote from '../../remote/activitypub/renderer/note';
 | 
				
			||||||
 | 
					import request from '../../remote/request';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default async (job: kue.Job, done): Promise<void> => {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						request(user, following.follower[0].account.inbox, create);
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
							
								
								
									
										55
									
								
								src/queue/processors/http/process-inbox.ts
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								src/queue/processors/http/process-inbox.ts
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
					@ -0,0 +1,55 @@
 | 
				
			||||||
 | 
					import * as kue from 'kue';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					import { verifySignature } from 'http-signature';
 | 
				
			||||||
 | 
					import parseAcct from '../../acct/parse';
 | 
				
			||||||
 | 
					import User, { IRemoteUser } from '../../models/user';
 | 
				
			||||||
 | 
					import act from '../../remote/activitypub/act';
 | 
				
			||||||
 | 
					import resolvePerson from '../../remote/activitypub/resolve-person';
 | 
				
			||||||
 | 
					import Resolver from '../../remote/activitypub/resolver';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// ユーザーのinboxにアクティビティが届いた時の処理
 | 
				
			||||||
 | 
					export default async (job: kue.Job, done): Promise<void> => {
 | 
				
			||||||
 | 
						const signature = job.data.signature;
 | 
				
			||||||
 | 
						const activity = job.data.activity;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						const keyIdLower = signature.keyId.toLowerCase();
 | 
				
			||||||
 | 
						let user;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (keyIdLower.startsWith('acct:')) {
 | 
				
			||||||
 | 
							const { username, host } = parseAcct(keyIdLower.slice('acct:'.length));
 | 
				
			||||||
 | 
							if (host === null) {
 | 
				
			||||||
 | 
								console.warn(`request was made by local user: @${username}`);
 | 
				
			||||||
 | 
								done();
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							user = await User.findOne({ usernameLower: username, hostLower: host }) as IRemoteUser;
 | 
				
			||||||
 | 
						} else {
 | 
				
			||||||
 | 
							user = await User.findOne({
 | 
				
			||||||
 | 
								host: { $ne: null },
 | 
				
			||||||
 | 
								'account.publicKey.id': signature.keyId
 | 
				
			||||||
 | 
							}) as IRemoteUser;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							// アクティビティを送信してきたユーザーがまだMisskeyサーバーに登録されていなかったら登録する
 | 
				
			||||||
 | 
							if (user === null) {
 | 
				
			||||||
 | 
								user = await resolvePerson(signature.keyId);
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (user === null) {
 | 
				
			||||||
 | 
							done(new Error('failed to resolve user'));
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						if (!verifySignature(signature, user.account.publicKey.publicKeyPem)) {
 | 
				
			||||||
 | 
							done(new Error('signature verification failed'));
 | 
				
			||||||
 | 
							return;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// アクティビティを処理
 | 
				
			||||||
 | 
						try {
 | 
				
			||||||
 | 
							await act(new Resolver(), user, activity);
 | 
				
			||||||
 | 
							done();
 | 
				
			||||||
 | 
						} catch (e) {
 | 
				
			||||||
 | 
							done(e);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					};
 | 
				
			||||||
| 
						 | 
					@ -1,10 +1,92 @@
 | 
				
			||||||
import create from '../create';
 | 
					import { JSDOM } from 'jsdom';
 | 
				
			||||||
import Resolver from '../resolver';
 | 
					const createDOMPurify = require('dompurify');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default (resolver: Resolver, actor, activity, distribute) => {
 | 
					import Resolver from '../resolver';
 | 
				
			||||||
 | 
					import DriveFile from '../../../models/drive-file';
 | 
				
			||||||
 | 
					import Post from '../../../models/post';
 | 
				
			||||||
 | 
					import uploadFromUrl from '../../../drive/upload-from-url';
 | 
				
			||||||
 | 
					import createPost from '../../../post/create';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					export default async (resolver: Resolver, actor, activity): Promise<void> => {
 | 
				
			||||||
	if ('actor' in activity && actor.account.uri !== activity.actor) {
 | 
						if ('actor' in activity && actor.account.uri !== activity.actor) {
 | 
				
			||||||
		throw new Error();
 | 
							throw new Error('invalid actor');
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return create(resolver, actor, activity.object, distribute);
 | 
						const uri = activity.id || activity;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						try {
 | 
				
			||||||
 | 
							await Promise.all([
 | 
				
			||||||
 | 
								DriveFile.findOne({ 'metadata.uri': uri }).then(file => {
 | 
				
			||||||
 | 
									if (file !== null) {
 | 
				
			||||||
 | 
										throw new Error();
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}, () => {}),
 | 
				
			||||||
 | 
								Post.findOne({ uri }).then(post => {
 | 
				
			||||||
 | 
									if (post !== null) {
 | 
				
			||||||
 | 
										throw new Error();
 | 
				
			||||||
 | 
									}
 | 
				
			||||||
 | 
								}, () => {})
 | 
				
			||||||
 | 
							]);
 | 
				
			||||||
 | 
						} catch (object) {
 | 
				
			||||||
 | 
							throw new Error(`already registered: ${uri}`);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						const object = await resolver.resolve(activity);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						switch (object.type) {
 | 
				
			||||||
 | 
						case 'Image':
 | 
				
			||||||
 | 
							createImage(resolver, object);
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						case 'Note':
 | 
				
			||||||
 | 
							createNote(resolver, object);
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						///
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						async function createImage(resolver: Resolver, image) {
 | 
				
			||||||
 | 
							if ('attributedTo' in image && actor.account.uri !== image.attributedTo) {
 | 
				
			||||||
 | 
								throw new Error('invalid image');
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							return await uploadFromUrl(image.url, actor);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						async function createNote(resolver: Resolver, note) {
 | 
				
			||||||
 | 
							if (
 | 
				
			||||||
 | 
								('attributedTo' in note && actor.account.uri !== note.attributedTo) ||
 | 
				
			||||||
 | 
								typeof note.id !== 'string'
 | 
				
			||||||
 | 
							) {
 | 
				
			||||||
 | 
								throw new Error('invalid note');
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							const mediaIds = [];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if ('attachment' in note) {
 | 
				
			||||||
 | 
								note.attachment.forEach(async media => {
 | 
				
			||||||
 | 
									const created = await createImage(resolver, media);
 | 
				
			||||||
 | 
									mediaIds.push(created._id);
 | 
				
			||||||
 | 
								});
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							const { window } = new JSDOM(note.content);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							await createPost(actor, {
 | 
				
			||||||
 | 
								channelId: undefined,
 | 
				
			||||||
 | 
								index: undefined,
 | 
				
			||||||
 | 
								createdAt: new Date(note.published),
 | 
				
			||||||
 | 
								mediaIds,
 | 
				
			||||||
 | 
								replyId: undefined,
 | 
				
			||||||
 | 
								repostId: undefined,
 | 
				
			||||||
 | 
								poll: undefined,
 | 
				
			||||||
 | 
								text: window.document.body.textContent,
 | 
				
			||||||
 | 
								textHtml: note.content && createDOMPurify(window).sanitize(note.content),
 | 
				
			||||||
 | 
								userId: actor._id,
 | 
				
			||||||
 | 
								appId: null,
 | 
				
			||||||
 | 
								viaMobile: false,
 | 
				
			||||||
 | 
								geo: undefined,
 | 
				
			||||||
 | 
								uri: note.id
 | 
				
			||||||
 | 
							}, null, null, []);
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -2,35 +2,29 @@ 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 createObject from '../create';
 | 
					 | 
				
			||||||
import Resolver from '../resolver';
 | 
					import Resolver from '../resolver';
 | 
				
			||||||
 | 
					import { IObject } from '../type';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default async (parentResolver: Resolver, actor, value, distribute?: boolean) => {
 | 
					export default async (parentResolver: Resolver, actor, activity: IObject): Promise<void> => {
 | 
				
			||||||
	const collection = await parentResolver.resolveCollection(value);
 | 
						switch (activity.type) {
 | 
				
			||||||
 | 
						case 'Create':
 | 
				
			||||||
 | 
							await create(parentResolver, actor, activity);
 | 
				
			||||||
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return collection.object.map(async element => {
 | 
						case 'Delete':
 | 
				
			||||||
		const { resolver, object } = await collection.resolver.resolveOne(element);
 | 
							await performDeleteActivity(parentResolver, actor, activity);
 | 
				
			||||||
		const created = await (await createObject(resolver, actor, [object], distribute))[0];
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		if (created !== null) {
 | 
						case 'Follow':
 | 
				
			||||||
			return created;
 | 
							await follow(parentResolver, actor, activity);
 | 
				
			||||||
		}
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		switch (object.type) {
 | 
						case 'Undo':
 | 
				
			||||||
		case 'Create':
 | 
							await undo(parentResolver, actor, activity);
 | 
				
			||||||
			return create(resolver, actor, object, distribute);
 | 
							break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		case 'Delete':
 | 
						default:
 | 
				
			||||||
			return performDeleteActivity(resolver, actor, object);
 | 
							console.warn(`unknown activity type: ${activity.type}`);
 | 
				
			||||||
 | 
							return null;
 | 
				
			||||||
		case 'Follow':
 | 
						}
 | 
				
			||||||
			return follow(resolver, actor, object, distribute);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		case 'Undo':
 | 
					 | 
				
			||||||
			return undo(resolver, actor, object);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		default:
 | 
					 | 
				
			||||||
			return null;
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
	});
 | 
					 | 
				
			||||||
};
 | 
					};
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,158 +0,0 @@
 | 
				
			||||||
import { JSDOM } from 'jsdom';
 | 
					 | 
				
			||||||
import { ObjectID } from 'mongodb';
 | 
					 | 
				
			||||||
import config from '../../config';
 | 
					 | 
				
			||||||
import DriveFile from '../../models/drive-file';
 | 
					 | 
				
			||||||
import Post from '../../models/post';
 | 
					 | 
				
			||||||
import { IRemoteUser } from '../../models/user';
 | 
					 | 
				
			||||||
import uploadFromUrl from '../../drive/upload-from-url';
 | 
					 | 
				
			||||||
import createPost from '../../post/create';
 | 
					 | 
				
			||||||
import distributePost from '../../post/distribute';
 | 
					 | 
				
			||||||
import Resolver from './resolver';
 | 
					 | 
				
			||||||
const createDOMPurify = require('dompurify');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
type IResult = {
 | 
					 | 
				
			||||||
	resolver: Resolver;
 | 
					 | 
				
			||||||
	object: {
 | 
					 | 
				
			||||||
		$ref: string;
 | 
					 | 
				
			||||||
		$id: ObjectID;
 | 
					 | 
				
			||||||
	};
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class Creator {
 | 
					 | 
				
			||||||
	private actor: IRemoteUser;
 | 
					 | 
				
			||||||
	private distribute: boolean;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	constructor(actor, distribute) {
 | 
					 | 
				
			||||||
		this.actor = actor;
 | 
					 | 
				
			||||||
		this.distribute = distribute;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	private async createImage(resolver: Resolver, image) {
 | 
					 | 
				
			||||||
		if ('attributedTo' in image && this.actor.account.uri !== image.attributedTo) {
 | 
					 | 
				
			||||||
			throw new Error();
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		const { _id } = await uploadFromUrl(image.url, this.actor, image.id || null);
 | 
					 | 
				
			||||||
		return {
 | 
					 | 
				
			||||||
			resolver,
 | 
					 | 
				
			||||||
			object: { $ref: 'driveFiles.files', $id: _id }
 | 
					 | 
				
			||||||
		};
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	private async createNote(resolver: Resolver, note) {
 | 
					 | 
				
			||||||
		if (
 | 
					 | 
				
			||||||
			('attributedTo' in note && this.actor.account.uri !== note.attributedTo) ||
 | 
					 | 
				
			||||||
			typeof note.id !== 'string'
 | 
					 | 
				
			||||||
		) {
 | 
					 | 
				
			||||||
			throw new Error();
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		const mediaIds = 'attachment' in note &&
 | 
					 | 
				
			||||||
			(await Promise.all(await this.create(resolver, note.attachment)))
 | 
					 | 
				
			||||||
				.filter(media => media !== null && media.object.$ref === 'driveFiles.files')
 | 
					 | 
				
			||||||
				.map(({ object }) => object.$id);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		const { window } = new JSDOM(note.content);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		const inserted = await createPost({
 | 
					 | 
				
			||||||
			channelId: undefined,
 | 
					 | 
				
			||||||
			index: undefined,
 | 
					 | 
				
			||||||
			createdAt: new Date(note.published),
 | 
					 | 
				
			||||||
			mediaIds,
 | 
					 | 
				
			||||||
			replyId: undefined,
 | 
					 | 
				
			||||||
			repostId: undefined,
 | 
					 | 
				
			||||||
			poll: undefined,
 | 
					 | 
				
			||||||
			text: window.document.body.textContent,
 | 
					 | 
				
			||||||
			textHtml: note.content && createDOMPurify(window).sanitize(note.content),
 | 
					 | 
				
			||||||
			userId: this.actor._id,
 | 
					 | 
				
			||||||
			appId: null,
 | 
					 | 
				
			||||||
			viaMobile: false,
 | 
					 | 
				
			||||||
			geo: undefined,
 | 
					 | 
				
			||||||
			uri: note.id
 | 
					 | 
				
			||||||
		}, null, null, []);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		const promises = [];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		if (this.distribute) {
 | 
					 | 
				
			||||||
			promises.push(distributePost(this.actor, inserted.mentions, inserted));
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		// Register to search database
 | 
					 | 
				
			||||||
		if (note.content && config.elasticsearch.enable) {
 | 
					 | 
				
			||||||
			const es = require('../../db/elasticsearch');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			promises.push(new Promise((resolve, reject) => {
 | 
					 | 
				
			||||||
				es.index({
 | 
					 | 
				
			||||||
					index: 'misskey',
 | 
					 | 
				
			||||||
					type: 'post',
 | 
					 | 
				
			||||||
					id: inserted._id.toString(),
 | 
					 | 
				
			||||||
					body: {
 | 
					 | 
				
			||||||
						text: window.document.body.textContent
 | 
					 | 
				
			||||||
					}
 | 
					 | 
				
			||||||
				}, resolve);
 | 
					 | 
				
			||||||
			}));
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		await Promise.all(promises);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return {
 | 
					 | 
				
			||||||
			resolver,
 | 
					 | 
				
			||||||
			object: { $ref: 'posts', id: inserted._id }
 | 
					 | 
				
			||||||
		};
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	public async create(parentResolver: Resolver, value): Promise<Array<Promise<IResult>>> {
 | 
					 | 
				
			||||||
		const collection = await parentResolver.resolveCollection(value);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return collection.object.map(async element => {
 | 
					 | 
				
			||||||
			const uri = element.id || element;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			try {
 | 
					 | 
				
			||||||
				await Promise.all([
 | 
					 | 
				
			||||||
					DriveFile.findOne({ 'metadata.uri': uri }).then(file => {
 | 
					 | 
				
			||||||
						if (file === null) {
 | 
					 | 
				
			||||||
							return;
 | 
					 | 
				
			||||||
						}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
						throw {
 | 
					 | 
				
			||||||
							$ref: 'driveFile.files',
 | 
					 | 
				
			||||||
							$id: file._id
 | 
					 | 
				
			||||||
						};
 | 
					 | 
				
			||||||
					}, () => {}),
 | 
					 | 
				
			||||||
					Post.findOne({ uri }).then(post => {
 | 
					 | 
				
			||||||
						if (post === null) {
 | 
					 | 
				
			||||||
							return;
 | 
					 | 
				
			||||||
						}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
						throw {
 | 
					 | 
				
			||||||
							$ref: 'posts',
 | 
					 | 
				
			||||||
							$id: post._id
 | 
					 | 
				
			||||||
						};
 | 
					 | 
				
			||||||
					}, () => {})
 | 
					 | 
				
			||||||
				]);
 | 
					 | 
				
			||||||
			} catch (object) {
 | 
					 | 
				
			||||||
				return {
 | 
					 | 
				
			||||||
					resolver: collection.resolver,
 | 
					 | 
				
			||||||
					object
 | 
					 | 
				
			||||||
				};
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			const { resolver, object } = await collection.resolver.resolveOne(element);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			switch (object.type) {
 | 
					 | 
				
			||||||
			case 'Image':
 | 
					 | 
				
			||||||
				return this.createImage(resolver, object);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			case 'Note':
 | 
					 | 
				
			||||||
				return this.createNote(resolver, object);
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
			return null;
 | 
					 | 
				
			||||||
		});
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export default (resolver: Resolver, actor, value, distribute?: boolean) => {
 | 
					 | 
				
			||||||
	const creator = new Creator(actor, distribute);
 | 
					 | 
				
			||||||
	return creator.create(resolver, value);
 | 
					 | 
				
			||||||
};
 | 
					 | 
				
			||||||
| 
						 | 
					@ -1,20 +1,45 @@
 | 
				
			||||||
 | 
					import { IObject } from "./type";
 | 
				
			||||||
 | 
					
 | 
				
			||||||
const request = require('request-promise-native');
 | 
					const request = require('request-promise-native');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export default class Resolver {
 | 
					export default class Resolver {
 | 
				
			||||||
	private requesting: Set<string>;
 | 
						private history: Set<string>;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	constructor(iterable?: Iterable<string>) {
 | 
						constructor() {
 | 
				
			||||||
		this.requesting = new Set(iterable);
 | 
							this.history = new Set();
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	private async resolveUnrequestedOne(value) {
 | 
						public async resolveCollection(value) {
 | 
				
			||||||
		if (typeof value !== 'string') {
 | 
							const collection = typeof value === 'string'
 | 
				
			||||||
			return { resolver: this, object: value };
 | 
								? await this.resolve(value)
 | 
				
			||||||
 | 
								: value;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							switch (collection.type) {
 | 
				
			||||||
 | 
							case 'Collection':
 | 
				
			||||||
 | 
								collection.objects = collection.object.items;
 | 
				
			||||||
 | 
								break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							case 'OrderedCollection':
 | 
				
			||||||
 | 
								collection.objects = collection.object.orderedItems;
 | 
				
			||||||
 | 
								break;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							default:
 | 
				
			||||||
 | 
								throw new Error(`unknown collection type: ${collection.type}`);
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		const resolver = new Resolver(this.requesting);
 | 
							return collection;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		resolver.requesting.add(value);
 | 
						public async resolve(value): Promise<IObject> {
 | 
				
			||||||
 | 
							if (typeof value !== 'string') {
 | 
				
			||||||
 | 
								return value;
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if (this.history.has(value)) {
 | 
				
			||||||
 | 
								throw new Error('cannot resolve already resolved one');
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							this.history.add(value);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		const object = await request({
 | 
							const object = await request({
 | 
				
			||||||
			url: value,
 | 
								url: value,
 | 
				
			||||||
| 
						 | 
					@ -29,41 +54,9 @@ export default class Resolver {
 | 
				
			||||||
				!object['@context'].includes('https://www.w3.org/ns/activitystreams') :
 | 
									!object['@context'].includes('https://www.w3.org/ns/activitystreams') :
 | 
				
			||||||
				object['@context'] !== 'https://www.w3.org/ns/activitystreams'
 | 
									object['@context'] !== 'https://www.w3.org/ns/activitystreams'
 | 
				
			||||||
		)) {
 | 
							)) {
 | 
				
			||||||
			throw new Error();
 | 
								throw new Error('invalid response');
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		return { resolver, object };
 | 
							return object;
 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	public async resolveCollection(value) {
 | 
					 | 
				
			||||||
		const resolved = typeof value === 'string' ?
 | 
					 | 
				
			||||||
			await this.resolveUnrequestedOne(value) :
 | 
					 | 
				
			||||||
			{ resolver: this, object: value };
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		switch (resolved.object.type) {
 | 
					 | 
				
			||||||
		case 'Collection':
 | 
					 | 
				
			||||||
			resolved.object = resolved.object.items;
 | 
					 | 
				
			||||||
			break;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		case 'OrderedCollection':
 | 
					 | 
				
			||||||
			resolved.object = resolved.object.orderedItems;
 | 
					 | 
				
			||||||
			break;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		default:
 | 
					 | 
				
			||||||
			if (!Array.isArray(value)) {
 | 
					 | 
				
			||||||
				resolved.object = [resolved.object];
 | 
					 | 
				
			||||||
			}
 | 
					 | 
				
			||||||
			break;
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return resolved;
 | 
					 | 
				
			||||||
	}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
	public resolveOne(value) {
 | 
					 | 
				
			||||||
		if (this.requesting.has(value)) {
 | 
					 | 
				
			||||||
			throw new Error();
 | 
					 | 
				
			||||||
		}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
		return this.resolveUnrequestedOne(value);
 | 
					 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -24,7 +24,7 @@ app.post('/@:user/inbox', bodyParser.json({
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	queue.create('http', {
 | 
						queue.create('http', {
 | 
				
			||||||
		type: 'processInbox',
 | 
							type: 'processInbox',
 | 
				
			||||||
		inbox: req.body,
 | 
							activity: req.body,
 | 
				
			||||||
		signature,
 | 
							signature,
 | 
				
			||||||
	}).save();
 | 
						}).save();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		
		Reference in a new issue