8d29df64fa
we were inconsistent with the colons, thanks to usedbunny for noticing!
110 lines
3.1 KiB
TypeScript
110 lines
3.1 KiB
TypeScript
/*
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
import { URL } from 'node:url';
|
|
import { Injectable } from '@nestjs/common';
|
|
import { XMLParser } from 'fast-xml-parser';
|
|
import { HttpRequestService } from '@/core/HttpRequestService.js';
|
|
import { bindThis } from '@/decorators.js';
|
|
|
|
export type ILink = {
|
|
href: string;
|
|
rel?: string;
|
|
};
|
|
|
|
export type IWebFinger = {
|
|
links: ILink[];
|
|
subject: string;
|
|
};
|
|
|
|
const urlRegex = /^https?:\/\//;
|
|
const mRegex = /^([^@]+)@(.*)/;
|
|
|
|
// we have the colons here, because URL.protocol does as well, so it's
|
|
// more uniform in the places we use both
|
|
const defaultProtocol = process.env.MISSKEY_WEBFINGER_USE_HTTP?.toLowerCase() === 'true' ? 'http:' : 'https:';
|
|
|
|
@Injectable()
|
|
export class WebfingerService {
|
|
constructor(
|
|
private httpRequestService: HttpRequestService,
|
|
) {
|
|
}
|
|
|
|
@bindThis
|
|
public async webfinger(query: string): Promise<IWebFinger> {
|
|
const hostMetaUrl = this.queryToHostMetaUrl(query);
|
|
const template = await this.fetchWebFingerTemplateFromHostMeta(hostMetaUrl) ?? this.queryToWebFingerTemplate(query);
|
|
const url = this.genUrl(query, template);
|
|
|
|
return await this.httpRequestService.getJson<IWebFinger>(url, 'application/jrd+json, application/json');
|
|
}
|
|
|
|
@bindThis
|
|
private genUrl(query: string, template: string): string {
|
|
if (template.indexOf('{uri}') < 0) throw new Error(`Invalid webFingerUrl: ${template}`);
|
|
|
|
if (query.match(urlRegex)) {
|
|
return template.replace('{uri}', encodeURIComponent(query));
|
|
}
|
|
|
|
const m = query.match(mRegex);
|
|
if (m) {
|
|
return template.replace('{uri}', encodeURIComponent(`acct:${query}`));
|
|
}
|
|
|
|
throw new Error(`Invalid query (${query})`);
|
|
}
|
|
|
|
@bindThis
|
|
private queryToWebFingerTemplate(query: string): string {
|
|
if (query.match(urlRegex)) {
|
|
const u = new URL(query);
|
|
return `${u.protocol}//${u.hostname}/.well-known/webfinger?resource={uri}`;
|
|
}
|
|
|
|
const m = query.match(mRegex);
|
|
if (m) {
|
|
const hostname = m[2];
|
|
return `${defaultProtocol}//${hostname}/.well-known/webfinger?resource={uri}`;
|
|
}
|
|
|
|
throw new Error(`Invalid query (${query})`);
|
|
}
|
|
|
|
@bindThis
|
|
private queryToHostMetaUrl(query: string): string {
|
|
if (query.match(urlRegex)) {
|
|
const u = new URL(query);
|
|
return `${u.protocol}//${u.hostname}/.well-known/host-meta`;
|
|
}
|
|
|
|
const m = query.match(mRegex);
|
|
if (m) {
|
|
const hostname = m[2];
|
|
return `${defaultProtocol}//${hostname}/.well-known/host-meta`;
|
|
}
|
|
|
|
throw new Error(`Invalid query (${query})`);
|
|
}
|
|
|
|
@bindThis
|
|
private async fetchWebFingerTemplateFromHostMeta(url: string): Promise<string | null> {
|
|
try {
|
|
const res = await this.httpRequestService.getHtml(url, 'application/xrd+xml');
|
|
const options = {
|
|
ignoreAttributes: false,
|
|
isArray: (_name: string, jpath: string) => jpath === 'XRD.Link',
|
|
};
|
|
const parser = new XMLParser(options);
|
|
const hostMeta = parser.parse(res);
|
|
const template = (hostMeta['XRD']['Link'] as Array<any>).filter(p => p['@_rel'] === 'lrdd')[0]['@_template'];
|
|
return template.indexOf('{uri}') < 0 ? null : template;
|
|
} catch (err) {
|
|
console.error(`error while request host-meta for ${url}: ${err}`);
|
|
return null;
|
|
}
|
|
}
|
|
}
|