2022-09-17 20:27:08 +02:00
|
|
|
import * as fs from 'node:fs';
|
|
|
|
import Ajv from 'ajv';
|
|
|
|
import type { Schema, SchemaType } from '@/misc/schema.js';
|
|
|
|
import type { CacheableLocalUser } from '@/models/entities/User.js';
|
|
|
|
import type { AccessToken } from '@/models/entities/AccessToken.js';
|
|
|
|
import { ApiError } from './error.js';
|
|
|
|
import type { IEndpointMeta } from './endpoints.js';
|
|
|
|
|
|
|
|
const ajv = new Ajv({
|
|
|
|
useDefaults: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
ajv.addFormat('misskey:id', /^[a-zA-Z0-9]+$/);
|
|
|
|
|
|
|
|
export type Response = Record<string, any> | void;
|
|
|
|
|
2022-12-03 11:42:05 +01:00
|
|
|
type File = {
|
|
|
|
name: string | null;
|
|
|
|
path: string;
|
|
|
|
};
|
|
|
|
|
2022-09-17 20:27:08 +02:00
|
|
|
// TODO: paramsの型をT['params']のスキーマ定義から推論する
|
|
|
|
type executor<T extends IEndpointMeta, Ps extends Schema> =
|
2022-12-03 11:42:05 +01:00
|
|
|
(params: SchemaType<Ps>, user: T['requireCredential'] extends true ? CacheableLocalUser : CacheableLocalUser | null, token: AccessToken | null, file?: File, cleanup?: () => any, ip?: string | null, headers?: Record<string, string> | null) =>
|
2022-09-17 20:27:08 +02:00
|
|
|
Promise<T['res'] extends undefined ? Response : SchemaType<NonNullable<T['res']>>>;
|
|
|
|
|
|
|
|
export abstract class Endpoint<T extends IEndpointMeta, Ps extends Schema> {
|
2022-12-03 11:42:05 +01:00
|
|
|
public exec: (params: any, user: T['requireCredential'] extends true ? CacheableLocalUser : CacheableLocalUser | null, token: AccessToken | null, file?: File, ip?: string | null, headers?: Record<string, string> | null) => Promise<any>;
|
2022-09-17 20:27:08 +02:00
|
|
|
|
|
|
|
constructor(meta: T, paramDef: Ps, cb: executor<T, Ps>) {
|
|
|
|
const validate = ajv.compile(paramDef);
|
|
|
|
|
2022-12-03 11:42:05 +01:00
|
|
|
this.exec = (params: any, user: T['requireCredential'] extends true ? CacheableLocalUser : CacheableLocalUser | null, token: AccessToken | null, file?: File, ip?: string | null, headers?: Record<string, string> | null) => {
|
2022-09-17 20:27:08 +02:00
|
|
|
let cleanup: undefined | (() => void) = undefined;
|
|
|
|
|
|
|
|
if (meta.requireFile) {
|
|
|
|
cleanup = () => {
|
2022-12-03 11:42:05 +01:00
|
|
|
if (file) fs.unlink(file.path, () => {});
|
2022-09-17 20:27:08 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
if (file == null) return Promise.reject(new ApiError({
|
|
|
|
message: 'File required.',
|
|
|
|
code: 'FILE_REQUIRED',
|
|
|
|
id: '4267801e-70d1-416a-b011-4ee502885d8b',
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
const valid = validate(params);
|
|
|
|
if (!valid) {
|
|
|
|
if (file) cleanup!();
|
|
|
|
|
|
|
|
const errors = validate.errors!;
|
|
|
|
const err = new ApiError({
|
|
|
|
message: 'Invalid param.',
|
|
|
|
code: 'INVALID_PARAM',
|
|
|
|
id: '3d81ceae-475f-4600-b2a8-2bc116157532',
|
|
|
|
}, {
|
|
|
|
param: errors[0].schemaPath,
|
|
|
|
reason: errors[0].message,
|
|
|
|
});
|
|
|
|
return Promise.reject(err);
|
|
|
|
}
|
|
|
|
|
|
|
|
return cb(params as SchemaType<Ps>, user, token, file, cleanup, ip, headers);
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|