extracted blobSlicer, fixed download cancel tests

This commit is contained in:
Danny Coates 2018-07-25 12:29:19 -07:00
parent 5483dc2506
commit 5e9e63944b
No known key found for this signature in database
GPG Key ID: 4C442633C62E00CB
6 changed files with 58 additions and 58 deletions

27
app/blobSlicer.js Normal file
View File

@ -0,0 +1,27 @@
export default class BlobSlicer {
constructor(blob, size) {
this.blob = blob;
this.index = 0;
this.chunkSize = size;
}
pull(controller) {
return new Promise((resolve, reject) => {
const bytesLeft = this.blob.size - this.index;
if (bytesLeft <= 0) {
controller.close();
return resolve();
}
const size = Math.min(this.chunkSize, bytesLeft);
const blob = this.blob.slice(this.index, this.index + size);
const reader = new FileReader();
reader.onload = () => {
controller.enqueue(new Uint8Array(reader.result));
resolve();
};
reader.onerror = reject;
reader.readAsArrayBuffer(blob);
this.index += size;
});
}
}

View File

@ -1,4 +1,5 @@
import 'buffer'; import 'buffer';
import BlobSlicer from './blobSlicer';
import { transformStream } from './streams'; import { transformStream } from './streams';
const NONCE_LENGTH = 12; const NONCE_LENGTH = 12;
@ -225,40 +226,6 @@ class ECETransformer {
} }
} }
export class BlobSlicer {
constructor(blob, rs, mode) {
this.blob = blob;
this.index = 0;
this.mode = mode;
this.chunkSize = mode === MODE_ENCRYPT ? rs - 17 : rs;
}
pull(controller) {
return new Promise((resolve, reject) => {
const bytesLeft = this.blob.size - this.index;
if (bytesLeft <= 0) {
controller.close();
return resolve();
}
let size = 1;
if (this.mode === MODE_DECRYPT && this.index === 0) {
size = Math.min(21, bytesLeft);
} else {
size = Math.min(this.chunkSize, bytesLeft);
}
const blob = this.blob.slice(this.index, this.index + size);
const reader = new FileReader();
reader.onload = () => {
controller.enqueue(new Uint8Array(reader.result));
resolve();
};
reader.onerror = reject;
reader.readAsArrayBuffer(blob);
this.index += size;
});
}
}
class StreamSlicer { class StreamSlicer {
constructor(rs, mode) { constructor(rs, mode) {
this.mode = mode; this.mode = mode;
@ -350,7 +317,7 @@ export default class ECE {
if (this.input instanceof Blob) { if (this.input instanceof Blob) {
inputStream = new ReadableStream( inputStream = new ReadableStream(
new BlobSlicer(this.input, this.rs, this.mode) new BlobSlicer(this.input, this.rs - 17)
); );
} else { } else {
inputStream = transformStream( inputStream = transformStream(

View File

@ -148,7 +148,7 @@ export default class FileReceiver extends Nanobus {
this.state = 'complete'; this.state = 'complete';
} catch (e) { } catch (e) {
this.downloadRequest = null; this.downloadRequest = null;
if (e === 'cancelled') { if (e === 'cancelled' || e.message === '400') {
throw new Error(0); throw new Error(0);
} }
throw e; throw e;

View File

@ -15,8 +15,11 @@ self.addEventListener('activate', event => {
}); });
async function decryptStream(id) { async function decryptStream(id) {
const file = map.get(id);
if (!file) {
return new Response(null, { status: 400 });
}
try { try {
const file = map.get(id);
const keychain = new Keychain(file.key, file.nonce); const keychain = new Keychain(file.key, file.nonce);
if (file.requiresPassword) { if (file.requiresPassword) {
keychain.setPassword(file.password, file.url); keychain.setPassword(file.password, file.url);

View File

@ -3,6 +3,7 @@ require('buffer');
import assert from 'assert'; import assert from 'assert';
import { b64ToArray } from '../../../app/utils'; import { b64ToArray } from '../../../app/utils';
import BlobSlicer from '../../../app/blobSlicer';
import ECE from '../../../app/ece.js'; import ECE from '../../../app/ece.js';
const rs = 36; const rs = 36;
@ -47,8 +48,10 @@ describe('Streaming', function() {
}); });
it('can decrypt', async function() { it('can decrypt', async function() {
const encBlob = new Blob([encrypted]); const blobStream = new ReadableStream(
const ece = new ECE(encBlob, key, 'decrypt', rs); new BlobSlicer(new Blob([encrypted]), rs)
);
const ece = new ECE(blobStream, key, 'decrypt', rs);
const decStream = await ece.transform(); const decStream = await ece.transform();
const reader = decStream.getReader(); const reader = decStream.getReader();

View File

@ -153,26 +153,26 @@ describe('Upload / Download flow', function() {
assert.equal(file.dtotal, 1); assert.equal(file.dtotal, 1);
}); });
// it('does not increase download count when download cancelled', async function() { it('does not increase download count when download cancelled', async function() {
// const fs = new FileSender(blob); const fs = new FileSender(blob);
// const file = await fs.upload(); const file = await fs.upload();
// const fr = new FileReceiver({ const fr = new FileReceiver({
// secretKey: file.toJSON().secretKey, secretKey: file.toJSON().secretKey,
// id: file.id, id: file.id,
// nonce: file.keychain.nonce, nonce: file.keychain.nonce,
// requiresPassword: false requiresPassword: false
// }); });
// await fr.getMetadata(); await fr.getMetadata();
// fr.once('progress', () => fr.cancel()); fr.once('progress', () => fr.cancel());
// try { try {
// await fr.download(noSave); await fr.download(noSave);
// assert.fail('not cancelled'); assert.fail('not cancelled');
// } catch (e) { } catch (e) {
// await file.updateDownloadCount(); await file.updateDownloadCount();
// assert.equal(file.dtotal, 0); assert.equal(file.dtotal, 0);
// } }
// }); });
it('can allow multiple downloads', async function() { it('can allow multiple downloads', async function() {
const fs = new FileSender(blob); const fs = new FileSender(blob);