Compare commits

...

2 Commits

Author SHA1 Message Date
Derek 2fc1098d39 Initial incoming post -> channel~
Also adds channel "username" like access via url/+channel-slug (a new, 
required, and currently unsettable, channel option)

Completely untested, and known unfinshed in a few places, good luck 
future me!
2022-05-31 03:33:36 -04:00
Derek f6f06dc98d Fix channel creation post actor conversion 2022-05-31 01:41:50 -04:00
11 changed files with 409 additions and 18 deletions

View File

@ -0,0 +1,45 @@
import { toPuny } from '../built/misc/convert-host.js';
export class channelBoxes1653969505036 {
name = 'channelBoxes1653969505036'
async up(queryRunner) {
await queryRunner.query(`ALTER TABLE "channel" ADD "slug" character varying(512)`);
await queryRunner.query(`COMMENT ON COLUMN "channel"."slug" IS 'URL safe channel name.'`);
const channels = await queryRunner.query(`SELECT id, name FROM "channel"`);
for (let i = 0; i < channels.length; i++) {
const channelId = channels[i].id;
const channelName = channels[i].name;
const slug = toPuny(channelName.replace(' ', '-'));
await queryRunner.query(`UPDATE "channel" SET "slug"='${slug}' WHERE "id" = '${channelId}'`);
}
await queryRunner.query(`ALTER TABLE "channel" ALTER COLUMN "slug" SET NOT NULL`);
await queryRunner.query(`ALTER TABLE "channel" ADD "host" character varying(128)`);
await queryRunner.query(`COMMENT ON COLUMN "channel"."host" IS 'The host of the Channel. It will be null if the origin of the channel is local.'`);
await queryRunner.query(`ALTER TABLE "channel" ADD "inbox" character varying(512)`);
await queryRunner.query(`COMMENT ON COLUMN "channel"."inbox" IS 'The inbox URL of the Channel. It will be null if the origin of the channel is local.'`);
await queryRunner.query(`ALTER TABLE "channel" ADD "sharedInbox" character varying(512)`);
await queryRunner.query(`COMMENT ON COLUMN "channel"."sharedInbox" IS 'The sharedInbox URL of the Channel. It will be null if the origin of the channel is local.'`);
await queryRunner.query(`ALTER TABLE "channel" ADD "uri" character varying(512)`);
await queryRunner.query(`COMMENT ON COLUMN "channel"."uri" IS 'The URI of the Channel. It will be null if the origin of the channel is local.'`);
await queryRunner.query(`ALTER TABLE "channel" ADD "followersUri" character varying(512)`);
await queryRunner.query(`COMMENT ON COLUMN "channel"."followersUri" IS 'The URI of the channel Follower Collection. It will be null if the origin of the channel is local.'`);
await queryRunner.query(`CREATE INDEX "IDX_8479b8802f0cd5d60c61a5ad20" ON "channel" ("slug") `);
await queryRunner.query(`CREATE INDEX "IDX_ddc763034a97bf9035c8a6bd12" ON "channel" ("host") `);
await queryRunner.query(`CREATE INDEX "IDX_bd0e56edb9ccb54f92b3693d14" ON "channel" ("uri") `);
await queryRunner.query(`CREATE UNIQUE INDEX "IDX_4bfbb3a1ec83af58d47b298942" ON "channel" ("slug", "host") `);
}
async down(queryRunner) {
await queryRunner.query(`ALTER TABLE "channel" DROP COLUMN "followersUri"`);
await queryRunner.query(`ALTER TABLE "channel" DROP COLUMN "uri"`);
await queryRunner.query(`ALTER TABLE "channel" DROP COLUMN "sharedInbox"`);
await queryRunner.query(`ALTER TABLE "channel" DROP COLUMN "inbox"`);
await queryRunner.query(`ALTER TABLE "channel" DROP COLUMN "host"`);
await queryRunner.query(`ALTER TABLE "channel" DROP COLUMN "slug"`);
}
}

