2018-04-13 00:06:18 +03:00
|
|
|
import * as Koa from 'koa';
|
|
|
|
import * as send from 'koa-send';
|
|
|
|
import * as mongodb from 'mongodb';
|
2018-05-03 14:03:14 +03:00
|
|
|
import DriveFile, { getDriveFileBucket } from '../../models/drive-file';
|
|
|
|
import DriveFileThumbnail, { getDriveFileThumbnailBucket } from '../../models/drive-file-thumbnail';
|
2018-11-25 21:25:48 +02:00
|
|
|
import DriveFileWebpublic, { getDriveFileWebpublicBucket } from '../../models/drive-file-webpublic';
|
2018-05-03 14:03:14 +03:00
|
|
|
|
2018-05-04 11:59:51 +03:00
|
|
|
const assets = `${__dirname}/../../server/file/assets/`;
|
|
|
|
|
2019-01-22 14:42:05 +02:00
|
|
|
const commonReadableHandlerGenerator = (ctx: Koa.BaseContext) => (e: Error): void => {
|
2018-05-03 14:03:14 +03:00
|
|
|
console.error(e);
|
|
|
|
ctx.status = 500;
|
|
|
|
};
|
2018-04-13 00:06:18 +03:00
|
|
|
|
2019-01-22 14:42:05 +02:00
|
|
|
export default async function(ctx: Koa.BaseContext) {
|
2018-04-13 00:06:18 +03:00
|
|
|
// Validate id
|
|
|
|
if (!mongodb.ObjectID.isValid(ctx.params.id)) {
|
|
|
|
ctx.throw(400, 'incorrect id');
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const fileId = new mongodb.ObjectID(ctx.params.id);
|
|
|
|
|
|
|
|
// Fetch drive file
|
|
|
|
const file = await DriveFile.findOne({ _id: fileId });
|
|
|
|
|
|
|
|
if (file == null) {
|
|
|
|
ctx.status = 404;
|
2019-01-22 14:42:05 +02:00
|
|
|
await send(ctx as any, '/dummy.png', { root: assets });
|
2018-04-13 00:06:18 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-17 14:04:19 +03:00
|
|
|
if (file.metadata.deletedAt) {
|
|
|
|
ctx.status = 410;
|
2019-01-22 14:42:05 +02:00
|
|
|
await send(ctx as any, '/tombstone.png', { root: assets });
|
2018-05-25 14:19:14 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-07-23 23:04:43 +03:00
|
|
|
if (file.metadata.withoutChunks) {
|
2018-05-25 14:19:14 +03:00
|
|
|
ctx.status = 204;
|
2018-04-17 14:04:19 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-03 14:12:08 +03:00
|
|
|
const sendRaw = async () => {
|
2018-11-25 21:25:48 +02:00
|
|
|
if (file.metadata && file.metadata.accessKey && file.metadata.accessKey != ctx.query['original']) {
|
|
|
|
ctx.status = 403;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-03 14:12:08 +03:00
|
|
|
const bucket = await getDriveFileBucket();
|
|
|
|
const readable = bucket.openDownloadStream(fileId);
|
|
|
|
readable.on('error', commonReadableHandlerGenerator(ctx));
|
|
|
|
ctx.set('Content-Type', file.contentType);
|
|
|
|
ctx.body = readable;
|
|
|
|
};
|
|
|
|
|
2018-05-03 14:03:14 +03:00
|
|
|
if ('thumbnail' in ctx.query) {
|
2018-08-16 01:17:04 +03:00
|
|
|
const thumb = await DriveFileThumbnail.findOne({
|
|
|
|
'metadata.originalId': fileId
|
|
|
|
});
|
|
|
|
|
|
|
|
if (thumb != null) {
|
|
|
|
ctx.set('Content-Type', 'image/jpeg');
|
|
|
|
const bucket = await getDriveFileThumbnailBucket();
|
|
|
|
ctx.body = bucket.openDownloadStream(thumb._id);
|
2018-05-03 14:03:14 +03:00
|
|
|
} else {
|
2018-08-16 01:17:04 +03:00
|
|
|
await sendRaw();
|
2018-05-03 14:03:14 +03:00
|
|
|
}
|
2018-11-25 21:25:48 +02:00
|
|
|
} else if ('web' in ctx.query) {
|
|
|
|
const web = await DriveFileWebpublic.findOne({
|
|
|
|
'metadata.originalId': fileId
|
|
|
|
});
|
|
|
|
|
|
|
|
if (web != null) {
|
|
|
|
ctx.set('Content-Type', file.contentType);
|
|
|
|
|
|
|
|
const bucket = await getDriveFileWebpublicBucket();
|
|
|
|
ctx.body = bucket.openDownloadStream(web._id);
|
|
|
|
} else {
|
|
|
|
await sendRaw();
|
|
|
|
}
|
2018-05-03 14:03:14 +03:00
|
|
|
} else {
|
|
|
|
if ('download' in ctx.query) {
|
|
|
|
ctx.set('Content-Disposition', 'attachment');
|
|
|
|
}
|
2018-04-13 00:06:18 +03:00
|
|
|
|
2018-05-03 14:12:08 +03:00
|
|
|
await sendRaw();
|
2018-05-03 14:03:14 +03:00
|
|
|
}
|
2018-04-13 00:06:18 +03:00
|
|
|
}
|