2022-03-26 15:34:00 +09:00
|
|
|
import { db } from '@/db/postgre.js';
|
2022-02-27 11:07:39 +09:00
|
|
|
import { DriveFile } from '@/models/entities/drive-file.js';
|
|
|
|
import { Users, DriveFolders } from '../index.js';
|
|
|
|
import { User } from '@/models/entities/user.js';
|
|
|
|
import { toPuny } from '@/misc/convert-host.js';
|
|
|
|
import { awaitAll, Promiseable } from '@/prelude/await-all.js';
|
|
|
|
import { Packed } from '@/misc/schema.js';
|
|
|
|
import config from '@/config/index.js';
|
|
|
|
import { query, appendQuery } from '@/prelude/url.js';
|
|
|
|
import { Meta } from '@/models/entities/meta.js';
|
|
|
|
import { fetchMeta } from '@/misc/fetch-meta.js';
|
2019-04-23 22:35:26 +09:00
|
|
|
|
2021-03-16 12:50:07 +09:00
|
|
|
type PackOptions = {
|
|
|
|
detail?: boolean,
|
|
|
|
self?: boolean,
|
|
|
|
withUser?: boolean,
|
|
|
|
};
|
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
export const DriveFileRepository = db.getRepository(DriveFile).extend({
|
|
|
|
validateFileName(name: string): boolean {
|
2019-04-07 21:50:36 +09:00
|
|
|
return (
|
|
|
|
(name.trim().length > 0) &&
|
|
|
|
(name.length <= 200) &&
|
|
|
|
(name.indexOf('\\') === -1) &&
|
|
|
|
(name.indexOf('/') === -1) &&
|
|
|
|
(name.indexOf('..') === -1)
|
|
|
|
);
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2019-04-07 21:50:36 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
getPublicProperties(file: DriveFile): DriveFile['properties'] {
|
2021-12-03 11:19:28 +09:00
|
|
|
if (file.properties.orientation != null) {
|
|
|
|
const properties = JSON.parse(JSON.stringify(file.properties));
|
|
|
|
if (file.properties.orientation >= 5) {
|
|
|
|
[properties.width, properties.height] = [properties.height, properties.width];
|
|
|
|
}
|
|
|
|
properties.orientation = undefined;
|
|
|
|
return properties;
|
|
|
|
}
|
|
|
|
|
|
|
|
return file.properties;
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2021-12-03 11:19:28 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
getPublicUrl(file: DriveFile, thumbnail = false): string | null {
|
2019-12-31 17:23:47 +09:00
|
|
|
// リモートかつメディアプロキシ
|
|
|
|
if (file.uri != null && file.userHost != null && config.mediaProxy != null) {
|
|
|
|
return appendQuery(config.mediaProxy, query({
|
|
|
|
url: file.uri,
|
2021-12-09 23:58:30 +09:00
|
|
|
thumbnail: thumbnail ? '1' : undefined,
|
2019-12-31 17:23:47 +09:00
|
|
|
}));
|
2019-12-20 01:54:28 +09:00
|
|
|
}
|
2019-12-31 17:23:47 +09:00
|
|
|
|
|
|
|
// リモートかつ期限切れはローカルプロキシを試みる
|
2022-02-27 13:59:10 +09:00
|
|
|
if (file.uri != null && file.isLink && config.proxyRemoteFiles) {
|
2019-12-31 17:23:47 +09:00
|
|
|
const key = thumbnail ? file.thumbnailAccessKey : file.webpublicAccessKey;
|
|
|
|
|
|
|
|
if (key && !key.match('/')) { // 古いものはここにオブジェクトストレージキーが入ってるので除外
|
2020-07-10 03:52:20 +09:00
|
|
|
return `${config.url}/files/${key}`;
|
2019-12-31 17:23:47 +09:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-02 02:45:05 +09:00
|
|
|
const isImage = file.type && ['image/png', 'image/apng', 'image/gif', 'image/jpeg', 'image/webp', 'image/svg+xml'].includes(file.type);
|
|
|
|
|
|
|
|
return thumbnail ? (file.thumbnailUrl || (isImage ? (file.webpublicUrl || file.url) : null)) : (file.webpublicUrl || file.url);
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2019-04-07 21:50:36 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
async calcDriveUsageOf(user: User['id'] | { id: User['id'] }): Promise<number> {
|
2019-04-07 21:50:36 +09:00
|
|
|
const id = typeof user === 'object' ? user.id : user;
|
|
|
|
|
|
|
|
const { sum } = await this
|
|
|
|
.createQueryBuilder('file')
|
|
|
|
.where('file.userId = :id', { id: id })
|
2021-05-30 13:48:23 +09:00
|
|
|
.andWhere('file.isLink = FALSE')
|
2019-04-07 21:50:36 +09:00
|
|
|
.select('SUM(file.size)', 'sum')
|
|
|
|
.getRawOne();
|
|
|
|
|
|
|
|
return parseInt(sum, 10) || 0;
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2019-04-07 21:50:36 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
async calcDriveUsageOfHost(host: string): Promise<number> {
|
2019-04-07 21:50:36 +09:00
|
|
|
const { sum } = await this
|
|
|
|
.createQueryBuilder('file')
|
2019-04-10 00:59:41 +09:00
|
|
|
.where('file.userHost = :host', { host: toPuny(host) })
|
2021-05-30 13:48:23 +09:00
|
|
|
.andWhere('file.isLink = FALSE')
|
2019-04-07 21:50:36 +09:00
|
|
|
.select('SUM(file.size)', 'sum')
|
|
|
|
.getRawOne();
|
|
|
|
|
|
|
|
return parseInt(sum, 10) || 0;
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2019-04-07 21:50:36 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
async calcDriveUsageOfLocal(): Promise<number> {
|
2019-04-07 21:50:36 +09:00
|
|
|
const { sum } = await this
|
|
|
|
.createQueryBuilder('file')
|
|
|
|
.where('file.userHost IS NULL')
|
2021-05-30 13:48:23 +09:00
|
|
|
.andWhere('file.isLink = FALSE')
|
2019-04-07 21:50:36 +09:00
|
|
|
.select('SUM(file.size)', 'sum')
|
|
|
|
.getRawOne();
|
|
|
|
|
|
|
|
return parseInt(sum, 10) || 0;
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2019-04-07 21:50:36 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
async calcDriveUsageOfRemote(): Promise<number> {
|
2019-04-07 21:50:36 +09:00
|
|
|
const { sum } = await this
|
|
|
|
.createQueryBuilder('file')
|
|
|
|
.where('file.userHost IS NOT NULL')
|
2021-05-30 13:48:23 +09:00
|
|
|
.andWhere('file.isLink = FALSE')
|
2019-04-07 21:50:36 +09:00
|
|
|
.select('SUM(file.size)', 'sum')
|
|
|
|
.getRawOne();
|
|
|
|
|
|
|
|
return parseInt(sum, 10) || 0;
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2019-04-07 21:50:36 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
async pack(
|
2019-04-07 21:50:36 +09:00
|
|
|
src: DriveFile['id'] | DriveFile,
|
2021-03-16 12:50:07 +09:00
|
|
|
options?: PackOptions
|
2021-09-22 22:35:55 +09:00
|
|
|
): Promise<Packed<'DriveFile'> | null> {
|
2019-04-07 21:50:36 +09:00
|
|
|
const opts = Object.assign({
|
|
|
|
detail: false,
|
2021-12-09 23:58:30 +09:00
|
|
|
self: false,
|
2019-04-07 21:50:36 +09:00
|
|
|
}, options);
|
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
const file = typeof src === 'object' ? src : await this.findOneBy({ id: src });
|
2021-03-16 12:50:07 +09:00
|
|
|
if (file == null) return null;
|
2019-04-07 21:50:36 +09:00
|
|
|
|
2022-01-18 22:27:10 +09:00
|
|
|
return await awaitAll<Packed<'DriveFile'>>({
|
2019-04-07 21:50:36 +09:00
|
|
|
id: file.id,
|
2019-04-23 22:35:26 +09:00
|
|
|
createdAt: file.createdAt.toISOString(),
|
2019-04-07 21:50:36 +09:00
|
|
|
name: file.name,
|
|
|
|
type: file.type,
|
|
|
|
md5: file.md5,
|
|
|
|
size: file.size,
|
|
|
|
isSensitive: file.isSensitive,
|
2020-07-19 00:24:07 +09:00
|
|
|
blurhash: file.blurhash,
|
2021-12-03 11:19:28 +09:00
|
|
|
properties: opts.self ? file.properties : this.getPublicProperties(file),
|
2022-02-27 13:59:10 +09:00
|
|
|
url: opts.self ? file.url : this.getPublicUrl(file, false),
|
|
|
|
thumbnailUrl: this.getPublicUrl(file, true),
|
2020-10-17 20:12:00 +09:00
|
|
|
comment: file.comment,
|
2019-04-07 21:50:36 +09:00
|
|
|
folderId: file.folderId,
|
|
|
|
folder: opts.detail && file.folderId ? DriveFolders.pack(file.folderId, {
|
2021-12-09 23:58:30 +09:00
|
|
|
detail: true,
|
2019-04-07 21:50:36 +09:00
|
|
|
}) : null,
|
2020-10-28 22:24:16 +09:00
|
|
|
userId: opts.withUser ? file.userId : null,
|
2021-12-09 23:58:30 +09:00
|
|
|
user: (opts.withUser && file.userId) ? Users.pack(file.userId) : null,
|
2019-04-07 21:50:36 +09:00
|
|
|
});
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
2019-04-25 13:27:07 +09:00
|
|
|
|
2022-03-26 15:34:00 +09:00
|
|
|
async packMany(
|
2021-03-19 18:22:14 +09:00
|
|
|
files: (DriveFile['id'] | DriveFile)[],
|
2021-03-16 12:50:07 +09:00
|
|
|
options?: PackOptions
|
2019-04-25 13:27:07 +09:00
|
|
|
) {
|
2021-03-16 12:50:07 +09:00
|
|
|
const items = await Promise.all(files.map(f => this.pack(f, options)));
|
|
|
|
return items.filter(x => x != null);
|
2022-03-26 15:34:00 +09:00
|
|
|
},
|
|
|
|
});
|