2023-01-13 05:40:33 +01:00
|
|
|
import * as fs from "node:fs";
|
|
|
|
import type { Schema, SchemaType } from "@/misc/schema.js";
|
|
|
|
import type { AccessToken } from "@/models/entities/access-token.js";
|
2023-11-26 21:33:46 +01:00
|
|
|
import type { CacheableLocalUser } from "@/models/entities/user.js";
|
|
|
|
import Ajv from "ajv";
|
2023-01-13 05:40:33 +01:00
|
|
|
import type { IEndpointMeta } from "./endpoints.js";
|
|
|
|
import { ApiError } from "./error.js";
|
2018-11-02 05:47:44 +01:00
|
|
|
|
2019-02-22 03:46:58 +01:00
|
|
|
export type Response = Record<string, any> | void;
|
|
|
|
|
2022-02-19 06:05:32 +01:00
|
|
|
// TODO: paramsの型をT['params']のスキーマ定義から推論する
|
2023-01-13 05:40:33 +01:00
|
|
|
type executor<T extends IEndpointMeta, Ps extends Schema> = (
|
|
|
|
params: SchemaType<Ps>,
|
|
|
|
user: T["requireCredential"] extends true
|
|
|
|
? CacheableLocalUser
|
|
|
|
: CacheableLocalUser | null,
|
|
|
|
token: AccessToken | null,
|
|
|
|
file?: any,
|
|
|
|
cleanup?: () => any,
|
|
|
|
ip?: string | null,
|
|
|
|
headers?: Record<string, string> | null,
|
|
|
|
) => Promise<
|
|
|
|
T["res"] extends undefined ? Response : SchemaType<NonNullable<T["res"]>>
|
|
|
|
>;
|
2019-04-23 15:35:26 +02:00
|
|
|
|
2022-02-19 06:05:32 +01:00
|
|
|
const ajv = new Ajv({
|
|
|
|
useDefaults: true,
|
|
|
|
});
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
ajv.addFormat("misskey:id", /^[a-zA-Z0-9]+$/);
|
2022-02-19 06:05:32 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
export default function <T extends IEndpointMeta, Ps extends Schema>(
|
|
|
|
meta: T,
|
|
|
|
paramDef: Ps,
|
|
|
|
cb: executor<T, Ps>,
|
|
|
|
): (
|
|
|
|
params: any,
|
|
|
|
user: T["requireCredential"] extends true
|
|
|
|
? CacheableLocalUser
|
|
|
|
: CacheableLocalUser | null,
|
|
|
|
token: AccessToken | null,
|
|
|
|
file?: any,
|
|
|
|
ip?: string | null,
|
|
|
|
headers?: Record<string, string> | null,
|
|
|
|
) => Promise<any> {
|
2022-02-19 06:05:32 +01:00
|
|
|
const validate = ajv.compile(paramDef);
|
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
return (
|
|
|
|
params: any,
|
|
|
|
user: T["requireCredential"] extends true
|
|
|
|
? CacheableLocalUser
|
|
|
|
: CacheableLocalUser | null,
|
|
|
|
token: AccessToken | null,
|
|
|
|
file?: any,
|
|
|
|
ip?: string | null,
|
|
|
|
headers?: Record<string, string> | null,
|
|
|
|
) => {
|
2022-07-02 08:12:11 +02:00
|
|
|
let cleanup: undefined | (() => void) = undefined;
|
|
|
|
|
|
|
|
if (meta.requireFile) {
|
|
|
|
cleanup = () => {
|
|
|
|
fs.unlink(file.path, () => {});
|
|
|
|
};
|
2018-11-02 05:47:44 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
if (file == null)
|
|
|
|
return Promise.reject(
|
|
|
|
new ApiError({
|
|
|
|
message: "File required.",
|
|
|
|
code: "FILE_REQUIRED",
|
|
|
|
id: "4267801e-70d1-416a-b011-4ee502885d8b",
|
|
|
|
}),
|
|
|
|
);
|
2022-07-02 08:12:11 +02:00
|
|
|
}
|
2018-11-02 05:47:44 +01:00
|
|
|
|
2022-02-19 06:05:32 +01:00
|
|
|
const valid = validate(params);
|
|
|
|
if (!valid) {
|
2022-07-02 08:12:11 +02:00
|
|
|
if (file) cleanup!();
|
2018-11-02 05:47:44 +01:00
|
|
|
|
2022-02-19 06:05:32 +01:00
|
|
|
const errors = validate.errors!;
|
2023-01-13 05:40:33 +01:00
|
|
|
const err = new ApiError(
|
|
|
|
{
|
|
|
|
message: "Invalid param.",
|
|
|
|
code: "INVALID_PARAM",
|
|
|
|
id: "3d81ceae-475f-4600-b2a8-2bc116157532",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
param: errors[0].schemaPath,
|
|
|
|
reason: errors[0].message,
|
|
|
|
},
|
|
|
|
);
|
2022-02-19 06:05:32 +01:00
|
|
|
return Promise.reject(err);
|
2018-11-02 05:47:44 +01:00
|
|
|
}
|
2022-02-19 06:05:32 +01:00
|
|
|
|
2023-01-13 05:40:33 +01:00
|
|
|
return cb(
|
|
|
|
params as SchemaType<Ps>,
|
|
|
|
user,
|
|
|
|
token,
|
|
|
|
file,
|
|
|
|
cleanup,
|
|
|
|
ip,
|
|
|
|
headers,
|
|
|
|
);
|
2022-02-19 06:05:32 +01:00
|
|
|
};
|
2018-11-02 05:47:44 +01:00
|
|
|
}
|