View File

@ -9,7 +9,7 @@ const userCache = new Cache<UserKeypair>(Infinity);
const channelCache = new Cache<ChannelKeypair>(Infinity); const channelCache = new Cache<ChannelKeypair>(Infinity);
export async function getUserKeypair(userId: User['id']): Promise<UserKeypair> { export async function getUserKeypair(userId: User['id']): Promise<UserKeypair> {
return await cache.fetch(userId, () => UserKeypairs.findOneByOrFail({ userId })); return await userCache.fetch(userId, () => UserKeypairs.findOneByOrFail({ userId }));
} }
export async function getChannelKeypair(channelId: Channel['id']): Promise<ChannelKeypair> { export async function getChannelKeypair(channelId: Channel['id']): Promise<ChannelKeypair> {

View File

@ -4,6 +4,7 @@ import { id } from '../id.js';
import { DriveFile } from './drive-file.js'; import { DriveFile } from './drive-file.js';
@Entity() @Entity()
@Index(['slug', 'host'], { unique: true })
export class Channel { export class Channel {
@PrimaryColumn(id()) @PrimaryColumn(id())
public id: string; public id: string;
@ -40,6 +41,13 @@ export class Channel {
}) })
public name: string; public name: string;
@Index()
@Column('varchar', {
length: 512,
comment: 'URL safe channel name.',
})
public slug: string;
@Column('varchar', { @Column('varchar', {
length: 2048, nullable: true, length: 2048, nullable: true,
comment: 'The description of the Channel.', comment: 'The description of the Channel.',
@ -83,4 +91,36 @@ export class Channel {
comment: 'The count of users.', comment: 'The count of users.',
}) })
public usersCount: number; public usersCount: number;
@Index()
@Column('varchar', {
length: 128, nullable: true,
comment: 'The host of the Channel. It will be null if the origin of the channel is local.',
})
public host: string | null;
@Column('varchar', {
length: 512, nullable: true,
comment: 'The inbox URL of the Channel. It will be null if the origin of the channel is local.',
})
public inbox: string | null;
@Column('varchar', {
length: 512, nullable: true,
comment: 'The sharedInbox URL of the Channel. It will be null if the origin of the channel is local.',
})
public sharedInbox: string | null;
@Index()
@Column('varchar', {
length: 512, nullable: true,
comment: 'The URI of the Channel. It will be null if the origin of the channel is local.',
})
public uri: string | null;
@Column('varchar', {
length: 512, nullable: true,
comment: 'The URI of the channel Follower Collection. It will be null if the origin of the channel is local.',
})
public followersUri: string | null;
} }

View File

