196 lines
5.1 KiB
TypeScript
196 lines
5.1 KiB
TypeScript
/*
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
import { Inject, Injectable, OnApplicationShutdown } from '@nestjs/common';
|
|
import { DI } from '@/di-symbols.js';
|
|
import type { NotesRepository, UserPublickeysRepository, UsersRepository } from '@/models/_.js';
|
|
import type { Config } from '@/config.js';
|
|
import { MemoryKVCache } from '@/misc/cache.js';
|
|
import type { MiUserPublickey } from '@/models/UserPublickey.js';
|
|
import { CacheService } from '@/core/CacheService.js';
|
|
import type { MiNote } from '@/models/Note.js';
|
|
import { bindThis } from '@/decorators.js';
|
|
import type { MiLocalUser, MiRemoteUser } from '@/models/User.js';
|
|
import { getApId } from './type.js';
|
|
import { ApPersonService } from './models/ApPersonService.js';
|
|
import type { IObject } from './type.js';
|
|
|
|
export type UriParseResult = {
|
|
/** wether the URI was generated by us */
|
|
local: true;
|
|
/** id in DB */
|
|
id: string;
|
|
/** hint of type, e.g. "notes", "users" */
|
|
type: string;
|
|
/** any remaining text after type and id, not including the slash after id. undefined if empty */
|
|
rest?: string;
|
|
} | {
|
|
/** wether the URI was generated by us */
|
|
local: false;
|
|
/** uri in DB */
|
|
uri: string;
|
|
};
|
|
|
|
@Injectable()
|
|
export class ApDbResolverService implements OnApplicationShutdown {
|
|
private publicKeyCache: MemoryKVCache<MiUserPublickey | null>;
|
|
private publicKeyByUserIdCache: MemoryKVCache<MiUserPublickey | null>;
|
|
|
|
constructor(
|
|
@Inject(DI.config)
|
|
private config: Config,
|
|
|
|
@Inject(DI.usersRepository)
|
|
private usersRepository: UsersRepository,
|
|
|
|
@Inject(DI.notesRepository)
|
|
private notesRepository: NotesRepository,
|
|
|
|
@Inject(DI.userPublickeysRepository)
|
|
private userPublickeysRepository: UserPublickeysRepository,
|
|
|
|
private cacheService: CacheService,
|
|
private apPersonService: ApPersonService,
|
|
) {
|
|
this.publicKeyCache = new MemoryKVCache<MiUserPublickey | null>(Infinity);
|
|
this.publicKeyByUserIdCache = new MemoryKVCache<MiUserPublickey | null>(Infinity);
|
|
}
|
|
|
|
@bindThis
|
|
public parseUri(value: string | IObject): UriParseResult {
|
|
const separator = '/';
|
|
|
|
const uri = new URL(getApId(value));
|
|
if (uri.origin !== this.config.url) return { local: false, uri: uri.href };
|
|
|
|
const [, type, id, ...rest] = uri.pathname.split(separator);
|
|
return {
|
|
local: true,
|
|
type,
|
|
id,
|
|
rest: rest.length === 0 ? undefined : rest.join(separator),
|
|
};
|
|
}
|
|
|
|
/**
|
|
* AP Note => Misskey Note in DB
|
|
*/
|
|
@bindThis
|
|
public async getNoteFromApId(value: string | IObject): Promise<MiNote | null> {
|
|
const parsed = this.parseUri(value);
|
|
|
|
if (parsed.local) {
|
|
if (parsed.type !== 'notes') return null;
|
|
|
|
return await this.notesRepository.findOneBy({
|
|
id: parsed.id,
|
|
});
|
|
} else {
|
|
return await this.notesRepository.findOneBy({
|
|
uri: parsed.uri,
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* AP Person => Misskey User in DB
|
|
*/
|
|
@bindThis
|
|
public async getUserFromApId(value: string | IObject): Promise<MiLocalUser | MiRemoteUser | null> {
|
|
const parsed = this.parseUri(value);
|
|
|
|
if (parsed.local) {
|
|
if (parsed.type !== 'users') return null;
|
|
|
|
return await this.cacheService.userByIdCache.fetchMaybe(
|
|
parsed.id,
|
|
() => this.usersRepository.findOneBy({ id: parsed.id, isDeleted: false }).then(x => x ?? undefined),
|
|
) as MiLocalUser | undefined ?? null;
|
|
} else {
|
|
return await this.cacheService.uriPersonCache.fetch(
|
|
parsed.uri,
|
|
() => this.usersRepository.findOneBy({ uri: parsed.uri, isDeleted: false }),
|
|
) as MiRemoteUser | null;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* AP KeyId => Misskey User and Key
|
|
*/
|
|
@bindThis
|
|
public async getAuthUserFromKeyId(keyId: string): Promise<{
|
|
user: MiRemoteUser;
|
|
key: MiUserPublickey;
|
|
} | null> {
|
|
const key = await this.publicKeyCache.fetch(keyId, async () => {
|
|
const key = await this.userPublickeysRepository.findOneBy({
|
|
keyId,
|
|
});
|
|
|
|
if (key == null) return null;
|
|
|
|
return key;
|
|
}, key => key != null);
|
|
|
|
if (key == null) return null;
|
|
|
|
const user = await this.cacheService.findUserById(key.userId).catch(() => null) as MiRemoteUser | null;
|
|
if (user == null) return null;
|
|
if (user.isDeleted) return null;
|
|
|
|
return {
|
|
user,
|
|
key,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* AP Actor id => Misskey User and Key
|
|
*/
|
|
@bindThis
|
|
public async getAuthUserFromApId(uri: string): Promise<{
|
|
user: MiRemoteUser;
|
|
key: MiUserPublickey | null;
|
|
} | null> {
|
|
const user = await this.apPersonService.resolvePerson(uri) as MiRemoteUser;
|
|
if (user.isDeleted) return null;
|
|
|
|
const key = await this.publicKeyByUserIdCache.fetch(
|
|
user.id,
|
|
() => this.userPublickeysRepository.findOneBy({ userId: user.id }),
|
|
v => v != null,
|
|
);
|
|
|
|
return {
|
|
user,
|
|
key,
|
|
};
|
|
}
|
|
|
|
/**
|
|
* Sharkey User -> Refetched Key
|
|
*/
|
|
@bindThis
|
|
public async refetchPublicKeyForApId(user: MiRemoteUser): Promise<MiUserPublickey | null> {
|
|
await this.apPersonService.updatePerson(user.uri);
|
|
const key = await this.userPublickeysRepository.findOneBy({ userId: user.id });
|
|
if (key != null) {
|
|
await this.publicKeyByUserIdCache.set(user.id, key);
|
|
}
|
|
return key;
|
|
}
|
|
|
|
@bindThis
|
|
public dispose(): void {
|
|
this.publicKeyCache.dispose();
|
|
this.publicKeyByUserIdCache.dispose();
|
|
}
|
|
|
|
@bindThis
|
|
public onApplicationShutdown(signal?: string | undefined): void {
|
|
this.dispose();
|
|
}
|
|
}
|