|
|
|
define([
|
|
|
|
'/bower_components/tweetnacl/nacl-fast.min.js',
|
|
|
|
], function () {
|
|
|
|
var Nacl = window.nacl;
|
|
|
|
var PARANOIA = true;
|
|
|
|
|
|
|
|
var plainChunkLength = 128 * 1024;
|
|
|
|
var cypherChunkLength = 131088;
|
|
|
|
|
|
|
|
var encodePrefix = function (p) {
|
|
|
|
return [
|
|
|
|
65280, // 255 << 8
|
|
|
|
255,
|
|
|
|
].map(function (n, i) {
|
|
|
|
return (p & n) >> ((1 - i) * 8);
|
|
|
|
});
|
|
|
|
};
|
|
|
|
var decodePrefix = function (A) {
|
|
|
|
return (A[0] << 8) | A[1];
|
|
|
|
};
|
|
|
|
|
|
|
|
var slice = function (A) {
|
|
|
|
return Array.prototype.slice.call(A);
|
|
|
|
};
|
|
|
|
|
|
|
|
var createNonce = function () {
|
|
|
|
return new Uint8Array(new Array(24).fill(0));
|
|
|
|
};
|
|
|
|
|
|
|
|
var increment = function (N) {
|
|
|
|
var l = N.length;
|
|
|
|
while (l-- > 1) {
|
|
|
|
if (PARANOIA) {
|
|
|
|
if (typeof(N[l]) !== 'number') {
|
|
|
|
throw new Error('E_UNSAFE_TYPE');
|
|
|
|
}
|
|
|
|
if (N[l] > 255) {
|
|
|
|
throw new Error('E_OUT_OF_BOUNDS');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/* jshint probably suspects this is unsafe because we lack types
|
|
|
|
but as long as this is only used on nonces, it should be safe */
|
|
|
|
if (N[l] !== 255) { return void N[l]++; } // jshint ignore:line
|
|
|
|
N[l] = 0;
|
|
|
|
|
|
|
|
// you don't need to worry about this running out.
|
|
|
|
// you'd need a REAAAALLY big file
|
|
|
|
if (l === 0) {
|
|
|
|
throw new Error('E_NONCE_TOO_LARGE');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var joinChunks = function (chunks) {
|
|
|
|
return new Uint8Array(chunks.reduce(function (A, B) {
|
|
|
|
return slice(A).concat(slice(B));
|
|
|
|
}, []));
|
|
|
|
};
|
|
|
|
|
|
|
|
var padChunk = function (A) {
|
|
|
|
var padding;
|
|
|
|
if (A.length === plainChunkLength) { return A; }
|
|
|
|
if (A.length < plainChunkLength) {
|
|
|
|
padding = new Array(plainChunkLength - A.length).fill(32);
|
|
|
|
return A.concat(padding);
|
|
|
|
}
|
|
|
|
if (A.length > plainChunkLength) {
|
|
|
|
// how many times larger is it?
|
|
|
|
var chunks = Math.ceil(A.length / plainChunkLength);
|
|
|
|
padding = new Array((plainChunkLength * chunks) - A.length).fill(32);
|
|
|
|
return A.concat(padding);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
var decrypt = function (u8, key, cb) {
|
|
|
|
var nonce = createNonce();
|
|
|
|
var i = 0;
|
|
|
|
|
|
|
|
decodePrefix([]); // TODO
|
|
|
|
var takeChunk = function () {
|
|
|
|
var start = i * cypherChunkLength;
|
|
|
|
var end = start + cypherChunkLength;
|
|
|
|
i++;
|
|
|
|
var box = new Uint8Array(u8.subarray(start, end));
|
|
|
|
|
|
|
|
// decrypt the chunk
|
|
|
|
var plaintext = Nacl.secretbox.open(box, nonce, key);
|
|
|
|
increment(nonce);
|
|
|
|
return plaintext;
|
|
|
|
};
|
|
|
|
|
|
|
|
var buffer = '';
|
|
|
|
|
|
|
|
var res = {
|
|
|
|
metadata: undefined,
|
|
|
|
};
|
|
|
|
|
|
|
|
// decrypt metadata
|
|
|
|
var chunk;
|
|
|
|
for (; !res.metadata && i * cypherChunkLength < u8.length;) {
|
|
|
|
chunk = takeChunk();
|
|
|
|
buffer += Nacl.util.encodeUTF8(chunk);
|
|
|
|
try {
|
|
|
|
res.metadata = JSON.parse(buffer);
|
|
|
|
//console.log(res.metadata);
|
|
|
|
} catch (e) {
|
|
|
|
console.log('buffering another chunk for metadata');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!res.metadata) {
|
|
|
|
return void setTimeout(function () {
|
|
|
|
cb('NO_METADATA');
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
var fail = function () {
|
|
|
|
cb("DECRYPTION_ERROR");
|
|
|
|
};
|
|
|
|
|
|
|
|
var chunks = [];
|
|
|
|
// decrypt file contents
|
|
|
|
for (;i * cypherChunkLength < u8.length;) {
|
|
|
|
chunk = takeChunk();
|
|
|
|
if (!chunk) {
|
|
|
|
return window.setTimeout(fail);
|
|
|
|
}
|
|
|
|
chunks.push(chunk);
|
|
|
|
}
|
|
|
|
|
|
|
|
// send chunks
|
|
|
|
res.content = joinChunks(chunks);
|
|
|
|
|
|
|
|
cb(void 0, res);
|
|
|
|
};
|
|
|
|
|
|
|
|
// metadata
|
|
|
|
/* { filename: 'raccoon.jpg', type: 'image/jpeg' } */
|
|
|
|
var encrypt = function (u8, metadata, key) {
|
|
|
|
var nonce = createNonce();
|
|
|
|
|
|
|
|
encodePrefix(); // TODO
|
|
|
|
|
|
|
|
// encode metadata
|
|
|
|
var metaBuffer = Array.prototype.slice
|
|
|
|
.call(Nacl.util.decodeUTF8(JSON.stringify(metadata)));
|
|
|
|
|
|
|
|
var plaintext = new Uint8Array(padChunk(metaBuffer));
|
|
|
|
|
|
|
|
var j = 0;
|
|
|
|
var i = 0;
|
|
|
|
|
|
|
|
/*
|
|
|
|
0: metadata
|
|
|
|
1: u8
|
|
|
|
2: done
|
|
|
|
*/
|
|
|
|
|
|
|
|
var state = 0;
|
|
|
|
|
|
|
|
var next = function (cb) {
|
|
|
|
var start;
|
|
|
|
var end;
|
|
|
|
var part;
|
|
|
|
var box;
|
|
|
|
|
|
|
|
if (state === 0) { // metadata...
|
|
|
|
start = j * plainChunkLength;
|
|
|
|
end = start + plainChunkLength;
|
|
|
|
|
|
|
|
part = plaintext.subarray(start, end);
|
|
|
|
box = Nacl.secretbox(part, nonce, key);
|
|
|
|
increment(nonce);
|
|
|
|
|
|
|
|
j++;
|
|
|
|
|
|
|
|
// metadata is done
|
|
|
|
if (j * plainChunkLength >= plaintext.length) {
|
|
|
|
return void cb(state++, box);
|
|
|
|
}
|
|
|
|
|
|
|
|
return void cb(state, box);
|
|
|
|
}
|
|
|
|
|
|
|
|
// encrypt the rest of the file...
|
|
|
|
start = i * plainChunkLength;
|
|
|
|
end = start + plainChunkLength;
|
|
|
|
|
|
|
|
part = u8.subarray(start, end);
|
|
|
|
box = Nacl.secretbox(part, nonce, key);
|
|
|
|
increment(nonce);
|
|
|
|
i++;
|
|
|
|
|
|
|
|
// regular data is done
|
|
|
|
if (i * plainChunkLength >= u8.length) { state = 2; }
|
|
|
|
|
|
|
|
return void cb(state, box);
|
|
|
|
};
|
|
|
|
|
|
|
|
return next;
|
|
|
|
};
|
|
|
|
|
|
|
|
return {
|
|
|
|
decrypt: decrypt,
|
|
|
|
encrypt: encrypt,
|
|
|
|
joinChunks: joinChunks,
|
|
|
|
};
|
|
|
|
});
|