@ -0,0 +1,250 @@
import { URL } from 'node:url';
import promiseLimit from 'promise-limit';
import $, { Context } from 'cafy';
import config from '@/config/index.js';
import Resolver from '../resolver.js';
import { resolveImage } from './image.js';
import { isCollectionOrOrderedCollection, isCollection, IActor, getApId, getOneApHrefNullable, IObject, isPropertyValue, IApPropertyValue, getApType, isActor } from '../type.js';
import { fromHtml } from '../../../mfm/from-html.js';
import { htmlToMfm } from '../misc/html-to-mfm.js';
import { resolveNote, extractEmojis } from './note.js';
import { registerOrFetchInstanceDoc } from '@/services/register-or-fetch-instance-doc.js';
import { extractApHashtags } from './tag.js';
import { apLogger } from '../logger.js';
import { Note } from '@/models/entities/note.js';
import { updateUsertags } from '@/services/update-hashtag.js';
import { Users, Instances, DriveFiles, Followings, UserProfiles, UserPublickeys } from '@/models/index.js';
import { User, IRemoteUser, CacheableUser } from '@/models/entities/user.js';
import { Emoji } from '@/models/entities/emoji.js';
import { UserNotePining } from '@/models/entities/user-note-pining.js';
import { genId } from '@/misc/gen-id.js';
import { instanceChart, usersChart } from '@/services/chart/index.js';
import { UserPublickey } from '@/models/entities/user-publickey.js';
import { isDuplicateKeyValueError } from '@/misc/is-duplicate-key-value-error.js';
import { toPuny } from '@/misc/convert-host.js';
import { UserProfile } from '@/models/entities/user-profile.js';
import { toArray } from '@/prelude/array.js';
import { fetchInstanceMetadata } from '@/services/fetch-instance-metadata.js';
import { normalizeForSearch } from '@/misc/normalize-for-search.js';
import { truncate } from '@/misc/truncate.js';
import { StatusError } from '@/misc/fetch.js';
import { uriPersonCache } from '@/services/user-cache.js';
import { publishInternalEvent } from '@/services/stream.js';
import { db } from '@/db/postgre.js';
const logger = apLogger;
const nameLength = 128;
const summaryLength = 2048;
/**
* Validate and convert to actor object
* @param x Fetched object
* @param uri Fetch target URI
*/
function validateActor(x: IObject, uri: string): IActor {
return x;
}
export async function fetchGroup(uri: string, resolver?: Resolver): Promise<CacheableGroup | null> {
if (typeof uri !== 'string') throw new Error('uri is not string');
const exist = await Channel.findOneBy({ uri });
if (exist) {
return exist;
}
return null;
}
export async function createGroup(uri: string, resolver?: Resolver): Promise<Channel> {
if (typeof uri !== 'string') throw new Error('uri is not string');
if (uri.startsWith(config.url)) {
throw new StatusError('cannot resolve local channel', 400, 'cannot resolve local channel');
}
if (resolver == null) resolver = new Resolver();
const object = await resolver.resolve(uri) as any;
const group = validateActor(object, uri);
logger.info(`Creating the Group: ${group.id}`);
const host = toPuny(new URL(group.id).hostname);
const tags = extractApHashtags(group.tag).map(tag => normalizeForSearch(tag)).splice(0, 32);
// Create channel
let channel: IRemoteChannel;
try {
// Start transaction
await db.transaction(async transactionalEntityManager => {
channel = await transactionalEntityManager.save(new Channel({
id: genId(),
name: truncate(person.name, nameLength),
host,
inbox: person.inbox,
sharedInbox: person.sharedInbox || (person.endpoints ? person.endpoints.sharedInbox : undefined),
followersUri: person.followers ? getApId(person.followers) : undefined,
uri: person.id,
tags,
})) as IRemoteChannel;
// if (person.publicKey) {
// await transactionalEntityManager.save(new UserPublickey({
// userId: user.id,
// keyId: person.publicKey.id,
// keyPem: person.publicKey.publicKeyPem,
// }));
// }
});
} catch (e) {
// duplicate key error
if (isDuplicateKeyValueError(e)) {
throw new Error('already registered');
} else {
logger.error(e instanceof Error ? e : new Error(e as string));
throw e;
}
}
// Register host
registerOrFetchInstanceDoc(host).then(i => {
fetchInstanceMetadata(i);
});
//#region アバターとヘッダー画像をフェッチ
let banner;
if (banner) {
banner = await resolveImage(channel!, group.image).catch(() => null)
}
const bannerId = banner ? banner.id : null;
await Channel.update(channel!.id, {
bannerId,
});
channel!.bannerId = bannerId;
//#endregion
//#region カスタム絵文字取得
const emojis = await extractEmojis(group.tag || [], host).catch(e => {
logger.info(`extractEmojis: ${e}`);
return [] as Emoji[];
});
const emojiNames = emojis.map(emoji => emoji.name);
await Channel.update(channel!.id, {
emojis: emojiNames,
});
//#endregion
return channel!;
}
/**
* Personの情報を更新します
* Misskeyに対象のPersonが登録されていなければ無視します
* @param uri URI of Person
* @param resolver Resolver
* @param hint Hint of Person object (Personの場合Remote resolveをせずに更新に利用します)
*/
export async function updateGroup(uri: string, resolver?: Resolver | null, hint?: Record<string, unknown>): Promise<void> {
if (typeof uri !== 'string') throw new Error('uri is not string');
// URIがこのサーバーを指しているならスキップ
if (uri.startsWith(config.url + '/')) {
return;
}
//#region このサーバーに既に登録されているか
const exist = await Channel.findOneBy({ uri }) as IRemoteChannel;
if (exist == null) {
return;
}
//#endregion
if (resolver == null) resolver = new Resolver();
const object = hint || await resolver.resolve(uri) as any;
const group = validateActor(object, uri);
logger.info(`Updating the Group: ${group.id}`);
// アバターとヘッダー画像をフェッチ
let banner;
if (banner) {
banner = await resolveImage(channel!, group.image).catch(() => null)
}
// カスタム絵文字取得
const emojis = await extractEmojis(group.tag || [], exist.host).catch(e => {
logger.info(`extractEmojis: ${e}`);
return [] as Emoji[];
});
const emojiNames = emojis.map(emoji => emoji.name);
const tags = extractApHashtags(group.tag).map(tag => normalizeForSearch(tag)).splice(0, 32);
const updates = {
inbox: group.inbox,
sharedInbox: group.sharedInbox || (group.endpoints ? group.endpoints.sharedInbox : undefined),
followersUri: group.followers ? getApId(group.followers) : undefined,
emojis: emojiNames,
name: truncate(group.name, nameLength),
tags,
} as Partial<User>;
if (banner) {
updates.bannerId = banner.id;
}
// Update user
await Channel.update(exist.id, updates);
// if (person.publicKey) {
// await UserPublickeys.update({ userId: exist.id }, {
// keyId: person.publicKey.id,
// keyPem: person.publicKey.publicKeyPem,
// });
// }
// TODO: investigate if this should happen for channels aswell
// publishInternalEvent('remoteUserUpdated', { id: exist.id });
await Followings.update({
followerId: exist.id,
}, {
followerSharedInbox: group.sharedInbox || (group.endpoints ? group.endpoints.sharedInbox : undefined),
});
}
/**
* Personを解決します
*
* Misskeyに対象のPersonが登録されていればそれを返し
* Misskeyに登録しそれを返します
*/
export async function resolveGroup(uri: string, resolver?: Resolver): Promise<CacheableGroup> {
if (typeof uri !== 'string') throw new Error('uri is not string');
//#region このサーバーに既に登録されていたらそれを返す
const exist = await fetchGroup(uri);
if (exist) {
return exist;
}
//#endregion
// リモートサーバーからフェッチしてきて登録
if (resolver == null) resolver = new Resolver();
return await createGroup(uri, resolver);
}

