Sharkey/src/models/drive-file.ts

239 lines
5.4 KiB
TypeScript
Raw Normal View History

2018-04-11 22:22:06 +03:00
import * as mongo from 'mongodb';
import * as deepcopy from 'deepcopy';
2018-02-02 01:06:01 +02:00
import { pack as packFolder } from './drive-folder';
import { pack as packUser, IUser } from './user';
import monkDb, { nativeDbConn, dbLogger } from '../db/mongodb';
2018-10-16 05:38:09 +03:00
import isObjectId from '../misc/is-objectid';
import getDriveFileUrl, { getOriginalUrl } from '../misc/get-drive-file-url';
import wrapUrl from '../misc/wrap-url';
2017-01-17 02:12:33 +02:00
2018-03-29 08:48:47 +03:00
const DriveFile = monkDb.get<IDriveFile>('driveFiles.files');
DriveFile.createIndex('md5');
2018-08-11 21:09:22 +03:00
DriveFile.createIndex('metadata.uri');
2018-10-12 19:00:43 +03:00
DriveFile.createIndex('metadata.userId');
2018-10-12 19:17:23 +03:00
DriveFile.createIndex('metadata.folderId');
2018-02-02 01:06:01 +02:00
export default DriveFile;
2016-12-29 00:49:51 +02:00
2018-04-17 13:55:58 +03:00
export const DriveFileChunk = monkDb.get('driveFiles.chunks');
export const getDriveFileBucket = async (): Promise<mongo.GridFSBucket> => {
2017-11-06 09:32:01 +02:00
const db = await nativeDbConn();
2018-04-11 22:22:06 +03:00
const bucket = new mongo.GridFSBucket(db, {
2018-03-29 08:48:47 +03:00
bucketName: 'driveFiles'
2017-11-06 09:32:01 +02:00
});
return bucket;
};
2017-11-06 07:37:00 +02:00
2018-04-03 17:45:13 +03:00
export type IMetadata = {
properties: any;
2018-04-11 22:22:06 +03:00
userId: mongo.ObjectID;
_user: any;
2018-04-11 22:22:06 +03:00
folderId: mongo.ObjectID;
2018-04-03 17:45:13 +03:00
comment: string;
/**
* URL
*/
uri?: string;
/**
* URL for web() or original
* * or
*/
url?: string;
/**
* URL for thumbnail (thumbnailがなければなし)
* * or
*/
2018-08-16 01:17:04 +03:00
thumbnailUrl?: string;
/**
* URL for original (web用が生成されてない場合はurlがoriginalを指す)
* * or
*/
webpublicUrl?: string;
accessKey?: string;
2018-07-23 23:04:43 +03:00
src?: string;
2018-04-17 14:04:19 +03:00
deletedAt?: Date;
2018-10-31 15:35:02 +02:00
/**
* MongoDB内に保存されていないか否か
2018-10-31 15:35:02 +02:00
* or
* true
2018-10-31 15:35:02 +02:00
*/
2018-07-23 23:04:43 +03:00
withoutChunks?: boolean;
2018-10-31 15:35:02 +02:00
2018-07-24 00:21:21 +03:00
storage?: string;
/***
* ObjectStorage
*/
storageProps?: IStorageProps;
2018-07-19 20:40:37 +03:00
isSensitive?: boolean;
2018-07-25 03:54:03 +03:00
2018-10-23 01:04:00 +03:00
/**
* 稿ID一覧
*/
attachedNoteIds?: mongo.ObjectID[];
2018-07-25 03:54:03 +03:00
/**
* ()URLへの直リンクか否か
*/
2018-07-24 18:29:18 +03:00
isRemote?: boolean;
2018-04-03 17:45:13 +03:00
};
export type IStorageProps = {
/**
* ObjectStorage key for original
*/
key: string;
/***
* ObjectStorage key for thumbnail (thumbnailがなければなし)
*/
thumbnailKey?: string;
/***
* ObjectStorage key for webpublic (webpublicがなければなし)
*/
webpublicKey?: string;
id?: string;
};
2018-02-02 01:06:01 +02:00
export type IDriveFile = {
2018-04-11 22:22:06 +03:00
_id: mongo.ObjectID;
2018-02-02 01:21:30 +02:00
uploadDate: Date;
md5: string;
filename: string;
2018-02-04 07:52:33 +02:00
contentType: string;
2018-04-03 17:45:13 +03:00
metadata: IMetadata;
2018-08-18 17:48:54 +03:00
/**
*
*/
length: number;
2018-02-02 01:06:01 +02:00
};
2016-12-29 00:49:51 +02:00
export function validateFileName(name: string): boolean {
return (
(name.trim().length > 0) &&
(name.length <= 200) &&
(name.indexOf('\\') === -1) &&
(name.indexOf('/') === -1) &&
(name.indexOf('..') === -1)
);
}
2018-02-02 01:06:01 +02:00
2018-10-31 04:22:49 +02:00
export const packMany = (
files: any[],
options?: {
detail?: boolean
self?: boolean,
2018-12-14 12:09:11 +02:00
withUser?: boolean,
me?: string | mongo.ObjectID | IUser,
}
) => {
2018-10-31 04:22:49 +02:00
return Promise.all(files.map(f => pack(f, options)));
};
2018-02-02 01:06:01 +02:00
/**
* Pack a drive file for API response
*/
export const pack = (
file: any,
options?: {
detail?: boolean,
self?: boolean,
2018-12-14 12:09:11 +02:00
withUser?: boolean,
me?: string | mongo.ObjectID | IUser,
2018-02-02 01:06:01 +02:00
}
) => new Promise<any>(async (resolve, reject) => {
const opts = Object.assign({
detail: false,
self: false
2018-02-02 01:06:01 +02:00
}, options);
let _file: any;
// Populate the file if 'file' is ID
2018-10-16 05:38:09 +03:00
if (isObjectId(file)) {
2018-02-02 01:06:01 +02:00
_file = await DriveFile.findOne({
_id: file
});
} else if (typeof file === 'string') {
_file = await DriveFile.findOne({
2018-04-12 00:20:53 +03:00
_id: new mongo.ObjectID(file)
2018-02-02 01:06:01 +02:00
});
} else {
_file = deepcopy(file);
}
// (データベースの欠損などで)ファイルがデータベース上に見つからなかったとき
if (_file == null) {
dbLogger.warn(`[DAMAGED DB] (missing) pkg: driveFile :: ${file}`);
2018-10-04 22:50:23 +03:00
return resolve(null);
}
2018-02-02 01:06:01 +02:00
// rendered target
let _target: any = {};
_target.id = _file._id;
2018-03-29 08:48:47 +03:00
_target.createdAt = _file.uploadDate;
2018-02-02 01:06:01 +02:00
_target.name = _file.filename;
_target.type = _file.contentType;
_target.datasize = _file.length;
_target.md5 = _file.md5;
_target = Object.assign(_target, _file.metadata);
2018-10-31 15:55:17 +02:00
_target.url = getDriveFileUrl(_file);
_target.thumbnailUrl = getDriveFileUrl(_file, true);
if (_target.thumbnailUrl != null) {
_target.thumbnailUrl = wrapUrl(_target.thumbnailUrl, options.me);
}
2018-07-24 18:29:18 +03:00
_target.isRemote = _file.metadata.isRemote;
2018-02-02 01:06:01 +02:00
if (_target.properties == null) _target.properties = {};
if (opts.detail) {
2018-03-29 08:48:47 +03:00
if (_target.folderId) {
2018-02-02 01:06:01 +02:00
// Populate folder
2018-03-29 08:48:47 +03:00
_target.folder = await packFolder(_target.folderId, {
2018-02-02 01:06:01 +02:00
detail: true
});
}
/*
if (_target.tags) {
// Populate tags
_target.tags = await _target.tags.map(async (tag: any) =>
await serializeDriveTag(tag)
);
}
*/
}
2018-12-14 12:09:11 +02:00
if (opts.withUser) {
// Populate user
_target.user = await packUser(_file.metadata.userId);
}
2018-09-02 03:31:11 +03:00
delete _target.withoutChunks;
delete _target.storage;
delete _target.storageProps;
delete _target.isRemote;
2018-11-01 02:02:54 +02:00
delete _target._user;
2018-09-02 03:31:11 +03:00
if (opts.self) {
_target.url = getOriginalUrl(_file);
}
2018-02-02 01:06:01 +02:00
resolve(_target);
});