wip
This commit is contained in:
parent
361ab296c7
commit
13fe20d47e
4 changed files with 135 additions and 148 deletions
|
@ -1,4 +1,4 @@
|
||||||
import type { Endpoints } from './api.types.js';
|
import type { Endpoints, SchemaOrUndefined, IEndpointMeta, EndpointDefines } from './endpoints.types';
|
||||||
|
|
||||||
const MK_API_ERROR = Symbol();
|
const MK_API_ERROR = Symbol();
|
||||||
|
|
||||||
|
@ -25,15 +25,8 @@ export type FetchLike = (input: string, init?: {
|
||||||
json(): Promise<any>;
|
json(): Promise<any>;
|
||||||
}>;
|
}>;
|
||||||
|
|
||||||
type IsNeverType<T> = [T] extends [never] ? true : false;
|
type Response<D extends IEndpointMeta, P extends SchemaOrUndefined<D['defines'][number]['req']>, DD extends EndpointDefines[number] = D['defines'][number]> =
|
||||||
|
P extends DD['req'] ? SchemaOrUndefined<DD['res']> : never;
|
||||||
type StrictExtract<Union, Cond> = Cond extends Union ? Union : never;
|
|
||||||
|
|
||||||
type IsCaseMatched<E extends keyof Endpoints, P extends Endpoints[E]['req'], C extends number> =
|
|
||||||
IsNeverType<StrictExtract<Endpoints[E]['res']['$switch']['$cases'][C], [P, any]>> extends false ? true : false;
|
|
||||||
|
|
||||||
type GetCaseResult<E extends keyof Endpoints, P extends Endpoints[E]['req'], C extends number> =
|
|
||||||
StrictExtract<Endpoints[E]['res']['$switch']['$cases'][C], [P, any]>[1];
|
|
||||||
|
|
||||||
export class APIClient {
|
export class APIClient {
|
||||||
public origin: string;
|
public origin: string;
|
||||||
|
@ -52,22 +45,9 @@ export class APIClient {
|
||||||
this.fetch = opts.fetch ?? ((...args) => fetch(...args));
|
this.fetch = opts.fetch ?? ((...args) => fetch(...args));
|
||||||
}
|
}
|
||||||
|
|
||||||
public request<E extends keyof Endpoints, P extends Endpoints[E]['req']>(
|
public request<E extends keyof Endpoints, P extends SchemaOrUndefined<D['defines'][number]['req']>, D extends IEndpointMeta = Endpoints[E], R = Response<D, P>>(
|
||||||
endpoint: E, params: P = {} as P, credential?: string | null | undefined,
|
endpoint: E, params: P = {} as P, credential?: string | null | undefined,
|
||||||
): Promise<Endpoints[E]['res'] extends { $switch: { $cases: [any, any][]; $default: any; }; }
|
): Promise<R>
|
||||||
?
|
|
||||||
IsCaseMatched<E, P, 0> extends true ? GetCaseResult<E, P, 0> :
|
|
||||||
IsCaseMatched<E, P, 1> extends true ? GetCaseResult<E, P, 1> :
|
|
||||||
IsCaseMatched<E, P, 2> extends true ? GetCaseResult<E, P, 2> :
|
|
||||||
IsCaseMatched<E, P, 3> extends true ? GetCaseResult<E, P, 3> :
|
|
||||||
IsCaseMatched<E, P, 4> extends true ? GetCaseResult<E, P, 4> :
|
|
||||||
IsCaseMatched<E, P, 5> extends true ? GetCaseResult<E, P, 5> :
|
|
||||||
IsCaseMatched<E, P, 6> extends true ? GetCaseResult<E, P, 6> :
|
|
||||||
IsCaseMatched<E, P, 7> extends true ? GetCaseResult<E, P, 7> :
|
|
||||||
IsCaseMatched<E, P, 8> extends true ? GetCaseResult<E, P, 8> :
|
|
||||||
IsCaseMatched<E, P, 9> extends true ? GetCaseResult<E, P, 9> :
|
|
||||||
Endpoints[E]['res']['$switch']['$default']
|
|
||||||
: Endpoints[E]['res']>
|
|
||||||
{
|
{
|
||||||
const promise = new Promise((resolve, reject) => {
|
const promise = new Promise((resolve, reject) => {
|
||||||
this.fetch(`${this.origin}/api/${endpoint}`, {
|
this.fetch(`${this.origin}/api/${endpoint}`, {
|
||||||
|
|
|
@ -1,126 +1,5 @@
|
||||||
|
import { IEndpointMeta } from "./endpoints.types";
|
||||||
import { localUsernameSchema, passwordSchema } from "./schemas/user";
|
import { localUsernameSchema, passwordSchema } from "./schemas/user";
|
||||||
import type { JSONSchema7 } from 'schema-type';
|
|
||||||
|
|
||||||
export type RolePolicies = {
|
|
||||||
gtlAvailable: boolean;
|
|
||||||
ltlAvailable: boolean;
|
|
||||||
canPublicNote: boolean;
|
|
||||||
canInvite: boolean;
|
|
||||||
canManageCustomEmojis: boolean;
|
|
||||||
canSearchNotes: boolean;
|
|
||||||
canHideAds: boolean;
|
|
||||||
driveCapacityMb: number;
|
|
||||||
pinLimit: number;
|
|
||||||
antennaLimit: number;
|
|
||||||
wordMuteLimit: number;
|
|
||||||
webhookLimit: number;
|
|
||||||
clipLimit: number;
|
|
||||||
noteEachClipsLimit: number;
|
|
||||||
userListLimit: number;
|
|
||||||
userEachUserListsLimit: number;
|
|
||||||
rateLimitFactor: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type EndpointDefines = ReadonlyArray<{ req: JSONSchema7 | undefined; res: JSONSchema7 | undefined; }>;
|
|
||||||
|
|
||||||
export interface IEndpointMeta {
|
|
||||||
readonly stability?: 'deprecated' | 'experimental' | 'stable';
|
|
||||||
|
|
||||||
readonly tags?: ReadonlyArray<string>;
|
|
||||||
|
|
||||||
readonly errors?: {
|
|
||||||
readonly [key: string]: {
|
|
||||||
readonly message: string;
|
|
||||||
readonly code: string;
|
|
||||||
readonly id: string;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
readonly defines: EndpointDefines;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* このエンドポイントにリクエストするのにユーザー情報が必須か否か
|
|
||||||
* 省略した場合は false として解釈されます。
|
|
||||||
*/
|
|
||||||
readonly requireCredential?: boolean;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* isModeratorなロールを必要とするか
|
|
||||||
*/
|
|
||||||
readonly requireModerator?: boolean;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* isAdministratorなロールを必要とするか
|
|
||||||
*/
|
|
||||||
readonly requireAdmin?: boolean;
|
|
||||||
|
|
||||||
readonly requireRolePolicy?: keyof RolePolicies;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 引っ越し済みのユーザーによるリクエストを禁止するか
|
|
||||||
* 省略した場合は false として解釈されます。
|
|
||||||
*/
|
|
||||||
readonly prohibitMoved?: boolean;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* エンドポイントのリミテーションに関するやつ
|
|
||||||
* 省略した場合はリミテーションは無いものとして解釈されます。
|
|
||||||
*/
|
|
||||||
readonly limit?: {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 複数のエンドポイントでリミットを共有したい場合に指定するキー
|
|
||||||
*/
|
|
||||||
readonly key?: string;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* リミットを適用する期間(ms)
|
|
||||||
* このプロパティを設定する場合、max プロパティも設定する必要があります。
|
|
||||||
*/
|
|
||||||
readonly duration?: number;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* durationで指定した期間内にいくつまでリクエストできるのか
|
|
||||||
* このプロパティを設定する場合、duration プロパティも設定する必要があります。
|
|
||||||
*/
|
|
||||||
readonly max?: number;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 最低でもどれくらいの間隔を開けてリクエストしなければならないか(ms)
|
|
||||||
*/
|
|
||||||
readonly minInterval?: number;
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* ファイルの添付を必要とするか否か
|
|
||||||
* 省略した場合は false として解釈されます。
|
|
||||||
*/
|
|
||||||
readonly requireFile?: boolean;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* サードパーティアプリからはリクエストすることができないか否か
|
|
||||||
* 省略した場合は false として解釈されます。
|
|
||||||
*/
|
|
||||||
readonly secure?: boolean;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* エンドポイントの種類
|
|
||||||
* パーミッションの実現に利用されます。
|
|
||||||
*/
|
|
||||||
readonly kind?: string;
|
|
||||||
|
|
||||||
readonly description?: string;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* GETでのリクエストを許容するか否か
|
|
||||||
*/
|
|
||||||
readonly allowGet?: boolean;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 正常応答をキャッシュ (Cache-Control: public) する秒数
|
|
||||||
*/
|
|
||||||
readonly cacheSec?: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const endpoints = {
|
export const endpoints = {
|
||||||
"admin/accounts/create": {
|
"admin/accounts/create": {
|
||||||
|
|
128
packages/misskey-js/src/endpoints.types.ts
Normal file
128
packages/misskey-js/src/endpoints.types.ts
Normal file
|
@ -0,0 +1,128 @@
|
||||||
|
import type { JSONSchema7, SchemaType } from 'schema-type';
|
||||||
|
import type { References } from './schemas';
|
||||||
|
import type { endpoints } from './endpoints';
|
||||||
|
|
||||||
|
export type RolePolicies = {
|
||||||
|
gtlAvailable: boolean;
|
||||||
|
ltlAvailable: boolean;
|
||||||
|
canPublicNote: boolean;
|
||||||
|
canInvite: boolean;
|
||||||
|
canManageCustomEmojis: boolean;
|
||||||
|
canSearchNotes: boolean;
|
||||||
|
canHideAds: boolean;
|
||||||
|
driveCapacityMb: number;
|
||||||
|
pinLimit: number;
|
||||||
|
antennaLimit: number;
|
||||||
|
wordMuteLimit: number;
|
||||||
|
webhookLimit: number;
|
||||||
|
clipLimit: number;
|
||||||
|
noteEachClipsLimit: number;
|
||||||
|
userListLimit: number;
|
||||||
|
userEachUserListsLimit: number;
|
||||||
|
rateLimitFactor: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type EndpointDefines = ReadonlyArray<{ req: JSONSchema7 | undefined; res: JSONSchema7 | undefined; }>;
|
||||||
|
|
||||||
|
export interface IEndpointMeta {
|
||||||
|
readonly stability?: 'deprecated' | 'experimental' | 'stable';
|
||||||
|
|
||||||
|
readonly tags?: ReadonlyArray<string>;
|
||||||
|
|
||||||
|
readonly errors?: {
|
||||||
|
readonly [key: string]: {
|
||||||
|
readonly message: string;
|
||||||
|
readonly code: string;
|
||||||
|
readonly id: string;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
readonly defines: EndpointDefines;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* このエンドポイントにリクエストするのにユーザー情報が必須か否か
|
||||||
|
* 省略した場合は false として解釈されます。
|
||||||
|
*/
|
||||||
|
readonly requireCredential?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* isModeratorなロールを必要とするか
|
||||||
|
*/
|
||||||
|
readonly requireModerator?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* isAdministratorなロールを必要とするか
|
||||||
|
*/
|
||||||
|
readonly requireAdmin?: boolean;
|
||||||
|
|
||||||
|
readonly requireRolePolicy?: keyof RolePolicies;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 引っ越し済みのユーザーによるリクエストを禁止するか
|
||||||
|
* 省略した場合は false として解釈されます。
|
||||||
|
*/
|
||||||
|
readonly prohibitMoved?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* エンドポイントのリミテーションに関するやつ
|
||||||
|
* 省略した場合はリミテーションは無いものとして解釈されます。
|
||||||
|
*/
|
||||||
|
readonly limit?: {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 複数のエンドポイントでリミットを共有したい場合に指定するキー
|
||||||
|
*/
|
||||||
|
readonly key?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* リミットを適用する期間(ms)
|
||||||
|
* このプロパティを設定する場合、max プロパティも設定する必要があります。
|
||||||
|
*/
|
||||||
|
readonly duration?: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* durationで指定した期間内にいくつまでリクエストできるのか
|
||||||
|
* このプロパティを設定する場合、duration プロパティも設定する必要があります。
|
||||||
|
*/
|
||||||
|
readonly max?: number;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 最低でもどれくらいの間隔を開けてリクエストしなければならないか(ms)
|
||||||
|
*/
|
||||||
|
readonly minInterval?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ファイルの添付を必要とするか否か
|
||||||
|
* 省略した場合は false として解釈されます。
|
||||||
|
*/
|
||||||
|
readonly requireFile?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* サードパーティアプリからはリクエストすることができないか否か
|
||||||
|
* 省略した場合は false として解釈されます。
|
||||||
|
*/
|
||||||
|
readonly secure?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* エンドポイントの種類
|
||||||
|
* パーミッションの実現に利用されます。
|
||||||
|
*/
|
||||||
|
readonly kind?: string;
|
||||||
|
|
||||||
|
readonly description?: string;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* GETでのリクエストを許容するか否か
|
||||||
|
*/
|
||||||
|
readonly allowGet?: boolean;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 正常応答をキャッシュ (Cache-Control: public) する秒数
|
||||||
|
*/
|
||||||
|
readonly cacheSec?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type SchemaOrUndefined<T extends JSONSchema7 | undefined> = T extends JSONSchema7 ? SchemaType<T, References> : never;
|
||||||
|
|
||||||
|
export type Endpoints = typeof endpoints;
|
|
@ -76,7 +76,7 @@ export const refs = {
|
||||||
Announcement: packedAnnouncementSchema,
|
Announcement: packedAnnouncementSchema,
|
||||||
} as const satisfies { [x: string]: JSONSchema7Definition };
|
} as const satisfies { [x: string]: JSONSchema7Definition };
|
||||||
|
|
||||||
type References = GetRefs<typeof refs>;
|
export type References = GetRefs<typeof refs>;
|
||||||
|
|
||||||
export type Packed<x extends GetKeys<References, 'https://misskey-hub.net/api/schemas/'>> = GetDef<References, x, 'https://misskey-hub.net/api/schemas/'>;
|
export type Packed<x extends GetKeys<References, 'https://misskey-hub.net/api/schemas/'>> = GetDef<References, x, 'https://misskey-hub.net/api/schemas/'>;
|
||||||
export type Def<x extends GetKeys<References>> = GetDef<References, x>;
|
export type Def<x extends GetKeys<References>> = GetDef<References, x>;
|
||||||
|
|
Loading…
Reference in a new issue