stream footer
This commit is contained in:
parent
a4cf46c0eb
commit
9d04514f8e
3 changed files with 36 additions and 8 deletions
13
app/api.js
13
app/api.js
|
@ -115,6 +115,7 @@ function listenForResponse(ws, canceller) {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
ws.close();
|
||||||
canceller.cancelled = true;
|
canceller.cancelled = true;
|
||||||
canceller.error = e;
|
canceller.error = e;
|
||||||
reject(e);
|
reject(e);
|
||||||
|
@ -134,7 +135,6 @@ async function upload(
|
||||||
const host = window.location.hostname;
|
const host = window.location.hostname;
|
||||||
const port = window.location.port;
|
const port = window.location.port;
|
||||||
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
const protocol = window.location.protocol === 'https:' ? 'wss:' : 'ws:';
|
||||||
const error = { cancelled: false };
|
|
||||||
const ws = await asyncInitWebSocket(`${protocol}//${host}:${port}/api/ws`);
|
const ws = await asyncInitWebSocket(`${protocol}//${host}:${port}/api/ws`);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
@ -144,7 +144,7 @@ async function upload(
|
||||||
authorization: `send-v1 ${verifierB64}`
|
authorization: `send-v1 ${verifierB64}`
|
||||||
};
|
};
|
||||||
|
|
||||||
const responsePromise = listenForResponse(ws, error);
|
const responsePromise = listenForResponse(ws, canceller);
|
||||||
|
|
||||||
ws.send(JSON.stringify(fileMeta));
|
ws.send(JSON.stringify(fileMeta));
|
||||||
|
|
||||||
|
@ -154,17 +154,17 @@ async function upload(
|
||||||
while (!state.done) {
|
while (!state.done) {
|
||||||
const buf = state.value;
|
const buf = state.value;
|
||||||
if (canceller.cancelled) {
|
if (canceller.cancelled) {
|
||||||
throw new Error(0);
|
throw canceller.error;
|
||||||
}
|
|
||||||
if (error.cancelled) {
|
|
||||||
throw new Error(error.error);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
ws.send(buf);
|
ws.send(buf);
|
||||||
|
|
||||||
onprogress([Math.min(streamInfo.fileSize, size), streamInfo.fileSize]);
|
onprogress([Math.min(streamInfo.fileSize, size), streamInfo.fileSize]);
|
||||||
size += streamInfo.recordSize;
|
size += streamInfo.recordSize;
|
||||||
state = await reader.read();
|
state = await reader.read();
|
||||||
}
|
}
|
||||||
|
const footer = new Uint8Array([0]);
|
||||||
|
ws.send(footer);
|
||||||
|
|
||||||
const response = await responsePromise; //promise only fufills if response is good
|
const response = await responsePromise; //promise only fufills if response is good
|
||||||
ws.close();
|
ws.close();
|
||||||
|
@ -180,6 +180,7 @@ export function uploadWs(encrypted, info, metadata, verifierB64, onprogress) {
|
||||||
|
|
||||||
return {
|
return {
|
||||||
cancel: function() {
|
cancel: function() {
|
||||||
|
canceller.error = new Error(0);
|
||||||
canceller.cancelled = true;
|
canceller.cancelled = true;
|
||||||
},
|
},
|
||||||
result: upload(
|
result: upload(
|
||||||
|
|
|
@ -3,6 +3,7 @@ const storage = require('../storage');
|
||||||
const config = require('../config');
|
const config = require('../config');
|
||||||
const mozlog = require('../log');
|
const mozlog = require('../log');
|
||||||
const Limiter = require('../limiter');
|
const Limiter = require('../limiter');
|
||||||
|
const Parser = require('../streamparser');
|
||||||
const wsStream = require('websocket-stream/stream');
|
const wsStream = require('websocket-stream/stream');
|
||||||
|
|
||||||
const log = mozlog('send.upload');
|
const log = mozlog('send.upload');
|
||||||
|
@ -45,7 +46,10 @@ module.exports = async function(ws, req) {
|
||||||
const url = `${protocol}://${req.get('host')}/download/${newId}/`;
|
const url = `${protocol}://${req.get('host')}/download/${newId}/`;
|
||||||
|
|
||||||
const limiter = new Limiter(config.max_file_size);
|
const limiter = new Limiter(config.max_file_size);
|
||||||
fileStream = wsStream(ws, { binary: true }).pipe(limiter);
|
const parser = new Parser();
|
||||||
|
fileStream = wsStream(ws, { binary: true })
|
||||||
|
.pipe(limiter)
|
||||||
|
.pipe(parser);
|
||||||
storage.set(newId, fileStream, meta);
|
storage.set(newId, fileStream, meta);
|
||||||
|
|
||||||
ws.send(
|
ws.send(
|
||||||
|
@ -60,7 +64,7 @@ module.exports = async function(ws, req) {
|
||||||
log.error('upload', e);
|
log.error('upload', e);
|
||||||
ws.send(
|
ws.send(
|
||||||
JSON.stringify({
|
JSON.stringify({
|
||||||
error: 500
|
error: e === 'limit' ? 413 : 500
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
ws.close();
|
ws.close();
|
||||||
|
|
23
server/streamparser.js
Normal file
23
server/streamparser.js
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
const { Transform } = require('stream');
|
||||||
|
|
||||||
|
class StreamParser extends Transform {
|
||||||
|
constructor() {
|
||||||
|
super();
|
||||||
|
let res;
|
||||||
|
this.promise = new Promise(resolve => {
|
||||||
|
res = resolve;
|
||||||
|
});
|
||||||
|
this.res = res;
|
||||||
|
}
|
||||||
|
|
||||||
|
_transform(chunk, encoding, callback) {
|
||||||
|
if (chunk.byteLength === 1 && chunk[0] === 0) {
|
||||||
|
this.res();
|
||||||
|
} else {
|
||||||
|
this.push(chunk);
|
||||||
|
}
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = StreamParser;
|
Loading…
Reference in a new issue