View File

@ -5,6 +5,7 @@ import Resolver from '../resolver.js';
import post from '@/services/note/create.js'; import post from '@/services/note/create.js';
import { resolvePerson, updatePerson } from './person.js'; import { resolvePerson, updatePerson } from './person.js';
import { resolveImage } from './image.js'; import { resolveImage } from './image.js';
import { resolveGroup, updateGroup } from './group.js';
import { CacheableRemoteUser, IRemoteUser } from '@/models/entities/user.js'; import { CacheableRemoteUser, IRemoteUser } from '@/models/entities/user.js';
import { htmlToMfm } from '../misc/html-to-mfm.js'; import { htmlToMfm } from '../misc/html-to-mfm.js';
import { extractApHashtags } from './tag.js'; import { extractApHashtags } from './tag.js';
@ -226,6 +227,11 @@ export async function createNote(value: string | IObject, resolver?: Resolver, s
return [] as Emoji[]; return [] as Emoji[];
}); });
const channel = await extractChannel(note.tag || [], actor.host).catch(e => {
logger.info(`extractChannel: ${e}`);
return null;
});
const apEmojis = emojis.map(emoji => emoji.name); const apEmojis = emojis.map(emoji => emoji.name);
const poll = await extractPollFromQuestion(note, resolver).catch(() => undefined); const poll = await extractPollFromQuestion(note, resolver).catch(() => undefined);
@ -252,6 +258,7 @@ export async function createNote(value: string | IObject, resolver?: Resolver, s
apHashtags, apHashtags,
apEmojis, apEmojis,
poll, poll,
channel,
uri: note.id, uri: note.id,
url: getOneApHrefNullable(note.url), url: getOneApHrefNullable(note.url),
}, silent); }, silent);
@ -350,3 +357,14 @@ export async function extractEmojis(tags: IObject | IObject[], host: string): Pr
} as Partial<Emoji>).then(x => Emojis.findOneByOrFail(x.identifiers[0])); } as Partial<Emoji>).then(x => Emojis.findOneByOrFail(x.identifiers[0]));
})); }));
} }
export async function extractChannel(tags: IObject | IObject[], host:string): Promise<Channel | null> {
const apGroup = toArray(tags).find(isGroup);
if (!apGroup) {
return null;
}
return await resolveGroup(getOneApId(apGroup), resolver) as CacheableRemoteGroup;
}

