Distribute posts from remote

This commit is contained in:
Akihiko Odaki
2018-04-02 17:11:14 +09:00
parent aa94035987
commit ce7efc4dbb
12 changed files with 417 additions and 353 deletions

View File

@ -1,9 +1,9 @@
import create from '../create';
export default (resolver, actor, activity) => {
export default (resolver, actor, activity, distribute) => {
if ('actor' in activity && actor.account.uri !== activity.actor) {
throw new Error();
}
return create(resolver, actor, activity.object);
return create(resolver, actor, activity.object, distribute);
};

View File

@ -2,10 +2,10 @@ import create from './create';
import createObject from '../create';
import Resolver from '../resolver';
export default (actor, value) => {
export default (actor, value, distribute) => {
return new Resolver().resolve(value).then(resolved => Promise.all(resolved.map(async promisedResult => {
const { resolver, object } = await promisedResult;
const created = await (await createObject(resolver, actor, [object]))[0];
const created = await (await createObject(resolver, actor, [object], distribute))[0];
if (created !== null) {
return created;
@ -13,7 +13,7 @@ export default (actor, value) => {
switch (object.type) {
case 'Create':
return create(resolver, actor, object);
return create(resolver, actor, object, distribute);
default:
return null;

View File

@ -1,8 +1,11 @@
import { JSDOM } from 'jsdom';
import config from '../../config';
import Post from '../../models/post';
import { pack as packPost } from '../../models/post';
import RemoteUserObject, { IRemoteUserObject } from '../../models/remote-user-object';
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');
@ -16,72 +19,98 @@ function createRemoteUserObject($ref, $id, { id }) {
return RemoteUserObject.insert({ uri: id, object });
}
async function createImage(actor, object) {
if ('attributedTo' in object && actor.account.uri !== object.attributedTo) {
throw new Error();
class Creator {
private actor: IRemoteUser;
private distribute: boolean;
constructor(actor, distribute) {
this.actor = actor;
this.distribute = distribute;
}
const { _id } = await uploadFromUrl(object.url, actor);
return createRemoteUserObject('driveFiles.files', _id, object);
}
async function createNote(resolver, actor, object) {
if ('attributedTo' in object && actor.account.uri !== object.attributedTo) {
throw new Error();
}
const mediaIds = 'attachment' in object &&
(await Promise.all(await create(resolver, actor, object.attachment)))
.filter(media => media !== null && media.object.$ref === 'driveFiles.files')
.map(({ object }) => object.$id);
const { window } = new JSDOM(object.content);
const { _id } = await Post.insert({
channelId: undefined,
index: undefined,
createdAt: new Date(object.published),
mediaIds,
replyId: undefined,
repostId: undefined,
poll: undefined,
text: window.document.body.textContent,
textHtml: object.content && createDOMPurify(window).sanitize(object.content),
userId: actor._id,
appId: null,
viaMobile: false,
geo: undefined
});
// Register to search database
if (object.content && config.elasticsearch.enable) {
const es = require('../../db/elasticsearch');
es.index({
index: 'misskey',
type: 'post',
id: _id.toString(),
body: {
text: window.document.body.textContent
}
});
}
return createRemoteUserObject('posts', _id, object);
}
export default async function create(parentResolver: Resolver, actor, value): Promise<Array<Promise<IRemoteUserObject>>> {
const results = await parentResolver.resolveRemoteUserObjects(value);
return results.map(promisedResult => promisedResult.then(({ resolver, object }) => {
switch (object.type) {
case 'Image':
return createImage(actor, object);
case 'Note':
return createNote(resolver, actor, object);
private async createImage(object) {
if ('attributedTo' in object && this.actor.account.uri !== object.attributedTo) {
throw new Error();
}
return null;
}));
const { _id } = await uploadFromUrl(object.url, this.actor);
return createRemoteUserObject('driveFiles.files', _id, object);
}
private async createNote(resolver, object) {
if ('attributedTo' in object && this.actor.account.uri !== object.attributedTo) {
throw new Error();
}
const mediaIds = 'attachment' in object &&
(await Promise.all(await this.create(resolver, object.attachment)))
.filter(media => media !== null && media.object.$ref === 'driveFiles.files')
.map(({ object }) => object.$id);
const { window } = new JSDOM(object.content);
const inserted = await createPost({
channelId: undefined,
index: undefined,
createdAt: new Date(object.published),
mediaIds,
replyId: undefined,
repostId: undefined,
poll: undefined,
text: window.document.body.textContent,
textHtml: object.content && createDOMPurify(window).sanitize(object.content),
userId: this.actor._id,
appId: null,
viaMobile: false,
geo: undefined
}, null, null, []);
const promisedRemoteUserObject = createRemoteUserObject('posts', inserted._id, object);
const promises = [];
if (this.distribute) {
promises.push(distributePost(this.actor, inserted.mentions, packPost(inserted)));
}
// Register to search database
if (object.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 promisedRemoteUserObject;
}
public async create(parentResolver, value): Promise<Array<Promise<IRemoteUserObject>>> {
const results = await parentResolver.resolveRemoteUserObjects(value);
return results.map(promisedResult => promisedResult.then(({ resolver, object }) => {
switch (object.type) {
case 'Image':
return this.createImage(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);
};

View File

@ -52,10 +52,10 @@ export default async (value, usernameLower, hostLower, acctLower) => {
bannerId: null,
createdAt: Date.parse(object.published),
description: summaryDOM.textContent,
followersCount: followers.totalItem,
followingCount: following.totalItem,
followersCount: followers ? followers.totalItem || 0 : 0,
followingCount: following ? following.totalItem || 0 : 0,
name: object.name,
postsCount: outbox.totalItem,
postsCount: outbox ? outbox.totalItem || 0 : 0,
driveCapacity: 1024 * 1024 * 8, // 8MiB
username: object.preferredUsername,
usernameLower,