7199e6f4e0
* Update reaction.vue
* fix bug
* wip
* wip
* wjio
* wip
* Revert "wip"
This reverts commit e427f2160a
.
* wip
* wip
* wip
* Update init.ts
* Update drive-window.vue
* wip
* wip
* Use PascalCase for components
* Use PascalCase for components
* update dep
* wip
* wip
* wip
* Update init.ts
* wip
* Update paging.ts
* Update test.vue
* watch deep
* wip
* lint
* wip
* wip
* wip
* wip
* wiop
* wip
* Update webpack.config.ts
* alllow null poll
* wip
* wip
* wip
* wiop
* UI redesign & refactor (#6714)
* wip
* wip
* wip
* wip
* wip
* Update drive.vue
* Update word-mute.vue
* wip
* wip
* wip
* clean up
* wip
* Update default.vue
* wip
* Update notes.vue
* Update mfm.ts
* Update index.home.vue
* Update post-form.vue
* Update post-form-attaches.vue
* wip
* Update post-form.vue
* Update sidebar.vue
* wip
* wip
* Update index.vue
* wip
* Update default.vue
* Update index.vue
* Update index.vue
* wip
* Update post-form-attaches.vue
* Update note.vue
* wip
* clean up
* Update notes.vue
* wip
* wip
* Update ja-JP.yml
* wip
* wip
* Update index.vue
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* wip
* Update default.vue
* wip
* Update _dark.json5
* wip
* wip
* wip
* clean up
* wip
* wip
* Update index.vue
* Update test.vue
* wip
* wip
* fix
* wip
* wip
* wip
* wip
* clena yop
* wip
* wip
* Update store.ts
* Update messaging-room.vue
* Update default.widgets.vue
* fix
* wip
* wip
* Update modal.vue
* wip
* Update os.ts
* Update os.ts
* Update deck.vue
* Update init.ts
* wip
* Update ja-JP.yml
* v-sizeは単にwindowのresizeを監視するだけで良いかもしれない
* Update modal.vue
* wip
* Update tooltip.ts
* wip
* wip
* wip
* wip
* wip
* Update image-viewer.vue
* wip
* wip
* Update style.scss
* Update style.scss
* Update visitor.vue
* wip
* Update init.ts
* Update init.ts
* wip
* wip
* Update visitor.vue
* Update visitor.vue
* Update visitor.vue
* Update visitor.vue
* wip
* wip
* Update modal.vue
* Update header.vue
* Update menu.vue
* Update about.vue
* Update about-misskey.vue
* wip
* wip
* Update visitor.vue
* Update tooltip.ts
* wip
* Update drive.vue
* wip
* Update style.scss
* Update header.vue
* wip
* wip
* Update users.user.vue
* Update announcements.vue
* wip
* wip
* wip
* Update emojis.vue
* wip
* Update emojis.vue
* Update style.scss
* Update users.vue
* wip
* Update style.scss
* wip
* Update welcome.entrance.vue
* Update radio.vue
* Update size.ts
* Update emoji-edit-dialog.vue
* wip
* Update emojis.vue
* wip
* Update emojis.vue
* Update emojis.vue
* Update emojis.vue
* wip
* wip
* wip
* wip
* Update file-dialog.vue
* wip
* wip
* Update token-generate-window.vue
* Update notification-setting-window.vue
* wip
* wip
* Update _error_.vue
* Update ja-JP.yml
* wip
* wip
* Update store.ts
* Update emojis.vue
* Update emojis.vue
* Update emojis.vue
* Update announcements.vue
* Update store.ts
* wip
* Update page-editor.vue
* wip
* wip
* Update modal.vue
* wip
* Update select-file.ts
* Update timeline.vue
* Update emojis.vue
* Update os.ts
* wip
* Update user-select.vue
* Update mfm.ts
* Update get-file-info.ts
* Update drive.vue
* Update init.ts
* Update mfm.ts
* wip
* wip
* Update window.vue
* Update note.vue
* wip
* wip
* Update user-info.vue
* wip
* wip
* wip
* wip
* wip
* Update header.vue
* Update header.vue
* wip
* Update explore.vue
* wip
* wip
* wip
* Update webpack.config.ts
* wip
* wip
* wip
* wip
* wip
* wip
* Update autocomplete.ts
* wip
* wip
* wip
* Update toast.vue
* wip
* Update post-form-dialog.vue
* wip
* wip
* wip
* wip
* wip
* Update users.vue
* wip
* Update explore.vue
* wip
* wip
* wip
* Update package.json
* wip
* Update icon-dialog.vue
* wip
* wip
* Update user-preview.ts
* wip
* wip
* wip
* wip
* wip
* Update instance.vue
* Update user-name.vue
* Update federation.vue
* Update instance.vue
* wip
* wip
* Update tag.vue
* wip
* wip
* wip
* wip
* wip
* Update instance.vue
* wip
* Update os.ts
* Update os.ts
* wip
* wip
* wip
* Update router.ts
* wip
* Update init.ts
* Update note.vue
* Update messages.vue
* wip
* wip
* wip
* wip
* wip
* google
* wip
* wip
* wip
* wip
* Update theme-editor.vue
* wip
* wip
* Update room.vue
* Update channel-editor.vue
* wip
* Update window.vue
* Update window.vue
* wip
* Update window.vue
* Update window.vue
* wip
* Update menu.vue
* wip
* wip
* wip
* wip
* Update messaging-room.vue
* wip
* Update post-form.vue
* Update default.widgets.vue
* Update window.vue
* wip
206 lines
5.9 KiB
TypeScript
206 lines
5.9 KiB
TypeScript
import { EntityRepository, Repository } from 'typeorm';
|
|
import { DriveFile } from '../entities/drive-file';
|
|
import { Users, DriveFolders } from '..';
|
|
import { User } from '../entities/user';
|
|
import { toPuny } from '../../misc/convert-host';
|
|
import { ensure } from '../../prelude/ensure';
|
|
import { awaitAll } from '../../prelude/await-all';
|
|
import { SchemaType } from '../../misc/schema';
|
|
import config from '../../config';
|
|
import { query, appendQuery } from '../../prelude/url';
|
|
import { Meta } from '../entities/meta';
|
|
import { fetchMeta } from '../../misc/fetch-meta';
|
|
|
|
export type PackedDriveFile = SchemaType<typeof packedDriveFileSchema>;
|
|
|
|
@EntityRepository(DriveFile)
|
|
export class DriveFileRepository extends Repository<DriveFile> {
|
|
public validateFileName(name: string): boolean {
|
|
return (
|
|
(name.trim().length > 0) &&
|
|
(name.length <= 200) &&
|
|
(name.indexOf('\\') === -1) &&
|
|
(name.indexOf('/') === -1) &&
|
|
(name.indexOf('..') === -1)
|
|
);
|
|
}
|
|
|
|
public getPublicUrl(file: DriveFile, thumbnail = false, meta?: Meta): string | null {
|
|
// リモートかつメディアプロキシ
|
|
if (file.uri != null && file.userHost != null && config.mediaProxy != null) {
|
|
return appendQuery(config.mediaProxy, query({
|
|
url: file.uri,
|
|
thumbnail: thumbnail ? '1' : undefined
|
|
}));
|
|
}
|
|
|
|
// リモートかつ期限切れはローカルプロキシを試みる
|
|
if (file.uri != null && file.isLink && meta && meta.proxyRemoteFiles) {
|
|
const key = thumbnail ? file.thumbnailAccessKey : file.webpublicAccessKey;
|
|
|
|
if (key && !key.match('/')) { // 古いものはここにオブジェクトストレージキーが入ってるので除外
|
|
return `${config.url}/files/${key}`;
|
|
}
|
|
}
|
|
|
|
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);
|
|
}
|
|
|
|
public async clacDriveUsageOf(user: User['id'] | User): Promise<number> {
|
|
const id = typeof user === 'object' ? user.id : user;
|
|
|
|
const { sum } = await this
|
|
.createQueryBuilder('file')
|
|
.where('file.userId = :id', { id: id })
|
|
.select('SUM(file.size)', 'sum')
|
|
.getRawOne();
|
|
|
|
return parseInt(sum, 10) || 0;
|
|
}
|
|
|
|
public async clacDriveUsageOfHost(host: string): Promise<number> {
|
|
const { sum } = await this
|
|
.createQueryBuilder('file')
|
|
.where('file.userHost = :host', { host: toPuny(host) })
|
|
.select('SUM(file.size)', 'sum')
|
|
.getRawOne();
|
|
|
|
return parseInt(sum, 10) || 0;
|
|
}
|
|
|
|
public async clacDriveUsageOfLocal(): Promise<number> {
|
|
const { sum } = await this
|
|
.createQueryBuilder('file')
|
|
.where('file.userHost IS NULL')
|
|
.select('SUM(file.size)', 'sum')
|
|
.getRawOne();
|
|
|
|
return parseInt(sum, 10) || 0;
|
|
}
|
|
|
|
public async clacDriveUsageOfRemote(): Promise<number> {
|
|
const { sum } = await this
|
|
.createQueryBuilder('file')
|
|
.where('file.userHost IS NOT NULL')
|
|
.select('SUM(file.size)', 'sum')
|
|
.getRawOne();
|
|
|
|
return parseInt(sum, 10) || 0;
|
|
}
|
|
|
|
public async pack(
|
|
src: DriveFile['id'] | DriveFile,
|
|
options?: {
|
|
detail?: boolean,
|
|
self?: boolean,
|
|
withUser?: boolean,
|
|
}
|
|
): Promise<PackedDriveFile> {
|
|
const opts = Object.assign({
|
|
detail: false,
|
|
self: false
|
|
}, options);
|
|
|
|
const file = typeof src === 'object' ? src : await this.findOne(src).then(ensure);
|
|
|
|
const meta = await fetchMeta();
|
|
|
|
return await awaitAll({
|
|
id: file.id,
|
|
createdAt: file.createdAt.toISOString(),
|
|
name: file.name,
|
|
type: file.type,
|
|
md5: file.md5,
|
|
size: file.size,
|
|
isSensitive: file.isSensitive,
|
|
blurhash: file.blurhash,
|
|
properties: file.properties,
|
|
url: opts.self ? file.url : this.getPublicUrl(file, false, meta),
|
|
thumbnailUrl: this.getPublicUrl(file, true, meta),
|
|
comment: file.comment,
|
|
folderId: file.folderId,
|
|
folder: opts.detail && file.folderId ? DriveFolders.pack(file.folderId, {
|
|
detail: true
|
|
}) : null,
|
|
userId: opts.withUser ? file.userId! : null,
|
|
user: opts.withUser ? Users.pack(file.userId!) : null
|
|
});
|
|
}
|
|
|
|
public packMany(
|
|
files: any[],
|
|
options?: {
|
|
detail?: boolean
|
|
self?: boolean,
|
|
withUser?: boolean,
|
|
}
|
|
) {
|
|
return Promise.all(files.map(f => this.pack(f, options)));
|
|
}
|
|
}
|
|
|
|
export const packedDriveFileSchema = {
|
|
type: 'object' as const,
|
|
optional: false as const, nullable: false as const,
|
|
properties: {
|
|
id: {
|
|
type: 'string' as const,
|
|
optional: false as const, nullable: false as const,
|
|
format: 'id',
|
|
description: 'The unique identifier for this Drive file.',
|
|
example: 'xxxxxxxxxx',
|
|
},
|
|
createdAt: {
|
|
type: 'string' as const,
|
|
optional: false as const, nullable: false as const,
|
|
format: 'date-time',
|
|
description: 'The date that the Drive file was created on Misskey.'
|
|
},
|
|
name: {
|
|
type: 'string' as const,
|
|
optional: false as const, nullable: false as const,
|
|
description: 'The file name with extension.',
|
|
example: 'lenna.jpg'
|
|
},
|
|
type: {
|
|
type: 'string' as const,
|
|
optional: false as const, nullable: false as const,
|
|
description: 'The MIME type of this Drive file.',
|
|
example: 'image/jpeg'
|
|
},
|
|
md5: {
|
|
type: 'string' as const,
|
|
optional: false as const, nullable: false as const,
|
|
format: 'md5',
|
|
description: 'The MD5 hash of this Drive file.',
|
|
example: '15eca7fba0480996e2245f5185bf39f2'
|
|
},
|
|
size: {
|
|
type: 'number' as const,
|
|
optional: false as const, nullable: false as const,
|
|
description: 'The size of this Drive file. (bytes)',
|
|
example: 51469
|
|
},
|
|
url: {
|
|
type: 'string' as const,
|
|
optional: false as const, nullable: true as const,
|
|
format: 'url',
|
|
description: 'The URL of this Drive file.',
|
|
},
|
|
folderId: {
|
|
type: 'string' as const,
|
|
optional: false as const, nullable: true as const,
|
|
format: 'id',
|
|
description: 'The parent folder ID of this Drive file.',
|
|
example: 'xxxxxxxxxx',
|
|
},
|
|
isSensitive: {
|
|
type: 'boolean' as const,
|
|
optional: false as const, nullable: false as const,
|
|
description: 'Whether this Drive file is sensitive.',
|
|
},
|
|
},
|
|
};
|