View File

@ -38,7 +38,7 @@ export async function renderGroup(channel: Channel) {
following: `${id}/following`, // this is required by spec. cant be empty. sadge following: `${id}/following`, // this is required by spec. cant be empty. sadge
sharedInbox: `${config.url}/inbox`, sharedInbox: `${config.url}/inbox`,
endpoints: { sharedInbox: `${config.url}/inbox` }, endpoints: { sharedInbox: `${config.url}/inbox` },
url: id, url: `${config.url}/+${channel.slug}`,
name: channel.name, name: channel.name,
summary: channel.description ? toHtml(mfm.parse(channel.description)) : null, summary: channel.description ? toHtml(mfm.parse(channel.description)) : null,
icon: null, icon: null,

View File

@ -2,6 +2,7 @@ import renderDocument from './document.js';
import renderHashtag from './hashtag.js'; import renderHashtag from './hashtag.js';
import renderMention from './mention.js'; import renderMention from './mention.js';
import renderEmoji from './emoji.js'; import renderEmoji from './emoji.js';
import renderGroup from './group.js';
import config from '@/config/index.js'; import config from '@/config/index.js';
import toHtml from '../misc/get-note-html.js'; import toHtml from '../misc/get-note-html.js';
import { Note, IMentionedRemoteUsers } from '@/models/entities/note.js'; import { Note, IMentionedRemoteUsers } from '@/models/entities/note.js';
@ -53,6 +54,12 @@ export default async function renderNote(note: Note, dive = true, isTalk = false
} }
} }
let channel: Channel | null;
if (note.channelId) {
channel = await Channel.findOneBy({ id: note.channelId });
}
const attributedTo = `${config.url}/users/${note.userId}`; const attributedTo = `${config.url}/users/${note.userId}`;
const mentions = (JSON.parse(note.mentionedRemoteUsers) as IMentionedRemoteUsers).map(x => x.uri); const mentions = (JSON.parse(note.mentionedRemoteUsers) as IMentionedRemoteUsers).map(x => x.uri);
@ -60,9 +67,10 @@ export default async function renderNote(note: Note, dive = true, isTalk = false
let to: string[] = []; let to: string[] = [];
let cc: string[] = []; let cc: string[] = [];
if (note.channelId) { if (channel) {
const channel = `${config.url}/channels/${note.channelId}`; const channelUrl = `${config.url}/channels/${channel.id}`;
to = [channel]; // to = [`${channelUrl}/followers`];
to = [channelUrl];
cc = [`${attributedTo}/followers`, 'https://www.w3.org/ns/activitystreams#Public'].concat(mentions); cc = [`${attributedTo}/followers`, 'https://www.w3.org/ns/activitystreams#Public'].concat(mentions);
} else { } else {
if (note.visibility === 'public') { if (note.visibility === 'public') {
@ -112,11 +120,15 @@ export default async function renderNote(note: Note, dive = true, isTalk = false
const emojis = await getEmojis(note.emojis); const emojis = await getEmojis(note.emojis);
const apemojis = emojis.map(emoji => renderEmoji(emoji)); const apemojis = emojis.map(emoji => renderEmoji(emoji));
const tag = [ let tag = [
...hashtagTags, ...hashtagTags,
...mentionTags, ...mentionTags,
...apemojis, ...apemojis,
]; ];
if (channel) {
const apGroup = await renderGroup(channel);
tag.push(apGroup);
}
const asPoll = poll ? { const asPoll = poll ? {
type: 'Question', type: 'Question',

View File

@ -216,6 +216,14 @@ export interface IApEmoji extends IObject {
export const isEmoji = (object: IObject): object is IApEmoji => export const isEmoji = (object: IObject): object is IApEmoji =>
getApType(object) === 'Emoji' && !Array.isArray(object.icon) && object.icon.url != null; getApType(object) === 'Emoji' && !Array.isArray(object.icon) && object.icon.url != null;
export interface IApGroup extends IObject {
type: 'Group';
id: String;
}
export const isGroup = (object: IObject): object is IApGroup =>
getApType(object) === 'Group' && typeof object.id === 'string';
export interface ICreate extends IActivity { export interface ICreate extends IActivity {
type: 'Create'; type: 'Create';
} }

View File

@ -226,21 +226,38 @@ router.get('/likes/:like', async ctx => {
}); });
// channel // channel
router.get('/channels/:channelId', async (ctx, next) => { async function channelInfo(ctx: Router.RouterContext, channel: Channel | null) {
if (!isActivityPubReq(ctx)) return await next();
const channel = await Channels.findOneBy({
id: ctx.params.channelId,
});
if (channel == null) { if (channel == null) {
ctx.status = 404; ctx.status = 404;
return; return;
} }
ctx.body = renderActivity(await renderGroup(channel)); ctx.body = renderActivity(await renderGroup(channel as ILocalChannel));
ctx.set('Cache-Control', 'public, max-age=180'); ctx.set('Cache-Control', 'public, max-age=180');
setResponseType(ctx); setResponseType(ctx);
}
router.get('/channels/:channelId', async (ctx, next) => {
if (!isActivityPubReq(ctx)) return await next();
const channel = await Channels.findOneBy({
id: ctx.params.channelId,
host: IsNull(),
});
await channelInfo(ctx, channel);
}); });
router.get('/\\+:channelSlug', async (ctx, next) => {
if (!isActivityPubReq(ctx)) return await next();
const channel = await Channels.findOneBy({
slug: ctx.params.channelSlug.toLowerCase(),
host: IsNull(),
});
await channelInfo(ctx, channel);
});
//#endregion
export default router; export default router;

View File

@ -54,17 +54,18 @@ export default define(meta, paramDef, async (ps, user) => {
} }
} }
const keyPair = genRsaKeyPair(4096); const keyPair = await genRsaKeyPair(4096);
let channel;
await db.transaction(async transactionalEntityManager => { await db.transaction(async transactionalEntityManager => {
const channel = await transactionalEntityManager.insert(Channel, { channel = await transactionalEntityManager.insert(Channel, {
id: genId(), id: genId(),
createdAt: new Date(), createdAt: new Date(),
userId: user.id, userId: user.id,
name: ps.name, name: ps.name,
description: ps.description || null, description: ps.description || null,
bannerId: banner ? banner.id : null, bannerId: banner ? banner.id : null,
}).then(x => Channels.findOneByOrFail(x.identifiers[0])); }).then(x => transactionalEntityManager.findOneByOrFail(Channel, x.identifiers[0]));
await transactionalEntityManager.insert(ChannelKeypair, { await transactionalEntityManager.insert(ChannelKeypair, {
publicKey: keyPair.publicKey, publicKey: keyPair.publicKey,