zip.js 30 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970
  1. /*
  2. Copyright (c) 2013 Gildas Lormeau. All rights reserved.
  3. Redistribution and use in source and binary forms, with or without
  4. modification, are permitted provided that the following conditions are met:
  5. 1. Redistributions of source code must retain the above copyright notice,
  6. this list of conditions and the following disclaimer.
  7. 2. Redistributions in binary form must reproduce the above copyright
  8. notice, this list of conditions and the following disclaimer in
  9. the documentation and/or other materials provided with the distribution.
  10. 3. The names of the authors may not be used to endorse or promote products
  11. derived from this software without specific prior written permission.
  12. THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED WARRANTIES,
  13. INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
  14. FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL JCRAFT,
  15. INC. OR ANY CONTRIBUTORS TO THIS SOFTWARE BE LIABLE FOR ANY DIRECT, INDIRECT,
  16. INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
  17. LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA,
  18. OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
  19. LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
  20. NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
  21. EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  22. */
  23. (function(obj) {
  24. "use strict";
  25. var ERR_BAD_FORMAT = "File format is not recognized.";
  26. var ERR_CRC = "CRC failed.";
  27. var ERR_ENCRYPTED = "File contains encrypted entry.";
  28. var ERR_ZIP64 = "File is using Zip64 (4gb+ file size).";
  29. var ERR_READ = "Error while reading zip file.";
  30. var ERR_WRITE = "Error while writing zip file.";
  31. var ERR_WRITE_DATA = "Error while writing file data.";
  32. var ERR_READ_DATA = "Error while reading file data.";
  33. var ERR_DUPLICATED_NAME = "File already exists.";
  34. var CHUNK_SIZE = 512 * 1024;
  35. var TEXT_PLAIN = "text/plain";
  36. var appendABViewSupported;
  37. try {
  38. appendABViewSupported = new Blob([ new DataView(new ArrayBuffer(0)) ]).size === 0;
  39. } catch (e) {
  40. }
  41. function Crc32() {
  42. this.crc = -1;
  43. }
  44. Crc32.prototype.append = function append(data) {
  45. var crc = this.crc | 0, table = this.table;
  46. for (var offset = 0, len = data.length | 0; offset < len; offset++)
  47. crc = (crc >>> 8) ^ table[(crc ^ data[offset]) & 0xFF];
  48. this.crc = crc;
  49. };
  50. Crc32.prototype.get = function get() {
  51. return ~this.crc;
  52. };
  53. Crc32.prototype.table = (function() {
  54. var i, j, t, table = []; // Uint32Array is actually slower than []
  55. for (i = 0; i < 256; i++) {
  56. t = i;
  57. for (j = 0; j < 8; j++)
  58. if (t & 1)
  59. t = (t >>> 1) ^ 0xEDB88320;
  60. else
  61. t = t >>> 1;
  62. table[i] = t;
  63. }
  64. return table;
  65. })();
  66. // "no-op" codec
  67. function NOOP() {}
  68. NOOP.prototype.append = function append(bytes, onprogress) {
  69. return bytes;
  70. };
  71. NOOP.prototype.flush = function flush() {};
  72. function blobSlice(blob, index, length) {
  73. if (index < 0 || length < 0 || index + length > blob.size)
  74. throw new RangeError('offset:' + index + ', length:' + length + ', size:' + blob.size);
  75. if (blob.slice)
  76. return blob.slice(index, index + length);
  77. else if (blob.webkitSlice)
  78. return blob.webkitSlice(index, index + length);
  79. else if (blob.mozSlice)
  80. return blob.mozSlice(index, index + length);
  81. else if (blob.msSlice)
  82. return blob.msSlice(index, index + length);
  83. }
  84. function getDataHelper(byteLength, bytes) {
  85. var dataBuffer, dataArray;
  86. dataBuffer = new ArrayBuffer(byteLength);
  87. dataArray = new Uint8Array(dataBuffer);
  88. if (bytes)
  89. dataArray.set(bytes, 0);
  90. return {
  91. buffer : dataBuffer,
  92. array : dataArray,
  93. view : new DataView(dataBuffer)
  94. };
  95. }
  96. // Readers
  97. function Reader() {
  98. }
  99. function TextReader(text) {
  100. var that = this, blobReader;
  101. function init(callback, onerror) {
  102. var blob = new Blob([ text ], {
  103. type : TEXT_PLAIN
  104. });
  105. blobReader = new BlobReader(blob);
  106. blobReader.init(function() {
  107. that.size = blobReader.size;
  108. callback();
  109. }, onerror);
  110. }
  111. function readUint8Array(index, length, callback, onerror) {
  112. blobReader.readUint8Array(index, length, callback, onerror);
  113. }
  114. that.size = 0;
  115. that.init = init;
  116. that.readUint8Array = readUint8Array;
  117. }
  118. TextReader.prototype = new Reader();
  119. TextReader.prototype.constructor = TextReader;
  120. function Data64URIReader(dataURI) {
  121. var that = this, dataStart;
  122. function init(callback) {
  123. var dataEnd = dataURI.length;
  124. while (dataURI.charAt(dataEnd - 1) == "=")
  125. dataEnd--;
  126. dataStart = dataURI.indexOf(",") + 1;
  127. that.size = Math.floor((dataEnd - dataStart) * 0.75);
  128. callback();
  129. }
  130. function readUint8Array(index, length, callback) {
  131. var i, data = getDataHelper(length);
  132. var start = Math.floor(index / 3) * 4;
  133. var end = Math.ceil((index + length) / 3) * 4;
  134. var bytes = obj.atob(dataURI.substring(start + dataStart, end + dataStart));
  135. var delta = index - Math.floor(start / 4) * 3;
  136. for (i = delta; i < delta + length; i++)
  137. data.array[i - delta] = bytes.charCodeAt(i);
  138. callback(data.array);
  139. }
  140. that.size = 0;
  141. that.init = init;
  142. that.readUint8Array = readUint8Array;
  143. }
  144. Data64URIReader.prototype = new Reader();
  145. Data64URIReader.prototype.constructor = Data64URIReader;
  146. function BlobReader(blob) {
  147. var that = this;
  148. function init(callback) {
  149. that.size = blob.size;
  150. callback();
  151. }
  152. function readUint8Array(index, length, callback, onerror) {
  153. var reader = new FileReader();
  154. reader.onload = function(e) {
  155. callback(new Uint8Array(e.target.result));
  156. };
  157. reader.onerror = onerror;
  158. try {
  159. reader.readAsArrayBuffer(blobSlice(blob, index, length));
  160. } catch (e) {
  161. onerror(e);
  162. }
  163. }
  164. that.size = 0;
  165. that.init = init;
  166. that.readUint8Array = readUint8Array;
  167. }
  168. BlobReader.prototype = new Reader();
  169. BlobReader.prototype.constructor = BlobReader;
  170. // Writers
  171. function Writer() {
  172. }
  173. Writer.prototype.getData = function(callback) {
  174. callback(this.data);
  175. };
  176. function TextWriter(encoding) {
  177. var that = this, blob;
  178. function init(callback) {
  179. blob = new Blob([], {
  180. type : TEXT_PLAIN
  181. });
  182. callback();
  183. }
  184. function writeUint8Array(array, callback) {
  185. blob = new Blob([ blob, appendABViewSupported ? array : array.buffer ], {
  186. type : TEXT_PLAIN
  187. });
  188. callback();
  189. }
  190. function getData(callback, onerror) {
  191. var reader = new FileReader();
  192. reader.onload = function(e) {
  193. callback(e.target.result);
  194. };
  195. reader.onerror = onerror;
  196. reader.readAsText(blob, encoding);
  197. }
  198. that.init = init;
  199. that.writeUint8Array = writeUint8Array;
  200. that.getData = getData;
  201. }
  202. TextWriter.prototype = new Writer();
  203. TextWriter.prototype.constructor = TextWriter;
  204. function Data64URIWriter(contentType) {
  205. var that = this, data = "", pending = "";
  206. function init(callback) {
  207. data += "data:" + (contentType || "") + ";base64,";
  208. callback();
  209. }
  210. function writeUint8Array(array, callback) {
  211. var i, delta = pending.length, dataString = pending;
  212. pending = "";
  213. for (i = 0; i < (Math.floor((delta + array.length) / 3) * 3) - delta; i++)
  214. dataString += String.fromCharCode(array[i]);
  215. for (; i < array.length; i++)
  216. pending += String.fromCharCode(array[i]);
  217. if (dataString.length > 2)
  218. data += obj.btoa(dataString);
  219. else
  220. pending = dataString;
  221. callback();
  222. }
  223. function getData(callback) {
  224. callback(data + obj.btoa(pending));
  225. }
  226. that.init = init;
  227. that.writeUint8Array = writeUint8Array;
  228. that.getData = getData;
  229. }
  230. Data64URIWriter.prototype = new Writer();
  231. Data64URIWriter.prototype.constructor = Data64URIWriter;
  232. function BlobWriter(contentType) {
  233. var blob, that = this;
  234. function init(callback) {
  235. blob = new Blob([], {
  236. type : contentType
  237. });
  238. callback();
  239. }
  240. function writeUint8Array(array, callback) {
  241. blob = new Blob([ blob, appendABViewSupported ? array : array.buffer ], {
  242. type : contentType
  243. });
  244. callback();
  245. }
  246. function getData(callback) {
  247. callback(blob);
  248. }
  249. that.init = init;
  250. that.writeUint8Array = writeUint8Array;
  251. that.getData = getData;
  252. }
  253. BlobWriter.prototype = new Writer();
  254. BlobWriter.prototype.constructor = BlobWriter;
  255. /**
  256. * inflate/deflate core functions
  257. * @param worker {Worker} web worker for the task.
  258. * @param initialMessage {Object} initial message to be sent to the worker. should contain
  259. * sn(serial number for distinguishing multiple tasks sent to the worker), and codecClass.
  260. * This function may add more properties before sending.
  261. */
  262. function launchWorkerProcess(worker, initialMessage, reader, writer, offset, size, onprogress, onend, onreaderror, onwriteerror) {
  263. var chunkIndex = 0, index, outputSize, sn = initialMessage.sn, crc;
  264. function onflush() {
  265. worker.removeEventListener('message', onmessage, false);
  266. onend(outputSize, crc);
  267. }
  268. function onmessage(event) {
  269. var message = event.data, data = message.data, err = message.error;
  270. if (err) {
  271. err.toString = function () { return 'Error: ' + this.message; };
  272. onreaderror(err);
  273. return;
  274. }
  275. if (message.sn !== sn)
  276. return;
  277. if (typeof message.codecTime === 'number')
  278. worker.codecTime += message.codecTime; // should be before onflush()
  279. if (typeof message.crcTime === 'number')
  280. worker.crcTime += message.crcTime;
  281. switch (message.type) {
  282. case 'append':
  283. if (data) {
  284. outputSize += data.length;
  285. writer.writeUint8Array(data, function() {
  286. step();
  287. }, onwriteerror);
  288. } else
  289. step();
  290. break;
  291. case 'flush':
  292. crc = message.crc;
  293. if (data) {
  294. outputSize += data.length;
  295. writer.writeUint8Array(data, function() {
  296. onflush();
  297. }, onwriteerror);
  298. } else
  299. onflush();
  300. break;
  301. case 'progress':
  302. if (onprogress)
  303. onprogress(index + message.loaded, size);
  304. break;
  305. case 'importScripts': //no need to handle here
  306. case 'newTask':
  307. case 'echo':
  308. break;
  309. default:
  310. console.warn('zip.js:launchWorkerProcess: unknown message: ', message);
  311. }
  312. }
  313. function step() {
  314. index = chunkIndex * CHUNK_SIZE;
  315. // use `<=` instead of `<`, because `size` may be 0.
  316. if (index <= size) {
  317. reader.readUint8Array(offset + index, Math.min(CHUNK_SIZE, size - index), function(array) {
  318. if (onprogress)
  319. onprogress(index, size);
  320. var msg = index === 0 ? initialMessage : {sn : sn};
  321. msg.type = 'append';
  322. msg.data = array;
  323. // posting a message with transferables will fail on IE10
  324. try {
  325. worker.postMessage(msg, [array.buffer]);
  326. } catch(ex) {
  327. worker.postMessage(msg); // retry without transferables
  328. }
  329. chunkIndex++;
  330. }, onreaderror);
  331. } else {
  332. worker.postMessage({
  333. sn: sn,
  334. type: 'flush'
  335. });
  336. }
  337. }
  338. outputSize = 0;
  339. worker.addEventListener('message', onmessage, false);
  340. step();
  341. }
  342. function launchProcess(process, reader, writer, offset, size, crcType, onprogress, onend, onreaderror, onwriteerror) {
  343. var chunkIndex = 0, index, outputSize = 0,
  344. crcInput = crcType === 'input',
  345. crcOutput = crcType === 'output',
  346. crc = new Crc32();
  347. function step() {
  348. var outputData;
  349. index = chunkIndex * CHUNK_SIZE;
  350. if (index < size)
  351. reader.readUint8Array(offset + index, Math.min(CHUNK_SIZE, size - index), function (inputData) {
  352. var outputData;
  353. try {
  354. outputData = process.append(inputData, function(loaded) {
  355. if (onprogress) {
  356. onprogress(index + loaded, size);
  357. }
  358. });
  359. } catch (e) {
  360. onreaderror(e);
  361. return;
  362. }
  363. if (outputData) {
  364. outputSize += outputData.length;
  365. writer.writeUint8Array(outputData, function() {
  366. chunkIndex++;
  367. setTimeout(step, 1);
  368. }, onwriteerror);
  369. if (crcOutput)
  370. crc.append(outputData);
  371. } else {
  372. chunkIndex++;
  373. setTimeout(step, 1);
  374. }
  375. if (crcInput)
  376. crc.append(inputData);
  377. if (onprogress)
  378. onprogress(index, size);
  379. }, onreaderror);
  380. else {
  381. try {
  382. outputData = process.flush();
  383. } catch (e) {
  384. onreaderror(e);
  385. return;
  386. }
  387. if (outputData) {
  388. if (crcOutput)
  389. crc.append(outputData);
  390. outputSize += outputData.length;
  391. writer.writeUint8Array(outputData, function() {
  392. onend(outputSize, crc.get());
  393. }, onwriteerror);
  394. } else
  395. onend(outputSize, crc.get());
  396. }
  397. }
  398. step();
  399. }
  400. function inflate(worker, sn, reader, writer, offset, size, computeCrc32, onend, onprogress, onreaderror, onwriteerror) {
  401. var crcType = computeCrc32 ? 'output' : 'none';
  402. if (obj.zip.useWebWorkers) {
  403. var initialMessage = {
  404. sn: sn,
  405. codecClass: 'Inflater',
  406. crcType: crcType,
  407. };
  408. launchWorkerProcess(worker, initialMessage, reader, writer, offset, size, onprogress, onend, onreaderror, onwriteerror);
  409. } else
  410. launchProcess(new obj.zip.Inflater(), reader, writer, offset, size, crcType, onprogress, onend, onreaderror, onwriteerror);
  411. }
  412. function deflate(worker, sn, reader, writer, level, onend, onprogress, onreaderror, onwriteerror) {
  413. var crcType = 'input';
  414. if (obj.zip.useWebWorkers) {
  415. var initialMessage = {
  416. sn: sn,
  417. options: {level: level},
  418. codecClass: 'Deflater',
  419. crcType: crcType,
  420. };
  421. launchWorkerProcess(worker, initialMessage, reader, writer, 0, reader.size, onprogress, onend, onreaderror, onwriteerror);
  422. } else
  423. launchProcess(new obj.zip.Deflater(), reader, writer, 0, reader.size, crcType, onprogress, onend, onreaderror, onwriteerror);
  424. }
  425. function copy(worker, sn, reader, writer, offset, size, computeCrc32, onend, onprogress, onreaderror, onwriteerror) {
  426. var crcType = 'input';
  427. if (obj.zip.useWebWorkers && computeCrc32) {
  428. var initialMessage = {
  429. sn: sn,
  430. codecClass: 'NOOP',
  431. crcType: crcType,
  432. };
  433. launchWorkerProcess(worker, initialMessage, reader, writer, offset, size, onprogress, onend, onreaderror, onwriteerror);
  434. } else
  435. launchProcess(new NOOP(), reader, writer, offset, size, crcType, onprogress, onend, onreaderror, onwriteerror);
  436. }
  437. // ZipReader
  438. function decodeASCII(str) {
  439. var i, out = "", charCode, extendedASCII = [ '\u00C7', '\u00FC', '\u00E9', '\u00E2', '\u00E4', '\u00E0', '\u00E5', '\u00E7', '\u00EA', '\u00EB',
  440. '\u00E8', '\u00EF', '\u00EE', '\u00EC', '\u00C4', '\u00C5', '\u00C9', '\u00E6', '\u00C6', '\u00F4', '\u00F6', '\u00F2', '\u00FB', '\u00F9',
  441. '\u00FF', '\u00D6', '\u00DC', '\u00F8', '\u00A3', '\u00D8', '\u00D7', '\u0192', '\u00E1', '\u00ED', '\u00F3', '\u00FA', '\u00F1', '\u00D1',
  442. '\u00AA', '\u00BA', '\u00BF', '\u00AE', '\u00AC', '\u00BD', '\u00BC', '\u00A1', '\u00AB', '\u00BB', '_', '_', '_', '\u00A6', '\u00A6',
  443. '\u00C1', '\u00C2', '\u00C0', '\u00A9', '\u00A6', '\u00A6', '+', '+', '\u00A2', '\u00A5', '+', '+', '-', '-', '+', '-', '+', '\u00E3',
  444. '\u00C3', '+', '+', '-', '-', '\u00A6', '-', '+', '\u00A4', '\u00F0', '\u00D0', '\u00CA', '\u00CB', '\u00C8', 'i', '\u00CD', '\u00CE',
  445. '\u00CF', '+', '+', '_', '_', '\u00A6', '\u00CC', '_', '\u00D3', '\u00DF', '\u00D4', '\u00D2', '\u00F5', '\u00D5', '\u00B5', '\u00FE',
  446. '\u00DE', '\u00DA', '\u00DB', '\u00D9', '\u00FD', '\u00DD', '\u00AF', '\u00B4', '\u00AD', '\u00B1', '_', '\u00BE', '\u00B6', '\u00A7',
  447. '\u00F7', '\u00B8', '\u00B0', '\u00A8', '\u00B7', '\u00B9', '\u00B3', '\u00B2', '_', ' ' ];
  448. for (i = 0; i < str.length; i++) {
  449. charCode = str.charCodeAt(i) & 0xFF;
  450. if (charCode > 127)
  451. out += extendedASCII[charCode - 128];
  452. else
  453. out += String.fromCharCode(charCode);
  454. }
  455. return out;
  456. }
  457. function decodeUTF8(string) {
  458. return decodeURIComponent(escape(string));
  459. }
  460. function getString(bytes) {
  461. var i, str = "";
  462. for (i = 0; i < bytes.length; i++)
  463. str += String.fromCharCode(bytes[i]);
  464. return str;
  465. }
  466. function getDate(timeRaw) {
  467. var date = (timeRaw & 0xffff0000) >> 16, time = timeRaw & 0x0000ffff;
  468. try {
  469. return new Date(1980 + ((date & 0xFE00) >> 9), ((date & 0x01E0) >> 5) - 1, date & 0x001F, (time & 0xF800) >> 11, (time & 0x07E0) >> 5,
  470. (time & 0x001F) * 2, 0);
  471. } catch (e) {
  472. }
  473. }
  474. function readCommonHeader(entry, data, index, centralDirectory, onerror) {
  475. entry.version = data.view.getUint16(index, true);
  476. entry.bitFlag = data.view.getUint16(index + 2, true);
  477. entry.compressionMethod = data.view.getUint16(index + 4, true);
  478. entry.lastModDateRaw = data.view.getUint32(index + 6, true);
  479. entry.lastModDate = getDate(entry.lastModDateRaw);
  480. if ((entry.bitFlag & 0x01) === 0x01) {
  481. onerror(ERR_ENCRYPTED);
  482. return;
  483. }
  484. if (centralDirectory || (entry.bitFlag & 0x0008) != 0x0008) {
  485. entry.crc32 = data.view.getUint32(index + 10, true);
  486. entry.compressedSize = data.view.getUint32(index + 14, true);
  487. entry.uncompressedSize = data.view.getUint32(index + 18, true);
  488. }
  489. if (entry.compressedSize === 0xFFFFFFFF || entry.uncompressedSize === 0xFFFFFFFF) {
  490. onerror(ERR_ZIP64);
  491. return;
  492. }
  493. entry.filenameLength = data.view.getUint16(index + 22, true);
  494. entry.extraFieldLength = data.view.getUint16(index + 24, true);
  495. }
  496. function createZipReader(reader, callback, onerror) {
  497. var inflateSN = 0;
  498. function Entry() {
  499. }
  500. Entry.prototype.getData = function(writer, onend, onprogress, checkCrc32) {
  501. var that = this;
  502. function testCrc32(crc32) {
  503. var dataCrc32 = getDataHelper(4);
  504. dataCrc32.view.setUint32(0, crc32);
  505. return that.crc32 == dataCrc32.view.getUint32(0);
  506. }
  507. function getWriterData(uncompressedSize, crc32) {
  508. if (checkCrc32 && !testCrc32(crc32))
  509. onerror(ERR_CRC);
  510. else
  511. writer.getData(function(data) {
  512. onend(data);
  513. });
  514. }
  515. function onreaderror(err) {
  516. onerror(err || ERR_READ_DATA);
  517. }
  518. function onwriteerror(err) {
  519. onerror(err || ERR_WRITE_DATA);
  520. }
  521. var xwsize = 30;
  522. reader.readUint8Array(that.offset, xwsize, function (bytes) {
  523. var data = getDataHelper(bytes.length, bytes), dataOffset;
  524. if (data.view.getUint32(0) != 0x504b0304) {
  525. onerror(ERR_BAD_FORMAT);
  526. return;
  527. }
  528. readCommonHeader(that, data, 4, false, onerror);
  529. dataOffset = that.offset + xwsize + that.filenameLength + that.extraFieldLength;
  530. writer.init(function () {
  531. if (that.compressionMethod === 0)
  532. copy(that._worker, inflateSN++, reader, writer, dataOffset, that.compressedSize, checkCrc32, getWriterData, onprogress, onreaderror, onwriteerror);
  533. else
  534. inflate(that._worker, inflateSN++, reader, writer, dataOffset, that.compressedSize, checkCrc32, getWriterData, onprogress, onreaderror, onwriteerror);
  535. }, onwriteerror);
  536. }, onreaderror);
  537. };
  538. function seekEOCDR(eocdrCallback) {
  539. // "End of central directory record" is the last part of a zip archive, and is at least 22 bytes long.
  540. // Zip file comment is the last part of EOCDR and has max length of 64KB,
  541. // so we only have to search the last 64K + 22 bytes of a archive for EOCDR signature (0x06054b50).
  542. var EOCDR_MIN = 22;
  543. if (reader.size < EOCDR_MIN) {
  544. onerror(ERR_BAD_FORMAT);
  545. return;
  546. }
  547. var ZIP_COMMENT_MAX = 256 * 256, EOCDR_MAX = EOCDR_MIN + ZIP_COMMENT_MAX;
  548. // In most cases, the EOCDR is EOCDR_MIN bytes long
  549. doSeek(EOCDR_MIN, function() {
  550. // If not found, try within EOCDR_MAX bytes
  551. doSeek(Math.min(EOCDR_MAX, reader.size), function() {
  552. onerror(ERR_BAD_FORMAT);
  553. });
  554. });
  555. // seek last length bytes of file for EOCDR
  556. function doSeek(length, eocdrNotFoundCallback) {
  557. reader.readUint8Array(reader.size - length, length, function(bytes) {
  558. for (var i = bytes.length - EOCDR_MIN; i >= 0; i--) {
  559. if (bytes[i] === 0x50 && bytes[i + 1] === 0x4b && bytes[i + 2] === 0x05 && bytes[i + 3] === 0x06) {
  560. eocdrCallback(new DataView(bytes.buffer, i, EOCDR_MIN));
  561. return;
  562. }
  563. }
  564. eocdrNotFoundCallback();
  565. }, function() {
  566. onerror(ERR_READ);
  567. });
  568. }
  569. }
  570. var zipReader = {
  571. getEntries : function(callback) {
  572. var worker = this._worker;
  573. // look for End of central directory record
  574. seekEOCDR(function(dataView) {
  575. var datalength, fileslength;
  576. datalength = dataView.getUint32(16, true);
  577. fileslength = dataView.getUint16(8, true);
  578. if (datalength < 0 || datalength >= reader.size) {
  579. onerror(ERR_BAD_FORMAT);
  580. return;
  581. }
  582. reader.readUint8Array(datalength, reader.size - datalength, function(bytes) {
  583. var i, index = 0, entries = [], entry, filename, comment, data = getDataHelper(bytes.length, bytes);
  584. for (i = 0; i < fileslength; i++) {
  585. entry = new Entry();
  586. entry._worker = worker;
  587. if (data.view.getUint32(index) != 0x504b0102) {
  588. onerror(ERR_BAD_FORMAT);
  589. return;
  590. }
  591. readCommonHeader(entry, data, index + 6, true, onerror);
  592. entry.commentLength = data.view.getUint16(index + 32, true);
  593. entry.directory = ((data.view.getUint8(index + 38) & 0x10) == 0x10);
  594. entry.offset = data.view.getUint32(index + 42, true);
  595. filename = getString(data.array.subarray(index + 46, index + 46 + entry.filenameLength));
  596. entry.filename = ((entry.bitFlag & 0x0800) === 0x0800) ? decodeUTF8(filename) : decodeASCII(filename);
  597. if (!entry.directory && entry.filename.charAt(entry.filename.length - 1) == "/")
  598. entry.directory = true;
  599. comment = getString(data.array.subarray(index + 46 + entry.filenameLength + entry.extraFieldLength, index + 46
  600. + entry.filenameLength + entry.extraFieldLength + entry.commentLength));
  601. entry.comment = ((entry.bitFlag & 0x0800) === 0x0800) ? decodeUTF8(comment) : decodeASCII(comment);
  602. entries.push(entry);
  603. index += 46 + entry.filenameLength + entry.extraFieldLength + entry.commentLength;
  604. }
  605. callback(entries);
  606. }, function() {
  607. onerror(ERR_READ);
  608. });
  609. });
  610. },
  611. close : function(callback) {
  612. if (this._worker) {
  613. this._worker.terminate();
  614. this._worker = null;
  615. }
  616. if (callback)
  617. callback();
  618. },
  619. _worker: null
  620. };
  621. if (!obj.zip.useWebWorkers)
  622. callback(zipReader);
  623. else {
  624. createWorker('inflater',
  625. function(worker) {
  626. zipReader._worker = worker;
  627. callback(zipReader);
  628. },
  629. function(err) {
  630. onerror(err);
  631. }
  632. );
  633. }
  634. }
  635. // ZipWriter
  636. function encodeUTF8(string) {
  637. return unescape(encodeURIComponent(string));
  638. }
  639. function getBytes(str) {
  640. var i, array = [];
  641. for (i = 0; i < str.length; i++)
  642. array.push(str.charCodeAt(i));
  643. return array;
  644. }
  645. function createZipWriter(writer, callback, onerror, dontDeflate) {
  646. var files = {}, filenames = [], datalength = 0;
  647. var deflateSN = 0;
  648. function onwriteerror(err) {
  649. onerror(err || ERR_WRITE);
  650. }
  651. function onreaderror(err) {
  652. onerror(err || ERR_READ_DATA);
  653. }
  654. var zipWriter = {
  655. add : function(name, reader, onend, onprogress, options) {
  656. var header, filename, date;
  657. var worker = this._worker;
  658. function writeHeader(callback) {
  659. var data;
  660. date = options.lastModDate || new Date();
  661. header = getDataHelper(26);
  662. files[name] = {
  663. headerArray : header.array,
  664. directory : options.directory,
  665. filename : filename,
  666. offset : datalength,
  667. comment : getBytes(encodeUTF8(options.comment || ""))
  668. };
  669. header.view.setUint32(0, 0x14000808);
  670. if (options.version)
  671. header.view.setUint8(0, options.version);
  672. if (!dontDeflate && options.level !== 0 && !options.directory)
  673. header.view.setUint16(4, 0x0800);
  674. header.view.setUint16(6, (((date.getHours() << 6) | date.getMinutes()) << 5) | date.getSeconds() / 2, true);
  675. header.view.setUint16(8, ((((date.getFullYear() - 1980) << 4) | (date.getMonth() + 1)) << 5) | date.getDate(), true);
  676. header.view.setUint16(22, filename.length, true);
  677. data = getDataHelper(30 + filename.length);
  678. data.view.setUint32(0, 0x504b0304);
  679. data.array.set(header.array, 4);
  680. data.array.set(filename, 30);
  681. datalength += data.array.length;
  682. writer.writeUint8Array(data.array, callback, onwriteerror);
  683. }
  684. function writeFooter(compressedLength, crc32) {
  685. var footer = getDataHelper(16);
  686. datalength += compressedLength || 0;
  687. footer.view.setUint32(0, 0x504b0708);
  688. if (typeof crc32 != "undefined") {
  689. header.view.setUint32(10, crc32, true);
  690. footer.view.setUint32(4, crc32, true);
  691. }
  692. if (reader) {
  693. footer.view.setUint32(8, compressedLength, true);
  694. header.view.setUint32(14, compressedLength, true);
  695. footer.view.setUint32(12, reader.size, true);
  696. header.view.setUint32(18, reader.size, true);
  697. }
  698. writer.writeUint8Array(footer.array, function() {
  699. datalength += 16;
  700. onend();
  701. }, onwriteerror);
  702. }
  703. function writeFile() {
  704. options = options || {};
  705. name = name.trim();
  706. if (options.directory && name.charAt(name.length - 1) != "/")
  707. name += "/";
  708. if (files.hasOwnProperty(name)) {
  709. onerror(ERR_DUPLICATED_NAME);
  710. return;
  711. }
  712. filename = getBytes(encodeUTF8(name));
  713. filenames.push(name);
  714. writeHeader(function() {
  715. if (reader)
  716. if (dontDeflate || options.level === 0)
  717. copy(worker, deflateSN++, reader, writer, 0, reader.size, true, writeFooter, onprogress, onreaderror, onwriteerror);
  718. else
  719. deflate(worker, deflateSN++, reader, writer, options.level, writeFooter, onprogress, onreaderror, onwriteerror);
  720. else
  721. writeFooter();
  722. }, onwriteerror);
  723. }
  724. if (reader)
  725. reader.init(writeFile, onreaderror);
  726. else
  727. writeFile();
  728. },
  729. close : function(callback) {
  730. if (this._worker) {
  731. this._worker.terminate();
  732. this._worker = null;
  733. }
  734. var data, length = 0, index = 0, indexFilename, file;
  735. for (indexFilename = 0; indexFilename < filenames.length; indexFilename++) {
  736. file = files[filenames[indexFilename]];
  737. length += 46 + file.filename.length + file.comment.length;
  738. }
  739. data = getDataHelper(length + 22);
  740. for (indexFilename = 0; indexFilename < filenames.length; indexFilename++) {
  741. file = files[filenames[indexFilename]];
  742. data.view.setUint32(index, 0x504b0102);
  743. data.view.setUint16(index + 4, 0x1400);
  744. data.array.set(file.headerArray, index + 6);
  745. data.view.setUint16(index + 32, file.comment.length, true);
  746. if (file.directory)
  747. data.view.setUint8(index + 38, 0x10);
  748. data.view.setUint32(index + 42, file.offset, true);
  749. data.array.set(file.filename, index + 46);
  750. data.array.set(file.comment, index + 46 + file.filename.length);
  751. index += 46 + file.filename.length + file.comment.length;
  752. }
  753. data.view.setUint32(index, 0x504b0506);
  754. data.view.setUint16(index + 8, filenames.length, true);
  755. data.view.setUint16(index + 10, filenames.length, true);
  756. data.view.setUint32(index + 12, length, true);
  757. data.view.setUint32(index + 16, datalength, true);
  758. writer.writeUint8Array(data.array, function() {
  759. writer.getData(callback);
  760. }, onwriteerror);
  761. },
  762. _worker: null
  763. };
  764. if (!obj.zip.useWebWorkers)
  765. callback(zipWriter);
  766. else {
  767. createWorker('deflater',
  768. function(worker) {
  769. zipWriter._worker = worker;
  770. callback(zipWriter);
  771. },
  772. function(err) {
  773. onerror(err);
  774. }
  775. );
  776. }
  777. }
  778. function resolveURLs(urls) {
  779. var a = document.createElement('a');
  780. return urls.map(function(url) {
  781. a.href = url;
  782. return a.href;
  783. });
  784. }
  785. var DEFAULT_WORKER_SCRIPTS = {
  786. deflater: ['z-worker.js', 'deflate.js'],
  787. inflater: ['z-worker.js', 'inflate.js']
  788. };
  789. function createWorker(type, callback, onerror) {
  790. if (obj.zip.workerScripts !== null && obj.zip.workerScriptsPath !== null) {
  791. onerror(new Error('Either zip.workerScripts or zip.workerScriptsPath may be set, not both.'));
  792. return;
  793. }
  794. var scripts;
  795. if (obj.zip.workerScripts) {
  796. scripts = obj.zip.workerScripts[type];
  797. if (!Array.isArray(scripts)) {
  798. onerror(new Error('zip.workerScripts.' + type + ' is not an array!'));
  799. return;
  800. }
  801. scripts = resolveURLs(scripts);
  802. } else {
  803. scripts = DEFAULT_WORKER_SCRIPTS[type].slice(0);
  804. scripts[0] = (obj.zip.workerScriptsPath || '') + scripts[0];
  805. }
  806. var worker = new Worker(scripts[0]);
  807. // record total consumed time by inflater/deflater/crc32 in this worker
  808. worker.codecTime = worker.crcTime = 0;
  809. worker.postMessage({ type: 'importScripts', scripts: scripts.slice(1) });
  810. worker.addEventListener('message', onmessage);
  811. function onmessage(ev) {
  812. var msg = ev.data;
  813. if (msg.error) {
  814. worker.terminate(); // should before onerror(), because onerror() may throw.
  815. onerror(msg.error);
  816. return;
  817. }
  818. if (msg.type === 'importScripts') {
  819. worker.removeEventListener('message', onmessage);
  820. worker.removeEventListener('error', errorHandler);
  821. callback(worker);
  822. }
  823. }
  824. // catch entry script loading error and other unhandled errors
  825. worker.addEventListener('error', errorHandler);
  826. function errorHandler(err) {
  827. worker.terminate();
  828. onerror(err);
  829. }
  830. }
  831. function onerror_default(error) {
  832. console.error(error);
  833. }
  834. obj.zip = {
  835. Reader : Reader,
  836. Writer : Writer,
  837. BlobReader : BlobReader,
  838. Data64URIReader : Data64URIReader,
  839. TextReader : TextReader,
  840. BlobWriter : BlobWriter,
  841. Data64URIWriter : Data64URIWriter,
  842. TextWriter : TextWriter,
  843. createReader : function(reader, callback, onerror) {
  844. onerror = onerror || onerror_default;
  845. reader.init(function() {
  846. createZipReader(reader, callback, onerror);
  847. }, onerror);
  848. },
  849. createWriter : function(writer, callback, onerror, dontDeflate) {
  850. onerror = onerror || onerror_default;
  851. dontDeflate = !!dontDeflate;
  852. writer.init(function() {
  853. createZipWriter(writer, callback, onerror, dontDeflate);
  854. }, onerror);
  855. },
  856. useWebWorkers : true,
  857. /**
  858. * Directory containing the default worker scripts (z-worker.js, deflate.js, and inflate.js), relative to current base url.
  859. * E.g.: zip.workerScripts = './';
  860. */
  861. workerScriptsPath : null,
  862. /**
  863. * Advanced option to control which scripts are loaded in the Web worker. If this option is specified, then workerScriptsPath must not be set.
  864. * workerScripts.deflater/workerScripts.inflater should be arrays of urls to scripts for deflater/inflater, respectively.
  865. * Scripts in the array are executed in order, and the first one should be z-worker.js, which is used to start the worker.
  866. * All urls are relative to current base url.
  867. * E.g.:
  868. * zip.workerScripts = {
  869. * deflater: ['z-worker.js', 'deflate.js'],
  870. * inflater: ['z-worker.js', 'inflate.js']
  871. * };
  872. */
  873. workerScripts : null,
  874. };
  875. })(this);