forked from mirror/misskey
173 lines
5.5 KiB
TypeScript
173 lines
5.5 KiB
TypeScript
/*
|
|
* SPDX-FileCopyrightText: syuilo and misskey-project
|
|
* SPDX-License-Identifier: AGPL-3.0-only
|
|
*/
|
|
|
|
import { Inject, Injectable } from '@nestjs/common';
|
|
import { DI } from '@/di-symbols.js';
|
|
import type { Config } from '@/config.js';
|
|
import { MetaService } from '@/core/MetaService.js';
|
|
import { MAX_NOTE_TEXT_LENGTH } from '@/const.js';
|
|
import { MemorySingleCache } from '@/misc/cache.js';
|
|
import { UserEntityService } from '@/core/entities/UserEntityService.js';
|
|
import { bindThis } from '@/decorators.js';
|
|
import NotesChart from '@/core/chart/charts/notes.js';
|
|
import UsersChart from '@/core/chart/charts/users.js';
|
|
import { DEFAULT_POLICIES } from '@/core/RoleService.js';
|
|
import type { FastifyInstance, FastifyPluginOptions } from 'fastify';
|
|
|
|
const nodeinfo2_1path = '/nodeinfo/2.1';
|
|
const nodeinfo2_0path = '/nodeinfo/2.0';
|
|
const nodeinfo_homepage = 'https://misskey-hub.net';
|
|
|
|
@Injectable()
|
|
export class NodeinfoServerService {
|
|
constructor(
|
|
@Inject(DI.config)
|
|
private config: Config,
|
|
|
|
private userEntityService: UserEntityService,
|
|
private metaService: MetaService,
|
|
private notesChart: NotesChart,
|
|
private usersChart: UsersChart,
|
|
) {
|
|
//this.createServer = this.createServer.bind(this);
|
|
}
|
|
|
|
@bindThis
|
|
public getLinks() {
|
|
return [{
|
|
rel: 'http://nodeinfo.diaspora.software/ns/schema/2.1',
|
|
href: this.config.url + nodeinfo2_1path
|
|
}, {
|
|
rel: 'http://nodeinfo.diaspora.software/ns/schema/2.0',
|
|
href: this.config.url + nodeinfo2_0path,
|
|
}];
|
|
}
|
|
|
|
@bindThis
|
|
public createServer(fastify: FastifyInstance, options: FastifyPluginOptions, done: (err?: Error) => void) {
|
|
const nodeinfo2 = async (version: number) => {
|
|
const now = Date.now();
|
|
|
|
const notesChart = await this.notesChart.getChart('hour', 1, null);
|
|
const localPosts = notesChart.local.total[0];
|
|
|
|
const usersChart = await this.usersChart.getChart('hour', 1, null);
|
|
const total = usersChart.local.total[0];
|
|
|
|
const [
|
|
meta,
|
|
//activeHalfyear,
|
|
//activeMonth,
|
|
] = await Promise.all([
|
|
this.metaService.fetch(true),
|
|
// 重い
|
|
//this.usersRepository.count({ where: { host: IsNull(), lastActiveDate: MoreThan(new Date(now - 15552000000)) } }),
|
|
//this.usersRepository.count({ where: { host: IsNull(), lastActiveDate: MoreThan(new Date(now - 2592000000)) } }),
|
|
]);
|
|
|
|
const activeHalfyear = null;
|
|
const activeMonth = null;
|
|
|
|
const proxyAccount = meta.proxyAccountId ? await this.userEntityService.pack(meta.proxyAccountId, null).catch(() => null) : null;
|
|
|
|
const basePolicies = { ...DEFAULT_POLICIES, ...meta.policies };
|
|
|
|
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
const document: any = {
|
|
software: {
|
|
name: 'misskey',
|
|
version: this.config.version,
|
|
homepage: nodeinfo_homepage,
|
|
repository: meta.repositoryUrl,
|
|
},
|
|
protocols: ['activitypub'],
|
|
services: {
|
|
inbound: [] as string[],
|
|
outbound: ['atom1.0', 'rss2.0'],
|
|
},
|
|
openRegistrations: !meta.disableRegistration,
|
|
usage: {
|
|
users: { total, activeHalfyear, activeMonth },
|
|
localPosts,
|
|
localComments: 0,
|
|
},
|
|
metadata: {
|
|
nodeName: meta.name,
|
|
nodeDescription: meta.description,
|
|
nodeAdmins: [{
|
|
name: meta.maintainerName,
|
|
email: meta.maintainerEmail,
|
|
}],
|
|
// deprecated
|
|
maintainer: {
|
|
name: meta.maintainerName,
|
|
email: meta.maintainerEmail,
|
|
},
|
|
langs: meta.langs,
|
|
tosUrl: meta.termsOfServiceUrl,
|
|
privacyPolicyUrl: meta.privacyPolicyUrl,
|
|
impressumUrl: meta.impressumUrl,
|
|
repositoryUrl: meta.repositoryUrl,
|
|
feedbackUrl: meta.feedbackUrl,
|
|
disableRegistration: meta.disableRegistration,
|
|
disableLocalTimeline: !basePolicies.ltlAvailable,
|
|
disableGlobalTimeline: !basePolicies.gtlAvailable,
|
|
emailRequiredForSignup: meta.emailRequiredForSignup,
|
|
enableHcaptcha: meta.enableHcaptcha,
|
|
enableRecaptcha: meta.enableRecaptcha,
|
|
enableMcaptcha: meta.enableMcaptcha,
|
|
enableTurnstile: meta.enableTurnstile,
|
|
maxNoteTextLength: MAX_NOTE_TEXT_LENGTH,
|
|
enableEmail: meta.enableEmail,
|
|
enableServiceWorker: meta.enableServiceWorker,
|
|
proxyAccountName: proxyAccount ? proxyAccount.username : null,
|
|
themeColor: meta.themeColor ?? '#86b300',
|
|
},
|
|
};
|
|
if (version >= 21) {
|
|
document.software.repository = meta.repositoryUrl;
|
|
document.software.homepage = meta.repositoryUrl;
|
|
}
|
|
return document;
|
|
};
|
|
|
|
const cache = new MemorySingleCache<Awaited<ReturnType<typeof nodeinfo2>>>(1000 * 60 * 10);
|
|
|
|
fastify.get(nodeinfo2_1path, async (request, reply) => {
|
|
const base = await cache.fetch(() => nodeinfo2(21));
|
|
|
|
reply
|
|
.type(
|
|
'application/json; profile="http://nodeinfo.diaspora.software/ns/schema/2.1#"',
|
|
)
|
|
.header('Cache-Control', 'public, max-age=600')
|
|
.header('Access-Control-Allow-Headers', 'Accept')
|
|
.header('Access-Control-Allow-Methods', 'GET, OPTIONS')
|
|
.header('Access-Control-Allow-Origin', '*')
|
|
.header('Access-Control-Expose-Headers', 'Vary');
|
|
return { version: '2.1', ...base };
|
|
});
|
|
|
|
fastify.get(nodeinfo2_0path, async (request, reply) => {
|
|
const base = await cache.fetch(() => nodeinfo2(20));
|
|
|
|
delete (base as any).software.repository;
|
|
|
|
reply
|
|
.type(
|
|
'application/json; profile="http://nodeinfo.diaspora.software/ns/schema/2.0#"',
|
|
)
|
|
.header('Cache-Control', 'public, max-age=600')
|
|
.header('Access-Control-Allow-Headers', 'Accept')
|
|
.header('Access-Control-Allow-Methods', 'GET, OPTIONS')
|
|
.header('Access-Control-Allow-Origin', '*')
|
|
.header('Access-Control-Expose-Headers', 'Vary');
|
|
return { version: '2.0', ...base };
|
|
});
|
|
|
|
done();
|
|
}
|
|
}
|