bcrypt.js 51 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224
  1. /*
  2. Copyright (c) 2012 Nevins Bartolomeo <nevins.bartolomeo@gmail.com>
  3. Copyright (c) 2012 Shane Girish <shaneGirish@gmail.com>
  4. Copyright (c) 2014 Daniel Wirtz <dcode@dcode.io>
  5. Redistribution and use in source and binary forms, with or without
  6. modification, are permitted provided that the following conditions
  7. are met:
  8. 1. Redistributions of source code must retain the above copyright
  9. notice, this list of conditions and the following disclaimer.
  10. 2. Redistributions in binary form must reproduce the above copyright
  11. notice, this list of conditions and the following disclaimer in the
  12. documentation and/or other materials provided with the distribution.
  13. 3. The name of the author may not be used to endorse or promote products
  14. derived from this software without specific prior written permission.
  15. THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
  16. IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
  17. OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
  18. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
  19. INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
  20. NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
  21. DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
  22. THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  23. (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
  24. THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  25. */
  26. /**
  27. * @license bcrypt.js (c) 2013 Daniel Wirtz <dcode@dcode.io>
  28. * Released under the Apache License, Version 2.0
  29. * see: https://github.com/dcodeIO/bcrypt.js for details
  30. */
  31. (function(global, factory) {
  32. /* AMD */ if (typeof define === 'function' && define["amd"])
  33. define([], factory);
  34. /* CommonJS */ else if (typeof require === 'function' && typeof module === "object" && module && module["exports"])
  35. module["exports"] = factory();
  36. /* Global */ else
  37. (global["dcodeIO"] = global["dcodeIO"] || {})["bcrypt"] = factory();
  38. }(this, function() {
  39. "use strict";
  40. /**
  41. * bcrypt namespace.
  42. * @type {Object.<string,*>}
  43. */
  44. var bcrypt = {};
  45. /**
  46. * The random implementation to use as a fallback.
  47. * @type {?function(number):!Array.<number>}
  48. * @inner
  49. */
  50. var randomFallback = null;
  51. /**
  52. * Generates cryptographically secure random bytes.
  53. * @function
  54. * @param {number} len Bytes length
  55. * @returns {!Array.<number>} Random bytes
  56. * @throws {Error} If no random implementation is available
  57. * @inner
  58. */
  59. function random(len) {
  60. /* node */ if (typeof module !== 'undefined' && module && module['exports'])
  61. try {
  62. return require("crypto")['randomBytes'](len);
  63. } catch (e) {}
  64. /* WCA */ try {
  65. var a; (self['crypto']||self['msCrypto'])['getRandomValues'](a = new Uint32Array(len));
  66. return Array.prototype.slice.call(a);
  67. } catch (e) {}
  68. /* fallback */ if (!randomFallback)
  69. throw Error("Neither WebCryptoAPI nor a crypto module is available. Use bcrypt.setRandomFallback to set an alternative");
  70. return randomFallback(len);
  71. }
  72. // Test if any secure randomness source is available
  73. var randomAvailable = false;
  74. try {
  75. random(1);
  76. randomAvailable = true;
  77. } catch (e) {}
  78. // Default fallback, if any
  79. randomFallback = null;
  80. /**
  81. * Sets the pseudo random number generator to use as a fallback if neither node's `crypto` module nor the Web Crypto
  82. * API is available. Please note: It is highly important that the PRNG used is cryptographically secure and that it
  83. * is seeded properly!
  84. * @param {?function(number):!Array.<number>} random Function taking the number of bytes to generate as its
  85. * sole argument, returning the corresponding array of cryptographically secure random byte values.
  86. * @see http://nodejs.org/api/crypto.html
  87. * @see http://www.w3.org/TR/WebCryptoAPI/
  88. */
  89. bcrypt.setRandomFallback = function(random) {
  90. randomFallback = random;
  91. };
  92. /**
  93. * Synchronously generates a salt.
  94. * @param {number=} rounds Number of rounds to use, defaults to 10 if omitted
  95. * @param {number=} seed_length Not supported.
  96. * @returns {string} Resulting salt
  97. * @throws {Error} If a random fallback is required but not set
  98. * @expose
  99. */
  100. bcrypt.genSaltSync = function(rounds, seed_length) {
  101. rounds = rounds || GENSALT_DEFAULT_LOG2_ROUNDS;
  102. if (typeof rounds !== 'number')
  103. throw Error("Illegal arguments: "+(typeof rounds)+", "+(typeof seed_length));
  104. if (rounds < 4)
  105. rounds = 4;
  106. else if (rounds > 31)
  107. rounds = 31;
  108. var salt = [];
  109. salt.push("$2a$");
  110. if (rounds < 10)
  111. salt.push("0");
  112. salt.push(rounds.toString());
  113. salt.push('$');
  114. salt.push(base64_encode(random(BCRYPT_SALT_LEN), BCRYPT_SALT_LEN)); // May throw
  115. return salt.join('');
  116. };
  117. /**
  118. * Asynchronously generates a salt.
  119. * @param {(number|function(Error, string=))=} rounds Number of rounds to use, defaults to 10 if omitted
  120. * @param {(number|function(Error, string=))=} seed_length Not supported.
  121. * @param {function(Error, string=)=} callback Callback receiving the error, if any, and the resulting salt
  122. * @expose
  123. */
  124. bcrypt.genSalt = function(rounds, seed_length, callback) {
  125. if (typeof seed_length === 'function')
  126. callback = seed_length,
  127. seed_length = undefined; // Not supported.
  128. if (typeof rounds === 'function')
  129. callback = rounds,
  130. rounds = GENSALT_DEFAULT_LOG2_ROUNDS;
  131. if (typeof callback !== 'function')
  132. throw Error("Illegal callback: "+typeof(callback));
  133. if (typeof rounds !== 'number') {
  134. nextTick(callback.bind(this, Error("Illegal arguments: "+(typeof rounds))));
  135. return;
  136. }
  137. nextTick(function() { // Pretty thin, but salting is fast enough
  138. try {
  139. callback(null, bcrypt.genSaltSync(rounds));
  140. } catch (err) {
  141. callback(err);
  142. }
  143. });
  144. };
  145. /**
  146. * Synchronously generates a hash for the given string.
  147. * @param {string} s String to hash
  148. * @param {(number|string)=} salt Salt length to generate or salt to use, default to 10
  149. * @returns {string} Resulting hash
  150. * @expose
  151. */
  152. bcrypt.hashSync = function(s, salt) {
  153. if (typeof salt === 'undefined')
  154. salt = GENSALT_DEFAULT_LOG2_ROUNDS;
  155. if (typeof salt === 'number')
  156. salt = bcrypt.genSaltSync(salt);
  157. if (typeof s !== 'string' || typeof salt !== 'string')
  158. throw Error("Illegal arguments: "+(typeof s)+', '+(typeof salt));
  159. return _hash(s, salt);
  160. };
  161. /**
  162. * Asynchronously generates a hash for the given string.
  163. * @param {string} s String to hash
  164. * @param {number|string} salt Salt length to generate or salt to use
  165. * @param {function(Error, string=)} callback Callback receiving the error, if any, and the resulting hash
  166. * @param {function(number)=} progressCallback Callback successively called with the percentage of rounds completed
  167. * (0.0 - 1.0), maximally once per `MAX_EXECUTION_TIME = 100` ms.
  168. * @expose
  169. */
  170. bcrypt.hash = function(s, salt, callback, progressCallback) {
  171. if (typeof callback !== 'function')
  172. throw Error("Illegal callback: "+typeof(callback));
  173. if (typeof s === 'string' && typeof salt === 'number')
  174. bcrypt.genSalt(salt, function(err, salt) {
  175. _hash(s, salt, callback, progressCallback);
  176. });
  177. else if (typeof s === 'string' && typeof salt === 'string')
  178. _hash(s, salt, callback, progressCallback);
  179. else
  180. nextTick(callback.bind(this, Error("Illegal arguments: "+(typeof s)+', '+(typeof salt))));
  181. };
  182. /**
  183. * Compares two strings of the same length in constant time.
  184. * @param {string} known Must be of the correct length
  185. * @param {string} unknown Must be the same length as `known`
  186. * @returns {boolean}
  187. * @inner
  188. */
  189. function safeStringCompare(known, unknown) {
  190. var right = 0,
  191. wrong = 0;
  192. for (var i=0, k=known.length; i<k; ++i) {
  193. if (known.charCodeAt(i) === unknown.charCodeAt(i))
  194. ++right;
  195. else
  196. ++wrong;
  197. }
  198. // Prevent removal of unused variables (never true, actually)
  199. if (right < 0)
  200. return false;
  201. return wrong === 0;
  202. }
  203. /**
  204. * Synchronously tests a string against a hash.
  205. * @param {string} s String to compare
  206. * @param {string} hash Hash to test against
  207. * @returns {boolean} true if matching, otherwise false
  208. * @throws {Error} If an argument is illegal
  209. * @expose
  210. */
  211. bcrypt.compareSync = function(s, hash) {
  212. if (typeof s !== "string" || typeof hash !== "string")
  213. throw Error("Illegal arguments: "+(typeof s)+', '+(typeof hash));
  214. if (hash.length !== 60)
  215. return false;
  216. return safeStringCompare(bcrypt.hashSync(s, hash.substr(0, hash.length-31)), hash);
  217. };
  218. /**
  219. * Asynchronously compares the given data against the given hash.
  220. * @param {string} s Data to compare
  221. * @param {string} hash Data to be compared to
  222. * @param {function(Error, boolean)} callback Callback receiving the error, if any, otherwise the result
  223. * @param {function(number)=} progressCallback Callback successively called with the percentage of rounds completed
  224. * (0.0 - 1.0), maximally once per `MAX_EXECUTION_TIME = 100` ms.
  225. * @throws {Error} If the callback argument is invalid
  226. * @expose
  227. */
  228. bcrypt.compare = function(s, hash, callback, progressCallback) {
  229. if (typeof callback !== 'function')
  230. throw Error("Illegal callback: "+typeof(callback));
  231. if (typeof s !== "string" || typeof hash !== "string") {
  232. nextTick(callback.bind(this, Error("Illegal arguments: "+(typeof s)+', '+(typeof hash))));
  233. return;
  234. }
  235. if (hash.length !== 60) {
  236. nextTick(callback.bind(this, null, false));
  237. return;
  238. }
  239. bcrypt.hash(s, hash.substr(0, 29), function(err, comp) {
  240. if (err)
  241. callback(err);
  242. else
  243. callback(null, safeStringCompare(comp, hash));
  244. }, progressCallback);
  245. };
  246. /**
  247. * Gets the number of rounds used to encrypt the specified hash.
  248. * @param {string} hash Hash to extract the used number of rounds from
  249. * @returns {number} Number of rounds used
  250. * @throws {Error} If hash is not a string
  251. * @expose
  252. */
  253. bcrypt.getRounds = function(hash) {
  254. if (typeof hash !== "string")
  255. throw Error("Illegal arguments: "+(typeof hash));
  256. return parseInt(hash.split("$")[2], 10);
  257. };
  258. /**
  259. * Gets the salt portion from a hash. Does not validate the hash.
  260. * @param {string} hash Hash to extract the salt from
  261. * @returns {string} Extracted salt part
  262. * @throws {Error} If `hash` is not a string or otherwise invalid
  263. * @expose
  264. */
  265. bcrypt.getSalt = function(hash) {
  266. if (typeof hash !== 'string')
  267. throw Error("Illegal arguments: "+(typeof hash));
  268. if (hash.length !== 60)
  269. throw Error("Illegal hash length: "+hash.length+" != 60");
  270. return hash.substring(0, 29);
  271. };
  272. /**
  273. * Continues with the callback on the next tick.
  274. * @function
  275. * @param {function(...[*])} callback Callback to execute
  276. * @inner
  277. */
  278. var nextTick = typeof process !== 'undefined' && process && typeof process.nextTick === 'function'
  279. ? (typeof setImmediate === 'function' ? setImmediate : process.nextTick)
  280. : setTimeout;
  281. /**
  282. * Converts a JavaScript string to UTF8 bytes.
  283. * @param {string} str String
  284. * @returns {!Array.<number>} UTF8 bytes
  285. * @inner
  286. */
  287. function stringToBytes(str) {
  288. var out = [],
  289. i = 0;
  290. utfx.encodeUTF16toUTF8(function() {
  291. if (i >= str.length) return null;
  292. return str.charCodeAt(i++);
  293. }, function(b) {
  294. out.push(b);
  295. });
  296. return out;
  297. }
  298. // A base64 implementation for the bcrypt algorithm. This is partly non-standard.
  299. /**
  300. * bcrypt's own non-standard base64 dictionary.
  301. * @type {!Array.<string>}
  302. * @const
  303. * @inner
  304. **/
  305. var BASE64_CODE = "./ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789".split('');
  306. /**
  307. * @type {!Array.<number>}
  308. * @const
  309. * @inner
  310. **/
  311. var BASE64_INDEX = [-1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  312. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1,
  313. -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, -1, 0,
  314. 1, 54, 55, 56, 57, 58, 59, 60, 61, 62, 63, -1, -1, -1, -1, -1, -1,
  315. -1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19,
  316. 20, 21, 22, 23, 24, 25, 26, 27, -1, -1, -1, -1, -1, -1, 28, 29, 30,
  317. 31, 32, 33, 34, 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47,
  318. 48, 49, 50, 51, 52, 53, -1, -1, -1, -1, -1];
  319. /**
  320. * @type {!function(...number):string}
  321. * @inner
  322. */
  323. var stringFromCharCode = String.fromCharCode;
  324. /**
  325. * Encodes a byte array to base64 with up to len bytes of input.
  326. * @param {!Array.<number>} b Byte array
  327. * @param {number} len Maximum input length
  328. * @returns {string}
  329. * @inner
  330. */
  331. function base64_encode(b, len) {
  332. var off = 0,
  333. rs = [],
  334. c1, c2;
  335. if (len <= 0 || len > b.length)
  336. throw Error("Illegal len: "+len);
  337. while (off < len) {
  338. c1 = b[off++] & 0xff;
  339. rs.push(BASE64_CODE[(c1 >> 2) & 0x3f]);
  340. c1 = (c1 & 0x03) << 4;
  341. if (off >= len) {
  342. rs.push(BASE64_CODE[c1 & 0x3f]);
  343. break;
  344. }
  345. c2 = b[off++] & 0xff;
  346. c1 |= (c2 >> 4) & 0x0f;
  347. rs.push(BASE64_CODE[c1 & 0x3f]);
  348. c1 = (c2 & 0x0f) << 2;
  349. if (off >= len) {
  350. rs.push(BASE64_CODE[c1 & 0x3f]);
  351. break;
  352. }
  353. c2 = b[off++] & 0xff;
  354. c1 |= (c2 >> 6) & 0x03;
  355. rs.push(BASE64_CODE[c1 & 0x3f]);
  356. rs.push(BASE64_CODE[c2 & 0x3f]);
  357. }
  358. return rs.join('');
  359. }
  360. /**
  361. * Decodes a base64 encoded string to up to len bytes of output.
  362. * @param {string} s String to decode
  363. * @param {number} len Maximum output length
  364. * @returns {!Array.<number>}
  365. * @inner
  366. */
  367. function base64_decode(s, len) {
  368. var off = 0,
  369. slen = s.length,
  370. olen = 0,
  371. rs = [],
  372. c1, c2, c3, c4, o, code;
  373. if (len <= 0)
  374. throw Error("Illegal len: "+len);
  375. while (off < slen - 1 && olen < len) {
  376. code = s.charCodeAt(off++);
  377. c1 = code < BASE64_INDEX.length ? BASE64_INDEX[code] : -1;
  378. code = s.charCodeAt(off++);
  379. c2 = code < BASE64_INDEX.length ? BASE64_INDEX[code] : -1;
  380. if (c1 == -1 || c2 == -1)
  381. break;
  382. o = (c1 << 2) >>> 0;
  383. o |= (c2 & 0x30) >> 4;
  384. rs.push(stringFromCharCode(o));
  385. if (++olen >= len || off >= slen)
  386. break;
  387. code = s.charCodeAt(off++);
  388. c3 = code < BASE64_INDEX.length ? BASE64_INDEX[code] : -1;
  389. if (c3 == -1)
  390. break;
  391. o = ((c2 & 0x0f) << 4) >>> 0;
  392. o |= (c3 & 0x3c) >> 2;
  393. rs.push(stringFromCharCode(o));
  394. if (++olen >= len || off >= slen)
  395. break;
  396. code = s.charCodeAt(off++);
  397. c4 = code < BASE64_INDEX.length ? BASE64_INDEX[code] : -1;
  398. o = ((c3 & 0x03) << 6) >>> 0;
  399. o |= c4;
  400. rs.push(stringFromCharCode(o));
  401. ++olen;
  402. }
  403. var res = [];
  404. for (off = 0; off<olen; off++)
  405. res.push(rs[off].charCodeAt(0));
  406. return res;
  407. }
  408. /**
  409. * utfx-embeddable (c) 2014 Daniel Wirtz <dcode@dcode.io>
  410. * Released under the Apache License, Version 2.0
  411. * see: https://github.com/dcodeIO/utfx for details
  412. */
  413. var utfx = function() {
  414. "use strict";
  415. /**
  416. * utfx namespace.
  417. * @inner
  418. * @type {!Object.<string,*>}
  419. */
  420. var utfx = {};
  421. /**
  422. * Maximum valid code point.
  423. * @type {number}
  424. * @const
  425. */
  426. utfx.MAX_CODEPOINT = 0x10FFFF;
  427. /**
  428. * Encodes UTF8 code points to UTF8 bytes.
  429. * @param {(!function():number|null) | number} src Code points source, either as a function returning the next code point
  430. * respectively `null` if there are no more code points left or a single numeric code point.
  431. * @param {!function(number)} dst Bytes destination as a function successively called with the next byte
  432. */
  433. utfx.encodeUTF8 = function(src, dst) {
  434. var cp = null;
  435. if (typeof src === 'number')
  436. cp = src,
  437. src = function() { return null; };
  438. while (cp !== null || (cp = src()) !== null) {
  439. if (cp < 0x80)
  440. dst(cp&0x7F);
  441. else if (cp < 0x800)
  442. dst(((cp>>6)&0x1F)|0xC0),
  443. dst((cp&0x3F)|0x80);
  444. else if (cp < 0x10000)
  445. dst(((cp>>12)&0x0F)|0xE0),
  446. dst(((cp>>6)&0x3F)|0x80),
  447. dst((cp&0x3F)|0x80);
  448. else
  449. dst(((cp>>18)&0x07)|0xF0),
  450. dst(((cp>>12)&0x3F)|0x80),
  451. dst(((cp>>6)&0x3F)|0x80),
  452. dst((cp&0x3F)|0x80);
  453. cp = null;
  454. }
  455. };
  456. /**
  457. * Decodes UTF8 bytes to UTF8 code points.
  458. * @param {!function():number|null} src Bytes source as a function returning the next byte respectively `null` if there
  459. * are no more bytes left.
  460. * @param {!function(number)} dst Code points destination as a function successively called with each decoded code point.
  461. * @throws {RangeError} If a starting byte is invalid in UTF8
  462. * @throws {Error} If the last sequence is truncated. Has an array property `bytes` holding the
  463. * remaining bytes.
  464. */
  465. utfx.decodeUTF8 = function(src, dst) {
  466. var a, b, c, d, fail = function(b) {
  467. b = b.slice(0, b.indexOf(null));
  468. var err = Error(b.toString());
  469. err.name = "TruncatedError";
  470. err['bytes'] = b;
  471. throw err;
  472. };
  473. while ((a = src()) !== null) {
  474. if ((a&0x80) === 0)
  475. dst(a);
  476. else if ((a&0xE0) === 0xC0)
  477. ((b = src()) === null) && fail([a, b]),
  478. dst(((a&0x1F)<<6) | (b&0x3F));
  479. else if ((a&0xF0) === 0xE0)
  480. ((b=src()) === null || (c=src()) === null) && fail([a, b, c]),
  481. dst(((a&0x0F)<<12) | ((b&0x3F)<<6) | (c&0x3F));
  482. else if ((a&0xF8) === 0xF0)
  483. ((b=src()) === null || (c=src()) === null || (d=src()) === null) && fail([a, b, c ,d]),
  484. dst(((a&0x07)<<18) | ((b&0x3F)<<12) | ((c&0x3F)<<6) | (d&0x3F));
  485. else throw RangeError("Illegal starting byte: "+a);
  486. }
  487. };
  488. /**
  489. * Converts UTF16 characters to UTF8 code points.
  490. * @param {!function():number|null} src Characters source as a function returning the next char code respectively
  491. * `null` if there are no more characters left.
  492. * @param {!function(number)} dst Code points destination as a function successively called with each converted code
  493. * point.
  494. */
  495. utfx.UTF16toUTF8 = function(src, dst) {
  496. var c1, c2 = null;
  497. while (true) {
  498. if ((c1 = c2 !== null ? c2 : src()) === null)
  499. break;
  500. if (c1 >= 0xD800 && c1 <= 0xDFFF) {
  501. if ((c2 = src()) !== null) {
  502. if (c2 >= 0xDC00 && c2 <= 0xDFFF) {
  503. dst((c1-0xD800)*0x400+c2-0xDC00+0x10000);
  504. c2 = null; continue;
  505. }
  506. }
  507. }
  508. dst(c1);
  509. }
  510. if (c2 !== null) dst(c2);
  511. };
  512. /**
  513. * Converts UTF8 code points to UTF16 characters.
  514. * @param {(!function():number|null) | number} src Code points source, either as a function returning the next code point
  515. * respectively `null` if there are no more code points left or a single numeric code point.
  516. * @param {!function(number)} dst Characters destination as a function successively called with each converted char code.
  517. * @throws {RangeError} If a code point is out of range
  518. */
  519. utfx.UTF8toUTF16 = function(src, dst) {
  520. var cp = null;
  521. if (typeof src === 'number')
  522. cp = src, src = function() { return null; };
  523. while (cp !== null || (cp = src()) !== null) {
  524. if (cp <= 0xFFFF)
  525. dst(cp);
  526. else
  527. cp -= 0x10000,
  528. dst((cp>>10)+0xD800),
  529. dst((cp%0x400)+0xDC00);
  530. cp = null;
  531. }
  532. };
  533. /**
  534. * Converts and encodes UTF16 characters to UTF8 bytes.
  535. * @param {!function():number|null} src Characters source as a function returning the next char code respectively `null`
  536. * if there are no more characters left.
  537. * @param {!function(number)} dst Bytes destination as a function successively called with the next byte.
  538. */
  539. utfx.encodeUTF16toUTF8 = function(src, dst) {
  540. utfx.UTF16toUTF8(src, function(cp) {
  541. utfx.encodeUTF8(cp, dst);
  542. });
  543. };
  544. /**
  545. * Decodes and converts UTF8 bytes to UTF16 characters.
  546. * @param {!function():number|null} src Bytes source as a function returning the next byte respectively `null` if there
  547. * are no more bytes left.
  548. * @param {!function(number)} dst Characters destination as a function successively called with each converted char code.
  549. * @throws {RangeError} If a starting byte is invalid in UTF8
  550. * @throws {Error} If the last sequence is truncated. Has an array property `bytes` holding the remaining bytes.
  551. */
  552. utfx.decodeUTF8toUTF16 = function(src, dst) {
  553. utfx.decodeUTF8(src, function(cp) {
  554. utfx.UTF8toUTF16(cp, dst);
  555. });
  556. };
  557. /**
  558. * Calculates the byte length of an UTF8 code point.
  559. * @param {number} cp UTF8 code point
  560. * @returns {number} Byte length
  561. */
  562. utfx.calculateCodePoint = function(cp) {
  563. return (cp < 0x80) ? 1 : (cp < 0x800) ? 2 : (cp < 0x10000) ? 3 : 4;
  564. };
  565. /**
  566. * Calculates the number of UTF8 bytes required to store UTF8 code points.
  567. * @param {(!function():number|null)} src Code points source as a function returning the next code point respectively
  568. * `null` if there are no more code points left.
  569. * @returns {number} The number of UTF8 bytes required
  570. */
  571. utfx.calculateUTF8 = function(src) {
  572. var cp, l=0;
  573. while ((cp = src()) !== null)
  574. l += utfx.calculateCodePoint(cp);
  575. return l;
  576. };
  577. /**
  578. * Calculates the number of UTF8 code points respectively UTF8 bytes required to store UTF16 char codes.
  579. * @param {(!function():number|null)} src Characters source as a function returning the next char code respectively
  580. * `null` if there are no more characters left.
  581. * @returns {!Array.<number>} The number of UTF8 code points at index 0 and the number of UTF8 bytes required at index 1.
  582. */
  583. utfx.calculateUTF16asUTF8 = function(src) {
  584. var n=0, l=0;
  585. utfx.UTF16toUTF8(src, function(cp) {
  586. ++n; l += utfx.calculateCodePoint(cp);
  587. });
  588. return [n,l];
  589. };
  590. return utfx;
  591. }();
  592. Date.now = Date.now || function() { return +new Date; };
  593. /**
  594. * @type {number}
  595. * @const
  596. * @inner
  597. */
  598. var BCRYPT_SALT_LEN = 16;
  599. /**
  600. * @type {number}
  601. * @const
  602. * @inner
  603. */
  604. var GENSALT_DEFAULT_LOG2_ROUNDS = 10;
  605. /**
  606. * @type {number}
  607. * @const
  608. * @inner
  609. */
  610. var BLOWFISH_NUM_ROUNDS = 16;
  611. /**
  612. * @type {number}
  613. * @const
  614. * @inner
  615. */
  616. var MAX_EXECUTION_TIME = 100;
  617. /**
  618. * @type {Array.<number>}
  619. * @const
  620. * @inner
  621. */
  622. var P_ORIG = [
  623. 0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822,
  624. 0x299f31d0, 0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377,
  625. 0xbe5466cf, 0x34e90c6c, 0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5,
  626. 0xb5470917, 0x9216d5d9, 0x8979fb1b
  627. ];
  628. /**
  629. * @type {Array.<number>}
  630. * @const
  631. * @inner
  632. */
  633. var S_ORIG = [
  634. 0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed,
  635. 0x6a267e96, 0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7,
  636. 0x0801f2e2, 0x858efc16, 0x636920d8, 0x71574e69, 0xa458fea3,
  637. 0xf4933d7e, 0x0d95748f, 0x728eb658, 0x718bcd58, 0x82154aee,
  638. 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013, 0xc5d1b023,
  639. 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e,
  640. 0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda,
  641. 0x55605c60, 0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440,
  642. 0x55ca396a, 0x2aab10b6, 0xb4cc5c34, 0x1141e8ce, 0xa15486af,
  643. 0x7c72e993, 0xb3ee1411, 0x636fbc2a, 0x2ba9c55d, 0x741831f6,
  644. 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c, 0x7a325381,
  645. 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193,
  646. 0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d,
  647. 0xe98575b1, 0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5,
  648. 0x0f6d6ff3, 0x83f44239, 0x2e0b4482, 0xa4842004, 0x69c8f04a,
  649. 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a, 0x670c9c61, 0xabd388f0,
  650. 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3, 0x6eef0b6c,
  651. 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176,
  652. 0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3,
  653. 0x3b8b5ebe, 0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6,
  654. 0x4ed3aa62, 0x363f7706, 0x1bfedf72, 0x429b023d, 0x37d0d724,
  655. 0xd00a1248, 0xdb0fead3, 0x49f1c09b, 0x075372c9, 0x80991b7b,
  656. 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b, 0x976ce0bd,
  657. 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463,
  658. 0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f,
  659. 0x9b30952c, 0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd,
  660. 0x660f2807, 0x192e4bb3, 0xc0cba857, 0x45c8740f, 0xd20b5f39,
  661. 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a, 0xd6a100c6, 0x402c7279,
  662. 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8, 0x3c7516df,
  663. 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760,
  664. 0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e,
  665. 0xdf1769db, 0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573,
  666. 0x695b27b0, 0xbbca58c8, 0xe1ffa35d, 0xb8f011a0, 0x10fa3d98,
  667. 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b, 0x9a53e479, 0xb6f84565,
  668. 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33, 0x62fb1341,
  669. 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4,
  670. 0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0,
  671. 0xafc725e0, 0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64,
  672. 0x8888b812, 0x900df01c, 0x4fad5ea0, 0x688fc31c, 0xd1cff191,
  673. 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777, 0xea752dfe, 0x8b021fa1,
  674. 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299, 0xb4a84fe0,
  675. 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705,
  676. 0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5,
  677. 0xfb9d35cf, 0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49,
  678. 0x00250e2d, 0x2071b35e, 0x226800bb, 0x57b8e0af, 0x2464369b,
  679. 0xf009b91e, 0x5563911d, 0x59dfa6aa, 0x78c14389, 0xd95a537f,
  680. 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9, 0x11c81968,
  681. 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915,
  682. 0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5,
  683. 0x571be91f, 0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6,
  684. 0xff34052e, 0xc5855664, 0x53b02d5d, 0xa99f8fa1, 0x08ba4799,
  685. 0x6e85076a, 0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623,
  686. 0xad6ea6b0, 0x49a7df7d, 0x9cee60b8, 0x8fedb266, 0xecaa8c71,
  687. 0x699a17ff, 0x5664526c, 0xc2b19ee1, 0x193602a5, 0x75094c29,
  688. 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65, 0x6b8fe4d6,
  689. 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1,
  690. 0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f,
  691. 0x3ebaefc9, 0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286,
  692. 0xb79c5305, 0xaa500737, 0x3e07841c, 0x7fdeae5c, 0x8e7d44ec,
  693. 0x5716f2b8, 0xb03ada37, 0xf0500c0d, 0xf01c1f04, 0x0200b3ff,
  694. 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd, 0xd19113f9,
  695. 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc,
  696. 0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e,
  697. 0xa4751e41, 0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331,
  698. 0x4e548b38, 0x4f6db908, 0x6f420d03, 0xf60a04bf, 0x2cb81290,
  699. 0x24977c79, 0x5679b072, 0xbcaf89af, 0xde9a771f, 0xd9930810,
  700. 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124, 0x501adde6,
  701. 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c,
  702. 0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847,
  703. 0x3215d908, 0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451,
  704. 0x50940002, 0x133ae4dd, 0x71dff89e, 0x10314e55, 0x81ac77d6,
  705. 0x5f11199b, 0x043556f1, 0xd7a3c76b, 0x3c11183b, 0x5924a509,
  706. 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e, 0x86e34570,
  707. 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa,
  708. 0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978,
  709. 0x9c10b36a, 0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4,
  710. 0xf2f74ea7, 0x361d2b3d, 0x1939260f, 0x19c27960, 0x5223a708,
  711. 0xf71312b6, 0xebadfe6e, 0xeac31f66, 0xe3bc4595, 0xa67bc883,
  712. 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5, 0x65582185,
  713. 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84,
  714. 0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830,
  715. 0xeb61bd96, 0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239,
  716. 0xd59e9e0b, 0xcbaade14, 0xeecc86bc, 0x60622ca7, 0x9cab5cab,
  717. 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca, 0xa02369b9, 0x655abb50,
  718. 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7, 0x9b540b19,
  719. 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77,
  720. 0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1,
  721. 0x7858ba99, 0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696,
  722. 0xcdb30aeb, 0x532e3054, 0x8fd948e4, 0x6dbc3128, 0x58ebf2ef,
  723. 0x34c6ffea, 0xfe28ed61, 0xee7c3c73, 0x5d4a14d9, 0xe864b7e3,
  724. 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea, 0xdb6c4f15,
  725. 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105,
  726. 0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2,
  727. 0x5b8d2646, 0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492,
  728. 0x47848a0b, 0x5692b285, 0x095bbf00, 0xad19489d, 0x1462b174,
  729. 0x23820e00, 0x58428d2a, 0x0c55f5ea, 0x1dadf43e, 0x233f7061,
  730. 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb, 0x7cde3759,
  731. 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e,
  732. 0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc,
  733. 0x800bcadc, 0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9,
  734. 0xdb73dbd3, 0x105588cd, 0x675fda79, 0xe3674340, 0xc5c43465,
  735. 0x713e38d8, 0x3d28f89e, 0xf16dff20, 0x153e21e7, 0x8fb03d4a,
  736. 0xe6e39f2b, 0xdb83adf7, 0xe93d5a68, 0x948140f7, 0xf64c261c,
  737. 0x94692934, 0x411520f7, 0x7602d4f7, 0xbcf46b2e, 0xd4a20068,
  738. 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af, 0x1e39f62e,
  739. 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af,
  740. 0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0,
  741. 0x31cb8504, 0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a,
  742. 0x28507825, 0x530429f4, 0x0a2c86da, 0xe9b66dfb, 0x68dc1462,
  743. 0xd7486900, 0x680ec0a4, 0x27a18dee, 0x4f3ffea2, 0xe887ad8c,
  744. 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec, 0xce78a399,
  745. 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b,
  746. 0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74,
  747. 0xdd5b4332, 0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397,
  748. 0x454056ac, 0xba489527, 0x55533a3a, 0x20838d87, 0xfe6ba9b7,
  749. 0xd096954b, 0x55a867bc, 0xa1159a58, 0xcca92963, 0x99e1db33,
  750. 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c, 0xfdf8e802,
  751. 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22,
  752. 0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4,
  753. 0x5d886e17, 0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564,
  754. 0x257b7834, 0x602a9c60, 0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2,
  755. 0x02e1329e, 0xaf664fd1, 0xcad18115, 0x6b2395e0, 0x333e92e1,
  756. 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99, 0xde720c8c,
  757. 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0,
  758. 0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341,
  759. 0x992eff74, 0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8,
  760. 0x991be14c, 0xdb6e6b0d, 0xc67b5510, 0x6d672c37, 0x2765d43b,
  761. 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3, 0xb5390f92, 0x690fed0b,
  762. 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3, 0xbb132f88,
  763. 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979,
  764. 0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc,
  765. 0x782ef11c, 0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350,
  766. 0x1a6b1018, 0x11caedfa, 0x3d25bdd8, 0xe2e1c3c9, 0x44421659,
  767. 0x0a121386, 0xd90cec6e, 0xd5abea2a, 0x64af674e, 0xda86a85f,
  768. 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086, 0x60787bf8,
  769. 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc,
  770. 0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be,
  771. 0xbde8ae24, 0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2,
  772. 0xf474ef38, 0x8789bdc2, 0x5366f9c3, 0xc8b38e74, 0xb475f255,
  773. 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84, 0x846a0e79, 0x915f95e2,
  774. 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c, 0xb90bace1,
  775. 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09,
  776. 0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025,
  777. 0x1d6efe10, 0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169,
  778. 0xdcb7da83, 0x573906fe, 0xa1e2ce9b, 0x4fcd7f52, 0x50115e01,
  779. 0xa70683fa, 0xa002b5c4, 0x0de6d027, 0x9af88c27, 0x773f8641,
  780. 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0, 0x006058aa,
  781. 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634,
  782. 0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409,
  783. 0x4b7c0188, 0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9,
  784. 0x1ac15bb4, 0xd39eb8fc, 0xed545578, 0x08fca5b5, 0xd83d7cd3,
  785. 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8, 0xa28514d9, 0x6c51133c,
  786. 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837, 0xd79a3234,
  787. 0x92638212, 0x670efa8e, 0x406000e0, 0x3a39ce37, 0xd3faf5cf,
  788. 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742, 0xd3822740,
  789. 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b,
  790. 0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f,
  791. 0xbc946e79, 0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d,
  792. 0xd5730a1d, 0x4cd04dc6, 0x2939bbdb, 0xa9ba4650, 0xac9526e8,
  793. 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a, 0x63ef8ce2, 0x9a86ee22,
  794. 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4, 0x83c061ba,
  795. 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1,
  796. 0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69,
  797. 0x77fa0a59, 0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593,
  798. 0xe990fd5a, 0x9e34d797, 0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a,
  799. 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28, 0x1f9f25cf, 0xadf2b89b,
  800. 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6, 0x47b0acfd,
  801. 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28,
  802. 0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4,
  803. 0x88f46dba, 0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2,
  804. 0x97271aec, 0xa93a072a, 0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb,
  805. 0x26dcf319, 0x7533d928, 0xb155fdf5, 0x03563482, 0x8aba3cbb,
  806. 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f, 0x4de81751,
  807. 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce,
  808. 0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369,
  809. 0x6413e680, 0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166,
  810. 0xb39a460a, 0x6445c0dd, 0x586cdecf, 0x1c20c8ae, 0x5bbef7dd,
  811. 0x1b588d40, 0xccd2017f, 0x6bb4e3bb, 0xdda26a7e, 0x3a59ff45,
  812. 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb, 0x8d6612ae,
  813. 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370,
  814. 0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08,
  815. 0x4eb4e2cc, 0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d,
  816. 0x06b89fb4, 0xce6ea048, 0x6f3f3b82, 0x3520ab82, 0x011a1d4b,
  817. 0x277227f8, 0x611560b1, 0xe7933fdc, 0xbb3a792b, 0x344525bd,
  818. 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9, 0xe01cc87e,
  819. 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a,
  820. 0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c,
  821. 0xe0b12b4f, 0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c,
  822. 0xbf97222c, 0x15e6fc2a, 0x0f91fc71, 0x9b941525, 0xfae59361,
  823. 0xceb69ceb, 0xc2a86459, 0x12baa8d1, 0xb6c1075e, 0xe3056a0c,
  824. 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b, 0x4c98a0be,
  825. 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e,
  826. 0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d,
  827. 0x9b992f2e, 0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891,
  828. 0xce6279cf, 0xcd3e7e6f, 0x1618b166, 0xfd2c1d05, 0x848fd2c5,
  829. 0xf6fb2299, 0xf523f357, 0xa6327623, 0x93a83531, 0x56cccd02,
  830. 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc, 0xde966292,
  831. 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a,
  832. 0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2,
  833. 0x35bdd2f6, 0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b,
  834. 0x53113ec0, 0x1640e3d3, 0x38abbd60, 0x2547adf0, 0xba38209c,
  835. 0xf746ce76, 0x77afa1c5, 0x20756060, 0x85cbfe4e, 0x8ae88dd8,
  836. 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c, 0x01c36ae4,
  837. 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f,
  838. 0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6
  839. ];
  840. /**
  841. * @type {Array.<number>}
  842. * @const
  843. * @inner
  844. */
  845. var C_ORIG = [
  846. 0x4f727068, 0x65616e42, 0x65686f6c, 0x64657253, 0x63727944,
  847. 0x6f756274
  848. ];
  849. /**
  850. * @param {Array.<number>} lr
  851. * @param {number} off
  852. * @param {Array.<number>} P
  853. * @param {Array.<number>} S
  854. * @returns {Array.<number>}
  855. * @inner
  856. */
  857. function _encipher(lr, off, P, S) { // This is our bottleneck: 1714/1905 ticks / 90% - see profile.txt
  858. var n,
  859. l = lr[off],
  860. r = lr[off + 1];
  861. l ^= P[0];
  862. for (var i=0, k=BLOWFISH_NUM_ROUNDS-2; i<=k;)
  863. // Feistel substitution on left word
  864. n = S[(l >> 24) & 0xff],
  865. n += S[0x100 | ((l >> 16) & 0xff)],
  866. n ^= S[0x200 | ((l >> 8) & 0xff)],
  867. n += S[0x300 | (l & 0xff)],
  868. r ^= n ^ P[++i],
  869. // Feistel substitution on right word
  870. n = S[(r >> 24) & 0xff],
  871. n += S[0x100 | ((r >> 16) & 0xff)],
  872. n ^= S[0x200 | ((r >> 8) & 0xff)],
  873. n += S[0x300 | (r & 0xff)],
  874. l ^= n ^ P[++i];
  875. lr[off] = r ^ P[BLOWFISH_NUM_ROUNDS + 1];
  876. lr[off + 1] = l;
  877. return lr;
  878. }
  879. /**
  880. * @param {Array.<number>} data
  881. * @param {number} offp
  882. * @returns {{key: number, offp: number}}
  883. * @inner
  884. */
  885. function _streamtoword(data, offp) {
  886. for (var i = 0, word = 0; i < 4; ++i)
  887. word = (word << 8) | (data[offp] & 0xff),
  888. offp = (offp + 1) % data.length;
  889. return { key: word, offp: offp };
  890. }
  891. /**
  892. * @param {Array.<number>} key
  893. * @param {Array.<number>} P
  894. * @param {Array.<number>} S
  895. * @inner
  896. */
  897. function _key(key, P, S) {
  898. var offset = 0,
  899. lr = [0, 0],
  900. plen = P.length,
  901. slen = S.length,
  902. sw;
  903. for (var i = 0; i < plen; i++)
  904. sw = _streamtoword(key, offset),
  905. offset = sw.offp,
  906. P[i] = P[i] ^ sw.key;
  907. for (i = 0; i < plen; i += 2)
  908. lr = _encipher(lr, 0, P, S),
  909. P[i] = lr[0],
  910. P[i + 1] = lr[1];
  911. for (i = 0; i < slen; i += 2)
  912. lr = _encipher(lr, 0, P, S),
  913. S[i] = lr[0],
  914. S[i + 1] = lr[1];
  915. }
  916. /**
  917. * Expensive key schedule Blowfish.
  918. * @param {Array.<number>} data
  919. * @param {Array.<number>} key
  920. * @param {Array.<number>} P
  921. * @param {Array.<number>} S
  922. * @inner
  923. */
  924. function _ekskey(data, key, P, S) {
  925. var offp = 0,
  926. lr = [0, 0],
  927. plen = P.length,
  928. slen = S.length,
  929. sw;
  930. for (var i = 0; i < plen; i++)
  931. sw = _streamtoword(key, offp),
  932. offp = sw.offp,
  933. P[i] = P[i] ^ sw.key;
  934. offp = 0;
  935. for (i = 0; i < plen; i += 2)
  936. sw = _streamtoword(data, offp),
  937. offp = sw.offp,
  938. lr[0] ^= sw.key,
  939. sw = _streamtoword(data, offp),
  940. offp = sw.offp,
  941. lr[1] ^= sw.key,
  942. lr = _encipher(lr, 0, P, S),
  943. P[i] = lr[0],
  944. P[i + 1] = lr[1];
  945. for (i = 0; i < slen; i += 2)
  946. sw = _streamtoword(data, offp),
  947. offp = sw.offp,
  948. lr[0] ^= sw.key,
  949. sw = _streamtoword(data, offp),
  950. offp = sw.offp,
  951. lr[1] ^= sw.key,
  952. lr = _encipher(lr, 0, P, S),
  953. S[i] = lr[0],
  954. S[i + 1] = lr[1];
  955. }
  956. /**
  957. * Internaly crypts a string.
  958. * @param {Array.<number>} b Bytes to crypt
  959. * @param {Array.<number>} salt Salt bytes to use
  960. * @param {number} rounds Number of rounds
  961. * @param {function(Error, Array.<number>=)=} callback Callback receiving the error, if any, and the resulting bytes. If
  962. * omitted, the operation will be performed synchronously.
  963. * @param {function(number)=} progressCallback Callback called with the current progress
  964. * @returns {!Array.<number>|undefined} Resulting bytes if callback has been omitted, otherwise `undefined`
  965. * @inner
  966. */
  967. function _crypt(b, salt, rounds, callback, progressCallback) {
  968. var cdata = C_ORIG.slice(),
  969. clen = cdata.length,
  970. err;
  971. // Validate
  972. if (rounds < 4 || rounds > 31) {
  973. err = Error("Illegal number of rounds (4-31): "+rounds);
  974. if (callback) {
  975. nextTick(callback.bind(this, err));
  976. return;
  977. } else
  978. throw err;
  979. }
  980. if (salt.length !== BCRYPT_SALT_LEN) {
  981. err =Error("Illegal salt length: "+salt.length+" != "+BCRYPT_SALT_LEN);
  982. if (callback) {
  983. nextTick(callback.bind(this, err));
  984. return;
  985. } else
  986. throw err;
  987. }
  988. rounds = (1 << rounds) >>> 0;
  989. var P = P_ORIG.slice(),
  990. S = S_ORIG.slice(),
  991. i = 0, j;
  992. _ekskey(salt, b, P, S);
  993. /**
  994. * Calcualtes the next round.
  995. * @returns {Array.<number>|undefined} Resulting array if callback has been omitted, otherwise `undefined`
  996. * @inner
  997. */
  998. function next() {
  999. if (progressCallback)
  1000. progressCallback(i / rounds);
  1001. if (i < rounds) {
  1002. var start = Date.now();
  1003. for (; i < rounds;) {
  1004. i = i + 1;
  1005. _key(b, P, S);
  1006. _key(salt, P, S);
  1007. if (Date.now() - start > MAX_EXECUTION_TIME)
  1008. break;
  1009. }
  1010. } else {
  1011. for (i = 0; i < 64; i++)
  1012. for (j = 0; j < (clen >> 1); j++)
  1013. _encipher(cdata, j << 1, P, S);
  1014. var ret = [];
  1015. for (i = 0; i < clen; i++)
  1016. ret.push(((cdata[i] >> 24) & 0xff) >>> 0),
  1017. ret.push(((cdata[i] >> 16) & 0xff) >>> 0),
  1018. ret.push(((cdata[i] >> 8) & 0xff) >>> 0),
  1019. ret.push((cdata[i] & 0xff) >>> 0);
  1020. if (callback) {
  1021. callback(null, ret);
  1022. return;
  1023. } else
  1024. return ret;
  1025. }
  1026. if (callback)
  1027. nextTick(next);
  1028. }
  1029. // Async
  1030. if (typeof callback !== 'undefined') {
  1031. next();
  1032. // Sync
  1033. } else {
  1034. var res;
  1035. while (true)
  1036. if (typeof(res = next()) !== 'undefined')
  1037. return res || [];
  1038. }
  1039. }
  1040. /**
  1041. * Internally hashes a string.
  1042. * @param {string} s String to hash
  1043. * @param {?string} salt Salt to use, actually never null
  1044. * @param {function(Error, string=)=} callback Callback receiving the error, if any, and the resulting hash. If omitted,
  1045. * hashing is perormed synchronously.
  1046. * @param {function(number)=} progressCallback Callback called with the current progress
  1047. * @returns {string|undefined} Resulting hash if callback has been omitted, otherwise `undefined`
  1048. * @inner
  1049. */
  1050. function _hash(s, salt, callback, progressCallback) {
  1051. var err;
  1052. if (typeof s !== 'string' || typeof salt !== 'string') {
  1053. err = Error("Invalid string / salt: Not a string");
  1054. if (callback) {
  1055. nextTick(callback.bind(this, err));
  1056. return;
  1057. }
  1058. else
  1059. throw err;
  1060. }
  1061. // Validate the salt
  1062. var minor, offset;
  1063. if (salt.charAt(0) !== '$' || salt.charAt(1) !== '2') {
  1064. err = Error("Invalid salt version: "+salt.substring(0,2));
  1065. if (callback) {
  1066. nextTick(callback.bind(this, err));
  1067. return;
  1068. }
  1069. else
  1070. throw err;
  1071. }
  1072. if (salt.charAt(2) === '$')
  1073. minor = String.fromCharCode(0),
  1074. offset = 3;
  1075. else {
  1076. minor = salt.charAt(2);
  1077. if ((minor !== 'a' && minor !== 'b' && minor !== 'y') || salt.charAt(3) !== '$') {
  1078. err = Error("Invalid salt revision: "+salt.substring(2,4));
  1079. if (callback) {
  1080. nextTick(callback.bind(this, err));
  1081. return;
  1082. } else
  1083. throw err;
  1084. }
  1085. offset = 4;
  1086. }
  1087. // Extract number of rounds
  1088. if (salt.charAt(offset + 2) > '$') {
  1089. err = Error("Missing salt rounds");
  1090. if (callback) {
  1091. nextTick(callback.bind(this, err));
  1092. return;
  1093. } else
  1094. throw err;
  1095. }
  1096. var r1 = parseInt(salt.substring(offset, offset + 1), 10) * 10,
  1097. r2 = parseInt(salt.substring(offset + 1, offset + 2), 10),
  1098. rounds = r1 + r2,
  1099. real_salt = salt.substring(offset + 3, offset + 25);
  1100. s += minor >= 'a' ? "\x00" : "";
  1101. var passwordb = stringToBytes(s),
  1102. saltb = base64_decode(real_salt, BCRYPT_SALT_LEN);
  1103. /**
  1104. * Finishes hashing.
  1105. * @param {Array.<number>} bytes Byte array
  1106. * @returns {string}
  1107. * @inner
  1108. */
  1109. function finish(bytes) {
  1110. var res = [];
  1111. res.push("$2");
  1112. if (minor >= 'a')
  1113. res.push(minor);
  1114. res.push("$");
  1115. if (rounds < 10)
  1116. res.push("0");
  1117. res.push(rounds.toString());
  1118. res.push("$");
  1119. res.push(base64_encode(saltb, saltb.length));
  1120. res.push(base64_encode(bytes, C_ORIG.length * 4 - 1));
  1121. return res.join('');
  1122. }
  1123. // Sync
  1124. if (typeof callback == 'undefined')
  1125. return finish(_crypt(passwordb, saltb, rounds));
  1126. // Async
  1127. else {
  1128. _crypt(passwordb, saltb, rounds, function(err, bytes) {
  1129. if (err)
  1130. callback(err, null);
  1131. else
  1132. callback(null, finish(bytes));
  1133. }, progressCallback);
  1134. }
  1135. }
  1136. /**
  1137. * Encodes a byte array to base64 with up to len bytes of input, using the custom bcrypt alphabet.
  1138. * @function
  1139. * @param {!Array.<number>} b Byte array
  1140. * @param {number} len Maximum input length
  1141. * @returns {string}
  1142. * @expose
  1143. */
  1144. bcrypt.encodeBase64 = base64_encode;
  1145. /**
  1146. * Decodes a base64 encoded string to up to len bytes of output, using the custom bcrypt alphabet.
  1147. * @function
  1148. * @param {string} s String to decode
  1149. * @param {number} len Maximum output length
  1150. * @returns {!Array.<number>}
  1151. * @expose
  1152. */
  1153. bcrypt.decodeBase64 = base64_decode;
  1154. return bcrypt;
  1155. }));