2019-04-29 09:11:57 +09:00
|
|
|
import { EntityRepository, Repository } from 'typeorm';
|
|
|
|
import { Page } from '../entities/page';
|
2019-06-27 18:04:09 +09:00
|
|
|
import { SchemaType } from '../../misc/schema';
|
2019-05-17 19:56:47 +09:00
|
|
|
import { Users, DriveFiles, PageLikes } from '..';
|
2019-04-29 09:11:57 +09:00
|
|
|
import { awaitAll } from '../../prelude/await-all';
|
|
|
|
import { DriveFile } from '../entities/drive-file';
|
2019-05-17 19:56:47 +09:00
|
|
|
import { User } from '../entities/user';
|
|
|
|
import { ensure } from '../../prelude/ensure';
|
2019-04-29 09:11:57 +09:00
|
|
|
|
|
|
|
export type PackedPage = SchemaType<typeof packedPageSchema>;
|
|
|
|
|
|
|
|
@EntityRepository(Page)
|
|
|
|
export class PageRepository extends Repository<Page> {
|
|
|
|
public async pack(
|
2019-05-17 19:56:47 +09:00
|
|
|
src: Page['id'] | Page,
|
|
|
|
me?: User['id'] | User | null | undefined,
|
2019-04-29 09:11:57 +09:00
|
|
|
): Promise<PackedPage> {
|
2019-05-17 19:56:47 +09:00
|
|
|
const meId = me ? typeof me === 'string' ? me : me.id : null;
|
|
|
|
const page = typeof src === 'object' ? src : await this.findOne(src).then(ensure);
|
|
|
|
|
2019-04-29 09:11:57 +09:00
|
|
|
const attachedFiles: Promise<DriveFile | undefined>[] = [];
|
|
|
|
const collectFile = (xs: any[]) => {
|
|
|
|
for (const x of xs) {
|
|
|
|
if (x.type === 'image') {
|
|
|
|
attachedFiles.push(DriveFiles.findOne({
|
|
|
|
id: x.fileId,
|
2019-05-17 19:56:47 +09:00
|
|
|
userId: page.userId
|
2019-04-29 09:11:57 +09:00
|
|
|
}));
|
|
|
|
}
|
|
|
|
if (x.children) {
|
|
|
|
collectFile(x.children);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2019-05-17 19:56:47 +09:00
|
|
|
collectFile(page.content);
|
2019-04-30 12:15:41 +09:00
|
|
|
|
|
|
|
// 後方互換性のため
|
|
|
|
let migrated = false;
|
|
|
|
const migrate = (xs: any[]) => {
|
|
|
|
for (const x of xs) {
|
|
|
|
if (x.type === 'input') {
|
|
|
|
if (x.inputType === 'text') {
|
|
|
|
x.type = 'textInput';
|
|
|
|
}
|
|
|
|
if (x.inputType === 'number') {
|
|
|
|
x.type = 'numberInput';
|
|
|
|
if (x.default) x.default = parseInt(x.default, 10);
|
|
|
|
}
|
|
|
|
migrated = true;
|
|
|
|
}
|
|
|
|
if (x.children) {
|
|
|
|
migrate(x.children);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
2019-05-17 19:56:47 +09:00
|
|
|
migrate(page.content);
|
2019-04-30 12:15:41 +09:00
|
|
|
if (migrated) {
|
2019-05-17 19:56:47 +09:00
|
|
|
this.update(page.id, {
|
|
|
|
content: page.content
|
2019-04-30 12:15:41 +09:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2019-04-29 09:11:57 +09:00
|
|
|
return await awaitAll({
|
2019-05-17 19:56:47 +09:00
|
|
|
id: page.id,
|
|
|
|
createdAt: page.createdAt.toISOString(),
|
|
|
|
updatedAt: page.updatedAt.toISOString(),
|
|
|
|
userId: page.userId,
|
2020-02-01 11:35:49 +09:00
|
|
|
user: Users.pack(page.user || page.userId, me), // { detail: true } すると無限ループするので注意
|
2019-05-17 19:56:47 +09:00
|
|
|
content: page.content,
|
|
|
|
variables: page.variables,
|
|
|
|
title: page.title,
|
|
|
|
name: page.name,
|
|
|
|
summary: page.summary,
|
2019-07-07 06:56:13 +09:00
|
|
|
hideTitleWhenPinned: page.hideTitleWhenPinned,
|
2019-05-17 19:56:47 +09:00
|
|
|
alignCenter: page.alignCenter,
|
|
|
|
font: page.font,
|
2020-04-13 03:23:23 +09:00
|
|
|
script: page.script,
|
2019-05-17 19:56:47 +09:00
|
|
|
eyeCatchingImageId: page.eyeCatchingImageId,
|
|
|
|
eyeCatchingImage: page.eyeCatchingImageId ? await DriveFiles.pack(page.eyeCatchingImageId) : null,
|
|
|
|
attachedFiles: DriveFiles.packMany(await Promise.all(attachedFiles)),
|
|
|
|
likedCount: page.likedCount,
|
|
|
|
isLiked: meId ? await PageLikes.findOne({ pageId: page.id, userId: meId }).then(x => x != null) : undefined,
|
2019-04-29 09:11:57 +09:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
public packMany(
|
|
|
|
pages: Page[],
|
|
|
|
) {
|
|
|
|
return Promise.all(pages.map(x => this.pack(x)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export const packedPageSchema = {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'object' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-04-29 09:11:57 +09:00
|
|
|
properties: {
|
2019-05-01 04:44:46 +09:00
|
|
|
id: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'string' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
format: 'id',
|
|
|
|
example: 'xxxxxxxxxx',
|
|
|
|
},
|
|
|
|
createdAt: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'string' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
format: 'date-time',
|
|
|
|
},
|
|
|
|
updatedAt: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'string' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
format: 'date-time',
|
|
|
|
},
|
|
|
|
title: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'string' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
},
|
|
|
|
name: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'string' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
},
|
|
|
|
summary: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'string' as const,
|
|
|
|
optional: false as const, nullable: true as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
},
|
|
|
|
content: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'array' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
},
|
|
|
|
variables: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'array' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
},
|
|
|
|
userId: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'string' as const,
|
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
format: 'id',
|
|
|
|
},
|
|
|
|
user: {
|
2019-06-27 18:04:09 +09:00
|
|
|
type: 'object' as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
ref: 'User',
|
2019-06-27 18:04:09 +09:00
|
|
|
optional: false as const, nullable: false as const,
|
2019-05-01 04:44:46 +09:00
|
|
|
},
|
2019-04-29 09:11:57 +09:00
|
|
|
}
|
|
|
|
};
|