summaryrefslogtreecommitdiff
path: root/src/server/file
diff options
context:
space:
mode:
authorsyuilo <syuilotan@yahoo.co.jp>2018-04-13 06:06:18 +0900
committersyuilo <syuilotan@yahoo.co.jp>2018-04-13 06:06:18 +0900
commit3368fe855249f45bdf1e4c1e509d325d44e80fbe (patch)
tree63c8bc61fb645b1d730b05120ab5117c0fdeee29 /src/server/file
parentwip (diff)
downloadsharkey-3368fe855249f45bdf1e4c1e509d325d44e80fbe.tar.gz
sharkey-3368fe855249f45bdf1e4c1e509d325d44e80fbe.tar.bz2
sharkey-3368fe855249f45bdf1e4c1e509d325d44e80fbe.zip
wip
Diffstat (limited to 'src/server/file')
-rw-r--r--src/server/file/index.ts172
-rw-r--r--src/server/file/pour.ts93
-rw-r--r--src/server/file/send-drive-file.ts30
3 files changed, 140 insertions, 155 deletions
diff --git a/src/server/file/index.ts b/src/server/file/index.ts
index 95e7867f01..d58939f1be 100644
--- a/src/server/file/index.ts
+++ b/src/server/file/index.ts
@@ -3,171 +3,33 @@
*/
import * as fs from 'fs';
-import * as express from 'express';
-import * as bodyParser from 'body-parser';
-import * as cors from 'cors';
-import * as mongodb from 'mongodb';
-import * as _gm from 'gm';
-import * as stream from 'stream';
+import * as Koa from 'koa';
+import * as cors from '@koa/cors';
+import * as Router from 'koa-router';
+import pour from './pour';
+import sendDriveFile from './send-drive-file';
-import DriveFile, { getGridFSBucket } from '../../models/drive-file';
-
-const gm = _gm.subClass({
- imageMagick: true
-});
-
-/**
- * Init app
- */
-const app = express();
-
-app.disable('x-powered-by');
-app.locals.cache = true;
-app.use(bodyParser.urlencoded({ extended: true }));
+// Init app
+const app = new Koa();
app.use(cors());
-/**
- * Statics
- */
-app.use('/assets', express.static(`${__dirname}/assets`, {
- maxAge: 1000 * 60 * 60 * 24 * 365 // 一年
-}));
+// Init router
+const router = new Router();
-app.get('/', (req, res) => {
- res.send('yee haw');
-});
-
-app.get('/default-avatar.jpg', (req, res) => {
+router.get('/default-avatar.jpg', ctx => {
const file = fs.createReadStream(`${__dirname}/assets/avatar.jpg`);
- send(file, 'image/jpeg', req, res);
+ pour(file, 'image/jpeg', ctx);
});
-app.get('/app-default.jpg', (req, res) => {
+router.get('/app-default.jpg', ctx => {
const file = fs.createReadStream(`${__dirname}/assets/dummy.png`);
- send(file, 'image/png', req, res);
+ pour(file, 'image/png', ctx);
});
-interface ISend {
- contentType: string;
- stream: stream.Readable;
-}
-
-function thumbnail(data: stream.Readable, type: string, resize: number): ISend {
- const readable: stream.Readable = (() => {
- // 動画であれば
- if (/^video\/.*$/.test(type)) {
- // TODO
- // 使わないことになったストリームはしっかり取り壊す
- data.destroy();
- return fs.createReadStream(`${__dirname}/assets/thumbnail-not-available.png`);
- // 画像であれば
- // Note: SVGはapplication/xml
- } else if (/^image\/.*$/.test(type) || type == 'application/xml') {
- // 0フレーム目を送る
- try {
- return gm(data).selectFrame(0).stream();
- // だめだったら
- } catch (e) {
- // 使わないことになったストリームはしっかり取り壊す
- data.destroy();
- return fs.createReadStream(`${__dirname}/assets/thumbnail-not-available.png`);
- }
- // 動画か画像以外
- } else {
- data.destroy();
- return fs.createReadStream(`${__dirname}/assets/not-an-image.png`);
- }
- })();
-
- let g = gm(readable);
-
- if (resize) {
- g = g.resize(resize, resize);
- }
-
- const stream = g
- .compress('jpeg')
- .quality(80)
- .interlace('line')
- .stream();
-
- return {
- contentType: 'image/jpeg',
- stream
- };
-}
-
-const commonReadableHandlerGenerator = (req: express.Request, res: express.Response) => (e: Error): void => {
- console.dir(e);
- req.destroy();
- res.destroy(e);
-};
-
-function send(readable: stream.Readable, type: string, req: express.Request, res: express.Response): void {
- readable.on('error', commonReadableHandlerGenerator(req, res));
-
- const data = ((): ISend => {
- if (req.query.thumbnail !== undefined) {
- return thumbnail(readable, type, req.query.size);
- }
- return {
- contentType: type,
- stream: readable
- };
- })();
-
- if (readable !== data.stream) {
- data.stream.on('error', commonReadableHandlerGenerator(req, res));
- }
-
- if (req.query.download !== undefined) {
- res.header('Content-Disposition', 'attachment');
- }
-
- res.header('Content-Type', data.contentType);
-
- data.stream.pipe(res);
-
- data.stream.on('end', () => {
- res.end();
- });
-}
-
-async function sendFileById(req: express.Request, res: express.Response): Promise<void> {
- // Validate id
- if (!mongodb.ObjectID.isValid(req.params.id)) {
- res.status(400).send('incorrect id');
- return;
- }
-
- const fileId = new mongodb.ObjectID(req.params.id);
-
- // Fetch (drive) file
- const file = await DriveFile.findOne({ _id: fileId });
-
- // validate name
- if (req.params.name !== undefined && req.params.name !== file.filename) {
- res.status(404).send('there is no file has given name');
- return;
- }
-
- if (file == null) {
- res.status(404).sendFile(`${__dirname}/assets/dummy.png`);
- return;
- }
-
- const bucket = await getGridFSBucket();
-
- const readable = bucket.openDownloadStream(fileId);
-
- send(readable, file.contentType, req, res);
-}
-
-/**
- * Routing
- */
+router.get('/:id', sendDriveFile);
+router.get('/:id/:name', sendDriveFile);
-app.get('/:id', sendFileById);
-app.get('/:id/:name', sendFileById);
+// Register router
+app.use(router.routes());
module.exports = app;
diff --git a/src/server/file/pour.ts b/src/server/file/pour.ts
new file mode 100644
index 0000000000..2a31cb5898
--- /dev/null
+++ b/src/server/file/pour.ts
@@ -0,0 +1,93 @@
+import * as fs from 'fs';
+import * as stream from 'stream';
+import * as Koa from 'koa';
+import * as Gm from 'gm';
+
+const gm = Gm.subClass({
+ imageMagick: true
+});
+
+interface ISend {
+ contentType: string;
+ stream: stream.Readable;
+}
+
+function thumbnail(data: stream.Readable, type: string, resize: number): ISend {
+ const readable: stream.Readable = (() => {
+ // 動画であれば
+ if (/^video\/.*$/.test(type)) {
+ // TODO
+ // 使わないことになったストリームはしっかり取り壊す
+ data.destroy();
+ return fs.createReadStream(`${__dirname}/assets/thumbnail-not-available.png`);
+ // 画像であれば
+ // Note: SVGはapplication/xml
+ } else if (/^image\/.*$/.test(type) || type == 'application/xml') {
+ // 0フレーム目を送る
+ try {
+ return gm(data).selectFrame(0).stream();
+ // だめだったら
+ } catch (e) {
+ // 使わないことになったストリームはしっかり取り壊す
+ data.destroy();
+ return fs.createReadStream(`${__dirname}/assets/thumbnail-not-available.png`);
+ }
+ // 動画か画像以外
+ } else {
+ data.destroy();
+ return fs.createReadStream(`${__dirname}/assets/not-an-image.png`);
+ }
+ })();
+
+ let g = gm(readable);
+
+ if (resize) {
+ g = g.resize(resize, resize);
+ }
+
+ const stream = g
+ .compress('jpeg')
+ .quality(80)
+ .interlace('line')
+ .stream();
+
+ return {
+ contentType: 'image/jpeg',
+ stream
+ };
+}
+
+const commonReadableHandlerGenerator = (ctx: Koa.Context) => (e: Error): void => {
+ console.error(e);
+ ctx.status = 500;
+};
+
+export default function(readable: stream.Readable, type: string, ctx: Koa.Context): void {
+ readable.on('error', commonReadableHandlerGenerator(ctx));
+
+ const data = ((): ISend => {
+ if (ctx.query.thumbnail !== undefined) {
+ return thumbnail(readable, type, ctx.query.size);
+ }
+ return {
+ contentType: type,
+ stream: readable
+ };
+ })();
+
+ if (readable !== data.stream) {
+ data.stream.on('error', commonReadableHandlerGenerator(ctx));
+ }
+
+ if (ctx.query.download !== undefined) {
+ ctx.header('Content-Disposition', 'attachment');
+ }
+
+ ctx.header('Content-Type', data.contentType);
+
+ data.stream.pipe(ctx.res);
+
+ data.stream.on('end', () => {
+ ctx.res.end();
+ });
+}
diff --git a/src/server/file/send-drive-file.ts b/src/server/file/send-drive-file.ts
new file mode 100644
index 0000000000..e6ee19ff1d
--- /dev/null
+++ b/src/server/file/send-drive-file.ts
@@ -0,0 +1,30 @@
+import * as Koa from 'koa';
+import * as send from 'koa-send';
+import * as mongodb from 'mongodb';
+import DriveFile, { getGridFSBucket } from '../../models/drive-file';
+import pour from './pour';
+
+export default async function(ctx: Koa.Context) {
+ // 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;
+ await send(ctx, `${__dirname}/assets/dummy.png`);
+ return;
+ }
+
+ const bucket = await getGridFSBucket();
+
+ const readable = bucket.openDownloadStream(fileId);
+
+ pour(readable, file.contentType, ctx);
+}