pageエンドポイント周りの改善
- featuredでページネーションが可能に - ページブロックのバリデーションを追加 - ページslugのバリデーションを追加 - ページブロックの容量制限を追加 - 未使用プロパティscriptとvariablesは変更が効かないように
This commit is contained in:
parent
a896c39dbf
commit
5b6146e348
|
@ -4,6 +4,7 @@
|
||||||
*/
|
*/
|
||||||
|
|
||||||
export const MAX_NOTE_TEXT_LENGTH = 3000;
|
export const MAX_NOTE_TEXT_LENGTH = 3000;
|
||||||
|
export const MAX_PAGE_CONTENT_BYTES = 1024 * 1024 * 1.5; // 1.5MB
|
||||||
|
|
||||||
export const USER_ONLINE_THRESHOLD = 1000 * 60 * 10; // 10min
|
export const USER_ONLINE_THRESHOLD = 1000 * 60 * 10; // 10min
|
||||||
export const USER_ACTIVE_THRESHOLD = 1000 * 60 * 60 * 24 * 3; // 3days
|
export const USER_ACTIVE_THRESHOLD = 1000 * 60 * 60 * 24 * 3; // 3days
|
||||||
|
|
|
@ -0,0 +1,40 @@
|
||||||
|
/*
|
||||||
|
* 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 PagesRepository } from '@/models/_.js';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ページ関係のService
|
||||||
|
*/
|
||||||
|
@Injectable()
|
||||||
|
export class PageService {
|
||||||
|
constructor(
|
||||||
|
@Inject(DI.pagesRepository)
|
||||||
|
private pagesRepository: PagesRepository,
|
||||||
|
) {
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 人気のあるページ一覧を取得する.
|
||||||
|
*/
|
||||||
|
public async featured(opts?: { offset?: number, limit: number }) {
|
||||||
|
const builder = this.pagesRepository.createQueryBuilder('page')
|
||||||
|
.andWhere('page.likedCount > 0')
|
||||||
|
.andWhere('page.visibility = :visibility', { visibility: 'public' })
|
||||||
|
.addOrderBy('page.likedCount', 'DESC')
|
||||||
|
.addOrderBy('page.updatedAt', 'DESC')
|
||||||
|
.addOrderBy('page.id', 'DESC');
|
||||||
|
|
||||||
|
if (opts?.offset) {
|
||||||
|
builder.skip(opts.offset);
|
||||||
|
}
|
||||||
|
|
||||||
|
builder.take(opts?.limit ?? 10);
|
||||||
|
|
||||||
|
return await builder.getMany();
|
||||||
|
}
|
||||||
|
}
|
|
@ -118,3 +118,5 @@ export class MiPage {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const pageNameSchema = { type: 'string', pattern: /^[a-zA-Z0-9_-]{1,256}$/.source } as const;
|
||||||
|
|
|
@ -33,6 +33,27 @@ const textBlockSchema = {
|
||||||
},
|
},
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
|
const headingBlockSchema = {
|
||||||
|
type: 'object',
|
||||||
|
properties: {
|
||||||
|
...blockBaseSchema.properties,
|
||||||
|
type: {
|
||||||
|
type: 'string',
|
||||||
|
optional: false, nullable: false,
|
||||||
|
enum: ['heading'],
|
||||||
|
},
|
||||||
|
level: {
|
||||||
|
type: 'number',
|
||||||
|
optional: false, nullable: false,
|
||||||
|
},
|
||||||
|
text: {
|
||||||
|
type: 'string',
|
||||||
|
optional: false, nullable: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
} as const;
|
||||||
|
|
||||||
|
/** @deprecated 要素を入れ子にする必要が(一旦)なくなったので非推奨。headingBlockを使用すること */
|
||||||
const sectionBlockSchema = {
|
const sectionBlockSchema = {
|
||||||
type: 'object',
|
type: 'object',
|
||||||
properties: {
|
properties: {
|
||||||
|
@ -100,6 +121,7 @@ export const packedPageBlockSchema = {
|
||||||
oneOf: [
|
oneOf: [
|
||||||
textBlockSchema,
|
textBlockSchema,
|
||||||
sectionBlockSchema,
|
sectionBlockSchema,
|
||||||
|
headingBlockSchema,
|
||||||
imageBlockSchema,
|
imageBlockSchema,
|
||||||
noteBlockSchema,
|
noteBlockSchema,
|
||||||
],
|
],
|
||||||
|
|
|
@ -7,11 +7,13 @@ import ms from 'ms';
|
||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Inject, Injectable } from '@nestjs/common';
|
||||||
import type { DriveFilesRepository, PagesRepository } from '@/models/_.js';
|
import type { DriveFilesRepository, PagesRepository } from '@/models/_.js';
|
||||||
import { IdService } from '@/core/IdService.js';
|
import { IdService } from '@/core/IdService.js';
|
||||||
import { MiPage } from '@/models/Page.js';
|
import { MiPage, pageNameSchema } from '@/models/Page.js';
|
||||||
import { Endpoint } from '@/server/api/endpoint-base.js';
|
import { Endpoint } from '@/server/api/endpoint-base.js';
|
||||||
import { PageEntityService } from '@/core/entities/PageEntityService.js';
|
import { PageEntityService } from '@/core/entities/PageEntityService.js';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import { ApiError } from '../../error.js';
|
import { ApiError } from '@/server/api/error.js';
|
||||||
|
import { MAX_PAGE_CONTENT_BYTES } from '@/const.js';
|
||||||
|
import { packedPageBlockSchema } from '@/models/json-schema/page.js';
|
||||||
|
|
||||||
export const meta = {
|
export const meta = {
|
||||||
tags: ['pages'],
|
tags: ['pages'],
|
||||||
|
@ -44,6 +46,11 @@ export const meta = {
|
||||||
code: 'NAME_ALREADY_EXISTS',
|
code: 'NAME_ALREADY_EXISTS',
|
||||||
id: '4650348e-301c-499a-83c9-6aa988c66bc1',
|
id: '4650348e-301c-499a-83c9-6aa988c66bc1',
|
||||||
},
|
},
|
||||||
|
contentTooLarge: {
|
||||||
|
message: 'Content is too large.',
|
||||||
|
code: 'CONTENT_TOO_LARGE',
|
||||||
|
id: '2a93fcc9-4cd7-4885-9e5b-be56ed8f4d4f',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
|
@ -51,10 +58,10 @@ export const paramDef = {
|
||||||
type: 'object',
|
type: 'object',
|
||||||
properties: {
|
properties: {
|
||||||
title: { type: 'string' },
|
title: { type: 'string' },
|
||||||
name: { type: 'string', minLength: 1 },
|
name: { ...pageNameSchema, minLength: 1 },
|
||||||
summary: { type: 'string', nullable: true },
|
summary: { type: 'string', nullable: true },
|
||||||
content: { type: 'array', items: {
|
content: { type: 'array', items: {
|
||||||
type: 'object', additionalProperties: true,
|
...packedPageBlockSchema,
|
||||||
} },
|
} },
|
||||||
variables: { type: 'array', items: {
|
variables: { type: 'array', items: {
|
||||||
type: 'object', additionalProperties: true,
|
type: 'object', additionalProperties: true,
|
||||||
|
@ -65,7 +72,7 @@ export const paramDef = {
|
||||||
alignCenter: { type: 'boolean', default: false },
|
alignCenter: { type: 'boolean', default: false },
|
||||||
hideTitleWhenPinned: { type: 'boolean', default: false },
|
hideTitleWhenPinned: { type: 'boolean', default: false },
|
||||||
},
|
},
|
||||||
required: ['title', 'name', 'content', 'variables', 'script'],
|
required: ['title', 'name', 'content'],
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
|
@ -81,6 +88,10 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
private idService: IdService,
|
private idService: IdService,
|
||||||
) {
|
) {
|
||||||
super(meta, paramDef, async (ps, me) => {
|
super(meta, paramDef, async (ps, me) => {
|
||||||
|
if (new Blob([JSON.stringify(ps.content)]).size > MAX_PAGE_CONTENT_BYTES) {
|
||||||
|
throw new ApiError(meta.errors.contentTooLarge);
|
||||||
|
}
|
||||||
|
|
||||||
let eyeCatchingImage = null;
|
let eyeCatchingImage = null;
|
||||||
if (ps.eyeCatchingImageId != null) {
|
if (ps.eyeCatchingImageId != null) {
|
||||||
eyeCatchingImage = await this.driveFilesRepository.findOneBy({
|
eyeCatchingImage = await this.driveFilesRepository.findOneBy({
|
||||||
|
@ -109,8 +120,8 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
name: ps.name,
|
name: ps.name,
|
||||||
summary: ps.summary,
|
summary: ps.summary,
|
||||||
content: ps.content,
|
content: ps.content,
|
||||||
variables: ps.variables,
|
//variables: ps.variables, もう使用されていない(動的ページ)
|
||||||
script: ps.script,
|
//script: ps.script, もう使用されていない(動的ページ)
|
||||||
eyeCatchingImageId: eyeCatchingImage ? eyeCatchingImage.id : null,
|
eyeCatchingImageId: eyeCatchingImage ? eyeCatchingImage.id : null,
|
||||||
userId: me.id,
|
userId: me.id,
|
||||||
visibility: 'public',
|
visibility: 'public',
|
||||||
|
|
|
@ -3,11 +3,10 @@
|
||||||
* SPDX-License-Identifier: AGPL-3.0-only
|
* SPDX-License-Identifier: AGPL-3.0-only
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { Inject, Injectable } from '@nestjs/common';
|
import { Injectable } from '@nestjs/common';
|
||||||
import type { PagesRepository } from '@/models/_.js';
|
|
||||||
import { Endpoint } from '@/server/api/endpoint-base.js';
|
import { Endpoint } from '@/server/api/endpoint-base.js';
|
||||||
|
import { PageService } from '@/core/PageService.js';
|
||||||
import { PageEntityService } from '@/core/entities/PageEntityService.js';
|
import { PageEntityService } from '@/core/entities/PageEntityService.js';
|
||||||
import { DI } from '@/di-symbols.js';
|
|
||||||
|
|
||||||
export const meta = {
|
export const meta = {
|
||||||
tags: ['pages'],
|
tags: ['pages'],
|
||||||
|
@ -27,27 +26,25 @@ export const meta = {
|
||||||
|
|
||||||
export const paramDef = {
|
export const paramDef = {
|
||||||
type: 'object',
|
type: 'object',
|
||||||
properties: {},
|
properties: {
|
||||||
|
offset: { type: 'integer', minimum: 0, default: 0 },
|
||||||
|
limit: { type: 'integer', minimum: 1, maximum: 100, default: 10 },
|
||||||
|
},
|
||||||
required: [],
|
required: [],
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
@Injectable()
|
@Injectable()
|
||||||
export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-disable-line import/no-default-export
|
export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-disable-line import/no-default-export
|
||||||
constructor(
|
constructor(
|
||||||
@Inject(DI.pagesRepository)
|
private pageService: PageService,
|
||||||
private pagesRepository: PagesRepository,
|
|
||||||
|
|
||||||
private pageEntityService: PageEntityService,
|
private pageEntityService: PageEntityService,
|
||||||
) {
|
) {
|
||||||
super(meta, paramDef, async (ps, me) => {
|
super(meta, paramDef, async (ps, me) => {
|
||||||
const query = this.pagesRepository.createQueryBuilder('page')
|
const result = await this.pageService.featured({
|
||||||
.where('page.visibility = \'public\'')
|
offset: ps.offset,
|
||||||
.andWhere('page.likedCount > 0')
|
limit: ps.limit,
|
||||||
.orderBy('page.likedCount', 'DESC');
|
});
|
||||||
|
return await this.pageEntityService.packMany(result, me);
|
||||||
const pages = await query.limit(10).getMany();
|
|
||||||
|
|
||||||
return await this.pageEntityService.packMany(pages, me);
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -9,7 +9,10 @@ import { Inject, Injectable } from '@nestjs/common';
|
||||||
import type { PagesRepository, DriveFilesRepository } from '@/models/_.js';
|
import type { PagesRepository, DriveFilesRepository } from '@/models/_.js';
|
||||||
import { Endpoint } from '@/server/api/endpoint-base.js';
|
import { Endpoint } from '@/server/api/endpoint-base.js';
|
||||||
import { DI } from '@/di-symbols.js';
|
import { DI } from '@/di-symbols.js';
|
||||||
import { ApiError } from '../../error.js';
|
import { ApiError } from '@/server/api/error.js';
|
||||||
|
import { MAX_PAGE_CONTENT_BYTES } from '@/const.js';
|
||||||
|
import { packedPageBlockSchema } from '@/models/json-schema/page.js';
|
||||||
|
import { pageNameSchema } from '@/models/Page.js';
|
||||||
|
|
||||||
export const meta = {
|
export const meta = {
|
||||||
tags: ['pages'],
|
tags: ['pages'],
|
||||||
|
@ -48,6 +51,11 @@ export const meta = {
|
||||||
code: 'NAME_ALREADY_EXISTS',
|
code: 'NAME_ALREADY_EXISTS',
|
||||||
id: '2298a392-d4a1-44c5-9ebb-ac1aeaa5a9ab',
|
id: '2298a392-d4a1-44c5-9ebb-ac1aeaa5a9ab',
|
||||||
},
|
},
|
||||||
|
contentTooLarge: {
|
||||||
|
message: 'Content is too large.',
|
||||||
|
code: 'CONTENT_TOO_LARGE',
|
||||||
|
id: '2a93fcc9-4cd7-4885-9e5b-be56ed8f4d4f',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
} as const;
|
} as const;
|
||||||
|
|
||||||
|
@ -56,10 +64,10 @@ export const paramDef = {
|
||||||
properties: {
|
properties: {
|
||||||
pageId: { type: 'string', format: 'misskey:id' },
|
pageId: { type: 'string', format: 'misskey:id' },
|
||||||
title: { type: 'string' },
|
title: { type: 'string' },
|
||||||
name: { type: 'string', minLength: 1 },
|
name: { ...pageNameSchema, minLength: 1 },
|
||||||
summary: { type: 'string', nullable: true },
|
summary: { type: 'string', nullable: true },
|
||||||
content: { type: 'array', items: {
|
content: { type: 'array', items: {
|
||||||
type: 'object', additionalProperties: true,
|
...packedPageBlockSchema,
|
||||||
} },
|
} },
|
||||||
variables: { type: 'array', items: {
|
variables: { type: 'array', items: {
|
||||||
type: 'object', additionalProperties: true,
|
type: 'object', additionalProperties: true,
|
||||||
|
@ -91,6 +99,10 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
throw new ApiError(meta.errors.accessDenied);
|
throw new ApiError(meta.errors.accessDenied);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (new Blob([JSON.stringify(ps.content)]).size > MAX_PAGE_CONTENT_BYTES) {
|
||||||
|
throw new ApiError(meta.errors.contentTooLarge);
|
||||||
|
}
|
||||||
|
|
||||||
if (ps.eyeCatchingImageId != null) {
|
if (ps.eyeCatchingImageId != null) {
|
||||||
const eyeCatchingImage = await this.driveFilesRepository.findOneBy({
|
const eyeCatchingImage = await this.driveFilesRepository.findOneBy({
|
||||||
id: ps.eyeCatchingImageId,
|
id: ps.eyeCatchingImageId,
|
||||||
|
@ -118,8 +130,8 @@ export default class extends Endpoint<typeof meta, typeof paramDef> { // eslint-
|
||||||
name: ps.name,
|
name: ps.name,
|
||||||
summary: ps.summary === undefined ? page.summary : ps.summary,
|
summary: ps.summary === undefined ? page.summary : ps.summary,
|
||||||
content: ps.content,
|
content: ps.content,
|
||||||
variables: ps.variables,
|
//variables: ps.variables, もう使用されていない(動的ページ)
|
||||||
script: ps.script,
|
//script: ps.script, もう使用されていない(動的ページ)
|
||||||
alignCenter: ps.alignCenter,
|
alignCenter: ps.alignCenter,
|
||||||
hideTitleWhenPinned: ps.hideTitleWhenPinned,
|
hideTitleWhenPinned: ps.hideTitleWhenPinned,
|
||||||
font: ps.font,
|
font: ps.font,
|
||||||
|
|
|
@ -1684,6 +1684,7 @@ declare namespace entities {
|
||||||
PagesCreateRequest,
|
PagesCreateRequest,
|
||||||
PagesCreateResponse,
|
PagesCreateResponse,
|
||||||
PagesDeleteRequest,
|
PagesDeleteRequest,
|
||||||
|
PagesFeaturedRequest,
|
||||||
PagesFeaturedResponse,
|
PagesFeaturedResponse,
|
||||||
PagesLikeRequest,
|
PagesLikeRequest,
|
||||||
PagesShowRequest,
|
PagesShowRequest,
|
||||||
|
@ -2841,6 +2842,9 @@ type PagesCreateResponse = operations['pages___create']['responses']['200']['con
|
||||||
// @public (undocumented)
|
// @public (undocumented)
|
||||||
type PagesDeleteRequest = operations['pages___delete']['requestBody']['content']['application/json'];
|
type PagesDeleteRequest = operations['pages___delete']['requestBody']['content']['application/json'];
|
||||||
|
|
||||||
|
// @public (undocumented)
|
||||||
|
type PagesFeaturedRequest = operations['pages___featured']['requestBody']['content']['application/json'];
|
||||||
|
|
||||||
// @public (undocumented)
|
// @public (undocumented)
|
||||||
type PagesFeaturedResponse = operations['pages___featured']['responses']['200']['content']['application/json'];
|
type PagesFeaturedResponse = operations['pages___featured']['responses']['200']['content']['application/json'];
|
||||||
|
|
||||||
|
|
|
@ -459,6 +459,7 @@ import type {
|
||||||
PagesCreateRequest,
|
PagesCreateRequest,
|
||||||
PagesCreateResponse,
|
PagesCreateResponse,
|
||||||
PagesDeleteRequest,
|
PagesDeleteRequest,
|
||||||
|
PagesFeaturedRequest,
|
||||||
PagesFeaturedResponse,
|
PagesFeaturedResponse,
|
||||||
PagesLikeRequest,
|
PagesLikeRequest,
|
||||||
PagesShowRequest,
|
PagesShowRequest,
|
||||||
|
@ -888,7 +889,7 @@ export type Endpoints = {
|
||||||
'page-push': { req: PagePushRequest; res: EmptyResponse };
|
'page-push': { req: PagePushRequest; res: EmptyResponse };
|
||||||
'pages/create': { req: PagesCreateRequest; res: PagesCreateResponse };
|
'pages/create': { req: PagesCreateRequest; res: PagesCreateResponse };
|
||||||
'pages/delete': { req: PagesDeleteRequest; res: EmptyResponse };
|
'pages/delete': { req: PagesDeleteRequest; res: EmptyResponse };
|
||||||
'pages/featured': { req: EmptyRequest; res: PagesFeaturedResponse };
|
'pages/featured': { req: PagesFeaturedRequest; res: PagesFeaturedResponse };
|
||||||
'pages/like': { req: PagesLikeRequest; res: EmptyResponse };
|
'pages/like': { req: PagesLikeRequest; res: EmptyResponse };
|
||||||
'pages/show': { req: PagesShowRequest; res: PagesShowResponse };
|
'pages/show': { req: PagesShowRequest; res: PagesShowResponse };
|
||||||
'pages/unlike': { req: PagesUnlikeRequest; res: EmptyResponse };
|
'pages/unlike': { req: PagesUnlikeRequest; res: EmptyResponse };
|
||||||
|
|
|
@ -462,6 +462,7 @@ export type PagePushRequest = operations['page-push']['requestBody']['content'][
|
||||||
export type PagesCreateRequest = operations['pages___create']['requestBody']['content']['application/json'];
|
export type PagesCreateRequest = operations['pages___create']['requestBody']['content']['application/json'];
|
||||||
export type PagesCreateResponse = operations['pages___create']['responses']['200']['content']['application/json'];
|
export type PagesCreateResponse = operations['pages___create']['responses']['200']['content']['application/json'];
|
||||||
export type PagesDeleteRequest = operations['pages___delete']['requestBody']['content']['application/json'];
|
export type PagesDeleteRequest = operations['pages___delete']['requestBody']['content']['application/json'];
|
||||||
|
export type PagesFeaturedRequest = operations['pages___featured']['requestBody']['content']['application/json'];
|
||||||
export type PagesFeaturedResponse = operations['pages___featured']['responses']['200']['content']['application/json'];
|
export type PagesFeaturedResponse = operations['pages___featured']['responses']['200']['content']['application/json'];
|
||||||
export type PagesLikeRequest = operations['pages___like']['requestBody']['content']['application/json'];
|
export type PagesLikeRequest = operations['pages___like']['requestBody']['content']['application/json'];
|
||||||
export type PagesShowRequest = operations['pages___show']['requestBody']['content']['application/json'];
|
export type PagesShowRequest = operations['pages___show']['requestBody']['content']['application/json'];
|
||||||
|
|
|
@ -4545,6 +4545,12 @@ export type components = {
|
||||||
type: 'section';
|
type: 'section';
|
||||||
title: string;
|
title: string;
|
||||||
children: components['schemas']['PageBlock'][];
|
children: components['schemas']['PageBlock'][];
|
||||||
|
}, {
|
||||||
|
id: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type: 'heading';
|
||||||
|
level: number;
|
||||||
|
text: string;
|
||||||
}, {
|
}, {
|
||||||
id: string;
|
id: string;
|
||||||
/** @enum {string} */
|
/** @enum {string} */
|
||||||
|
@ -23426,13 +23432,39 @@ export type operations = {
|
||||||
title: string;
|
title: string;
|
||||||
name: string;
|
name: string;
|
||||||
summary?: string | null;
|
summary?: string | null;
|
||||||
content: {
|
content: (OneOf<[{
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'text';
|
||||||
|
text?: string;
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'section';
|
||||||
|
title?: string;
|
||||||
|
children?: components['schemas']['PageBlock'][];
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'heading';
|
||||||
|
level?: number;
|
||||||
|
text?: string;
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'image';
|
||||||
|
fileId?: string | null;
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'note';
|
||||||
|
detailed?: boolean;
|
||||||
|
note?: string | null;
|
||||||
|
}]>)[];
|
||||||
|
variables?: {
|
||||||
[key: string]: unknown;
|
[key: string]: unknown;
|
||||||
}[];
|
}[];
|
||||||
variables: {
|
script?: string;
|
||||||
[key: string]: unknown;
|
|
||||||
}[];
|
|
||||||
script: string;
|
|
||||||
/** Format: misskey:id */
|
/** Format: misskey:id */
|
||||||
eyeCatchingImageId?: string | null;
|
eyeCatchingImageId?: string | null;
|
||||||
/**
|
/**
|
||||||
|
@ -23551,6 +23583,16 @@ export type operations = {
|
||||||
* **Credential required**: *No*
|
* **Credential required**: *No*
|
||||||
*/
|
*/
|
||||||
pages___featured: {
|
pages___featured: {
|
||||||
|
requestBody: {
|
||||||
|
content: {
|
||||||
|
'application/json': {
|
||||||
|
/** @default 0 */
|
||||||
|
offset?: number;
|
||||||
|
/** @default 10 */
|
||||||
|
limit?: number;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
};
|
||||||
responses: {
|
responses: {
|
||||||
/** @description OK (with results) */
|
/** @description OK (with results) */
|
||||||
200: {
|
200: {
|
||||||
|
@ -23765,9 +23807,35 @@ export type operations = {
|
||||||
title?: string;
|
title?: string;
|
||||||
name?: string;
|
name?: string;
|
||||||
summary?: string | null;
|
summary?: string | null;
|
||||||
content?: {
|
content?: (OneOf<[{
|
||||||
[key: string]: unknown;
|
id?: string;
|
||||||
}[];
|
/** @enum {string} */
|
||||||
|
type?: 'text';
|
||||||
|
text?: string;
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'section';
|
||||||
|
title?: string;
|
||||||
|
children?: components['schemas']['PageBlock'][];
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'heading';
|
||||||
|
level?: number;
|
||||||
|
text?: string;
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'image';
|
||||||
|
fileId?: string | null;
|
||||||
|
}, {
|
||||||
|
id?: string;
|
||||||
|
/** @enum {string} */
|
||||||
|
type?: 'note';
|
||||||
|
detailed?: boolean;
|
||||||
|
note?: string | null;
|
||||||
|
}]>)[];
|
||||||
variables?: {
|
variables?: {
|
||||||
[key: string]: unknown;
|
[key: string]: unknown;
|
||||||
}[];
|
}[];
|
||||||
|
|
Loading…
Reference in New Issue