diff --git a/resources/openpgp.js b/resources/openpgp.js index 8862c76e..32b90efe 100644 --- a/resources/openpgp.js +++ b/resources/openpgp.js @@ -1643,7 +1643,7 @@ module.exports = { } } -},{"../util":55,"./cipher":8}],4:[function(require,module,exports){ +},{"../util":56,"./cipher":8}],4:[function(require,module,exports){ /* Rijndael (AES) Encryption * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de * version 1.1, check www.haneWIN.de for the latest version @@ -2152,7 +2152,7 @@ for (var i in types) { module.exports[types[i]] = makeClass(types[i]); } -},{"../../util":55}],5:[function(require,module,exports){ +},{"../../util":56}],5:[function(require,module,exports){ /* Modified by Recurity Labs GmbH * * Originally written by nklein software (nklein.com) @@ -2566,7 +2566,7 @@ module.exports = BF; module.exports.keySize = BF.prototype.keySize = 16; module.exports.blockSize = BF.prototype.blockSize = 16; -},{"../../util":55}],6:[function(require,module,exports){ +},{"../../util":56}],6:[function(require,module,exports){ // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. @@ -3172,7 +3172,7 @@ module.exports = cast5; module.exports.blockSize = cast5.prototype.blockSize = 8; module.exports.keySize = cast5.prototype.keySize = 16; -},{"../../util":55}],7:[function(require,module,exports){ +},{"../../util":56}],7:[function(require,module,exports){ //Paul Tero, July 2001 //http://www.tero.co.uk/des/ // @@ -3579,7 +3579,7 @@ module.exports = { originalDes: OriginalDes } -},{"../../util":55}],8:[function(require,module,exports){ +},{"../../util":56}],8:[function(require,module,exports){ var desModule = require('./des.js'); module.exports = { @@ -3976,7 +3976,7 @@ module.exports = TF; module.exports.keySize = TF.prototype.keySize = 32; module.exports.blockSize = TF.prototype.blockSize = 16; -},{"../../util":55}],10:[function(require,module,exports){ +},{"../../util":56}],10:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -4194,7 +4194,7 @@ module.exports = { }; -},{"../type/mpi.js":53,"./cipher":8,"./public_key":19,"./random.js":22}],11:[function(require,module,exports){ +},{"../type/mpi.js":54,"./cipher":8,"./public_key":19,"./random.js":22}],11:[function(require,module,exports){ var sha = require('./sha.js'); module.exports = { @@ -4486,7 +4486,7 @@ if (md5('hello') != '5d41402abc4b2a76b9719d911017c592') { module.exports = MD5 -},{"../../util/util.js":55}],13:[function(require,module,exports){ +},{"../../util/util.js":56}],13:[function(require,module,exports){ /* * CryptoMX Tools * Copyright (C) 2004 - 2006 Derek Buitenhuis @@ -6051,7 +6051,7 @@ module.exports = { } } -},{"../util":55,"./crypto.js":10,"./hash":11,"./public_key/jsbn.js":20,"./random.js":22}],17:[function(require,module,exports){ +},{"../util":56,"./crypto.js":10,"./hash":11,"./public_key/jsbn.js":20,"./random.js":22}],17:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -6215,7 +6215,7 @@ function DSA() { module.exports = DSA; -},{"../../util":55,"../hash":11,"../random.js":22,"./jsbn.js":20}],18:[function(require,module,exports){ +},{"../../util":56,"../hash":11,"../random.js":22,"./jsbn.js":20}],18:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -6270,7 +6270,7 @@ function Elgamal() { module.exports = Elgamal; -},{"../../util":55,"../random.js":22,"./jsbn.js":20}],19:[function(require,module,exports){ +},{"../../util":56,"../random.js":22,"./jsbn.js":20}],19:[function(require,module,exports){ module.exports = { rsa: require('./rsa.js'), elgamal: require('./elgamal.js'), @@ -7984,7 +7984,7 @@ BigInteger.prototype.toMPI = bnToMPI; // JSBN-specific extension BigInteger.prototype.square = bnSquare; -},{"../../util":55,"./jsbn.js":20}],21:[function(require,module,exports){ +},{"../../util":56,"./jsbn.js":20}],21:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -8131,7 +8131,7 @@ function RSA() { module.exports = RSA; -},{"../../util":55,"../random.js":22,"./jsbn.js":20}],22:[function(require,module,exports){ +},{"../../util":56,"../random.js":22,"./jsbn.js":20}],22:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -8237,7 +8237,7 @@ module.exports = { }; -},{"../type/mpi.js":53}],23:[function(require,module,exports){ +},{"../type/mpi.js":54}],23:[function(require,module,exports){ var publicKey = require('./public_key'), pkcs1 = require('./pkcs1.js'), hashModule = require('./hash'); @@ -9022,10 +9022,13 @@ var enums = { module.exports = enums; +},{}],"openpgp":[function(require,module,exports){ +module.exports=require('IKGp/G'); },{}],"IKGp/G":[function(require,module,exports){ module.exports = require('./openpgp.js'); module.exports.key = require('./key.js'); +module.exports.keyring = require('./keyring.js'); module.exports.message = require('./message.js'); module.exports.util = require('./util'); module.exports.packet = require('./packet'); @@ -9037,9 +9040,251 @@ module.exports.enums = require('./enums.js'); module.exports.config = require('./config'); module.exports.crypto = require('./crypto'); -},{"./config":2,"./crypto":15,"./encoding/armor.js":24,"./enums.js":26,"./key.js":29,"./message.js":30,"./openpgp.js":31,"./packet":34,"./type/keyid.js":52,"./type/mpi.js":53,"./type/s2k.js":54,"./util":55}],"openpgp":[function(require,module,exports){ -module.exports=require('IKGp/G'); -},{}],29:[function(require,module,exports){ +},{"./config":2,"./crypto":15,"./encoding/armor.js":24,"./enums.js":26,"./key.js":29,"./keyring.js":30,"./message.js":31,"./openpgp.js":32,"./packet":35,"./type/keyid.js":53,"./type/mpi.js":54,"./type/s2k.js":55,"./util":56}],29:[function(require,module,exports){ +// GPG4Browsers - An OpenPGP implementation in javascript +// Copyright (C) 2011 Recurity Labs GmbH +// +// This library is free software; you can redistribute it and/or +// modify it under the terms of the GNU Lesser General Public +// License as published by the Free Software Foundation; either +// version 2.1 of the License, or (at your option) any later version. +// +// This library is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +// Lesser General Public License for more details. +// +// You should have received a copy of the GNU Lesser General Public +// License along with this library; if not, write to the Free Software +// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + +var packet = require('./packet'); +var enums = require('./enums.js'); +var armor = require('./encoding/armor.js'); +var config = require('./config'); + +/** + * @class + * @classdesc Class that represents an OpenPGP key. Must contain a master key. + * @param {packetlist} packetlist [description] + * Can contain additional subkeys, signatures, + * user ids, user attributes. + */ + + function key(packetlist) { + + this.packets = packetlist || new packet.list(); + + + /** + * Returns the primary key packet (secret or public) + * @returns {packet_secret_key|packet_public_key|null} + */ + this.getKeyPacket = function() { + for (var i = 0; i < this.packets.length; i++) { + if (this.packets[i].tag == enums.packet.public_key || + this.packets[i].tag == enums.packet.secret_key) { + return this.packets[i]; + } + } + return null; + } + + /** + * Returns all the private and public subkey packets + * @returns {[public_subkey|secret_subkey]} + */ + this.getSubkeyPackets = function() { + + var subkeys = []; + + for (var i = 0; i < this.packets.length; i++) { + if (this.packets[i].tag == enums.packet.public_subkey || + this.packets[i].tag == enums.packet.secret_subkey) { + subkeys.push(this.packets[i]); + } + } + + return subkeys; + } + + /** + * Returns all the private and public key and subkey packets + * @returns {[public_subkey|secret_subkey|packet_secret_key|packet_public_key]} + */ + this.getAllKeyPackets = function() { + return [this.getKeyPacket()].concat(this.getSubkeyPackets()); + } + + /** + * Returns key IDs of all key packets + * @returns {[keyid]} + */ + this.getKeyIds = function() { + var keyIds = []; + var keys = this.getAllKeyPackets(); + for (var i = 0; i < keys.length; i++) { + keyIds.push(keys[i].getKeyId()); + } + return keyIds; + } + + /** + * Returns key IDs of all key packets in hex + * @returns {[String]} + */ + this.getKeyIdsHex = function() { + return this.getKeyIds().map(function(keyId) { + return keyId.toHex(); + }); + } + + /** + * Returns first key packet which match to an array of key IDs + * @param {[keyid]} keyIds + * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} + */ + this.getKeyPacketByIds = function(keyIds) { + var keys = this.getAllKeyPackets(); + for (var i = 0; i < keys.length; i++) { + var keyId = keys[i].getKeyId(); + for (var j = 0; j < keyIds.length; j++) { + if (keyId.equals(keyIds[j])) { + //TODO return only verified keys + return keys[i]; + } + } + } + return null; + } + + /** + * Returns true if this is a public key + * @return {Boolean} + */ + this.isPublic = function() { + var publicKeyPackets = this.packets.filterByTag(enums.packet.public_key); + return publicKeyPackets.length !== 0 ? true : false; + } + + /** + * Returns true if this is a private key + * @return {Boolean} + */ + this.isPrivate = function() { + var privateKeyPackets = this.packets.filterByTag(enums.packet.private_key); + return privateKeyPackets.length !== 0 ? true : false; + } + + /** + * Returns first key packet that is available for signing + * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} + */ + this.getSigningKeyPacket = function() { + + var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa]; + + signing = signing.map(function(s) { + return enums.read(enums.publicKey, s); + }); + + var keys = this.getAllKeyPackets(); + + for (var i = 0; i < keys.length; i++) { + if (signing.indexOf(keys[i].algorithm) !== -1) { + return keys[i]; + } + } + + return null; + } + + /** + * Returns preferred signature hash algorithm of this key + * @return {String} + */ + function getPreferredSignatureHashAlgorithm() { + //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8 + //separate private key preference from digest preferences + return config.prefer_hash_algorithm; + } + + /** + * Returns the first valid encryption key packet for this key + * @returns {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} key packet or null if no encryption key has been found + */ + this.getEncryptionKeyPacket = function() { + // V4: by convention subkeys are prefered for encryption service + // V3: keys MUST NOT have subkeys + var isValidEncryptionKey = function(key) { + //TODO evaluate key flags: http://tools.ietf.org/html/rfc4880#section-5.2.3.21 + return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey, + enums.publicKey.rsa_sign); + //TODO verify key + //&& keys.verifyKey() + }; + + var subkeys = this.getSubkeyPackets(); + + for (var j = 0; j < subkeys.length; j++) { + if (isValidEncryptionKey(subkeys[j])) { + return subkeys[j]; + } + } + // if no valid subkey for encryption, use primary key + var primaryKey = this.getKeyPacket(); + if (isValidEncryptionKey(primaryKey)) { + return primaryKey; + } + return null; + } + + /** + * Decrypts all secret key and subkey packets + * @param {String} passphrase + * @return {undefined} + */ + this.decrypt = function(passphrase) { + //TODO boolean return value + var keys = this.getAllKeyPackets(); + + for (var i in keys) + if (keys[i].tag == enums.packet.secret_subkey || + keys[i].tag == enums.packet.secret_key) { + keys[i].decrypt(passphrase); + } + } + + + // TODO + this.verify = function() { + + } + // TODO + this.revoke = function() { + + } + +} + +/** + * reads an OpenPGP armored text and returns a key object + * @param {String} armoredText text to be parsed + * @return {key} new key object + */ +key.readArmored = function(armoredText) { + //TODO how do we want to handle bad text? Exception throwing + //TODO don't accept non-key armored texts + var input = armor.decode(armoredText).openpgp; + var packetlist = new packet.list(); + packetlist.read(input); + var newKey = new key(packetlist); + return newKey; +} + +module.exports = key; + +},{"./config":2,"./encoding/armor.js":24,"./enums.js":26,"./packet":35}],30:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -9063,166 +9308,174 @@ var armor = require('./encoding/armor.js'); /** * @class - * @classdesc Class that represents an OpenPGP key. Must contain a master key. - * @param {packetlist} packetlist [description] - * Can contain additional subkeys, signatures, - * user ids, user attributes. + * @classdesc The class that deals with storage of the keyring. Currently the only option is to use HTML5 local storage. */ - - function key(packetlist) { - - this.packets = packetlist || new packet.list(); - - - /** Returns the primary key (secret or public) - * @returns {packet_secret_key|packet_public_key|null} */ - this.getKey = function() { - for (var i = 0; i < this.packets.length; i++) { - if (this.packets[i].tag == enums.packet.public_key || - this.packets[i].tag == enums.packet.secret_key) { - return this.packets[i]; - } - } - return null; - }; - - /** Returns all the private and public subkeys - * @returns {[public_subkey|secret_subkey]} */ - this.getSubkeys = function() { - - var subkeys = []; - - for (var i = 0; i < this.packets.length; i++) { - if (this.packets[i].tag == enums.packet.public_subkey || - this.packets[i].tag == enums.packet.secret_subkey) { - subkeys.push(this.packets[i]); - } - } - - return subkeys; - }; - - this.getAllKeys = function() { - return [this.getKey()].concat(this.getSubkeys()); - }; - - this.getKeyids = function() { - var keyids = []; - var keys = this.getAllKeys(); - for (var i = 0; i < keys.length; i++) { - keyids.push(keys[i].getKeyId()); - } - return keyids; - }; - - this.getKeyById = function(keyid) { - var keys = this.getAllKeys(); - for (var i = 0; i < keys.length; i++) { - if (keys[i].getKeyId().equals(keyid)) { - return keys[i]; - } - } - } - - this.getSigningKey = function() { - - var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa]; - - signing = signing.map(function(s) { - return enums.read(enums.publicKey, s); - }); - - var keys = this.getAllKeys(); - - for (var i = 0; i < keys.length; i++) { - if (signing.indexOf(keys[i].algorithm) !== -1) { - return keys[i]; - } - } - - return null; - }; - - function getPreferredSignatureHashAlgorithm() { - var pkey = this.getSigningKey(); - if (pkey === null) { - util.print_error("private key is for encryption only! Cannot create a signature."); - return null; - } - if (pkey.publicKey.publicKeyAlgorithm == 17) { - var dsa = new DSA(); - return dsa.select_hash_algorithm(pkey.publicKey.MPIs[1].toBigInteger()); // q - } - //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8 - //separate private key preference from digest preferences - return openpgp.config.config.prefer_hash_algorithm; - } +var keyring = function() { + this.armoredPacketlists = []; + this.parsedPacketlists = []; /** - * Finds an encryption key for this key - * @returns null if no encryption key has been found + * Initialization routine for the keyring. This method reads the + * keyring from HTML5 local storage and initializes this instance. + * This method is called by openpgp.init(). */ - this.getEncryptionKey = function() { - // V4: by convention subkeys are prefered for encryption service - // V3: keys MUST NOT have subkeys - var isValidEncryptionKey = function(key) { - return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey, - enums.publicKey.rsa_sign); - //TODO verify key - //&& keys.verifyKey() - }; - - var subkeys = this.getSubkeys(); - - for (var j = 0; j < subkeys.length; j++) { - if (isValidEncryptionKey(subkeys[j])) { - return subkeys[j]; - } + function init() { + var armoredPacketlists = JSON.parse(window.localStorage.getItem("armoredPacketlists")); + if (armoredPacketlists === null || armoredPacketlists.length === 0) { + armoredPacketlists = []; } - // if no valid subkey for encryption, use primary key - var primaryKey = this.getKey(); - if (isValidEncryptionKey(primaryKey)) { - return primaryKey; + this.armoredPacketlists = armoredPacketlists; + + var packetlist; + for (var i = 0; i < armoredPacketlists.length; i++) { + packetlist = new packet.list(); + packetlist.read(armoredPacketlists[i]); + this.parsedPacketlists.push(packetlist); } - return null; - }; - - this.decrypt = function(passphrase) { - var keys = this.getAllKeys(); - - for (var i in keys) - if (keys[i].tag == enums.packet.secret_subkey || - keys[i].tag == enums.packet.secret_key) { - keys[i].decrypt(passphrase); - } - }; - - - // TODO need to implement this - - function revoke() { - } + this.init = init; + + /** + * Saves the current state of the keyring to HTML5 local storage. + * The privateKeys array and publicKeys array gets Stringified using JSON + */ + function store() { + window.localStorage.setItem("armoredPacketlists", JSON.stringify(this.armoredPacketlists)); + } + this.store = store; + + function emailPacketCheck(packet, email) { + var emailMatch = false; + var packetEmail; + email = email.toLowerCase(); + if (packet.tag == enums.packet.userid) { + packetEmail = packet.userid; + //we need to get just the email from the userid packet + packetEmail = packetEmail.split('<')[1].split('<')[0].trim.toLowerCase(); + if (packetEmail == email) { + emailMatch = true; + } + } + return emailMatch; + } + + function idPacketCheck(packet, id) { + if (packet.getKeyId && packet.getKeyId().write() == id) { + return true; + } + return false; + } + + function helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist) { + var packet; + for (var l = 0; l < packetlist.length; l++) { + packet = packetlist[l]; + identityMatch = identityFunction(packet, identityInput); + if (!packetType) { + packetMatch = true; + } else if (packet.tag == packetType) { + packetMatch = true; + } + if (packetMatch && identityMatch) { + return true; + } + } + return false; + } + + function checkForIdentityAndPacketMatch(identityFunction, identityInput, packetType) { + var results = []; + var packetlist; + var identityMatch; + var packetMatch; + for (var p = 0; p < this.parsedPacketlists.length; p++) { + identityMatch = false; + packetMatch = false; + packetlist = this.parsedPacketlists[p]; + if (helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist)) { + results.push(packetlist); + } + } + return results; + } + this.checkForIdentityAndPacketMatch = checkForIdentityAndPacketMatch; + + /** + * searches all public keys in the keyring matching the address or address part of the user ids + * @param {String} email_address + * @return {openpgp_msg_publickey[]} The public keys associated with provided email address. + */ + function getPublicKeyForAddress(email) { + return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.public_key); + } + this.getPublicKeyForAddress = getPublicKeyForAddress; + + /** + * Searches the keyring for a private key containing the specified email address + * @param {String} email_address email address to search for + * @return {openpgp_msg_privatekey[]} private keys found + */ + function getPrivateKeyForAddress(email_address) { + return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.secret_key); + } + this.getPrivateKeyForAddress = getPrivateKeyForAddress; + + /** + * Searches the keyring for public keys having the specified key id + * @param {String} keyId provided as string of hex number (lowercase) + * @return {openpgp_msg_privatekey[]} public keys found + */ + function getPacketlistsForKeyId(keyId) { + return this.checkForIdentityAndPacketMatch(idPacketCheck, keyId); + } + this.getPacketlistsForKeyId = getPacketlistsForKeyId; + + /** + * Imports a packet list (public or private key block) from an ascii armored message + * @param {String} armored message to read the packets/key from + */ + function importPacketlist(armored) { + this.armoredPacketlists.push(armored); + + var dearmored = armor.decode(armored.replace(/\r/g, '')).openpgp; + + packetlist = new packet.list(); + packetlist.read(dearmored); + this.parsedPacketlists.push(packetlist); + + return true; + } + this.importPacketlist = importPacketlist; + + /** + * TODO + * returns the openpgp_msg_privatekey representation of the public key at public key ring index + * @param {Integer} index the index of the public key within the publicKeys array + * @return {openpgp_msg_privatekey} the public key object + */ + function exportPublicKey(index) { + return this.publicKey[index]; + } + this.exportPublicKey = exportPublicKey; + + /** + * TODO + * Removes a public key from the public key keyring at the specified index + * @param {Integer} index the index of the public key within the publicKeys array + * @return {openpgp_msg_privatekey} The public key object which has been removed + */ + function removePublicKey(index) { + var removed = this.publicKeys.splice(index, 1); + this.store(); + return removed; + } + this.removePublicKey = removePublicKey; }; -/** - * reads an OpenPGP armored text and returns a key object - * @param {String} armoredText text to be parsed - * @return {key} new key object - */ -key.readArmored = function(armoredText) { - //TODO how do we want to handle bad text? Exception throwing - var input = armor.decode(armoredText).openpgp; - var packetlist = new packet.list(); - packetlist.read(input); - var newKey = new key(packetlist); - return newKey; -} +module.exports = new keyring(); -module.exports = key; - -},{"./encoding/armor.js":24,"./enums.js":26,"./packet":34}],30:[function(require,module,exports){ +},{"./encoding/armor.js":24,"./enums.js":26,"./packet":35}],31:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -9255,67 +9508,60 @@ function message(packetlist) { /** * Returns the key IDs of the public keys to which the session key is encrypted - * @return {[keyId]} keyId provided as string of hex number (lowercase) + * @return {[keyId]} array of keyid objects */ this.getKeyIds = function() { var keyIds = []; var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key); pkESKeyPacketlist.forEach(function(packet) { - keyIds.push(packet.publicKeyId.toHex()); + keyIds.push(packet.publicKeyId); }); return keyIds; } - function generic_decrypt(packets, passphrase) { - var sessionkey; - - for (var i = 0; i < packets.length; i++) { - if (packets[i].tag == openpgp_packet.tags.public_key_encrypted_session_key) { - var key = openpgp.keyring.getKeyById(packets[i].public_key_id); - - } - } - + /** + * Returns the key IDs in hex of the public keys to which the session key is encrypted + * @return {[String]} keyId provided as string of hex numbers (lowercase) + */ + this.getKeyIdsHex = function() { + return this.getKeyIds().map(function(keyId) { + return keyId.toHex(); + }); } /** - * Decrypts a message and generates user interface message out of the found. - * MDC will be verified as well as message signatures - * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key) - * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message - * @return {} plaintext of the message or null on error + * Decrypts the message + * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with (corresponding to the session key) + * @return {[String]} array with plaintext of decrypted messages */ - this.decrypt = function(key) { + this.decrypt = function(privateKeyPacket) { var decryptedMessages = []; - var keyids = key.getKeyids(); var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key); - outer: for (var i = 0; i < pkESKeyPacketlist.length; i++) { + for (var i = 0; i < pkESKeyPacketlist.length; i++) { var pkESKeyPacket = pkESKeyPacketlist[i]; - for (var j = 0; j < keyids.length; j++) { - if (pkESKeyPacket.publicKeyId.equals(keyids[j])) { - pkESKeyPacket.decrypt(key.getKeyById(pkESKeyPacket.publicKeyId)); - var symEncryptedPacketlist = this.packets.filter(function(packet) { - return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected; - }); - for (var k = 0; k < symEncryptedPacketlist.length; k++) { - var symEncryptedPacket = symEncryptedPacketlist[k]; - symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey); - for (var l = 0; l < symEncryptedPacket.packets.length; l++) { - var dataPacket = symEncryptedPacket.packets[l]; - switch (dataPacket.tag) { - case enums.packet.literal: - decryptedMessages.push(dataPacket.getBytes()); - break; - case enums.packet.compressed: - //TODO - break; - default: - //TODO - } + if (pkESKeyPacket.publicKeyId.equals(privateKeyPacket.getKeyId())) { + pkESKeyPacket.decrypt(privateKeyPacket); + var symEncryptedPacketlist = this.packets.filter(function(packet) { + return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected; + }); + for (var k = 0; k < symEncryptedPacketlist.length; k++) { + var symEncryptedPacket = symEncryptedPacketlist[k]; + symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey); + for (var l = 0; l < symEncryptedPacket.packets.length; l++) { + var dataPacket = symEncryptedPacket.packets[l]; + switch (dataPacket.tag) { + case enums.packet.literal: + decryptedMessages.push(dataPacket.getBytes()); + break; + case enums.packet.compressed: + //TODO + break; + default: + //TODO } } - break outer; } + break; } } return decryptedMessages; @@ -9406,6 +9652,7 @@ function message(packetlist) { */ message.readArmored = function(armoredText) { //TODO how do we want to handle bad text? Exception throwing + //TODO don't accept non-message armored texts var input = armor.decode(armoredText).openpgp; var packetlist = new packet.list(); packetlist.read(input); @@ -9414,7 +9661,7 @@ message.readArmored = function(armoredText) { } module.exports = message; -},{"./encoding/armor.js":24,"./enums.js":26,"./packet":34}],31:[function(require,module,exports){ +},{"./encoding/armor.js":24,"./enums.js":26,"./packet":35}],32:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -9469,15 +9716,15 @@ function _openpgp() { var sessionKey = crypto.generateSessionKey(enums.read(enums.symmetric, config.encryption_cipher)); keys.forEach(function(key) { - var encryptionKey = key.getEncryptionKey(); - if (encryptionKey) { + var encryptionKeyPacket = key.getEncryptionKeyPacket(); + if (encryptionKeyPacket) { var pkESKeyPacket = new packet.public_key_encrypted_session_key(); - pkESKeyPacket.publicKeyId = encryptionKey.getKeyId(); - pkESKeyPacket.publicKeyAlgorithm = encryptionKey.algorithm; + pkESKeyPacket.publicKeyId = encryptionKeyPacket.getKeyId(); + pkESKeyPacket.publicKeyAlgorithm = encryptionKeyPacket.algorithm; pkESKeyPacket.sessionKey = sessionKey; //TODO get preferred algo from signature pkESKeyPacket.sessionKeyAlgorithm = enums.read(enums.symmetric, config.encryption_cipher); - pkESKeyPacket.encrypt(encryptionKey); + pkESKeyPacket.encrypt(encryptionKeyPacket); messagePacketlist.push(pkESKeyPacket); } }); @@ -9508,12 +9755,12 @@ function _openpgp() { /** * decrypts message - * @param {[key]} decrypted privateKey + * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with * @param {message} message the message object with the encrypted data * @return {String} decrypted message as as native JavaScript string */ - function decryptMessage(privateKey, message) { - return message.decrypt(privateKey); + function decryptMessage(privateKeyPacket, message) { + return message.decrypt(privateKeyPacket); } function decryptAndVerifyMessage(privateKey, publicKeys, messagePacketlist) { @@ -9732,7 +9979,7 @@ function _openpgp() { module.exports = new _openpgp(); -},{"./config":2,"./crypto":15,"./encoding/armor.js":24,"./enums.js":26,"./packet":34,"./util":55}],32:[function(require,module,exports){ +},{"./config":2,"./crypto":15,"./encoding/armor.js":24,"./enums.js":26,"./packet":35,"./util":56}],33:[function(require,module,exports){ var enums = require('../enums.js'); // This is pretty ugly, but browserify needs to have the requires explicitly written. @@ -9762,7 +10009,7 @@ for (var i in enums.packet) { packetClass.prototype.tag = enums.packet[i]; } -},{"../enums.js":26,"./compressed.js":33,"./literal.js":35,"./marker.js":36,"./one_pass_signature.js":37,"./public_key.js":40,"./public_key_encrypted_session_key.js":41,"./public_subkey.js":42,"./secret_key.js":43,"./secret_subkey.js":44,"./signature.js":45,"./sym_encrypted_integrity_protected.js":46,"./sym_encrypted_session_key.js":47,"./symmetrically_encrypted.js":48,"./trust.js":49,"./user_attribute.js":50,"./userid.js":51}],33:[function(require,module,exports){ +},{"../enums.js":26,"./compressed.js":34,"./literal.js":36,"./marker.js":37,"./one_pass_signature.js":38,"./public_key.js":41,"./public_key_encrypted_session_key.js":42,"./public_subkey.js":43,"./secret_key.js":44,"./secret_subkey.js":45,"./signature.js":46,"./sym_encrypted_integrity_protected.js":47,"./sym_encrypted_session_key.js":48,"./symmetrically_encrypted.js":49,"./trust.js":50,"./user_attribute.js":51,"./userid.js":52}],34:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -9927,7 +10174,7 @@ module.exports = function packet_compressed() { } }; -},{"../compression/jxg.js":1,"../encoding/base64.js":25,"../enums.js":26}],34:[function(require,module,exports){ +},{"../compression/jxg.js":1,"../encoding/base64.js":25,"../enums.js":26}],35:[function(require,module,exports){ var enums = require('../enums.js'); module.exports = { @@ -9939,7 +10186,7 @@ var packets = require('./all_packets.js'); for (var i in packets) module.exports[i] = packets[i]; -},{"../enums.js":26,"./all_packets.js":32,"./packetlist.js":39}],35:[function(require,module,exports){ +},{"../enums.js":26,"./all_packets.js":33,"./packetlist.js":40}],36:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -10060,7 +10307,7 @@ module.exports = function packet_literal() { } } -},{"../enums.js":26,"../util":55}],36:[function(require,module,exports){ +},{"../enums.js":26,"../util":56}],37:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -10113,7 +10360,7 @@ function packet_marker() { module.exports = packet_marker; -},{}],37:[function(require,module,exports){ +},{}],38:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -10217,7 +10464,7 @@ module.exports = function packet_one_pass_signature() { } }; -},{"../enums.js":26,"../type/keyid.js":52}],38:[function(require,module,exports){ +},{"../enums.js":26,"../type/keyid.js":53}],39:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -10478,7 +10725,7 @@ module.exports = { } } -},{"../enums.js":26,"../util":55}],39:[function(require,module,exports){ +},{"../enums.js":26,"../util":56}],40:[function(require,module,exports){ var packetParser = require('./packet.js'), packets = require('./all_packets.js'), enums = require('../enums.js'); @@ -10579,7 +10826,7 @@ module.exports = function packetlist() { } -},{"../enums.js":26,"./all_packets.js":32,"./packet.js":38}],40:[function(require,module,exports){ +},{"../enums.js":26,"./all_packets.js":33,"./packet.js":39}],41:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -10721,7 +10968,7 @@ module.exports = function packet_public_key() { } -},{"../crypto":15,"../enums.js":26,"../type/keyid.js":52,"../type/mpi.js":53,"../util":55}],41:[function(require,module,exports){ +},{"../crypto":15,"../enums.js":26,"../type/keyid.js":53,"../type/mpi.js":54,"../util":56}],42:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -10899,7 +11146,7 @@ module.exports = function packet_public_key_encrypted_session_key() { } }; -},{"../crypto":15,"../enums.js":26,"../type/keyid.js":52,"../type/mpi.js":53,"../util":55}],42:[function(require,module,exports){ +},{"../crypto":15,"../enums.js":26,"../type/keyid.js":53,"../type/mpi.js":54,"../util":56}],43:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -10923,7 +11170,7 @@ module.exports = function public_subkey() { public_key.call(this); } -},{"./public_key.js":40}],43:[function(require,module,exports){ +},{"./public_key.js":41}],44:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -11187,7 +11434,7 @@ packet_secret_key.prototype = new publicKey; module.exports = packet_secret_key; -},{"../crypto":15,"../enums.js":26,"../type/mpi.js":53,"../type/s2k.js":54,"../util":55,"./public_key.js":40}],44:[function(require,module,exports){ +},{"../crypto":15,"../enums.js":26,"../type/mpi.js":54,"../type/s2k.js":55,"../util":56,"./public_key.js":41}],45:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -11211,7 +11458,7 @@ module.exports = function secret_subkey() { secret_key.call(this); } -},{"./secret_key.js":43}],45:[function(require,module,exports){ +},{"./secret_key.js":44}],46:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -11727,7 +11974,7 @@ module.exports = function packet_signature() { } } -},{"../crypto":15,"../enums.js":26,"../type/mpi.js":53,"../util":55,"./packet.js":38}],46:[function(require,module,exports){ +},{"../crypto":15,"../enums.js":26,"../type/mpi.js":54,"../util":56,"./packet.js":39}],47:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -11845,7 +12092,7 @@ module.exports = function packet_sym_encrypted_integrity_protected() { } }; -},{"../crypto":15,"../util":55}],47:[function(require,module,exports){ +},{"../crypto":15,"../util":56}],48:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -11984,7 +12231,7 @@ module.exports = function packet_sym_encrypted_session_key() { } }; -},{"../crypto":15,"../enums.js":26,"../type/s2k.js":54}],48:[function(require,module,exports){ +},{"../crypto":15,"../enums.js":26,"../type/s2k.js":55}],49:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -12054,12 +12301,12 @@ module.exports = function packet_symmetrically_encrypted() { } }; -},{"../crypto":15}],49:[function(require,module,exports){ +},{"../crypto":15}],50:[function(require,module,exports){ module.exports = function packet_trust() { }; -},{}],50:[function(require,module,exports){ +},{}],51:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -12117,7 +12364,7 @@ module.exports = function packet_user_attribute() { } }; -},{}],51:[function(require,module,exports){ +},{}],52:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -12175,7 +12422,7 @@ module.exports = function packet_userid() { } } -},{"../util":55}],52:[function(require,module,exports){ +},{"../util":56}],53:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -12234,7 +12481,7 @@ module.exports = function keyid() { } }; -},{"../util":55}],53:[function(require,module,exports){ +},{"../util":56}],54:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -12333,7 +12580,7 @@ module.exports = function mpi() { } } -},{"../crypto/public_key/jsbn.js":20,"../util":55}],54:[function(require,module,exports){ +},{"../crypto/public_key/jsbn.js":20,"../util":56}],55:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -12506,7 +12753,7 @@ module.exports = function s2k() { } } -},{"../crypto":15,"../enums.js":26,"../util":55}],55:[function(require,module,exports){ +},{"../crypto":15,"../enums.js":26,"../util":56}],56:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -12649,7 +12896,11 @@ var Util = function() { * @return {String} A native javascript string */ this.decode_utf8 = function(utf8) { - return decodeURIComponent(escape(utf8)); + try { + return decodeURIComponent(escape(utf8)); + } catch (e) { + return utf8; + } }; var str2bin = function(str, result) { @@ -12855,5 +13106,5 @@ var Util = function() { module.exports = new Util(); },{}]},{},[]) -//@ sourceMappingURL=data:application/json;base64,{"version":3,"file":"generated.js","sources":["/home/toberndo/dev/openpgpjs-devel/src/compression/jxg.js","/home/toberndo/dev/openpgpjs-devel/src/config/config.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cfb.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/aes.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/blowfish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/cast5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/des.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/twofish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/crypto.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/md5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/ripe-md.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/sha.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/pkcs1.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/dsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/elgamal.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/jsbn.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/rsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/random.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/signature.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/armor.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/base64.js","/home/toberndo/dev/openpgpjs-devel/src/enums.js","/home/toberndo/dev/openpgpjs-devel/src/index.js","/home/toberndo/dev/openpgpjs-devel/src/key.js","/home/toberndo/dev/openpgpjs-devel/src/message.js","/home/toberndo/dev/openpgpjs-devel/src/openpgp.js","/home/toberndo/dev/openpgpjs-devel/src/packet/all_packets.js","/home/toberndo/dev/openpgpjs-devel/src/packet/compressed.js","/home/toberndo/dev/openpgpjs-devel/src/packet/index.js","/home/toberndo/dev/openpgpjs-devel/src/packet/literal.js","/home/toberndo/dev/openpgpjs-devel/src/packet/marker.js","/home/toberndo/dev/openpgpjs-devel/src/packet/one_pass_signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packet.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packetlist.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_integrity_protected.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/symmetrically_encrypted.js","/home/toberndo/dev/openpgpjs-devel/src/packet/trust.js","/home/toberndo/dev/openpgpjs-devel/src/packet/user_attribute.js","/home/toberndo/dev/openpgpjs-devel/src/packet/userid.js","/home/toberndo/dev/openpgpjs-devel/src/type/keyid.js","/home/toberndo/dev/openpgpjs-devel/src/type/mpi.js","/home/toberndo/dev/openpgpjs-devel/src/type/s2k.js","/home/toberndo/dev/openpgpjs-devel/src/util/util.js"],"names":[],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3fA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzqDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpEA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourcesContent":["JXG = {\n  exists: (function(undefined) {\n    return function(v) {\n      return !(v === undefined || v === null);\n    }\n  })()\n};\nJXG.decompress = function(str) {\n  return unescape((new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(str))).unzip()[0][0]);\n};\n/*\n    Copyright 2008-2012\n        Matthias Ehmann,\n        Michael Gerhaeuser,\n        Carsten Miller,\n        Bianca Valentin,\n        Alfred Wassermann,\n        Peter Wilfahrt\n\n    This file is part of JSXGraph.\n    \n    Dual licensed under the Apache License Version 2.0, or LGPL Version 3 licenses.\n\n    You should have received a copy of the GNU Lesser General Public License\n    along with JSXCompressor.  If not, see <http://www.gnu.org/licenses/>.\n    \n    You should have received a copy of the Apache License along with JSXCompressor.  \n    If not, see <http://www.apache.org/licenses/>.\n\n*/\n\n/**\n * @class Util class\n * @classdesc Utilities for uncompressing and base64 decoding\n * Class for gunzipping, unzipping and base64 decoding of files.\n * It is used for reading GEONExT, Geogebra and Intergeo files.\n *\n * Only Huffman codes are decoded in gunzip.\n * The code is based on the source code for gunzip.c by Pasi Ojala \n * {@link http://www.cs.tut.fi/~albert/Dev/gunzip/gunzip.c}\n * {@link http://www.cs.tut.fi/~albert}\n */\nJXG.Util = {};\n\n/**\n * Unzip zip files\n */\nJXG.Util.Unzip = function(barray) {\n  var outputArr = [],\n    output = \"\",\n    debug = false,\n    gpflags,\n    files = 0,\n    unzipped = [],\n    crc,\n    buf32k = new Array(32768),\n    bIdx = 0,\n    modeZIP = false,\n\n    CRC, SIZE,\n\n    bitReverse = [\n        0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0,\n        0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0,\n        0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,\n        0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8,\n        0x04, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4,\n        0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,\n        0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec,\n        0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc,\n        0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,\n        0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2,\n        0x0a, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea,\n        0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,\n        0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6,\n        0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6,\n        0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,\n        0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe,\n        0x01, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1,\n        0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,\n        0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9,\n        0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9,\n        0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,\n        0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5,\n        0x0d, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed,\n        0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,\n        0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3,\n        0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3,\n        0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,\n        0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb,\n        0x07, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7,\n        0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,\n        0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef,\n        0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff\n    ],\n\n    cplens = [\n        3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31,\n        35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0\n    ],\n\n    cplext = [\n        0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2,\n        3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 99, 99\n    ],\n    /* 99==invalid */\n\n    cpdist = [\n        0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0007, 0x0009, 0x000d,\n        0x0011, 0x0019, 0x0021, 0x0031, 0x0041, 0x0061, 0x0081, 0x00c1,\n        0x0101, 0x0181, 0x0201, 0x0301, 0x0401, 0x0601, 0x0801, 0x0c01,\n        0x1001, 0x1801, 0x2001, 0x3001, 0x4001, 0x6001\n    ],\n\n    cpdext = [\n        0, 0, 0, 0, 1, 1, 2, 2,\n        3, 3, 4, 4, 5, 5, 6, 6,\n        7, 7, 8, 8, 9, 9, 10, 10,\n        11, 11, 12, 12, 13, 13\n    ],\n\n    border = [16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15],\n\n    bA = barray,\n\n    bytepos = 0,\n    bitpos = 0,\n    bb = 1,\n    bits = 0,\n\n    NAMEMAX = 256,\n\n    nameBuf = [],\n\n    fileout;\n\n  function readByte() {\n    bits += 8;\n    if (bytepos < bA.length) {\n      //if (debug)\n      //    document.write(bytepos+\": \"+bA[bytepos]+\"<br>\");\n      return bA[bytepos++];\n    } else\n      return -1;\n  };\n\n  function byteAlign() {\n    bb = 1;\n  };\n\n  function readBit() {\n    var carry;\n    bits++;\n    carry = (bb & 1);\n    bb >>= 1;\n    if (bb == 0) {\n      bb = readByte();\n      carry = (bb & 1);\n      bb = (bb >> 1) | 0x80;\n    }\n    return carry;\n  };\n\n  function readBits(a) {\n    var res = 0,\n      i = a;\n\n    while (i--) {\n      res = (res << 1) | readBit();\n    }\n    if (a) {\n      res = bitReverse[res] >> (8 - a);\n    }\n    return res;\n  };\n\n  function flushBuffer() {\n    //document.write('FLUSHBUFFER:'+buf32k);\n    bIdx = 0;\n  };\n\n  function addBuffer(a) {\n    SIZE++;\n    //CRC=updcrc(a,crc);\n    buf32k[bIdx++] = a;\n    outputArr.push(String.fromCharCode(a));\n    //output+=String.fromCharCode(a);\n    if (bIdx == 0x8000) {\n      //document.write('ADDBUFFER:'+buf32k);\n      bIdx = 0;\n    }\n  };\n\n  function HufNode() {\n    this.b0 = 0;\n    this.b1 = 0;\n    this.jump = null;\n    this.jumppos = -1;\n  };\n\n  var LITERALS = 288;\n\n  var literalTree = new Array(LITERALS);\n  var distanceTree = new Array(32);\n  var treepos = 0;\n  var Places = null;\n  var Places2 = null;\n\n  var impDistanceTree = new Array(64);\n  var impLengthTree = new Array(64);\n\n  var len = 0;\n  var fpos = new Array(17);\n  fpos[0] = 0;\n  var flens;\n  var fmax;\n\n  function IsPat() {\n    while (1) {\n      if (fpos[len] >= fmax)\n        return -1;\n      if (flens[fpos[len]] == len)\n        return fpos[len]++;\n      fpos[len]++;\n    }\n  };\n\n  function Rec() {\n    var curplace = Places[treepos];\n    var tmp;\n    if (debug)\n      document.write(\"<br>len:\" + len + \" treepos:\" + treepos);\n    if (len == 17) { //war 17\n      return -1;\n    }\n    treepos++;\n    len++;\n\n    tmp = IsPat();\n    if (debug)\n      document.write(\"<br>IsPat \" + tmp);\n    if (tmp >= 0) {\n      curplace.b0 = tmp; /* leaf cell for 0-bit */\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n    } else {\n      /* Not a Leaf cell */\n      curplace.b0 = 0x8000;\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n      if (Rec())\n        return -1;\n    }\n    tmp = IsPat();\n    if (tmp >= 0) {\n      curplace.b1 = tmp; /* leaf cell for 1-bit */\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = null; /* Just for the display routine */\n    } else {\n      /* Not a Leaf cell */\n      curplace.b1 = 0x8000;\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = Places[treepos];\n      curplace.jumppos = treepos;\n      if (Rec())\n        return -1;\n    }\n    len--;\n    return 0;\n  };\n\n  function CreateTree(currentTree, numval, lengths, show) {\n    var i;\n    /* Create the Huffman decode tree/table */\n    //document.write(\"<br>createtree<br>\");\n    if (debug)\n      document.write(\"currentTree \" + currentTree + \" numval \" + numval + \" lengths \" + lengths + \" show \" + show);\n    Places = currentTree;\n    treepos = 0;\n    flens = lengths;\n    fmax = numval;\n    for (i = 0; i < 17; i++)\n      fpos[i] = 0;\n    len = 0;\n    if (Rec()) {\n      //fprintf(stderr, \"invalid huffman tree\\n\");\n      if (debug)\n        alert(\"invalid huffman tree\\n\");\n      return -1;\n    }\n    if (debug) {\n      document.write('<br>Tree: ' + Places.length);\n      for (var a = 0; a < 32; a++) {\n        document.write(\"Places[\" + a + \"].b0=\" + Places[a].b0 + \"<br>\");\n        document.write(\"Places[\" + a + \"].b1=\" + Places[a].b1 + \"<br>\");\n      }\n    }\n\n    /*if(show) {\n            var tmp;\n            for(tmp=currentTree;tmp<Places;tmp++) {\n                fprintf(stdout, \"0x%03x  0x%03x (0x%04x)\",tmp-currentTree, tmp->jump?tmp->jump-currentTree:0,(tmp->jump?tmp->jump-currentTree:0)*6+0xcf0);\n                if(!(tmp.b0 & 0x8000)) {\n                    //fprintf(stdout, \"  0x%03x (%c)\", tmp->b0,(tmp->b0<256 && isprint(tmp->b0))?tmp->b0:'�');\n                }\n                if(!(tmp.b1 & 0x8000)) {\n                    if((tmp.b0 & 0x8000))\n                        fprintf(stdout, \"           \");\n                    fprintf(stdout, \"  0x%03x (%c)\", tmp->b1,(tmp->b1<256 && isprint(tmp->b1))?tmp->b1:'�');\n                }\n                fprintf(stdout, \"\\n\");\n            }\n        }*/\n    return 0;\n  };\n\n  function DecodeValue(currentTree) {\n    var len, i,\n      xtreepos = 0,\n      X = currentTree[xtreepos],\n      b;\n\n    /* decode one symbol of the data */\n    while (1) {\n      b = readBit();\n      if (debug)\n        document.write(\"b=\" + b);\n      if (b) {\n        if (!(X.b1 & 0x8000)) {\n          if (debug)\n            document.write(\"ret1\");\n          return X.b1; /* If leaf node, return data */\n        }\n        X = X.jump;\n        len = currentTree.length;\n        for (i = 0; i < len; i++) {\n          if (currentTree[i] === X) {\n            xtreepos = i;\n            break;\n          }\n        }\n        //xtreepos++;\n      } else {\n        if (!(X.b0 & 0x8000)) {\n          if (debug)\n            document.write(\"ret2\");\n          return X.b0; /* If leaf node, return data */\n        }\n        //X++; //??????????????????\n        xtreepos++;\n        X = currentTree[xtreepos];\n      }\n    }\n    if (debug)\n      document.write(\"ret3\");\n    return -1;\n  };\n\n  function DeflateLoop() {\n    var last, c, type, i, len;\n\n    do {\n      /*if((last = readBit())){\n            fprintf(errfp, \"Last Block: \");\n        } else {\n            fprintf(errfp, \"Not Last Block: \");\n        }*/\n      last = readBit();\n      type = readBits(2);\n      switch (type) {\n        case 0:\n          if (debug)\n            alert(\"Stored\\n\");\n          break;\n        case 1:\n          if (debug)\n            alert(\"Fixed Huffman codes\\n\");\n          break;\n        case 2:\n          if (debug)\n            alert(\"Dynamic Huffman codes\\n\");\n          break;\n        case 3:\n          if (debug)\n            alert(\"Reserved block type!!\\n\");\n          break;\n        default:\n          if (debug)\n            alert(\"Unexpected value %d!\\n\", type);\n          break;\n      }\n\n      if (type == 0) {\n        var blockLen, cSum;\n\n        // Stored \n        byteAlign();\n        blockLen = readByte();\n        blockLen |= (readByte() << 8);\n\n        cSum = readByte();\n        cSum |= (readByte() << 8);\n\n        if (((blockLen ^ ~cSum) & 0xffff)) {\n          document.write(\"BlockLen checksum mismatch\\n\");\n        }\n        while (blockLen--) {\n          c = readByte();\n          addBuffer(c);\n        }\n      } else if (type == 1) {\n        var j;\n\n        /* Fixed Huffman tables -- fixed decode routine */\n        while (1) {\n          /*\n                256    0000000        0\n                :   :     :\n                279    0010111        23\n                0   00110000    48\n                :    :      :\n                143    10111111    191\n                280 11000000    192\n                :    :      :\n                287 11000111    199\n                144    110010000    400\n                :    :       :\n                255    111111111    511\n    \n                Note the bit order!\n                */\n\n          j = (bitReverse[readBits(7)] >> 1);\n          if (j > 23) {\n            j = (j << 1) | readBit(); /* 48..255 */\n\n            if (j > 199) { /* 200..255 */\n              j -= 128; /*  72..127 */\n              j = (j << 1) | readBit(); /* 144..255 << */\n            } else { /*  48..199 */\n              j -= 48; /*   0..151 */\n              if (j > 143) {\n                j = j + 136; /* 280..287 << */\n                /*   0..143 << */\n              }\n            }\n          } else { /*   0..23 */\n            j += 256; /* 256..279 << */\n          }\n          if (j < 256) {\n            addBuffer(j);\n            //document.write(\"out:\"+String.fromCharCode(j));\n            /*fprintf(errfp, \"@%d %02x\\n\", SIZE, j);*/\n          } else if (j == 256) {\n            /* EOF */\n            break;\n          } else {\n            var len, dist;\n\n            j -= 256 + 1; /* bytes + EOF */\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = bitReverse[readBits(5)] >> 3;\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n\n            /*fprintf(errfp, \"@%d (l%02x,d%04x)\\n\", SIZE, len, dist);*/\n            for (j = 0; j < len; j++) {\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          }\n        } // while\n      } else if (type == 2) {\n        var j, n, literalCodes, distCodes, lenCodes;\n        var ll = new Array(288 + 32); // \"static\" just to preserve stack\n\n        // Dynamic Huffman tables \n\n        literalCodes = 257 + readBits(5);\n        distCodes = 1 + readBits(5);\n        lenCodes = 4 + readBits(4);\n        //document.write(\"<br>param: \"+literalCodes+\" \"+distCodes+\" \"+lenCodes+\"<br>\");\n        for (j = 0; j < 19; j++) {\n          ll[j] = 0;\n        }\n\n        // Get the decode tree code lengths\n\n        //document.write(\"<br>\");\n        for (j = 0; j < lenCodes; j++) {\n          ll[border[j]] = readBits(3);\n          //document.write(ll[border[j]]+\" \");\n        }\n        //fprintf(errfp, \"\\n\");\n        //document.write('<br>ll:'+ll);\n        len = distanceTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        if (CreateTree(distanceTree, 19, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug) {\n          document.write(\"<br>distanceTree\");\n          for (var a = 0; a < distanceTree.length; a++) {\n            document.write(\"<br>\" + distanceTree[a].b0 + \" \" + distanceTree[a].b1 + \" \" + distanceTree[a].jump + \" \" +\n              distanceTree[a].jumppos);\n            /*if (distanceTree[a].jumppos!=-1)\n                    \tdocument.write(\" \"+distanceTree[a].jump.b0+\" \"+distanceTree[a].jump.b1);\n                \t*/\n          }\n        }\n        //document.write('<BR>tree created');\n\n        //read in literal and distance code lengths\n        n = literalCodes + distCodes;\n        i = 0;\n        var z = -1;\n        if (debug)\n          document.write(\"<br>n=\" + n + \" bits: \" + bits + \"<br>\");\n        while (i < n) {\n          z++;\n          j = DecodeValue(distanceTree);\n          if (debug)\n            document.write(\"<br>\" + z + \" i:\" + i + \" decode: \" + j + \"    bits \" + bits + \"<br>\");\n          if (j < 16) { // length of code in bits (0..15)\n            ll[i++] = j;\n          } else if (j == 16) { // repeat last length 3 to 6 times \n            var l;\n            j = 3 + readBits(2);\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            l = i ? ll[i - 1] : 0;\n            while (j--) {\n              ll[i++] = l;\n            }\n          } else {\n            if (j == 17) { // 3 to 10 zero length codes\n              j = 3 + readBits(3);\n            } else { // j == 18: 11 to 138 zero length codes \n              j = 11 + readBits(7);\n            }\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            while (j--) {\n              ll[i++] = 0;\n            }\n          }\n        }\n        /*for(j=0; j<literalCodes+distCodes; j++) {\n                //fprintf(errfp, \"%d \", ll[j]);\n                if ((j&7)==7)\n                    fprintf(errfp, \"\\n\");\n            }\n            fprintf(errfp, \"\\n\");*/\n        // Can overwrite tree decode tree as it is not used anymore\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          literalTree[i] = new HufNode();\n        if (CreateTree(literalTree, literalCodes, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        var ll2 = new Array();\n        for (i = literalCodes; i < ll.length; i++) {\n          ll2[i - literalCodes] = ll[i];\n        }\n        if (CreateTree(distanceTree, distCodes, ll2, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug)\n          document.write(\"<br>literalTree\");\n        outer: while (1) {\n          j = DecodeValue(literalTree);\n          if (j >= 256) { // In C64: if carry set\n            var len, dist;\n            j -= 256;\n            if (j == 0) {\n              // EOF\n              break;\n            }\n            j--;\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = DecodeValue(distanceTree);\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n            while (len--) {\n              if (bIdx - dist < 0) {\n                break outer;\n              }\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          } else {\n            addBuffer(j);\n          }\n        }\n      }\n    } while (!last);\n    flushBuffer();\n\n    byteAlign();\n    return 0;\n  };\n\n  JXG.Util.Unzip.prototype.unzipFile = function(name) {\n    var i;\n    this.unzip();\n    //alert(unzipped[0][1]);\n    for (i = 0; i < unzipped.length; i++) {\n      if (unzipped[i][1] == name) {\n        return unzipped[i][0];\n      }\n    }\n\n  };\n\n  JXG.Util.Unzip.prototype.deflate = function() {\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    DeflateLoop();\n    if (debug)\n      alert(outputArr.join(''));\n    unzipped[files] = new Array(2);\n    unzipped[files][0] = outputArr.join('');\n    unzipped[files][1] = \"DEFLATE\";\n    files++;\n    return unzipped;\n  }\n\n  JXG.Util.Unzip.prototype.unzip = function() {\n    //convertToByteArray(input);\n    if (debug)\n      alert(bA);\n    /*for (i=0;i<bA.length*8;i++){\n\t\tdocument.write(readBit());\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}*/\n    /*for (i=0;i<bA.length;i++){\n\t\tdocument.write(readByte()+\" \");\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}\n\tfor (i=0;i<bA.length;i++){\n\t\tdocument.write(bA[i]+\" \");\n\t\tif ((i+1)%16==0)\n\t\t\tdocument.write(\"<br>\");\n\t}\t\n\t*/\n    //alert(bA);\n    nextFile();\n    return unzipped;\n  };\n\n  function nextFile() {\n    if (debug)\n      alert(\"NEXTFILE\");\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    tmp[0] = readByte();\n    tmp[1] = readByte();\n    if (debug)\n      alert(\"type: \" + tmp[0] + \" \" + tmp[1]);\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"da\", 16)) { //GZIP\n      if (debug)\n        alert(\"GEONExT-GZIP\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"geonext.gxt\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"9c\", 16)) { //ZLIB\n      if (debug)\n        alert(\"ZLIB\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"ZLIB\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"1f\", 16) && tmp[1] == parseInt(\"8b\", 16)) { //GZIP\n      if (debug)\n        alert(\"GZIP\");\n      //DeflateLoop();\n      skipdir();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"file\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"50\", 16) && tmp[1] == parseInt(\"4b\", 16)) { //ZIP\n      modeZIP = true;\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n      if (tmp[2] == parseInt(\"3\", 16) && tmp[3] == parseInt(\"4\", 16)) {\n        //MODE_ZIP\n        tmp[0] = readByte();\n        tmp[1] = readByte();\n        if (debug)\n          alert(\"ZIP-Version: \" + tmp[1] + \" \" + tmp[0] / 10 + \".\" + tmp[0] % 10);\n\n        gpflags = readByte();\n        gpflags |= (readByte() << 8);\n        if (debug)\n          alert(\"gpflags: \" + gpflags);\n\n        var method = readByte();\n        method |= (readByte() << 8);\n        if (debug)\n          alert(\"method: \" + method);\n\n        readByte();\n        readByte();\n        readByte();\n        readByte();\n\n        var crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n\n        var compSize = readByte();\n        compSize |= (readByte() << 8);\n        compSize |= (readByte() << 16);\n        compSize |= (readByte() << 24);\n\n        var size = readByte();\n        size |= (readByte() << 8);\n        size |= (readByte() << 16);\n        size |= (readByte() << 24);\n\n        if (debug)\n          alert(\"local CRC: \" + crc + \"\\nlocal Size: \" + size + \"\\nlocal CompSize: \" + compSize);\n\n        var filelen = readByte();\n        filelen |= (readByte() << 8);\n\n        var extralen = readByte();\n        extralen |= (readByte() << 8);\n\n        if (debug)\n          alert(\"filelen \" + filelen);\n        i = 0;\n        nameBuf = [];\n        while (filelen--) {\n          var c = readByte();\n          if (c == \"/\" | c == \":\") {\n            i = 0;\n          } else if (i < NAMEMAX - 1)\n            nameBuf[i++] = String.fromCharCode(c);\n        }\n        if (debug)\n          alert(\"nameBuf: \" + nameBuf);\n\n        //nameBuf[i] = \"\\0\";\n        if (!fileout)\n          fileout = nameBuf;\n\n        var i = 0;\n        while (i < extralen) {\n          c = readByte();\n          i++;\n        }\n\n        CRC = 0xffffffff;\n        SIZE = 0;\n\n        if (size = 0 && fileOut.charAt(fileout.length - 1) == \"/\") {\n          //skipdir\n          if (debug)\n            alert(\"skipdir\");\n        }\n        if (method == 8) {\n          DeflateLoop();\n          if (debug)\n            alert(outputArr.join(''));\n          unzipped[files] = new Array(2);\n          unzipped[files][0] = outputArr.join('');\n          unzipped[files][1] = nameBuf.join('');\n          files++;\n          //return outputArr.join('');\n        }\n        skipdir();\n      }\n    }\n  };\n\n  function skipdir() {\n    var crc,\n      tmp = [],\n      compSize, size, os, i, c;\n\n    if ((gpflags & 8)) {\n      tmp[0] = readByte();\n      tmp[1] = readByte();\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n\n      if (tmp[0] == parseInt(\"50\", 16) &&\n        tmp[1] == parseInt(\"4b\", 16) &&\n        tmp[2] == parseInt(\"07\", 16) &&\n        tmp[3] == parseInt(\"08\", 16)) {\n        crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n      } else {\n        crc = tmp[0] | (tmp[1] << 8) | (tmp[2] << 16) | (tmp[3] << 24);\n      }\n\n      compSize = readByte();\n      compSize |= (readByte() << 8);\n      compSize |= (readByte() << 16);\n      compSize |= (readByte() << 24);\n\n      size = readByte();\n      size |= (readByte() << 8);\n      size |= (readByte() << 16);\n      size |= (readByte() << 24);\n\n      if (debug)\n        alert(\"CRC:\");\n    }\n\n    if (modeZIP)\n      nextFile();\n\n    tmp[0] = readByte();\n    if (tmp[0] != 8) {\n      if (debug)\n        alert(\"Unknown compression method!\");\n      return 0;\n    }\n\n    gpflags = readByte();\n    if (debug) {\n      if ((gpflags & ~(parseInt(\"1f\", 16))))\n        alert(\"Unknown flags set!\");\n    }\n\n    readByte();\n    readByte();\n    readByte();\n    readByte();\n\n    readByte();\n    os = readByte();\n\n    if ((gpflags & 4)) {\n      tmp[0] = readByte();\n      tmp[2] = readByte();\n      len = tmp[0] + 256 * tmp[1];\n      if (debug)\n        alert(\"Extra field size: \" + len);\n      for (i = 0; i < len; i++)\n        readByte();\n    }\n\n    if ((gpflags & 8)) {\n      i = 0;\n      nameBuf = [];\n      while (c = readByte()) {\n        if (c == \"7\" || c == \":\")\n          i = 0;\n        if (i < NAMEMAX - 1)\n          nameBuf[i++] = c;\n      }\n      //nameBuf[i] = \"\\0\";\n      if (debug)\n        alert(\"original file name: \" + nameBuf);\n    }\n\n    if ((gpflags & 16)) {\n      while (c = readByte()) {\n        //FILE COMMENT\n      }\n    }\n\n    if ((gpflags & 2)) {\n      readByte();\n      readByte();\n    }\n\n    DeflateLoop();\n\n    crc = readByte();\n    crc |= (readByte() << 8);\n    crc |= (readByte() << 16);\n    crc |= (readByte() << 24);\n\n    size = readByte();\n    size |= (readByte() << 8);\n    size |= (readByte() << 16);\n    size |= (readByte() << 24);\n\n    if (modeZIP)\n      nextFile();\n\n  };\n\n};\n\n/**\n *  Base64 encoding / decoding\n *  {@link http://www.webtoolkit.info/}\n */\nJXG.Util.Base64 = {\n\n  // private property\n  _keyStr: \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=\",\n\n  // public method for encoding\n  encode: function(input) {\n    var output = [],\n      chr1, chr2, chr3, enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = JXG.Util.Base64._utf8_encode(input);\n\n    while (i < input.length) {\n\n      chr1 = input.charCodeAt(i++);\n      chr2 = input.charCodeAt(i++);\n      chr3 = input.charCodeAt(i++);\n\n      enc1 = chr1 >> 2;\n      enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);\n      enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);\n      enc4 = chr3 & 63;\n\n      if (isNaN(chr2)) {\n        enc3 = enc4 = 64;\n      } else if (isNaN(chr3)) {\n        enc4 = 64;\n      }\n\n      output.push([this._keyStr.charAt(enc1),\n          this._keyStr.charAt(enc2),\n          this._keyStr.charAt(enc3),\n          this._keyStr.charAt(enc4)\n      ].join(''));\n    }\n\n    return output.join('');\n  },\n\n  // public method for decoding\n  decode: function(input, utf8) {\n    var output = [],\n      chr1, chr2, chr3,\n      enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = input.replace(/[^A-Za-z0-9\\+\\/\\=]/g, \"\");\n\n    while (i < input.length) {\n\n      enc1 = this._keyStr.indexOf(input.charAt(i++));\n      enc2 = this._keyStr.indexOf(input.charAt(i++));\n      enc3 = this._keyStr.indexOf(input.charAt(i++));\n      enc4 = this._keyStr.indexOf(input.charAt(i++));\n\n      chr1 = (enc1 << 2) | (enc2 >> 4);\n      chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);\n      chr3 = ((enc3 & 3) << 6) | enc4;\n\n      output.push(String.fromCharCode(chr1));\n\n      if (enc3 != 64) {\n        output.push(String.fromCharCode(chr2));\n      }\n      if (enc4 != 64) {\n        output.push(String.fromCharCode(chr3));\n      }\n    }\n\n    output = output.join('');\n\n    if (utf8) {\n      output = JXG.Util.Base64._utf8_decode(output);\n    }\n    return output;\n\n  },\n\n  // private method for UTF-8 encoding\n  _utf8_encode: function(string) {\n    string = string.replace(/\\r\\n/g, \"\\n\");\n    var utftext = \"\";\n\n    for (var n = 0; n < string.length; n++) {\n\n      var c = string.charCodeAt(n);\n\n      if (c < 128) {\n        utftext += String.fromCharCode(c);\n      } else if ((c > 127) && (c < 2048)) {\n        utftext += String.fromCharCode((c >> 6) | 192);\n        utftext += String.fromCharCode((c & 63) | 128);\n      } else {\n        utftext += String.fromCharCode((c >> 12) | 224);\n        utftext += String.fromCharCode(((c >> 6) & 63) | 128);\n        utftext += String.fromCharCode((c & 63) | 128);\n      }\n\n    }\n\n    return utftext;\n  },\n\n  // private method for UTF-8 decoding\n  _utf8_decode: function(utftext) {\n    var string = [],\n      i = 0,\n      c = 0,\n      c2 = 0,\n      c3 = 0;\n\n    while (i < utftext.length) {\n      c = utftext.charCodeAt(i);\n      if (c < 128) {\n        string.push(String.fromCharCode(c));\n        i++;\n      } else if ((c > 191) && (c < 224)) {\n        c2 = utftext.charCodeAt(i + 1);\n        string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n        i += 2;\n      } else {\n        c2 = utftext.charCodeAt(i + 1);\n        c3 = utftext.charCodeAt(i + 2);\n        string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n        i += 3;\n      }\n    }\n    return string.join('');\n  },\n\n  _destrip: function(stripped, wrap) {\n    var lines = [],\n      lineno, i,\n      destripped = [];\n\n    if (wrap == null)\n      wrap = 76;\n\n    stripped.replace(/ /g, \"\");\n    lineno = stripped.length / wrap;\n    for (i = 0; i < lineno; i++)\n      lines[i] = stripped.substr(i * wrap, wrap);\n    if (lineno != stripped.length / wrap)\n      lines[lines.length] = stripped.substr(lineno * wrap, stripped.length - (lineno * wrap));\n\n    for (i = 0; i < lines.length; i++)\n      destripped.push(lines[i]);\n    return destripped.join('\\n');\n  },\n\n  decodeAsArray: function(input) {\n    var dec = this.decode(input),\n      ar = [],\n      i;\n    for (i = 0; i < dec.length; i++) {\n      ar[i] = dec.charCodeAt(i);\n    }\n    return ar;\n  },\n\n  decodeGEONExT: function(input) {\n    return decodeAsArray(destrip(input), false);\n  }\n};\n\n/**\n * @private\n */\nJXG.Util.asciiCharCodeAt = function(str, i) {\n  var c = str.charCodeAt(i);\n  if (c > 255) {\n    switch (c) {\n      case 8364:\n        c = 128;\n        break;\n      case 8218:\n        c = 130;\n        break;\n      case 402:\n        c = 131;\n        break;\n      case 8222:\n        c = 132;\n        break;\n      case 8230:\n        c = 133;\n        break;\n      case 8224:\n        c = 134;\n        break;\n      case 8225:\n        c = 135;\n        break;\n      case 710:\n        c = 136;\n        break;\n      case 8240:\n        c = 137;\n        break;\n      case 352:\n        c = 138;\n        break;\n      case 8249:\n        c = 139;\n        break;\n      case 338:\n        c = 140;\n        break;\n      case 381:\n        c = 142;\n        break;\n      case 8216:\n        c = 145;\n        break;\n      case 8217:\n        c = 146;\n        break;\n      case 8220:\n        c = 147;\n        break;\n      case 8221:\n        c = 148;\n        break;\n      case 8226:\n        c = 149;\n        break;\n      case 8211:\n        c = 150;\n        break;\n      case 8212:\n        c = 151;\n        break;\n      case 732:\n        c = 152;\n        break;\n      case 8482:\n        c = 153;\n        break;\n      case 353:\n        c = 154;\n        break;\n      case 8250:\n        c = 155;\n        break;\n      case 339:\n        c = 156;\n        break;\n      case 382:\n        c = 158;\n        break;\n      case 376:\n        c = 159;\n        break;\n      default:\n        break;\n    }\n  }\n  return c;\n};\n\n/**\n * Decoding string into utf-8\n * @param {String} string to decode\n * @return {String} utf8 decoded string\n */\nJXG.Util.utf8Decode = function(utftext) {\n  var string = [];\n  var i = 0;\n  var c = 0,\n    c1 = 0,\n    c2 = 0,\n    c3;\n  if (!JXG.exists(utftext)) return '';\n\n  while (i < utftext.length) {\n    c = utftext.charCodeAt(i);\n\n    if (c < 128) {\n      string.push(String.fromCharCode(c));\n      i++;\n    } else if ((c > 191) && (c < 224)) {\n      c2 = utftext.charCodeAt(i + 1);\n      string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n      i += 2;\n    } else {\n      c2 = utftext.charCodeAt(i + 1);\n      c3 = utftext.charCodeAt(i + 2);\n      string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n      i += 3;\n    }\n  };\n  return string.join('');\n};\n\n/**\n * Generate a random uuid.\n * http://www.broofa.com\n * mailto:robert@broofa.com\n *\n * Copyright (c) 2010 Robert Kieffer\n * Dual licensed under the MIT and GPL licenses.\n *\n * EXAMPLES:\n *   >>> Math.uuid()\n *   \"92329D39-6F5C-4520-ABFC-AAB64544E172\"\n */\nJXG.Util.genUUID = function() {\n  // Private array of chars to use\n  var chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''),\n    uuid = new Array(36),\n    rnd = 0,\n    r;\n\n  for (var i = 0; i < 36; i++) {\n    if (i == 8 || i == 13 || i == 18 || i == 23) {\n      uuid[i] = '-';\n    } else if (i == 14) {\n      uuid[i] = '4';\n    } else {\n      if (rnd <= 0x02) rnd = 0x2000000 + (Math.random() * 0x1000000) | 0;\n      r = rnd & 0xf;\n      rnd = rnd >> 4;\n      uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r];\n    }\n  }\n\n  return uuid.join('');\n};\n\n\nmodule.exports = JXG;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js');\n\n/**\n *\n * This object contains configuration values and implements\n * storing and retrieving configuration them from HTML5 local storage.\n *\n * This object can be accessed after calling openpgp.init()\n * using openpgp.config\n * Stored config parameters can be accessed using\n * openpgp.config.config\n * @class\n * @classdesc Implementation of the GPG4Browsers config object\n */\nvar config = function() {\n  /**\n   * @property {Integer} prefer_hash_algorithm\n   * @property {Integer} encryption_cipher\n   * @property {Integer} compression\n   * @property {Boolean} show_version\n   * @property {Boolean} show_comment\n   * @property {Boolean} integrity_protect\n   * @property {String} keyserver\n   */\n  this.prefer_hash_algorithm = enums.hash.sha256;\n  this.encryption_cipher = enums.symmetric.aes256;\n  this.compression = enums.compression.zip;\n  this.show_version = true;\n  this.show_comment = true;\n  this.integrity_protect = true;\n  this.keyserver = \"keyserver.linux.it\"; // \"pgp.mit.edu:11371\"\n\n  this.versionstring = \"OpenPGP.js VERSION\";\n  this.commentstring = \"http://openpgpjs.org\";\n\n  /**\n   * If enabled, debug messages will be printed\n   */\n  this.debug = false;\n\n};\n\nmodule.exports = new config();\n","// Modified by Recurity Labs GmbH \n\n// modified version of http://www.hanewin.net/encrypt/PGdecode.js:\n\n/* OpenPGP encryption using RSA/AES\n * Copyright 2005-2006 Herbert Hanewinkel, www.haneWIN.de\n * version 2.0, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../util'),\n  cipher = require('./cipher');\n\nmodule.exports = {\n\n  /**\n   * An array of bytes, that is integers with values from 0 to 255\n   * @typedef {(Array|Uint8Array)} openpgp_byte_array\n   */\n\n  /**\n   * Block cipher function\n   * @callback openpgp_cipher_block_fn\n   * @param {openpgp_byte_array} block A block to perform operations on\n   * @param {openpgp_byte_array} key to use in encryption/decryption\n   * @return {openpgp_byte_array} Encrypted/decrypted block\n   */\n\n\n  // --------------------------------------\n  /**\n   * This function encrypts a given with the specified prefixrandom \n   * using the specified blockcipher to encrypt a message\n   * @param {String} prefixrandom random bytes of block_size length provided \n   *  as a string to be used in prefixing the data\n   * @param {openpgp_cipher_block_fn} blockcipherfn the algorithm encrypt function to encrypt\n   *  data in one block_size encryption. \n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext data to be encrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to encrypt the data. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Encryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the encrypted data\n   */\n  encrypt: function(prefixrandom, cipherfn, plaintext, key, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var FR = new Array(block_size);\n    var FRE = new Array(block_size);\n\n    prefixrandom = prefixrandom + prefixrandom.charAt(block_size - 2) + prefixrandom.charAt(block_size - 1);\n    util.print_debug(\"prefixrandom:\" + util.hexstrdump(prefixrandom));\n    var ciphertext = \"\";\n    // 1.  The feedback register (FR) is set to the IV, which is all zeros.\n    for (var i = 0; i < block_size; i++) FR[i] = 0;\n\n    // 2.  FR is encrypted to produce FRE (FR Encrypted).  This is the\n    //     encryption of an all-zero value.\n    FRE = cipherfn.encrypt(FR);\n    // 3.  FRE is xored with the first BS octets of random data prefixed to\n    //     the plaintext to produce C[1] through C[BS], the first BS octets\n    //     of ciphertext.\n    for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ prefixrandom.charCodeAt(i));\n\n    // 4.  FR is loaded with C[1] through C[BS].\n    for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n\n    // 5.  FR is encrypted to produce FRE, the encryption of the first BS\n    // \t   octets of ciphertext.\n    FRE = cipherfn.encrypt(FR);\n\n    // 6.  The left two octets of FRE get xored with the next two octets of\n    //     data that were prefixed to the plaintext.  This produces C[BS+1]\n    //     and C[BS+2], the next two octets of ciphertext.\n    ciphertext += String.fromCharCode(FRE[0] ^ prefixrandom.charCodeAt(block_size));\n    ciphertext += String.fromCharCode(FRE[1] ^ prefixrandom.charCodeAt(block_size + 1));\n\n    if (resync) {\n      // 7.  (The resync step) FR is loaded with C3-C10.\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i + 2);\n    } else {\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n    }\n    // 8.  FR is encrypted to produce FRE.\n    FRE = cipherfn.encrypt(FR, key);\n\n    if (resync) {\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 0; i < block_size; i++)\n        ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      for (n = block_size + 2; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(n + i);\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        // next 8 octets of ciphertext.  These are loaded into FR and the\n        // process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt((n - 2) +\n            i));\n      }\n    } else {\n      plaintext = \"  \" + plaintext;\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 2; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      var tempCiphertext = ciphertext.substring(0, 2 * block_size).split('');\n      var tempCiphertextString = ciphertext.substring(block_size);\n      for (n = block_size; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = tempCiphertextString.charCodeAt(i);\n        tempCiphertextString = '';\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        //     next 8 octets of ciphertext.  These are loaded into FR and the\n        //     process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) {\n          tempCiphertext.push(String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i)));\n          tempCiphertextString += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i));\n        }\n      }\n      ciphertext = tempCiphertext.join('');\n\n    }\n\n    ciphertext = ciphertext.substring(0, plaintext.length + 2 + block_size);\n\n    return ciphertext;\n  },\n\n  /**\n   * Decrypts the prefixed data for the Modification Detection Code (MDC) computation\n   * @param {openpgp_block_cipher_fn} cipherfn.encrypt Cipher function to use\n   * @param {Integer} block_size Blocksize of the algorithm\n   * @param {openpgp_byte_array} key The key for encryption\n   * @param {String} ciphertext The encrypted data\n   * @return {String} plaintext Data of D(ciphertext) with blocksize length +2\n   */\n  mdc: function(cipherfn, key, ciphertext) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i;\n\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock);\n\n    return util.bin2str(iblock) +\n      String.fromCharCode(ablock[0] ^ ciphertext.charCodeAt(block_size)) +\n      String.fromCharCode(ablock[1] ^ ciphertext.charCodeAt(block_size + 1));\n  },\n  /**\n   * This function decrypts a given plaintext using the specified\n   * blockcipher to decrypt a message\n   * @param {openpgp_cipher_block_fn} blockcipherfn The algorithm _encrypt_ function to encrypt\n   *  data in one block_size encryption.\n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext ciphertext to be decrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to decrypt the ciphertext. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Decryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the plaintext data\n   */\n\n  decrypt: function(cipherfn, key, ciphertext, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i, n = '';\n    var text = [];\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock, key);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock, key);\n\n    // test check octets\n    if (iblock[block_size - 2] != (ablock[0] ^ ciphertext.charCodeAt(block_size)) || iblock[block_size - 1] != (ablock[\n      1] ^ ciphertext.charCodeAt(block_size + 1))) {\n      throw new Error('Invalid data.');\n    }\n\n    /*  RFC4880: Tag 18 and Resync:\n\t\t *  [...] Unlike the Symmetrically Encrypted Data Packet, no\n\t\t *  special CFB resynchronization is done after encrypting this prefix\n\t\t *  data.  See \"OpenPGP CFB Mode\" below for more details.\n\n\t\t */\n\n    if (resync) {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i + 2);\n      for (n = block_size + 2; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    } else {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i);\n      for (n = block_size; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    }\n\n    var n = resync ? 0 : 2;\n\n    text = text.join('');\n\n    text = text.substring(n, ciphertext.length - block_size - 2 + n);\n\n\n    return text;\n  },\n\n\n  normalEncrypt: function(cipherfn, key, plaintext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blocki = \"\";\n    var blockc = \"\";\n    var pos = 0;\n    var cyphertext = [];\n    var tempBlock = [];\n    blockc = iv.substring(0, block_size);\n    while (plaintext.length > block_size * pos) {\n      var encblock = cipherfn.encrypt(util.str2bin(blockc));\n      blocki = plaintext.substring((pos * block_size), (pos * block_size) + block_size);\n      for (var i = 0; i < blocki.length; i++)\n        tempBlock.push(String.fromCharCode(blocki.charCodeAt(i) ^ encblock[i]));\n      blockc = tempBlock.join('');\n      tempBlock = [];\n      cyphertext.push(blockc);\n      pos++;\n    }\n    return cyphertext.join('');\n  },\n\n  normalDecrypt: function(cipherfn, key, ciphertext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blockp = \"\";\n    var pos = 0;\n    var plaintext = [];\n    var offset = 0;\n    if (iv == null)\n      for (var i = 0; i < block_size; i++) blockp += String.fromCharCode(0);\n    else\n      blockp = iv.substring(0, block_size);\n    while (ciphertext.length > (block_size * pos)) {\n      var decblock = cipherfn.encrypt(util.str2bin(blockp));\n      blockp = ciphertext.substring((pos * (block_size)) + offset, (pos * (block_size)) + (block_size) + offset);\n      for (var i = 0; i < blockp.length; i++) {\n        plaintext.push(String.fromCharCode(blockp.charCodeAt(i) ^ decblock[i]));\n      }\n      pos++;\n    }\n\n    return plaintext.join('');\n  }\n}\n","/* Rijndael (AES) Encryption\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.1, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../../util');\n\n// The round constants used in subkey expansion\nvar Rcon = [\n    0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36, 0x6c, 0xd8,\n    0xab, 0x4d, 0x9a, 0x2f, 0x5e, 0xbc, 0x63, 0xc6, 0x97, 0x35, 0x6a, 0xd4,\n    0xb3, 0x7d, 0xfa, 0xef, 0xc5, 0x91\n];\n\n// Precomputed lookup table for the SBox\nvar S = [\n    99, 124, 119, 123, 242, 107, 111, 197, 48, 1, 103, 43, 254, 215, 171,\n    118, 202, 130, 201, 125, 250, 89, 71, 240, 173, 212, 162, 175, 156, 164,\n    114, 192, 183, 253, 147, 38, 54, 63, 247, 204, 52, 165, 229, 241, 113,\n    216, 49, 21, 4, 199, 35, 195, 24, 150, 5, 154, 7, 18, 128, 226,\n    235, 39, 178, 117, 9, 131, 44, 26, 27, 110, 90, 160, 82, 59, 214,\n    179, 41, 227, 47, 132, 83, 209, 0, 237, 32, 252, 177, 91, 106, 203,\n    190, 57, 74, 76, 88, 207, 208, 239, 170, 251, 67, 77, 51, 133, 69,\n    249, 2, 127, 80, 60, 159, 168, 81, 163, 64, 143, 146, 157, 56, 245,\n    188, 182, 218, 33, 16, 255, 243, 210, 205, 12, 19, 236, 95, 151, 68,\n    23, 196, 167, 126, 61, 100, 93, 25, 115, 96, 129, 79, 220, 34, 42,\n    144, 136, 70, 238, 184, 20, 222, 94, 11, 219, 224, 50, 58, 10, 73,\n    6, 36, 92, 194, 211, 172, 98, 145, 149, 228, 121, 231, 200, 55, 109,\n    141, 213, 78, 169, 108, 86, 244, 234, 101, 122, 174, 8, 186, 120, 37,\n    46, 28, 166, 180, 198, 232, 221, 116, 31, 75, 189, 139, 138, 112, 62,\n    181, 102, 72, 3, 246, 14, 97, 53, 87, 185, 134, 193, 29, 158, 225,\n    248, 152, 17, 105, 217, 142, 148, 155, 30, 135, 233, 206, 85, 40, 223,\n    140, 161, 137, 13, 191, 230, 66, 104, 65, 153, 45, 15, 176, 84, 187,\n    22\n];\n\nvar T1 = [\n    0xa56363c6, 0x847c7cf8, 0x997777ee, 0x8d7b7bf6,\n    0x0df2f2ff, 0xbd6b6bd6, 0xb16f6fde, 0x54c5c591,\n    0x50303060, 0x03010102, 0xa96767ce, 0x7d2b2b56,\n    0x19fefee7, 0x62d7d7b5, 0xe6abab4d, 0x9a7676ec,\n    0x45caca8f, 0x9d82821f, 0x40c9c989, 0x877d7dfa,\n    0x15fafaef, 0xeb5959b2, 0xc947478e, 0x0bf0f0fb,\n    0xecadad41, 0x67d4d4b3, 0xfda2a25f, 0xeaafaf45,\n    0xbf9c9c23, 0xf7a4a453, 0x967272e4, 0x5bc0c09b,\n    0xc2b7b775, 0x1cfdfde1, 0xae93933d, 0x6a26264c,\n    0x5a36366c, 0x413f3f7e, 0x02f7f7f5, 0x4fcccc83,\n    0x5c343468, 0xf4a5a551, 0x34e5e5d1, 0x08f1f1f9,\n    0x937171e2, 0x73d8d8ab, 0x53313162, 0x3f15152a,\n    0x0c040408, 0x52c7c795, 0x65232346, 0x5ec3c39d,\n    0x28181830, 0xa1969637, 0x0f05050a, 0xb59a9a2f,\n    0x0907070e, 0x36121224, 0x9b80801b, 0x3de2e2df,\n    0x26ebebcd, 0x6927274e, 0xcdb2b27f, 0x9f7575ea,\n    0x1b090912, 0x9e83831d, 0x742c2c58, 0x2e1a1a34,\n    0x2d1b1b36, 0xb26e6edc, 0xee5a5ab4, 0xfba0a05b,\n    0xf65252a4, 0x4d3b3b76, 0x61d6d6b7, 0xceb3b37d,\n    0x7b292952, 0x3ee3e3dd, 0x712f2f5e, 0x97848413,\n    0xf55353a6, 0x68d1d1b9, 0x00000000, 0x2cededc1,\n    0x60202040, 0x1ffcfce3, 0xc8b1b179, 0xed5b5bb6,\n    0xbe6a6ad4, 0x46cbcb8d, 0xd9bebe67, 0x4b393972,\n    0xde4a4a94, 0xd44c4c98, 0xe85858b0, 0x4acfcf85,\n    0x6bd0d0bb, 0x2aefefc5, 0xe5aaaa4f, 0x16fbfbed,\n    0xc5434386, 0xd74d4d9a, 0x55333366, 0x94858511,\n    0xcf45458a, 0x10f9f9e9, 0x06020204, 0x817f7ffe,\n    0xf05050a0, 0x443c3c78, 0xba9f9f25, 0xe3a8a84b,\n    0xf35151a2, 0xfea3a35d, 0xc0404080, 0x8a8f8f05,\n    0xad92923f, 0xbc9d9d21, 0x48383870, 0x04f5f5f1,\n    0xdfbcbc63, 0xc1b6b677, 0x75dadaaf, 0x63212142,\n    0x30101020, 0x1affffe5, 0x0ef3f3fd, 0x6dd2d2bf,\n    0x4ccdcd81, 0x140c0c18, 0x35131326, 0x2fececc3,\n    0xe15f5fbe, 0xa2979735, 0xcc444488, 0x3917172e,\n    0x57c4c493, 0xf2a7a755, 0x827e7efc, 0x473d3d7a,\n    0xac6464c8, 0xe75d5dba, 0x2b191932, 0x957373e6,\n    0xa06060c0, 0x98818119, 0xd14f4f9e, 0x7fdcdca3,\n    0x66222244, 0x7e2a2a54, 0xab90903b, 0x8388880b,\n    0xca46468c, 0x29eeeec7, 0xd3b8b86b, 0x3c141428,\n    0x79dedea7, 0xe25e5ebc, 0x1d0b0b16, 0x76dbdbad,\n    0x3be0e0db, 0x56323264, 0x4e3a3a74, 0x1e0a0a14,\n    0xdb494992, 0x0a06060c, 0x6c242448, 0xe45c5cb8,\n    0x5dc2c29f, 0x6ed3d3bd, 0xefacac43, 0xa66262c4,\n    0xa8919139, 0xa4959531, 0x37e4e4d3, 0x8b7979f2,\n    0x32e7e7d5, 0x43c8c88b, 0x5937376e, 0xb76d6dda,\n    0x8c8d8d01, 0x64d5d5b1, 0xd24e4e9c, 0xe0a9a949,\n    0xb46c6cd8, 0xfa5656ac, 0x07f4f4f3, 0x25eaeacf,\n    0xaf6565ca, 0x8e7a7af4, 0xe9aeae47, 0x18080810,\n    0xd5baba6f, 0x887878f0, 0x6f25254a, 0x722e2e5c,\n    0x241c1c38, 0xf1a6a657, 0xc7b4b473, 0x51c6c697,\n    0x23e8e8cb, 0x7cdddda1, 0x9c7474e8, 0x211f1f3e,\n    0xdd4b4b96, 0xdcbdbd61, 0x868b8b0d, 0x858a8a0f,\n    0x907070e0, 0x423e3e7c, 0xc4b5b571, 0xaa6666cc,\n    0xd8484890, 0x05030306, 0x01f6f6f7, 0x120e0e1c,\n    0xa36161c2, 0x5f35356a, 0xf95757ae, 0xd0b9b969,\n    0x91868617, 0x58c1c199, 0x271d1d3a, 0xb99e9e27,\n    0x38e1e1d9, 0x13f8f8eb, 0xb398982b, 0x33111122,\n    0xbb6969d2, 0x70d9d9a9, 0x898e8e07, 0xa7949433,\n    0xb69b9b2d, 0x221e1e3c, 0x92878715, 0x20e9e9c9,\n    0x49cece87, 0xff5555aa, 0x78282850, 0x7adfdfa5,\n    0x8f8c8c03, 0xf8a1a159, 0x80898909, 0x170d0d1a,\n    0xdabfbf65, 0x31e6e6d7, 0xc6424284, 0xb86868d0,\n    0xc3414182, 0xb0999929, 0x772d2d5a, 0x110f0f1e,\n    0xcbb0b07b, 0xfc5454a8, 0xd6bbbb6d, 0x3a16162c\n];\n\nvar T2 = [\n    0x6363c6a5, 0x7c7cf884, 0x7777ee99, 0x7b7bf68d,\n    0xf2f2ff0d, 0x6b6bd6bd, 0x6f6fdeb1, 0xc5c59154,\n    0x30306050, 0x01010203, 0x6767cea9, 0x2b2b567d,\n    0xfefee719, 0xd7d7b562, 0xabab4de6, 0x7676ec9a,\n    0xcaca8f45, 0x82821f9d, 0xc9c98940, 0x7d7dfa87,\n    0xfafaef15, 0x5959b2eb, 0x47478ec9, 0xf0f0fb0b,\n    0xadad41ec, 0xd4d4b367, 0xa2a25ffd, 0xafaf45ea,\n    0x9c9c23bf, 0xa4a453f7, 0x7272e496, 0xc0c09b5b,\n    0xb7b775c2, 0xfdfde11c, 0x93933dae, 0x26264c6a,\n    0x36366c5a, 0x3f3f7e41, 0xf7f7f502, 0xcccc834f,\n    0x3434685c, 0xa5a551f4, 0xe5e5d134, 0xf1f1f908,\n    0x7171e293, 0xd8d8ab73, 0x31316253, 0x15152a3f,\n    0x0404080c, 0xc7c79552, 0x23234665, 0xc3c39d5e,\n    0x18183028, 0x969637a1, 0x05050a0f, 0x9a9a2fb5,\n    0x07070e09, 0x12122436, 0x80801b9b, 0xe2e2df3d,\n    0xebebcd26, 0x27274e69, 0xb2b27fcd, 0x7575ea9f,\n    0x0909121b, 0x83831d9e, 0x2c2c5874, 0x1a1a342e,\n    0x1b1b362d, 0x6e6edcb2, 0x5a5ab4ee, 0xa0a05bfb,\n    0x5252a4f6, 0x3b3b764d, 0xd6d6b761, 0xb3b37dce,\n    0x2929527b, 0xe3e3dd3e, 0x2f2f5e71, 0x84841397,\n    0x5353a6f5, 0xd1d1b968, 0x00000000, 0xededc12c,\n    0x20204060, 0xfcfce31f, 0xb1b179c8, 0x5b5bb6ed,\n    0x6a6ad4be, 0xcbcb8d46, 0xbebe67d9, 0x3939724b,\n    0x4a4a94de, 0x4c4c98d4, 0x5858b0e8, 0xcfcf854a,\n    0xd0d0bb6b, 0xefefc52a, 0xaaaa4fe5, 0xfbfbed16,\n    0x434386c5, 0x4d4d9ad7, 0x33336655, 0x85851194,\n    0x45458acf, 0xf9f9e910, 0x02020406, 0x7f7ffe81,\n    0x5050a0f0, 0x3c3c7844, 0x9f9f25ba, 0xa8a84be3,\n    0x5151a2f3, 0xa3a35dfe, 0x404080c0, 0x8f8f058a,\n    0x92923fad, 0x9d9d21bc, 0x38387048, 0xf5f5f104,\n    0xbcbc63df, 0xb6b677c1, 0xdadaaf75, 0x21214263,\n    0x10102030, 0xffffe51a, 0xf3f3fd0e, 0xd2d2bf6d,\n    0xcdcd814c, 0x0c0c1814, 0x13132635, 0xececc32f,\n    0x5f5fbee1, 0x979735a2, 0x444488cc, 0x17172e39,\n    0xc4c49357, 0xa7a755f2, 0x7e7efc82, 0x3d3d7a47,\n    0x6464c8ac, 0x5d5dbae7, 0x1919322b, 0x7373e695,\n    0x6060c0a0, 0x81811998, 0x4f4f9ed1, 0xdcdca37f,\n    0x22224466, 0x2a2a547e, 0x90903bab, 0x88880b83,\n    0x46468cca, 0xeeeec729, 0xb8b86bd3, 0x1414283c,\n    0xdedea779, 0x5e5ebce2, 0x0b0b161d, 0xdbdbad76,\n    0xe0e0db3b, 0x32326456, 0x3a3a744e, 0x0a0a141e,\n    0x494992db, 0x06060c0a, 0x2424486c, 0x5c5cb8e4,\n    0xc2c29f5d, 0xd3d3bd6e, 0xacac43ef, 0x6262c4a6,\n    0x919139a8, 0x959531a4, 0xe4e4d337, 0x7979f28b,\n    0xe7e7d532, 0xc8c88b43, 0x37376e59, 0x6d6ddab7,\n    0x8d8d018c, 0xd5d5b164, 0x4e4e9cd2, 0xa9a949e0,\n    0x6c6cd8b4, 0x5656acfa, 0xf4f4f307, 0xeaeacf25,\n    0x6565caaf, 0x7a7af48e, 0xaeae47e9, 0x08081018,\n    0xbaba6fd5, 0x7878f088, 0x25254a6f, 0x2e2e5c72,\n    0x1c1c3824, 0xa6a657f1, 0xb4b473c7, 0xc6c69751,\n    0xe8e8cb23, 0xdddda17c, 0x7474e89c, 0x1f1f3e21,\n    0x4b4b96dd, 0xbdbd61dc, 0x8b8b0d86, 0x8a8a0f85,\n    0x7070e090, 0x3e3e7c42, 0xb5b571c4, 0x6666ccaa,\n    0x484890d8, 0x03030605, 0xf6f6f701, 0x0e0e1c12,\n    0x6161c2a3, 0x35356a5f, 0x5757aef9, 0xb9b969d0,\n    0x86861791, 0xc1c19958, 0x1d1d3a27, 0x9e9e27b9,\n    0xe1e1d938, 0xf8f8eb13, 0x98982bb3, 0x11112233,\n    0x6969d2bb, 0xd9d9a970, 0x8e8e0789, 0x949433a7,\n    0x9b9b2db6, 0x1e1e3c22, 0x87871592, 0xe9e9c920,\n    0xcece8749, 0x5555aaff, 0x28285078, 0xdfdfa57a,\n    0x8c8c038f, 0xa1a159f8, 0x89890980, 0x0d0d1a17,\n    0xbfbf65da, 0xe6e6d731, 0x424284c6, 0x6868d0b8,\n    0x414182c3, 0x999929b0, 0x2d2d5a77, 0x0f0f1e11,\n    0xb0b07bcb, 0x5454a8fc, 0xbbbb6dd6, 0x16162c3a\n];\n\nvar T3 = [\n    0x63c6a563, 0x7cf8847c, 0x77ee9977, 0x7bf68d7b,\n    0xf2ff0df2, 0x6bd6bd6b, 0x6fdeb16f, 0xc59154c5,\n    0x30605030, 0x01020301, 0x67cea967, 0x2b567d2b,\n    0xfee719fe, 0xd7b562d7, 0xab4de6ab, 0x76ec9a76,\n    0xca8f45ca, 0x821f9d82, 0xc98940c9, 0x7dfa877d,\n    0xfaef15fa, 0x59b2eb59, 0x478ec947, 0xf0fb0bf0,\n    0xad41ecad, 0xd4b367d4, 0xa25ffda2, 0xaf45eaaf,\n    0x9c23bf9c, 0xa453f7a4, 0x72e49672, 0xc09b5bc0,\n    0xb775c2b7, 0xfde11cfd, 0x933dae93, 0x264c6a26,\n    0x366c5a36, 0x3f7e413f, 0xf7f502f7, 0xcc834fcc,\n    0x34685c34, 0xa551f4a5, 0xe5d134e5, 0xf1f908f1,\n    0x71e29371, 0xd8ab73d8, 0x31625331, 0x152a3f15,\n    0x04080c04, 0xc79552c7, 0x23466523, 0xc39d5ec3,\n    0x18302818, 0x9637a196, 0x050a0f05, 0x9a2fb59a,\n    0x070e0907, 0x12243612, 0x801b9b80, 0xe2df3de2,\n    0xebcd26eb, 0x274e6927, 0xb27fcdb2, 0x75ea9f75,\n    0x09121b09, 0x831d9e83, 0x2c58742c, 0x1a342e1a,\n    0x1b362d1b, 0x6edcb26e, 0x5ab4ee5a, 0xa05bfba0,\n    0x52a4f652, 0x3b764d3b, 0xd6b761d6, 0xb37dceb3,\n    0x29527b29, 0xe3dd3ee3, 0x2f5e712f, 0x84139784,\n    0x53a6f553, 0xd1b968d1, 0x00000000, 0xedc12ced,\n    0x20406020, 0xfce31ffc, 0xb179c8b1, 0x5bb6ed5b,\n    0x6ad4be6a, 0xcb8d46cb, 0xbe67d9be, 0x39724b39,\n    0x4a94de4a, 0x4c98d44c, 0x58b0e858, 0xcf854acf,\n    0xd0bb6bd0, 0xefc52aef, 0xaa4fe5aa, 0xfbed16fb,\n    0x4386c543, 0x4d9ad74d, 0x33665533, 0x85119485,\n    0x458acf45, 0xf9e910f9, 0x02040602, 0x7ffe817f,\n    0x50a0f050, 0x3c78443c, 0x9f25ba9f, 0xa84be3a8,\n    0x51a2f351, 0xa35dfea3, 0x4080c040, 0x8f058a8f,\n    0x923fad92, 0x9d21bc9d, 0x38704838, 0xf5f104f5,\n    0xbc63dfbc, 0xb677c1b6, 0xdaaf75da, 0x21426321,\n    0x10203010, 0xffe51aff, 0xf3fd0ef3, 0xd2bf6dd2,\n    0xcd814ccd, 0x0c18140c, 0x13263513, 0xecc32fec,\n    0x5fbee15f, 0x9735a297, 0x4488cc44, 0x172e3917,\n    0xc49357c4, 0xa755f2a7, 0x7efc827e, 0x3d7a473d,\n    0x64c8ac64, 0x5dbae75d, 0x19322b19, 0x73e69573,\n    0x60c0a060, 0x81199881, 0x4f9ed14f, 0xdca37fdc,\n    0x22446622, 0x2a547e2a, 0x903bab90, 0x880b8388,\n    0x468cca46, 0xeec729ee, 0xb86bd3b8, 0x14283c14,\n    0xdea779de, 0x5ebce25e, 0x0b161d0b, 0xdbad76db,\n    0xe0db3be0, 0x32645632, 0x3a744e3a, 0x0a141e0a,\n    0x4992db49, 0x060c0a06, 0x24486c24, 0x5cb8e45c,\n    0xc29f5dc2, 0xd3bd6ed3, 0xac43efac, 0x62c4a662,\n    0x9139a891, 0x9531a495, 0xe4d337e4, 0x79f28b79,\n    0xe7d532e7, 0xc88b43c8, 0x376e5937, 0x6ddab76d,\n    0x8d018c8d, 0xd5b164d5, 0x4e9cd24e, 0xa949e0a9,\n    0x6cd8b46c, 0x56acfa56, 0xf4f307f4, 0xeacf25ea,\n    0x65caaf65, 0x7af48e7a, 0xae47e9ae, 0x08101808,\n    0xba6fd5ba, 0x78f08878, 0x254a6f25, 0x2e5c722e,\n    0x1c38241c, 0xa657f1a6, 0xb473c7b4, 0xc69751c6,\n    0xe8cb23e8, 0xdda17cdd, 0x74e89c74, 0x1f3e211f,\n    0x4b96dd4b, 0xbd61dcbd, 0x8b0d868b, 0x8a0f858a,\n    0x70e09070, 0x3e7c423e, 0xb571c4b5, 0x66ccaa66,\n    0x4890d848, 0x03060503, 0xf6f701f6, 0x0e1c120e,\n    0x61c2a361, 0x356a5f35, 0x57aef957, 0xb969d0b9,\n    0x86179186, 0xc19958c1, 0x1d3a271d, 0x9e27b99e,\n    0xe1d938e1, 0xf8eb13f8, 0x982bb398, 0x11223311,\n    0x69d2bb69, 0xd9a970d9, 0x8e07898e, 0x9433a794,\n    0x9b2db69b, 0x1e3c221e, 0x87159287, 0xe9c920e9,\n    0xce8749ce, 0x55aaff55, 0x28507828, 0xdfa57adf,\n    0x8c038f8c, 0xa159f8a1, 0x89098089, 0x0d1a170d,\n    0xbf65dabf, 0xe6d731e6, 0x4284c642, 0x68d0b868,\n    0x4182c341, 0x9929b099, 0x2d5a772d, 0x0f1e110f,\n    0xb07bcbb0, 0x54a8fc54, 0xbb6dd6bb, 0x162c3a16\n];\n\nvar T4 = [\n    0xc6a56363, 0xf8847c7c, 0xee997777, 0xf68d7b7b,\n    0xff0df2f2, 0xd6bd6b6b, 0xdeb16f6f, 0x9154c5c5,\n    0x60503030, 0x02030101, 0xcea96767, 0x567d2b2b,\n    0xe719fefe, 0xb562d7d7, 0x4de6abab, 0xec9a7676,\n    0x8f45caca, 0x1f9d8282, 0x8940c9c9, 0xfa877d7d,\n    0xef15fafa, 0xb2eb5959, 0x8ec94747, 0xfb0bf0f0,\n    0x41ecadad, 0xb367d4d4, 0x5ffda2a2, 0x45eaafaf,\n    0x23bf9c9c, 0x53f7a4a4, 0xe4967272, 0x9b5bc0c0,\n    0x75c2b7b7, 0xe11cfdfd, 0x3dae9393, 0x4c6a2626,\n    0x6c5a3636, 0x7e413f3f, 0xf502f7f7, 0x834fcccc,\n    0x685c3434, 0x51f4a5a5, 0xd134e5e5, 0xf908f1f1,\n    0xe2937171, 0xab73d8d8, 0x62533131, 0x2a3f1515,\n    0x080c0404, 0x9552c7c7, 0x46652323, 0x9d5ec3c3,\n    0x30281818, 0x37a19696, 0x0a0f0505, 0x2fb59a9a,\n    0x0e090707, 0x24361212, 0x1b9b8080, 0xdf3de2e2,\n    0xcd26ebeb, 0x4e692727, 0x7fcdb2b2, 0xea9f7575,\n    0x121b0909, 0x1d9e8383, 0x58742c2c, 0x342e1a1a,\n    0x362d1b1b, 0xdcb26e6e, 0xb4ee5a5a, 0x5bfba0a0,\n    0xa4f65252, 0x764d3b3b, 0xb761d6d6, 0x7dceb3b3,\n    0x527b2929, 0xdd3ee3e3, 0x5e712f2f, 0x13978484,\n    0xa6f55353, 0xb968d1d1, 0x00000000, 0xc12ceded,\n    0x40602020, 0xe31ffcfc, 0x79c8b1b1, 0xb6ed5b5b,\n    0xd4be6a6a, 0x8d46cbcb, 0x67d9bebe, 0x724b3939,\n    0x94de4a4a, 0x98d44c4c, 0xb0e85858, 0x854acfcf,\n    0xbb6bd0d0, 0xc52aefef, 0x4fe5aaaa, 0xed16fbfb,\n    0x86c54343, 0x9ad74d4d, 0x66553333, 0x11948585,\n    0x8acf4545, 0xe910f9f9, 0x04060202, 0xfe817f7f,\n    0xa0f05050, 0x78443c3c, 0x25ba9f9f, 0x4be3a8a8,\n    0xa2f35151, 0x5dfea3a3, 0x80c04040, 0x058a8f8f,\n    0x3fad9292, 0x21bc9d9d, 0x70483838, 0xf104f5f5,\n    0x63dfbcbc, 0x77c1b6b6, 0xaf75dada, 0x42632121,\n    0x20301010, 0xe51affff, 0xfd0ef3f3, 0xbf6dd2d2,\n    0x814ccdcd, 0x18140c0c, 0x26351313, 0xc32fecec,\n    0xbee15f5f, 0x35a29797, 0x88cc4444, 0x2e391717,\n    0x9357c4c4, 0x55f2a7a7, 0xfc827e7e, 0x7a473d3d,\n    0xc8ac6464, 0xbae75d5d, 0x322b1919, 0xe6957373,\n    0xc0a06060, 0x19988181, 0x9ed14f4f, 0xa37fdcdc,\n    0x44662222, 0x547e2a2a, 0x3bab9090, 0x0b838888,\n    0x8cca4646, 0xc729eeee, 0x6bd3b8b8, 0x283c1414,\n    0xa779dede, 0xbce25e5e, 0x161d0b0b, 0xad76dbdb,\n    0xdb3be0e0, 0x64563232, 0x744e3a3a, 0x141e0a0a,\n    0x92db4949, 0x0c0a0606, 0x486c2424, 0xb8e45c5c,\n    0x9f5dc2c2, 0xbd6ed3d3, 0x43efacac, 0xc4a66262,\n    0x39a89191, 0x31a49595, 0xd337e4e4, 0xf28b7979,\n    0xd532e7e7, 0x8b43c8c8, 0x6e593737, 0xdab76d6d,\n    0x018c8d8d, 0xb164d5d5, 0x9cd24e4e, 0x49e0a9a9,\n    0xd8b46c6c, 0xacfa5656, 0xf307f4f4, 0xcf25eaea,\n    0xcaaf6565, 0xf48e7a7a, 0x47e9aeae, 0x10180808,\n    0x6fd5baba, 0xf0887878, 0x4a6f2525, 0x5c722e2e,\n    0x38241c1c, 0x57f1a6a6, 0x73c7b4b4, 0x9751c6c6,\n    0xcb23e8e8, 0xa17cdddd, 0xe89c7474, 0x3e211f1f,\n    0x96dd4b4b, 0x61dcbdbd, 0x0d868b8b, 0x0f858a8a,\n    0xe0907070, 0x7c423e3e, 0x71c4b5b5, 0xccaa6666,\n    0x90d84848, 0x06050303, 0xf701f6f6, 0x1c120e0e,\n    0xc2a36161, 0x6a5f3535, 0xaef95757, 0x69d0b9b9,\n    0x17918686, 0x9958c1c1, 0x3a271d1d, 0x27b99e9e,\n    0xd938e1e1, 0xeb13f8f8, 0x2bb39898, 0x22331111,\n    0xd2bb6969, 0xa970d9d9, 0x07898e8e, 0x33a79494,\n    0x2db69b9b, 0x3c221e1e, 0x15928787, 0xc920e9e9,\n    0x8749cece, 0xaaff5555, 0x50782828, 0xa57adfdf,\n    0x038f8c8c, 0x59f8a1a1, 0x09808989, 0x1a170d0d,\n    0x65dabfbf, 0xd731e6e6, 0x84c64242, 0xd0b86868,\n    0x82c34141, 0x29b09999, 0x5a772d2d, 0x1e110f0f,\n    0x7bcbb0b0, 0xa8fc5454, 0x6dd6bbbb, 0x2c3a1616\n];\n\nfunction B0(x) {\n  return (x & 255);\n}\n\nfunction B1(x) {\n  return ((x >> 8) & 255);\n}\n\nfunction B2(x) {\n  return ((x >> 16) & 255);\n}\n\nfunction B3(x) {\n  return ((x >> 24) & 255);\n}\n\nfunction F1(x0, x1, x2, x3) {\n  return B1(T1[x0 & 255]) | (B1(T1[(x1 >> 8) & 255]) << 8) | (B1(T1[(x2 >> 16) & 255]) << 16) | (B1(T1[x3 >>> 24]) <<\n    24);\n}\n\nfunction packBytes(octets) {\n  var i, j;\n  var len = octets.length;\n  var b = new Array(len / 4);\n\n  if (!octets || len % 4) return;\n\n  for (i = 0, j = 0; j < len; j += 4)\n    b[i++] = octets[j] | (octets[j + 1] << 8) | (octets[j + 2] << 16) | (octets[j + 3] << 24);\n\n  return b;\n}\n\nfunction unpackBytes(packed) {\n  var j;\n  var i = 0,\n    l = packed.length;\n  var r = new Array(l * 4);\n\n  for (j = 0; j < l; j++) {\n    r[i++] = B0(packed[j]);\n    r[i++] = B1(packed[j]);\n    r[i++] = B2(packed[j]);\n    r[i++] = B3(packed[j]);\n  }\n  return r;\n}\n\n// ------------------------------------------------\n\nvar maxkc = 8;\nvar maxrk = 14;\n\nfunction keyExpansion(key) {\n  var kc, i, j, r, t;\n  var rounds;\n  var keySched = new Array(maxrk + 1);\n  var keylen = key.length;\n  var k = new Array(maxkc);\n  var tk = new Array(maxkc);\n  var rconpointer = 0;\n\n  if (keylen == 16) {\n    rounds = 10;\n    kc = 4;\n  } else if (keylen == 24) {\n    rounds = 12;\n    kc = 6;\n  } else if (keylen == 32) {\n    rounds = 14;\n    kc = 8;\n  } else {\n    util.print_error('aes.js: Invalid key-length for AES key:' + keylen);\n    return;\n  }\n\n  for (i = 0; i < maxrk + 1; i++) keySched[i] = new Array(4);\n\n  for (i = 0, j = 0; j < keylen; j++, i += 4)\n    k[j] = key.charCodeAt(i) | (key.charCodeAt(i + 1) << 8) | (key.charCodeAt(i + 2) << 16) | (key.charCodeAt(i + 3) <<\n      24);\n\n  for (j = kc - 1; j >= 0; j--) tk[j] = k[j];\n\n  r = 0;\n  t = 0;\n  for (j = 0;\n  (j < kc) && (r < rounds + 1);) {\n    for (;\n    (j < kc) && (t < 4); j++, t++) {\n      keySched[r][t] = tk[j];\n    }\n    if (t == 4) {\n      r++;\n      t = 0;\n    }\n  }\n\n  while (r < rounds + 1) {\n    var temp = tk[kc - 1];\n\n    tk[0] ^= S[B1(temp)] | (S[B2(temp)] << 8) | (S[B3(temp)] << 16) | (S[B0(temp)] << 24);\n    tk[0] ^= Rcon[rconpointer++];\n\n    if (kc != 8) {\n      for (j = 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    } else {\n      for (j = 1; j < kc / 2; j++) tk[j] ^= tk[j - 1];\n\n      temp = tk[kc / 2 - 1];\n      tk[kc / 2] ^= S[B0(temp)] | (S[B1(temp)] << 8) | (S[B2(temp)] << 16) | (S[B3(temp)] << 24);\n\n      for (j = kc / 2 + 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    }\n\n    for (j = 0;\n    (j < kc) && (r < rounds + 1);) {\n      for (;\n      (j < kc) && (t < 4); j++, t++) {\n        keySched[r][t] = tk[j];\n      }\n      if (t == 4) {\n        r++;\n        t = 0;\n      }\n    }\n  }\n  this.rounds = rounds;\n  this.rk = keySched;\n  return this;\n}\n\nfunction AESencrypt(block, ctx) {\n  var r;\n  var t0, t1, t2, t3;\n\n  var b = packBytes(block);\n  var rounds = ctx.rounds;\n  var b0 = b[0];\n  var b1 = b[1];\n  var b2 = b[2];\n  var b3 = b[3];\n\n  for (r = 0; r < rounds - 1; r++) {\n    t0 = b0 ^ ctx.rk[r][0];\n    t1 = b1 ^ ctx.rk[r][1];\n    t2 = b2 ^ ctx.rk[r][2];\n    t3 = b3 ^ ctx.rk[r][3];\n\n    b0 = T1[t0 & 255] ^ T2[(t1 >> 8) & 255] ^ T3[(t2 >> 16) & 255] ^ T4[t3 >>> 24];\n    b1 = T1[t1 & 255] ^ T2[(t2 >> 8) & 255] ^ T3[(t3 >> 16) & 255] ^ T4[t0 >>> 24];\n    b2 = T1[t2 & 255] ^ T2[(t3 >> 8) & 255] ^ T3[(t0 >> 16) & 255] ^ T4[t1 >>> 24];\n    b3 = T1[t3 & 255] ^ T2[(t0 >> 8) & 255] ^ T3[(t1 >> 16) & 255] ^ T4[t2 >>> 24];\n  }\n\n  // last round is special\n  r = rounds - 1;\n\n  t0 = b0 ^ ctx.rk[r][0];\n  t1 = b1 ^ ctx.rk[r][1];\n  t2 = b2 ^ ctx.rk[r][2];\n  t3 = b3 ^ ctx.rk[r][3];\n\n  b[0] = F1(t0, t1, t2, t3) ^ ctx.rk[rounds][0];\n  b[1] = F1(t1, t2, t3, t0) ^ ctx.rk[rounds][1];\n  b[2] = F1(t2, t3, t0, t1) ^ ctx.rk[rounds][2];\n  b[3] = F1(t3, t0, t1, t2) ^ ctx.rk[rounds][3];\n\n  return unpackBytes(b);\n}\n\nfunction makeClass(length) {\n\n  var c = function(key) {\n    this.key = keyExpansion(key);\n\n    this.encrypt = function(block) {\n      return AESencrypt(block, this.key);\n    }\n  }\n\n  c.blockSize = c.prototype.blockSize = 16;\n  c.keySize = c.prototype.keySize = length / 8;\n\n  return c;\n}\n\nmodule.exports = {}\n\nvar types = [128, 192, 256];\n\nfor (var i in types) {\n  module.exports[types[i]] = makeClass(types[i]);\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Originally written by nklein software (nklein.com)\n */\n\n/* \n * Javascript implementation based on Bruce Schneier's reference implementation.\n *\n *\n * The constructor doesn't do much of anything.  It's just here\n * so we can start defining properties and methods and such.\n */\nfunction Blowfish() {};\n\n/*\n * Declare the block size so that protocols know what size\n * Initialization Vector (IV) they will need.\n */\nBlowfish.prototype.BLOCKSIZE = 8;\n\n/*\n * These are the default SBOXES.\n */\nBlowfish.prototype.SBOXES = [\n  [\n      0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96,\n      0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16,\n      0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658,\n      0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013,\n      0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e,\n      0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60,\n      0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6,\n      0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a,\n      0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c,\n      0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193,\n      0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1,\n      0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239,\n      0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a,\n      0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3,\n      0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176,\n      0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe,\n      0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706,\n      0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b,\n      0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b,\n      0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463,\n      0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c,\n      0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3,\n      0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a,\n      0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8,\n      0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760,\n      0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db,\n      0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8,\n      0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b,\n      0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33,\n      0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4,\n      0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0,\n      0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c,\n      0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777,\n      0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299,\n      0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705,\n      0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf,\n      0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e,\n      0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa,\n      0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9,\n      0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915,\n      0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f,\n      0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664,\n      0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a\n  ],\n  [\n      0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d,\n      0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1,\n      0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65,\n      0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1,\n      0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9,\n      0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737,\n      0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d,\n      0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd,\n      0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc,\n      0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41,\n      0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908,\n      0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af,\n      0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124,\n      0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c,\n      0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908,\n      0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd,\n      0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b,\n      0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e,\n      0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa,\n      0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a,\n      0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d,\n      0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66,\n      0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5,\n      0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84,\n      0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96,\n      0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14,\n      0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca,\n      0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7,\n      0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77,\n      0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99,\n      0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054,\n      0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73,\n      0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea,\n      0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105,\n      0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646,\n      0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285,\n      0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea,\n      0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb,\n      0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e,\n      0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc,\n      0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd,\n      0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20,\n      0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7\n  ],\n  [\n      0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7,\n      0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af,\n      0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af,\n      0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504,\n      0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4,\n      0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee,\n      0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec,\n      0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b,\n      0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332,\n      0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527,\n      0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58,\n      0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c,\n      0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22,\n      0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17,\n      0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60,\n      0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115,\n      0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99,\n      0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0,\n      0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74,\n      0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d,\n      0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3,\n      0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3,\n      0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979,\n      0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c,\n      0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa,\n      0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a,\n      0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086,\n      0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc,\n      0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24,\n      0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2,\n      0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84,\n      0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c,\n      0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09,\n      0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10,\n      0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe,\n      0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027,\n      0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0,\n      0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634,\n      0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188,\n      0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc,\n      0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8,\n      0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837,\n      0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0\n  ],\n  [\n      0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742,\n      0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b,\n      0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79,\n      0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6,\n      0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a,\n      0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4,\n      0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1,\n      0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59,\n      0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797,\n      0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28,\n      0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6,\n      0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28,\n      0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba,\n      0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a,\n      0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5,\n      0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f,\n      0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce,\n      0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680,\n      0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd,\n      0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb,\n      0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb,\n      0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370,\n      0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc,\n      0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048,\n      0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc,\n      0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9,\n      0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a,\n      0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f,\n      0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a,\n      0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1,\n      0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b,\n      0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e,\n      0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e,\n      0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f,\n      0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623,\n      0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc,\n      0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a,\n      0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6,\n      0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3,\n      0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060,\n      0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c,\n      0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f,\n      0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6\n  ]\n];\n\n//*\n//* This is the default PARRAY\n//*\nBlowfish.prototype.PARRAY = [\n    0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0,\n    0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c,\n    0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b\n];\n\n//*\n//* This is the number of rounds the cipher will go\n//*\nBlowfish.prototype.NN = 16;\n\n//*\n//* This function is needed to get rid of problems\n//* with the high-bit getting set.  If we don't do\n//* this, then sometimes ( aa & 0x00FFFFFFFF ) is not\n//* equal to ( bb & 0x00FFFFFFFF ) even when they\n//* agree bit-for-bit for the first 32 bits.\n//*\nBlowfish.prototype._clean = function(xx) {\n  if (xx < 0) {\n    var yy = xx & 0x7FFFFFFF;\n    xx = yy + 0x80000000;\n  }\n  return xx;\n};\n\n//*\n//* This is the mixing function that uses the sboxes\n//*\nBlowfish.prototype._F = function(xx) {\n  var aa;\n  var bb;\n  var cc;\n  var dd;\n  var yy;\n\n  dd = xx & 0x00FF;\n  xx >>>= 8;\n  cc = xx & 0x00FF;\n  xx >>>= 8;\n  bb = xx & 0x00FF;\n  xx >>>= 8;\n  aa = xx & 0x00FF;\n\n  yy = this.sboxes[0][aa] + this.sboxes[1][bb];\n  yy = yy ^ this.sboxes[2][cc];\n  yy = yy + this.sboxes[3][dd];\n\n  return yy;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and does NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._encrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = 0; ii < this.NN; ++ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[this.NN + 0];\n  dataR = dataR ^ this.parray[this.NN + 1];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a vector of numbers and turns them\n//* into long words so that they can be processed by the\n//* real algorithm.\n//*\n//* Maybe I should make the real algorithm above take a vector\n//* instead.  That will involve more looping, but it won't require\n//* the F() method to deconstruct the vector.\n//*\nBlowfish.prototype.encrypt_block = function(vector) {\n  var ii;\n  var vals = [0, 0];\n  var off = this.BLOCKSIZE / 2;\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    vals[0] = (vals[0] << 8) | (vector[ii + 0] & 0x00FF);\n    vals[1] = (vals[1] << 8) | (vector[ii + off] & 0x00FF);\n  }\n\n  this._encrypt_block(vals);\n\n  var ret = [];\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    ret[ii + 0] = (vals[0] >>> (24 - 8 * (ii)) & 0x00FF);\n    ret[ii + off] = (vals[1] >>> (24 - 8 * (ii)) & 0x00FF);\n    // vals[ 0 ] = ( vals[ 0 ] >>> 8 );\n    // vals[ 1 ] = ( vals[ 1 ] >>> 8 );\n  }\n\n  return ret;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and undoes NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._decrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = this.NN + 1; ii > 1; --ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[1];\n  dataR = dataR ^ this.parray[0];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a key array and initializes the\n//* sboxes and parray for this encryption.\n//*\nBlowfish.prototype.init = function(key) {\n  var ii;\n  var jj = 0;\n\n  this.parray = [];\n  for (ii = 0; ii < this.NN + 2; ++ii) {\n    var data = 0x00000000;\n    var kk;\n    for (kk = 0; kk < 4; ++kk) {\n      data = (data << 8) | (key[jj] & 0x00FF);\n      if (++jj >= key.length) {\n        jj = 0;\n      }\n    }\n    this.parray[ii] = this.PARRAY[ii] ^ data;\n  }\n\n  this.sboxes = [];\n  for (ii = 0; ii < 4; ++ii) {\n    this.sboxes[ii] = [];\n    for (jj = 0; jj < 256; ++jj) {\n      this.sboxes[ii][jj] = this.SBOXES[ii][jj];\n    }\n  }\n\n  var vals = [0x00000000, 0x00000000];\n\n  for (ii = 0; ii < this.NN + 2; ii += 2) {\n    this._encrypt_block(vals);\n    this.parray[ii + 0] = vals[0];\n    this.parray[ii + 1] = vals[1];\n  }\n\n  for (ii = 0; ii < 4; ++ii) {\n    for (jj = 0; jj < 256; jj += 2) {\n      this._encrypt_block(vals);\n      this.sboxes[ii][jj + 0] = vals[0];\n      this.sboxes[ii][jj + 1] = vals[1];\n    }\n  }\n};\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction BFencrypt(block, key) {\n  var bf = new Blowfish();\n  bf.init(util.str2bin(key));\n  return bf.encrypt_block(block);\n}\n\nfunction BF(key) {\n  this.bf = new Blowfish();\n  this.bf.init(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.bf.encrypt_block(block);\n  }\n}\n\n\nmodule.exports = BF;\nmodule.exports.keySize = BF.prototype.keySize = 16;\nmodule.exports.blockSize = BF.prototype.blockSize = 16;\n","// Use of this source code is governed by a BSD-style\n// license that can be found in the LICENSE file.\n\n// Copyright 2010 pjacobs@xeekr.com . All rights reserved.\n\n// Modified by Recurity Labs GmbH\n\n// fixed/modified by Herbert Hanewinkel, www.haneWIN.de\n// check www.haneWIN.de for the latest version\n\n// cast5.js is a Javascript implementation of CAST-128, as defined in RFC 2144.\n// CAST-128 is a common OpenPGP cipher.\n\n\n// CAST5 constructor\n\n\n\nfunction openpgp_symenc_cast5() {\n  this.BlockSize = 8;\n  this.KeySize = 16;\n\n  this.setKey = function(key) {\n    this.masking = new Array(16);\n    this.rotate = new Array(16);\n\n    this.reset();\n\n    if (key.length == this.KeySize) {\n      this.keySchedule(key);\n    } else {\n      util.print_error('cast5.js: CAST-128: keys must be 16 bytes');\n      return false;\n    }\n    return true;\n  };\n\n  this.reset = function() {\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = 0;\n      this.rotate[i] = 0;\n    }\n  };\n\n  this.getBlockSize = function() {\n    return BlockSize;\n  };\n\n  this.encrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >>> 16) & 255;\n      dst[i + 6] = (l >>> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n\n  this.decrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >> 16) & 255;\n      dst[i + 6] = (l >> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n  var scheduleA = new Array(4);\n\n  scheduleA[0] = new Array(4);\n  scheduleA[0][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 0x8);\n  scheduleA[0][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[0][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[0][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n  scheduleA[1] = new Array(4);\n  scheduleA[1][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[1][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[1][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[1][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  scheduleA[2] = new Array(4);\n  scheduleA[2][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 8);\n  scheduleA[2][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[2][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[2][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n\n  scheduleA[3] = new Array(4);\n  scheduleA[3][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[3][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[3][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[3][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  var scheduleB = new Array(4);\n\n  scheduleB[0] = new Array(4);\n  scheduleB[0][0] = new Array(16 + 8, 16 + 9, 16 + 7, 16 + 6, 16 + 2);\n  scheduleB[0][1] = new Array(16 + 0xa, 16 + 0xb, 16 + 5, 16 + 4, 16 + 6);\n  scheduleB[0][2] = new Array(16 + 0xc, 16 + 0xd, 16 + 3, 16 + 2, 16 + 9);\n  scheduleB[0][3] = new Array(16 + 0xe, 16 + 0xf, 16 + 1, 16 + 0, 16 + 0xc);\n\n  scheduleB[1] = new Array(4);\n  scheduleB[1][0] = new Array(3, 2, 0xc, 0xd, 8);\n  scheduleB[1][1] = new Array(1, 0, 0xe, 0xf, 0xd);\n  scheduleB[1][2] = new Array(7, 6, 8, 9, 3);\n  scheduleB[1][3] = new Array(5, 4, 0xa, 0xb, 7);\n\n\n  scheduleB[2] = new Array(4);\n  scheduleB[2][0] = new Array(16 + 3, 16 + 2, 16 + 0xc, 16 + 0xd, 16 + 9);\n  scheduleB[2][1] = new Array(16 + 1, 16 + 0, 16 + 0xe, 16 + 0xf, 16 + 0xc);\n  scheduleB[2][2] = new Array(16 + 7, 16 + 6, 16 + 8, 16 + 9, 16 + 2);\n  scheduleB[2][3] = new Array(16 + 5, 16 + 4, 16 + 0xa, 16 + 0xb, 16 + 6);\n\n\n  scheduleB[3] = new Array(4);\n  scheduleB[3][0] = new Array(8, 9, 7, 6, 3);\n  scheduleB[3][1] = new Array(0xa, 0xb, 5, 4, 7);\n  scheduleB[3][2] = new Array(0xc, 0xd, 3, 2, 8);\n  scheduleB[3][3] = new Array(0xe, 0xf, 1, 0, 0xd);\n\n  // changed 'in' to 'inn' (in javascript 'in' is a reserved word)\n  this.keySchedule = function(inn) {\n    var t = new Array(8);\n    var k = new Array(32);\n\n    for (var i = 0; i < 4; i++) {\n      var j = i * 4;\n      t[i] = inn[j] << 24 | inn[j + 1] << 16 | inn[j + 2] << 8 | inn[j + 3];\n    }\n\n    var x = [6, 7, 4, 5];\n    var ki = 0;\n\n    for (var half = 0; half < 2; half++) {\n      for (var round = 0; round < 4; round++) {\n        for (var j = 0; j < 4; j++) {\n          var a = scheduleA[round][j];\n          var w = t[a[1]];\n\n          w ^= sBox[4][(t[a[2] >>> 2] >>> (24 - 8 * (a[2] & 3))) & 0xff];\n          w ^= sBox[5][(t[a[3] >>> 2] >>> (24 - 8 * (a[3] & 3))) & 0xff];\n          w ^= sBox[6][(t[a[4] >>> 2] >>> (24 - 8 * (a[4] & 3))) & 0xff];\n          w ^= sBox[7][(t[a[5] >>> 2] >>> (24 - 8 * (a[5] & 3))) & 0xff];\n          w ^= sBox[x[j]][(t[a[6] >>> 2] >>> (24 - 8 * (a[6] & 3))) & 0xff];\n          t[a[0]] = w;\n        }\n\n        for (var j = 0; j < 4; j++) {\n          var b = scheduleB[round][j];\n          var w = sBox[4][(t[b[0] >>> 2] >>> (24 - 8 * (b[0] & 3))) & 0xff];\n\n          w ^= sBox[5][(t[b[1] >>> 2] >>> (24 - 8 * (b[1] & 3))) & 0xff];\n          w ^= sBox[6][(t[b[2] >>> 2] >>> (24 - 8 * (b[2] & 3))) & 0xff];\n          w ^= sBox[7][(t[b[3] >>> 2] >>> (24 - 8 * (b[3] & 3))) & 0xff];\n          w ^= sBox[4 + j][(t[b[4] >>> 2] >>> (24 - 8 * (b[4] & 3))) & 0xff];\n          k[ki] = w;\n          ki++;\n        }\n      }\n    }\n\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = k[i];\n      this.rotate[i] = k[16 + i] & 0x1f;\n    }\n  };\n\n  // These are the three 'f' functions. See RFC 2144, section 2.2.\n\n  function f1(d, m, r) {\n    var t = m + d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] ^ sBox[1][(I >>> 16) & 255]) - sBox[2][(I >>> 8) & 255]) + sBox[3][I & 255];\n  }\n\n  function f2(d, m, r) {\n    var t = m ^ d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] - sBox[1][(I >>> 16) & 255]) + sBox[2][(I >>> 8) & 255]) ^ sBox[3][I & 255];\n  }\n\n  function f3(d, m, r) {\n    var t = m - d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] + sBox[1][(I >>> 16) & 255]) ^ sBox[2][(I >>> 8) & 255]) - sBox[3][I & 255];\n  }\n\n  var sBox = new Array(8);\n  sBox[0] = new Array(\n    0x30fb40d4, 0x9fa0ff0b, 0x6beccd2f, 0x3f258c7a, 0x1e213f2f, 0x9c004dd3, 0x6003e540, 0xcf9fc949,\n    0xbfd4af27, 0x88bbbdb5, 0xe2034090, 0x98d09675, 0x6e63a0e0, 0x15c361d2, 0xc2e7661d, 0x22d4ff8e,\n    0x28683b6f, 0xc07fd059, 0xff2379c8, 0x775f50e2, 0x43c340d3, 0xdf2f8656, 0x887ca41a, 0xa2d2bd2d,\n    0xa1c9e0d6, 0x346c4819, 0x61b76d87, 0x22540f2f, 0x2abe32e1, 0xaa54166b, 0x22568e3a, 0xa2d341d0,\n    0x66db40c8, 0xa784392f, 0x004dff2f, 0x2db9d2de, 0x97943fac, 0x4a97c1d8, 0x527644b7, 0xb5f437a7,\n    0xb82cbaef, 0xd751d159, 0x6ff7f0ed, 0x5a097a1f, 0x827b68d0, 0x90ecf52e, 0x22b0c054, 0xbc8e5935,\n    0x4b6d2f7f, 0x50bb64a2, 0xd2664910, 0xbee5812d, 0xb7332290, 0xe93b159f, 0xb48ee411, 0x4bff345d,\n    0xfd45c240, 0xad31973f, 0xc4f6d02e, 0x55fc8165, 0xd5b1caad, 0xa1ac2dae, 0xa2d4b76d, 0xc19b0c50,\n    0x882240f2, 0x0c6e4f38, 0xa4e4bfd7, 0x4f5ba272, 0x564c1d2f, 0xc59c5319, 0xb949e354, 0xb04669fe,\n    0xb1b6ab8a, 0xc71358dd, 0x6385c545, 0x110f935d, 0x57538ad5, 0x6a390493, 0xe63d37e0, 0x2a54f6b3,\n    0x3a787d5f, 0x6276a0b5, 0x19a6fcdf, 0x7a42206a, 0x29f9d4d5, 0xf61b1891, 0xbb72275e, 0xaa508167,\n    0x38901091, 0xc6b505eb, 0x84c7cb8c, 0x2ad75a0f, 0x874a1427, 0xa2d1936b, 0x2ad286af, 0xaa56d291,\n    0xd7894360, 0x425c750d, 0x93b39e26, 0x187184c9, 0x6c00b32d, 0x73e2bb14, 0xa0bebc3c, 0x54623779,\n    0x64459eab, 0x3f328b82, 0x7718cf82, 0x59a2cea6, 0x04ee002e, 0x89fe78e6, 0x3fab0950, 0x325ff6c2,\n    0x81383f05, 0x6963c5c8, 0x76cb5ad6, 0xd49974c9, 0xca180dcf, 0x380782d5, 0xc7fa5cf6, 0x8ac31511,\n    0x35e79e13, 0x47da91d0, 0xf40f9086, 0xa7e2419e, 0x31366241, 0x051ef495, 0xaa573b04, 0x4a805d8d,\n    0x548300d0, 0x00322a3c, 0xbf64cddf, 0xba57a68e, 0x75c6372b, 0x50afd341, 0xa7c13275, 0x915a0bf5,\n    0x6b54bfab, 0x2b0b1426, 0xab4cc9d7, 0x449ccd82, 0xf7fbf265, 0xab85c5f3, 0x1b55db94, 0xaad4e324,\n    0xcfa4bd3f, 0x2deaa3e2, 0x9e204d02, 0xc8bd25ac, 0xeadf55b3, 0xd5bd9e98, 0xe31231b2, 0x2ad5ad6c,\n    0x954329de, 0xadbe4528, 0xd8710f69, 0xaa51c90f, 0xaa786bf6, 0x22513f1e, 0xaa51a79b, 0x2ad344cc,\n    0x7b5a41f0, 0xd37cfbad, 0x1b069505, 0x41ece491, 0xb4c332e6, 0x032268d4, 0xc9600acc, 0xce387e6d,\n    0xbf6bb16c, 0x6a70fb78, 0x0d03d9c9, 0xd4df39de, 0xe01063da, 0x4736f464, 0x5ad328d8, 0xb347cc96,\n    0x75bb0fc3, 0x98511bfb, 0x4ffbcc35, 0xb58bcf6a, 0xe11f0abc, 0xbfc5fe4a, 0xa70aec10, 0xac39570a,\n    0x3f04442f, 0x6188b153, 0xe0397a2e, 0x5727cb79, 0x9ceb418f, 0x1cacd68d, 0x2ad37c96, 0x0175cb9d,\n    0xc69dff09, 0xc75b65f0, 0xd9db40d8, 0xec0e7779, 0x4744ead4, 0xb11c3274, 0xdd24cb9e, 0x7e1c54bd,\n    0xf01144f9, 0xd2240eb1, 0x9675b3fd, 0xa3ac3755, 0xd47c27af, 0x51c85f4d, 0x56907596, 0xa5bb15e6,\n    0x580304f0, 0xca042cf1, 0x011a37ea, 0x8dbfaadb, 0x35ba3e4a, 0x3526ffa0, 0xc37b4d09, 0xbc306ed9,\n    0x98a52666, 0x5648f725, 0xff5e569d, 0x0ced63d0, 0x7c63b2cf, 0x700b45e1, 0xd5ea50f1, 0x85a92872,\n    0xaf1fbda7, 0xd4234870, 0xa7870bf3, 0x2d3b4d79, 0x42e04198, 0x0cd0ede7, 0x26470db8, 0xf881814c,\n    0x474d6ad7, 0x7c0c5e5c, 0xd1231959, 0x381b7298, 0xf5d2f4db, 0xab838653, 0x6e2f1e23, 0x83719c9e,\n    0xbd91e046, 0x9a56456e, 0xdc39200c, 0x20c8c571, 0x962bda1c, 0xe1e696ff, 0xb141ab08, 0x7cca89b9,\n    0x1a69e783, 0x02cc4843, 0xa2f7c579, 0x429ef47d, 0x427b169c, 0x5ac9f049, 0xdd8f0f00, 0x5c8165bf);\n\n  sBox[1] = new Array(\n    0x1f201094, 0xef0ba75b, 0x69e3cf7e, 0x393f4380, 0xfe61cf7a, 0xeec5207a, 0x55889c94, 0x72fc0651,\n    0xada7ef79, 0x4e1d7235, 0xd55a63ce, 0xde0436ba, 0x99c430ef, 0x5f0c0794, 0x18dcdb7d, 0xa1d6eff3,\n    0xa0b52f7b, 0x59e83605, 0xee15b094, 0xe9ffd909, 0xdc440086, 0xef944459, 0xba83ccb3, 0xe0c3cdfb,\n    0xd1da4181, 0x3b092ab1, 0xf997f1c1, 0xa5e6cf7b, 0x01420ddb, 0xe4e7ef5b, 0x25a1ff41, 0xe180f806,\n    0x1fc41080, 0x179bee7a, 0xd37ac6a9, 0xfe5830a4, 0x98de8b7f, 0x77e83f4e, 0x79929269, 0x24fa9f7b,\n    0xe113c85b, 0xacc40083, 0xd7503525, 0xf7ea615f, 0x62143154, 0x0d554b63, 0x5d681121, 0xc866c359,\n    0x3d63cf73, 0xcee234c0, 0xd4d87e87, 0x5c672b21, 0x071f6181, 0x39f7627f, 0x361e3084, 0xe4eb573b,\n    0x602f64a4, 0xd63acd9c, 0x1bbc4635, 0x9e81032d, 0x2701f50c, 0x99847ab4, 0xa0e3df79, 0xba6cf38c,\n    0x10843094, 0x2537a95e, 0xf46f6ffe, 0xa1ff3b1f, 0x208cfb6a, 0x8f458c74, 0xd9e0a227, 0x4ec73a34,\n    0xfc884f69, 0x3e4de8df, 0xef0e0088, 0x3559648d, 0x8a45388c, 0x1d804366, 0x721d9bfd, 0xa58684bb,\n    0xe8256333, 0x844e8212, 0x128d8098, 0xfed33fb4, 0xce280ae1, 0x27e19ba5, 0xd5a6c252, 0xe49754bd,\n    0xc5d655dd, 0xeb667064, 0x77840b4d, 0xa1b6a801, 0x84db26a9, 0xe0b56714, 0x21f043b7, 0xe5d05860,\n    0x54f03084, 0x066ff472, 0xa31aa153, 0xdadc4755, 0xb5625dbf, 0x68561be6, 0x83ca6b94, 0x2d6ed23b,\n    0xeccf01db, 0xa6d3d0ba, 0xb6803d5c, 0xaf77a709, 0x33b4a34c, 0x397bc8d6, 0x5ee22b95, 0x5f0e5304,\n    0x81ed6f61, 0x20e74364, 0xb45e1378, 0xde18639b, 0x881ca122, 0xb96726d1, 0x8049a7e8, 0x22b7da7b,\n    0x5e552d25, 0x5272d237, 0x79d2951c, 0xc60d894c, 0x488cb402, 0x1ba4fe5b, 0xa4b09f6b, 0x1ca815cf,\n    0xa20c3005, 0x8871df63, 0xb9de2fcb, 0x0cc6c9e9, 0x0beeff53, 0xe3214517, 0xb4542835, 0x9f63293c,\n    0xee41e729, 0x6e1d2d7c, 0x50045286, 0x1e6685f3, 0xf33401c6, 0x30a22c95, 0x31a70850, 0x60930f13,\n    0x73f98417, 0xa1269859, 0xec645c44, 0x52c877a9, 0xcdff33a6, 0xa02b1741, 0x7cbad9a2, 0x2180036f,\n    0x50d99c08, 0xcb3f4861, 0xc26bd765, 0x64a3f6ab, 0x80342676, 0x25a75e7b, 0xe4e6d1fc, 0x20c710e6,\n    0xcdf0b680, 0x17844d3b, 0x31eef84d, 0x7e0824e4, 0x2ccb49eb, 0x846a3bae, 0x8ff77888, 0xee5d60f6,\n    0x7af75673, 0x2fdd5cdb, 0xa11631c1, 0x30f66f43, 0xb3faec54, 0x157fd7fa, 0xef8579cc, 0xd152de58,\n    0xdb2ffd5e, 0x8f32ce19, 0x306af97a, 0x02f03ef8, 0x99319ad5, 0xc242fa0f, 0xa7e3ebb0, 0xc68e4906,\n    0xb8da230c, 0x80823028, 0xdcdef3c8, 0xd35fb171, 0x088a1bc8, 0xbec0c560, 0x61a3c9e8, 0xbca8f54d,\n    0xc72feffa, 0x22822e99, 0x82c570b4, 0xd8d94e89, 0x8b1c34bc, 0x301e16e6, 0x273be979, 0xb0ffeaa6,\n    0x61d9b8c6, 0x00b24869, 0xb7ffce3f, 0x08dc283b, 0x43daf65a, 0xf7e19798, 0x7619b72f, 0x8f1c9ba4,\n    0xdc8637a0, 0x16a7d3b1, 0x9fc393b7, 0xa7136eeb, 0xc6bcc63e, 0x1a513742, 0xef6828bc, 0x520365d6,\n    0x2d6a77ab, 0x3527ed4b, 0x821fd216, 0x095c6e2e, 0xdb92f2fb, 0x5eea29cb, 0x145892f5, 0x91584f7f,\n    0x5483697b, 0x2667a8cc, 0x85196048, 0x8c4bacea, 0x833860d4, 0x0d23e0f9, 0x6c387e8a, 0x0ae6d249,\n    0xb284600c, 0xd835731d, 0xdcb1c647, 0xac4c56ea, 0x3ebd81b3, 0x230eabb0, 0x6438bc87, 0xf0b5b1fa,\n    0x8f5ea2b3, 0xfc184642, 0x0a036b7a, 0x4fb089bd, 0x649da589, 0xa345415e, 0x5c038323, 0x3e5d3bb9,\n    0x43d79572, 0x7e6dd07c, 0x06dfdf1e, 0x6c6cc4ef, 0x7160a539, 0x73bfbe70, 0x83877605, 0x4523ecf1);\n\n  sBox[2] = new Array(\n    0x8defc240, 0x25fa5d9f, 0xeb903dbf, 0xe810c907, 0x47607fff, 0x369fe44b, 0x8c1fc644, 0xaececa90,\n    0xbeb1f9bf, 0xeefbcaea, 0xe8cf1950, 0x51df07ae, 0x920e8806, 0xf0ad0548, 0xe13c8d83, 0x927010d5,\n    0x11107d9f, 0x07647db9, 0xb2e3e4d4, 0x3d4f285e, 0xb9afa820, 0xfade82e0, 0xa067268b, 0x8272792e,\n    0x553fb2c0, 0x489ae22b, 0xd4ef9794, 0x125e3fbc, 0x21fffcee, 0x825b1bfd, 0x9255c5ed, 0x1257a240,\n    0x4e1a8302, 0xbae07fff, 0x528246e7, 0x8e57140e, 0x3373f7bf, 0x8c9f8188, 0xa6fc4ee8, 0xc982b5a5,\n    0xa8c01db7, 0x579fc264, 0x67094f31, 0xf2bd3f5f, 0x40fff7c1, 0x1fb78dfc, 0x8e6bd2c1, 0x437be59b,\n    0x99b03dbf, 0xb5dbc64b, 0x638dc0e6, 0x55819d99, 0xa197c81c, 0x4a012d6e, 0xc5884a28, 0xccc36f71,\n    0xb843c213, 0x6c0743f1, 0x8309893c, 0x0feddd5f, 0x2f7fe850, 0xd7c07f7e, 0x02507fbf, 0x5afb9a04,\n    0xa747d2d0, 0x1651192e, 0xaf70bf3e, 0x58c31380, 0x5f98302e, 0x727cc3c4, 0x0a0fb402, 0x0f7fef82,\n    0x8c96fdad, 0x5d2c2aae, 0x8ee99a49, 0x50da88b8, 0x8427f4a0, 0x1eac5790, 0x796fb449, 0x8252dc15,\n    0xefbd7d9b, 0xa672597d, 0xada840d8, 0x45f54504, 0xfa5d7403, 0xe83ec305, 0x4f91751a, 0x925669c2,\n    0x23efe941, 0xa903f12e, 0x60270df2, 0x0276e4b6, 0x94fd6574, 0x927985b2, 0x8276dbcb, 0x02778176,\n    0xf8af918d, 0x4e48f79e, 0x8f616ddf, 0xe29d840e, 0x842f7d83, 0x340ce5c8, 0x96bbb682, 0x93b4b148,\n    0xef303cab, 0x984faf28, 0x779faf9b, 0x92dc560d, 0x224d1e20, 0x8437aa88, 0x7d29dc96, 0x2756d3dc,\n    0x8b907cee, 0xb51fd240, 0xe7c07ce3, 0xe566b4a1, 0xc3e9615e, 0x3cf8209d, 0x6094d1e3, 0xcd9ca341,\n    0x5c76460e, 0x00ea983b, 0xd4d67881, 0xfd47572c, 0xf76cedd9, 0xbda8229c, 0x127dadaa, 0x438a074e,\n    0x1f97c090, 0x081bdb8a, 0x93a07ebe, 0xb938ca15, 0x97b03cff, 0x3dc2c0f8, 0x8d1ab2ec, 0x64380e51,\n    0x68cc7bfb, 0xd90f2788, 0x12490181, 0x5de5ffd4, 0xdd7ef86a, 0x76a2e214, 0xb9a40368, 0x925d958f,\n    0x4b39fffa, 0xba39aee9, 0xa4ffd30b, 0xfaf7933b, 0x6d498623, 0x193cbcfa, 0x27627545, 0x825cf47a,\n    0x61bd8ba0, 0xd11e42d1, 0xcead04f4, 0x127ea392, 0x10428db7, 0x8272a972, 0x9270c4a8, 0x127de50b,\n    0x285ba1c8, 0x3c62f44f, 0x35c0eaa5, 0xe805d231, 0x428929fb, 0xb4fcdf82, 0x4fb66a53, 0x0e7dc15b,\n    0x1f081fab, 0x108618ae, 0xfcfd086d, 0xf9ff2889, 0x694bcc11, 0x236a5cae, 0x12deca4d, 0x2c3f8cc5,\n    0xd2d02dfe, 0xf8ef5896, 0xe4cf52da, 0x95155b67, 0x494a488c, 0xb9b6a80c, 0x5c8f82bc, 0x89d36b45,\n    0x3a609437, 0xec00c9a9, 0x44715253, 0x0a874b49, 0xd773bc40, 0x7c34671c, 0x02717ef6, 0x4feb5536,\n    0xa2d02fff, 0xd2bf60c4, 0xd43f03c0, 0x50b4ef6d, 0x07478cd1, 0x006e1888, 0xa2e53f55, 0xb9e6d4bc,\n    0xa2048016, 0x97573833, 0xd7207d67, 0xde0f8f3d, 0x72f87b33, 0xabcc4f33, 0x7688c55d, 0x7b00a6b0,\n    0x947b0001, 0x570075d2, 0xf9bb88f8, 0x8942019e, 0x4264a5ff, 0x856302e0, 0x72dbd92b, 0xee971b69,\n    0x6ea22fde, 0x5f08ae2b, 0xaf7a616d, 0xe5c98767, 0xcf1febd2, 0x61efc8c2, 0xf1ac2571, 0xcc8239c2,\n    0x67214cb8, 0xb1e583d1, 0xb7dc3e62, 0x7f10bdce, 0xf90a5c38, 0x0ff0443d, 0x606e6dc6, 0x60543a49,\n    0x5727c148, 0x2be98a1d, 0x8ab41738, 0x20e1be24, 0xaf96da0f, 0x68458425, 0x99833be5, 0x600d457d,\n    0x282f9350, 0x8334b362, 0xd91d1120, 0x2b6d8da0, 0x642b1e31, 0x9c305a00, 0x52bce688, 0x1b03588a,\n    0xf7baefd5, 0x4142ed9c, 0xa4315c11, 0x83323ec5, 0xdfef4636, 0xa133c501, 0xe9d3531c, 0xee353783);\n\n  sBox[3] = new Array(\n    0x9db30420, 0x1fb6e9de, 0xa7be7bef, 0xd273a298, 0x4a4f7bdb, 0x64ad8c57, 0x85510443, 0xfa020ed1,\n    0x7e287aff, 0xe60fb663, 0x095f35a1, 0x79ebf120, 0xfd059d43, 0x6497b7b1, 0xf3641f63, 0x241e4adf,\n    0x28147f5f, 0x4fa2b8cd, 0xc9430040, 0x0cc32220, 0xfdd30b30, 0xc0a5374f, 0x1d2d00d9, 0x24147b15,\n    0xee4d111a, 0x0fca5167, 0x71ff904c, 0x2d195ffe, 0x1a05645f, 0x0c13fefe, 0x081b08ca, 0x05170121,\n    0x80530100, 0xe83e5efe, 0xac9af4f8, 0x7fe72701, 0xd2b8ee5f, 0x06df4261, 0xbb9e9b8a, 0x7293ea25,\n    0xce84ffdf, 0xf5718801, 0x3dd64b04, 0xa26f263b, 0x7ed48400, 0x547eebe6, 0x446d4ca0, 0x6cf3d6f5,\n    0x2649abdf, 0xaea0c7f5, 0x36338cc1, 0x503f7e93, 0xd3772061, 0x11b638e1, 0x72500e03, 0xf80eb2bb,\n    0xabe0502e, 0xec8d77de, 0x57971e81, 0xe14f6746, 0xc9335400, 0x6920318f, 0x081dbb99, 0xffc304a5,\n    0x4d351805, 0x7f3d5ce3, 0xa6c866c6, 0x5d5bcca9, 0xdaec6fea, 0x9f926f91, 0x9f46222f, 0x3991467d,\n    0xa5bf6d8e, 0x1143c44f, 0x43958302, 0xd0214eeb, 0x022083b8, 0x3fb6180c, 0x18f8931e, 0x281658e6,\n    0x26486e3e, 0x8bd78a70, 0x7477e4c1, 0xb506e07c, 0xf32d0a25, 0x79098b02, 0xe4eabb81, 0x28123b23,\n    0x69dead38, 0x1574ca16, 0xdf871b62, 0x211c40b7, 0xa51a9ef9, 0x0014377b, 0x041e8ac8, 0x09114003,\n    0xbd59e4d2, 0xe3d156d5, 0x4fe876d5, 0x2f91a340, 0x557be8de, 0x00eae4a7, 0x0ce5c2ec, 0x4db4bba6,\n    0xe756bdff, 0xdd3369ac, 0xec17b035, 0x06572327, 0x99afc8b0, 0x56c8c391, 0x6b65811c, 0x5e146119,\n    0x6e85cb75, 0xbe07c002, 0xc2325577, 0x893ff4ec, 0x5bbfc92d, 0xd0ec3b25, 0xb7801ab7, 0x8d6d3b24,\n    0x20c763ef, 0xc366a5fc, 0x9c382880, 0x0ace3205, 0xaac9548a, 0xeca1d7c7, 0x041afa32, 0x1d16625a,\n    0x6701902c, 0x9b757a54, 0x31d477f7, 0x9126b031, 0x36cc6fdb, 0xc70b8b46, 0xd9e66a48, 0x56e55a79,\n    0x026a4ceb, 0x52437eff, 0x2f8f76b4, 0x0df980a5, 0x8674cde3, 0xedda04eb, 0x17a9be04, 0x2c18f4df,\n    0xb7747f9d, 0xab2af7b4, 0xefc34d20, 0x2e096b7c, 0x1741a254, 0xe5b6a035, 0x213d42f6, 0x2c1c7c26,\n    0x61c2f50f, 0x6552daf9, 0xd2c231f8, 0x25130f69, 0xd8167fa2, 0x0418f2c8, 0x001a96a6, 0x0d1526ab,\n    0x63315c21, 0x5e0a72ec, 0x49bafefd, 0x187908d9, 0x8d0dbd86, 0x311170a7, 0x3e9b640c, 0xcc3e10d7,\n    0xd5cad3b6, 0x0caec388, 0xf73001e1, 0x6c728aff, 0x71eae2a1, 0x1f9af36e, 0xcfcbd12f, 0xc1de8417,\n    0xac07be6b, 0xcb44a1d8, 0x8b9b0f56, 0x013988c3, 0xb1c52fca, 0xb4be31cd, 0xd8782806, 0x12a3a4e2,\n    0x6f7de532, 0x58fd7eb6, 0xd01ee900, 0x24adffc2, 0xf4990fc5, 0x9711aac5, 0x001d7b95, 0x82e5e7d2,\n    0x109873f6, 0x00613096, 0xc32d9521, 0xada121ff, 0x29908415, 0x7fbb977f, 0xaf9eb3db, 0x29c9ed2a,\n    0x5ce2a465, 0xa730f32c, 0xd0aa3fe8, 0x8a5cc091, 0xd49e2ce7, 0x0ce454a9, 0xd60acd86, 0x015f1919,\n    0x77079103, 0xdea03af6, 0x78a8565e, 0xdee356df, 0x21f05cbe, 0x8b75e387, 0xb3c50651, 0xb8a5c3ef,\n    0xd8eeb6d2, 0xe523be77, 0xc2154529, 0x2f69efdf, 0xafe67afb, 0xf470c4b2, 0xf3e0eb5b, 0xd6cc9876,\n    0x39e4460c, 0x1fda8538, 0x1987832f, 0xca007367, 0xa99144f8, 0x296b299e, 0x492fc295, 0x9266beab,\n    0xb5676e69, 0x9bd3ddda, 0xdf7e052f, 0xdb25701c, 0x1b5e51ee, 0xf65324e6, 0x6afce36c, 0x0316cc04,\n    0x8644213e, 0xb7dc59d0, 0x7965291f, 0xccd6fd43, 0x41823979, 0x932bcdf6, 0xb657c34d, 0x4edfd282,\n    0x7ae5290c, 0x3cb9536b, 0x851e20fe, 0x9833557e, 0x13ecf0b0, 0xd3ffb372, 0x3f85c5c1, 0x0aef7ed2);\n\n  sBox[4] = new Array(\n    0x7ec90c04, 0x2c6e74b9, 0x9b0e66df, 0xa6337911, 0xb86a7fff, 0x1dd358f5, 0x44dd9d44, 0x1731167f,\n    0x08fbf1fa, 0xe7f511cc, 0xd2051b00, 0x735aba00, 0x2ab722d8, 0x386381cb, 0xacf6243a, 0x69befd7a,\n    0xe6a2e77f, 0xf0c720cd, 0xc4494816, 0xccf5c180, 0x38851640, 0x15b0a848, 0xe68b18cb, 0x4caadeff,\n    0x5f480a01, 0x0412b2aa, 0x259814fc, 0x41d0efe2, 0x4e40b48d, 0x248eb6fb, 0x8dba1cfe, 0x41a99b02,\n    0x1a550a04, 0xba8f65cb, 0x7251f4e7, 0x95a51725, 0xc106ecd7, 0x97a5980a, 0xc539b9aa, 0x4d79fe6a,\n    0xf2f3f763, 0x68af8040, 0xed0c9e56, 0x11b4958b, 0xe1eb5a88, 0x8709e6b0, 0xd7e07156, 0x4e29fea7,\n    0x6366e52d, 0x02d1c000, 0xc4ac8e05, 0x9377f571, 0x0c05372a, 0x578535f2, 0x2261be02, 0xd642a0c9,\n    0xdf13a280, 0x74b55bd2, 0x682199c0, 0xd421e5ec, 0x53fb3ce8, 0xc8adedb3, 0x28a87fc9, 0x3d959981,\n    0x5c1ff900, 0xfe38d399, 0x0c4eff0b, 0x062407ea, 0xaa2f4fb1, 0x4fb96976, 0x90c79505, 0xb0a8a774,\n    0xef55a1ff, 0xe59ca2c2, 0xa6b62d27, 0xe66a4263, 0xdf65001f, 0x0ec50966, 0xdfdd55bc, 0x29de0655,\n    0x911e739a, 0x17af8975, 0x32c7911c, 0x89f89468, 0x0d01e980, 0x524755f4, 0x03b63cc9, 0x0cc844b2,\n    0xbcf3f0aa, 0x87ac36e9, 0xe53a7426, 0x01b3d82b, 0x1a9e7449, 0x64ee2d7e, 0xcddbb1da, 0x01c94910,\n    0xb868bf80, 0x0d26f3fd, 0x9342ede7, 0x04a5c284, 0x636737b6, 0x50f5b616, 0xf24766e3, 0x8eca36c1,\n    0x136e05db, 0xfef18391, 0xfb887a37, 0xd6e7f7d4, 0xc7fb7dc9, 0x3063fcdf, 0xb6f589de, 0xec2941da,\n    0x26e46695, 0xb7566419, 0xf654efc5, 0xd08d58b7, 0x48925401, 0xc1bacb7f, 0xe5ff550f, 0xb6083049,\n    0x5bb5d0e8, 0x87d72e5a, 0xab6a6ee1, 0x223a66ce, 0xc62bf3cd, 0x9e0885f9, 0x68cb3e47, 0x086c010f,\n    0xa21de820, 0xd18b69de, 0xf3f65777, 0xfa02c3f6, 0x407edac3, 0xcbb3d550, 0x1793084d, 0xb0d70eba,\n    0x0ab378d5, 0xd951fb0c, 0xded7da56, 0x4124bbe4, 0x94ca0b56, 0x0f5755d1, 0xe0e1e56e, 0x6184b5be,\n    0x580a249f, 0x94f74bc0, 0xe327888e, 0x9f7b5561, 0xc3dc0280, 0x05687715, 0x646c6bd7, 0x44904db3,\n    0x66b4f0a3, 0xc0f1648a, 0x697ed5af, 0x49e92ff6, 0x309e374f, 0x2cb6356a, 0x85808573, 0x4991f840,\n    0x76f0ae02, 0x083be84d, 0x28421c9a, 0x44489406, 0x736e4cb8, 0xc1092910, 0x8bc95fc6, 0x7d869cf4,\n    0x134f616f, 0x2e77118d, 0xb31b2be1, 0xaa90b472, 0x3ca5d717, 0x7d161bba, 0x9cad9010, 0xaf462ba2,\n    0x9fe459d2, 0x45d34559, 0xd9f2da13, 0xdbc65487, 0xf3e4f94e, 0x176d486f, 0x097c13ea, 0x631da5c7,\n    0x445f7382, 0x175683f4, 0xcdc66a97, 0x70be0288, 0xb3cdcf72, 0x6e5dd2f3, 0x20936079, 0x459b80a5,\n    0xbe60e2db, 0xa9c23101, 0xeba5315c, 0x224e42f2, 0x1c5c1572, 0xf6721b2c, 0x1ad2fff3, 0x8c25404e,\n    0x324ed72f, 0x4067b7fd, 0x0523138e, 0x5ca3bc78, 0xdc0fd66e, 0x75922283, 0x784d6b17, 0x58ebb16e,\n    0x44094f85, 0x3f481d87, 0xfcfeae7b, 0x77b5ff76, 0x8c2302bf, 0xaaf47556, 0x5f46b02a, 0x2b092801,\n    0x3d38f5f7, 0x0ca81f36, 0x52af4a8a, 0x66d5e7c0, 0xdf3b0874, 0x95055110, 0x1b5ad7a8, 0xf61ed5ad,\n    0x6cf6e479, 0x20758184, 0xd0cefa65, 0x88f7be58, 0x4a046826, 0x0ff6f8f3, 0xa09c7f70, 0x5346aba0,\n    0x5ce96c28, 0xe176eda3, 0x6bac307f, 0x376829d2, 0x85360fa9, 0x17e3fe2a, 0x24b79767, 0xf5a96b20,\n    0xd6cd2595, 0x68ff1ebf, 0x7555442c, 0xf19f06be, 0xf9e0659a, 0xeeb9491d, 0x34010718, 0xbb30cab8,\n    0xe822fe15, 0x88570983, 0x750e6249, 0xda627e55, 0x5e76ffa8, 0xb1534546, 0x6d47de08, 0xefe9e7d4);\n\n  sBox[5] = new Array(\n    0xf6fa8f9d, 0x2cac6ce1, 0x4ca34867, 0xe2337f7c, 0x95db08e7, 0x016843b4, 0xeced5cbc, 0x325553ac,\n    0xbf9f0960, 0xdfa1e2ed, 0x83f0579d, 0x63ed86b9, 0x1ab6a6b8, 0xde5ebe39, 0xf38ff732, 0x8989b138,\n    0x33f14961, 0xc01937bd, 0xf506c6da, 0xe4625e7e, 0xa308ea99, 0x4e23e33c, 0x79cbd7cc, 0x48a14367,\n    0xa3149619, 0xfec94bd5, 0xa114174a, 0xeaa01866, 0xa084db2d, 0x09a8486f, 0xa888614a, 0x2900af98,\n    0x01665991, 0xe1992863, 0xc8f30c60, 0x2e78ef3c, 0xd0d51932, 0xcf0fec14, 0xf7ca07d2, 0xd0a82072,\n    0xfd41197e, 0x9305a6b0, 0xe86be3da, 0x74bed3cd, 0x372da53c, 0x4c7f4448, 0xdab5d440, 0x6dba0ec3,\n    0x083919a7, 0x9fbaeed9, 0x49dbcfb0, 0x4e670c53, 0x5c3d9c01, 0x64bdb941, 0x2c0e636a, 0xba7dd9cd,\n    0xea6f7388, 0xe70bc762, 0x35f29adb, 0x5c4cdd8d, 0xf0d48d8c, 0xb88153e2, 0x08a19866, 0x1ae2eac8,\n    0x284caf89, 0xaa928223, 0x9334be53, 0x3b3a21bf, 0x16434be3, 0x9aea3906, 0xefe8c36e, 0xf890cdd9,\n    0x80226dae, 0xc340a4a3, 0xdf7e9c09, 0xa694a807, 0x5b7c5ecc, 0x221db3a6, 0x9a69a02f, 0x68818a54,\n    0xceb2296f, 0x53c0843a, 0xfe893655, 0x25bfe68a, 0xb4628abc, 0xcf222ebf, 0x25ac6f48, 0xa9a99387,\n    0x53bddb65, 0xe76ffbe7, 0xe967fd78, 0x0ba93563, 0x8e342bc1, 0xe8a11be9, 0x4980740d, 0xc8087dfc,\n    0x8de4bf99, 0xa11101a0, 0x7fd37975, 0xda5a26c0, 0xe81f994f, 0x9528cd89, 0xfd339fed, 0xb87834bf,\n    0x5f04456d, 0x22258698, 0xc9c4c83b, 0x2dc156be, 0x4f628daa, 0x57f55ec5, 0xe2220abe, 0xd2916ebf,\n    0x4ec75b95, 0x24f2c3c0, 0x42d15d99, 0xcd0d7fa0, 0x7b6e27ff, 0xa8dc8af0, 0x7345c106, 0xf41e232f,\n    0x35162386, 0xe6ea8926, 0x3333b094, 0x157ec6f2, 0x372b74af, 0x692573e4, 0xe9a9d848, 0xf3160289,\n    0x3a62ef1d, 0xa787e238, 0xf3a5f676, 0x74364853, 0x20951063, 0x4576698d, 0xb6fad407, 0x592af950,\n    0x36f73523, 0x4cfb6e87, 0x7da4cec0, 0x6c152daa, 0xcb0396a8, 0xc50dfe5d, 0xfcd707ab, 0x0921c42f,\n    0x89dff0bb, 0x5fe2be78, 0x448f4f33, 0x754613c9, 0x2b05d08d, 0x48b9d585, 0xdc049441, 0xc8098f9b,\n    0x7dede786, 0xc39a3373, 0x42410005, 0x6a091751, 0x0ef3c8a6, 0x890072d6, 0x28207682, 0xa9a9f7be,\n    0xbf32679d, 0xd45b5b75, 0xb353fd00, 0xcbb0e358, 0x830f220a, 0x1f8fb214, 0xd372cf08, 0xcc3c4a13,\n    0x8cf63166, 0x061c87be, 0x88c98f88, 0x6062e397, 0x47cf8e7a, 0xb6c85283, 0x3cc2acfb, 0x3fc06976,\n    0x4e8f0252, 0x64d8314d, 0xda3870e3, 0x1e665459, 0xc10908f0, 0x513021a5, 0x6c5b68b7, 0x822f8aa0,\n    0x3007cd3e, 0x74719eef, 0xdc872681, 0x073340d4, 0x7e432fd9, 0x0c5ec241, 0x8809286c, 0xf592d891,\n    0x08a930f6, 0x957ef305, 0xb7fbffbd, 0xc266e96f, 0x6fe4ac98, 0xb173ecc0, 0xbc60b42a, 0x953498da,\n    0xfba1ae12, 0x2d4bd736, 0x0f25faab, 0xa4f3fceb, 0xe2969123, 0x257f0c3d, 0x9348af49, 0x361400bc,\n    0xe8816f4a, 0x3814f200, 0xa3f94043, 0x9c7a54c2, 0xbc704f57, 0xda41e7f9, 0xc25ad33a, 0x54f4a084,\n    0xb17f5505, 0x59357cbe, 0xedbd15c8, 0x7f97c5ab, 0xba5ac7b5, 0xb6f6deaf, 0x3a479c3a, 0x5302da25,\n    0x653d7e6a, 0x54268d49, 0x51a477ea, 0x5017d55b, 0xd7d25d88, 0x44136c76, 0x0404a8c8, 0xb8e5a121,\n    0xb81a928a, 0x60ed5869, 0x97c55b96, 0xeaec991b, 0x29935913, 0x01fdb7f1, 0x088e8dfa, 0x9ab6f6f5,\n    0x3b4cbf9f, 0x4a5de3ab, 0xe6051d35, 0xa0e1d855, 0xd36b4cf1, 0xf544edeb, 0xb0e93524, 0xbebb8fbd,\n    0xa2d762cf, 0x49c92f54, 0x38b5f331, 0x7128a454, 0x48392905, 0xa65b1db8, 0x851c97bd, 0xd675cf2f);\n\n  sBox[6] = new Array(\n    0x85e04019, 0x332bf567, 0x662dbfff, 0xcfc65693, 0x2a8d7f6f, 0xab9bc912, 0xde6008a1, 0x2028da1f,\n    0x0227bce7, 0x4d642916, 0x18fac300, 0x50f18b82, 0x2cb2cb11, 0xb232e75c, 0x4b3695f2, 0xb28707de,\n    0xa05fbcf6, 0xcd4181e9, 0xe150210c, 0xe24ef1bd, 0xb168c381, 0xfde4e789, 0x5c79b0d8, 0x1e8bfd43,\n    0x4d495001, 0x38be4341, 0x913cee1d, 0x92a79c3f, 0x089766be, 0xbaeeadf4, 0x1286becf, 0xb6eacb19,\n    0x2660c200, 0x7565bde4, 0x64241f7a, 0x8248dca9, 0xc3b3ad66, 0x28136086, 0x0bd8dfa8, 0x356d1cf2,\n    0x107789be, 0xb3b2e9ce, 0x0502aa8f, 0x0bc0351e, 0x166bf52a, 0xeb12ff82, 0xe3486911, 0xd34d7516,\n    0x4e7b3aff, 0x5f43671b, 0x9cf6e037, 0x4981ac83, 0x334266ce, 0x8c9341b7, 0xd0d854c0, 0xcb3a6c88,\n    0x47bc2829, 0x4725ba37, 0xa66ad22b, 0x7ad61f1e, 0x0c5cbafa, 0x4437f107, 0xb6e79962, 0x42d2d816,\n    0x0a961288, 0xe1a5c06e, 0x13749e67, 0x72fc081a, 0xb1d139f7, 0xf9583745, 0xcf19df58, 0xbec3f756,\n    0xc06eba30, 0x07211b24, 0x45c28829, 0xc95e317f, 0xbc8ec511, 0x38bc46e9, 0xc6e6fa14, 0xbae8584a,\n    0xad4ebc46, 0x468f508b, 0x7829435f, 0xf124183b, 0x821dba9f, 0xaff60ff4, 0xea2c4e6d, 0x16e39264,\n    0x92544a8b, 0x009b4fc3, 0xaba68ced, 0x9ac96f78, 0x06a5b79a, 0xb2856e6e, 0x1aec3ca9, 0xbe838688,\n    0x0e0804e9, 0x55f1be56, 0xe7e5363b, 0xb3a1f25d, 0xf7debb85, 0x61fe033c, 0x16746233, 0x3c034c28,\n    0xda6d0c74, 0x79aac56c, 0x3ce4e1ad, 0x51f0c802, 0x98f8f35a, 0x1626a49f, 0xeed82b29, 0x1d382fe3,\n    0x0c4fb99a, 0xbb325778, 0x3ec6d97b, 0x6e77a6a9, 0xcb658b5c, 0xd45230c7, 0x2bd1408b, 0x60c03eb7,\n    0xb9068d78, 0xa33754f4, 0xf430c87d, 0xc8a71302, 0xb96d8c32, 0xebd4e7be, 0xbe8b9d2d, 0x7979fb06,\n    0xe7225308, 0x8b75cf77, 0x11ef8da4, 0xe083c858, 0x8d6b786f, 0x5a6317a6, 0xfa5cf7a0, 0x5dda0033,\n    0xf28ebfb0, 0xf5b9c310, 0xa0eac280, 0x08b9767a, 0xa3d9d2b0, 0x79d34217, 0x021a718d, 0x9ac6336a,\n    0x2711fd60, 0x438050e3, 0x069908a8, 0x3d7fedc4, 0x826d2bef, 0x4eeb8476, 0x488dcf25, 0x36c9d566,\n    0x28e74e41, 0xc2610aca, 0x3d49a9cf, 0xbae3b9df, 0xb65f8de6, 0x92aeaf64, 0x3ac7d5e6, 0x9ea80509,\n    0xf22b017d, 0xa4173f70, 0xdd1e16c3, 0x15e0d7f9, 0x50b1b887, 0x2b9f4fd5, 0x625aba82, 0x6a017962,\n    0x2ec01b9c, 0x15488aa9, 0xd716e740, 0x40055a2c, 0x93d29a22, 0xe32dbf9a, 0x058745b9, 0x3453dc1e,\n    0xd699296e, 0x496cff6f, 0x1c9f4986, 0xdfe2ed07, 0xb87242d1, 0x19de7eae, 0x053e561a, 0x15ad6f8c,\n    0x66626c1c, 0x7154c24c, 0xea082b2a, 0x93eb2939, 0x17dcb0f0, 0x58d4f2ae, 0x9ea294fb, 0x52cf564c,\n    0x9883fe66, 0x2ec40581, 0x763953c3, 0x01d6692e, 0xd3a0c108, 0xa1e7160e, 0xe4f2dfa6, 0x693ed285,\n    0x74904698, 0x4c2b0edd, 0x4f757656, 0x5d393378, 0xa132234f, 0x3d321c5d, 0xc3f5e194, 0x4b269301,\n    0xc79f022f, 0x3c997e7e, 0x5e4f9504, 0x3ffafbbd, 0x76f7ad0e, 0x296693f4, 0x3d1fce6f, 0xc61e45be,\n    0xd3b5ab34, 0xf72bf9b7, 0x1b0434c0, 0x4e72b567, 0x5592a33d, 0xb5229301, 0xcfd2a87f, 0x60aeb767,\n    0x1814386b, 0x30bcc33d, 0x38a0c07d, 0xfd1606f2, 0xc363519b, 0x589dd390, 0x5479f8e6, 0x1cb8d647,\n    0x97fd61a9, 0xea7759f4, 0x2d57539d, 0x569a58cf, 0xe84e63ad, 0x462e1b78, 0x6580f87e, 0xf3817914,\n    0x91da55f4, 0x40a230f3, 0xd1988f35, 0xb6e318d2, 0x3ffa50bc, 0x3d40f021, 0xc3c0bdae, 0x4958c24c,\n    0x518f36b2, 0x84b1d370, 0x0fedce83, 0x878ddada, 0xf2a279c7, 0x94e01be8, 0x90716f4b, 0x954b8aa3);\n\n  sBox[7] = new Array(\n    0xe216300d, 0xbbddfffc, 0xa7ebdabd, 0x35648095, 0x7789f8b7, 0xe6c1121b, 0x0e241600, 0x052ce8b5,\n    0x11a9cfb0, 0xe5952f11, 0xece7990a, 0x9386d174, 0x2a42931c, 0x76e38111, 0xb12def3a, 0x37ddddfc,\n    0xde9adeb1, 0x0a0cc32c, 0xbe197029, 0x84a00940, 0xbb243a0f, 0xb4d137cf, 0xb44e79f0, 0x049eedfd,\n    0x0b15a15d, 0x480d3168, 0x8bbbde5a, 0x669ded42, 0xc7ece831, 0x3f8f95e7, 0x72df191b, 0x7580330d,\n    0x94074251, 0x5c7dcdfa, 0xabbe6d63, 0xaa402164, 0xb301d40a, 0x02e7d1ca, 0x53571dae, 0x7a3182a2,\n    0x12a8ddec, 0xfdaa335d, 0x176f43e8, 0x71fb46d4, 0x38129022, 0xce949ad4, 0xb84769ad, 0x965bd862,\n    0x82f3d055, 0x66fb9767, 0x15b80b4e, 0x1d5b47a0, 0x4cfde06f, 0xc28ec4b8, 0x57e8726e, 0x647a78fc,\n    0x99865d44, 0x608bd593, 0x6c200e03, 0x39dc5ff6, 0x5d0b00a3, 0xae63aff2, 0x7e8bd632, 0x70108c0c,\n    0xbbd35049, 0x2998df04, 0x980cf42a, 0x9b6df491, 0x9e7edd53, 0x06918548, 0x58cb7e07, 0x3b74ef2e,\n    0x522fffb1, 0xd24708cc, 0x1c7e27cd, 0xa4eb215b, 0x3cf1d2e2, 0x19b47a38, 0x424f7618, 0x35856039,\n    0x9d17dee7, 0x27eb35e6, 0xc9aff67b, 0x36baf5b8, 0x09c467cd, 0xc18910b1, 0xe11dbf7b, 0x06cd1af8,\n    0x7170c608, 0x2d5e3354, 0xd4de495a, 0x64c6d006, 0xbcc0c62c, 0x3dd00db3, 0x708f8f34, 0x77d51b42,\n    0x264f620f, 0x24b8d2bf, 0x15c1b79e, 0x46a52564, 0xf8d7e54e, 0x3e378160, 0x7895cda5, 0x859c15a5,\n    0xe6459788, 0xc37bc75f, 0xdb07ba0c, 0x0676a3ab, 0x7f229b1e, 0x31842e7b, 0x24259fd7, 0xf8bef472,\n    0x835ffcb8, 0x6df4c1f2, 0x96f5b195, 0xfd0af0fc, 0xb0fe134c, 0xe2506d3d, 0x4f9b12ea, 0xf215f225,\n    0xa223736f, 0x9fb4c428, 0x25d04979, 0x34c713f8, 0xc4618187, 0xea7a6e98, 0x7cd16efc, 0x1436876c,\n    0xf1544107, 0xbedeee14, 0x56e9af27, 0xa04aa441, 0x3cf7c899, 0x92ecbae6, 0xdd67016d, 0x151682eb,\n    0xa842eedf, 0xfdba60b4, 0xf1907b75, 0x20e3030f, 0x24d8c29e, 0xe139673b, 0xefa63fb8, 0x71873054,\n    0xb6f2cf3b, 0x9f326442, 0xcb15a4cc, 0xb01a4504, 0xf1e47d8d, 0x844a1be5, 0xbae7dfdc, 0x42cbda70,\n    0xcd7dae0a, 0x57e85b7a, 0xd53f5af6, 0x20cf4d8c, 0xcea4d428, 0x79d130a4, 0x3486ebfb, 0x33d3cddc,\n    0x77853b53, 0x37effcb5, 0xc5068778, 0xe580b3e6, 0x4e68b8f4, 0xc5c8b37e, 0x0d809ea2, 0x398feb7c,\n    0x132a4f94, 0x43b7950e, 0x2fee7d1c, 0x223613bd, 0xdd06caa2, 0x37df932b, 0xc4248289, 0xacf3ebc3,\n    0x5715f6b7, 0xef3478dd, 0xf267616f, 0xc148cbe4, 0x9052815e, 0x5e410fab, 0xb48a2465, 0x2eda7fa4,\n    0xe87b40e4, 0xe98ea084, 0x5889e9e1, 0xefd390fc, 0xdd07d35b, 0xdb485694, 0x38d7e5b2, 0x57720101,\n    0x730edebc, 0x5b643113, 0x94917e4f, 0x503c2fba, 0x646f1282, 0x7523d24a, 0xe0779695, 0xf9c17a8f,\n    0x7a5b2121, 0xd187b896, 0x29263a4d, 0xba510cdf, 0x81f47c9f, 0xad1163ed, 0xea7b5965, 0x1a00726e,\n    0x11403092, 0x00da6d77, 0x4a0cdd61, 0xad1f4603, 0x605bdfb0, 0x9eedc364, 0x22ebe6a8, 0xcee7d28a,\n    0xa0e736a0, 0x5564a6b9, 0x10853209, 0xc7eb8f37, 0x2de705ca, 0x8951570f, 0xdf09822b, 0xbd691a6c,\n    0xaa12e4f2, 0x87451c0f, 0xe0f6a27a, 0x3ada4819, 0x4cf1764f, 0x0d771c2b, 0x67cdb156, 0x350d8384,\n    0x5938fa0f, 0x42399ef3, 0x36997b07, 0x0e84093d, 0x4aa93e61, 0x8360d87b, 0x1fa98b0c, 0x1149382c,\n    0xe97625a5, 0x0614d1b7, 0x0e25244b, 0x0c768347, 0x589e8d82, 0x0d2059d1, 0xa466bb1e, 0xf8da0a82,\n    0x04f19130, 0xba6e4ec0, 0x99265164, 0x1ee7230d, 0x50b2ad80, 0xeaee6801, 0x8db2a283, 0xea8bf59e);\n\n};\n\nvar util = require('../../util');\n\nfunction cast5(key) {\n  this.cast5 = new openpgp_symenc_cast5();\n  this.cast5.setKey(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.cast5.encrypt(block);\n  }\n}\n\nmodule.exports = cast5;\nmodule.exports.blockSize = cast5.prototype.blockSize = 8;\nmodule.exports.keySize = cast5.prototype.keySize = 16;\n","//Paul Tero, July 2001\n//http://www.tero.co.uk/des/\n//\n//Optimised for performance with large blocks by Michael Hayworth, November 2001\n//http://www.netdealing.com\n//\n// Modified by Recurity Labs GmbH\n\n//THIS SOFTWARE IS PROVIDED \"AS IS\" AND\n//ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE\n//IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE\n//ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE\n//FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL\n//DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS\n//OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)\n//HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT\n//LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY\n//OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF\n//SUCH DAMAGE.\n\n//des\n//this takes the key, the message, and whether to encrypt or decrypt\n\n\n\n\nfunction des(keys, message, encrypt, mode, iv, padding) {\n  //declaring this locally speeds things up a bit\n  var spfunction1 = new Array(0x1010400, 0, 0x10000, 0x1010404, 0x1010004, 0x10404, 0x4, 0x10000, 0x400, 0x1010400,\n    0x1010404, 0x400, 0x1000404, 0x1010004, 0x1000000, 0x4, 0x404, 0x1000400, 0x1000400, 0x10400, 0x10400, 0x1010000,\n    0x1010000, 0x1000404, 0x10004, 0x1000004, 0x1000004, 0x10004, 0, 0x404, 0x10404, 0x1000000, 0x10000, 0x1010404, 0x4,\n    0x1010000, 0x1010400, 0x1000000, 0x1000000, 0x400, 0x1010004, 0x10000, 0x10400, 0x1000004, 0x400, 0x4, 0x1000404,\n    0x10404, 0x1010404, 0x10004, 0x1010000, 0x1000404, 0x1000004, 0x404, 0x10404, 0x1010400, 0x404, 0x1000400,\n    0x1000400, 0, 0x10004, 0x10400, 0, 0x1010004);\n  var spfunction2 = new Array(-0x7fef7fe0, -0x7fff8000, 0x8000, 0x108020, 0x100000, 0x20, -0x7fefffe0, -0x7fff7fe0, -\n    0x7fffffe0, -0x7fef7fe0, -0x7fef8000, -0x80000000, -0x7fff8000, 0x100000, 0x20, -0x7fefffe0, 0x108000, 0x100020, -\n    0x7fff7fe0, 0, -0x80000000, 0x8000, 0x108020, -0x7ff00000, 0x100020, -0x7fffffe0, 0, 0x108000, 0x8020, -0x7fef8000, -\n    0x7ff00000, 0x8020, 0, 0x108020, -0x7fefffe0, 0x100000, -0x7fff7fe0, -0x7ff00000, -0x7fef8000, 0x8000, -0x7ff00000, -\n    0x7fff8000, 0x20, -0x7fef7fe0, 0x108020, 0x20, 0x8000, -0x80000000, 0x8020, -0x7fef8000, 0x100000, -0x7fffffe0,\n    0x100020, -0x7fff7fe0, -0x7fffffe0, 0x100020, 0x108000, 0, -0x7fff8000, 0x8020, -0x80000000, -0x7fefffe0, -\n    0x7fef7fe0, 0x108000);\n  var spfunction3 = new Array(0x208, 0x8020200, 0, 0x8020008, 0x8000200, 0, 0x20208, 0x8000200, 0x20008, 0x8000008,\n    0x8000008, 0x20000, 0x8020208, 0x20008, 0x8020000, 0x208, 0x8000000, 0x8, 0x8020200, 0x200, 0x20200, 0x8020000,\n    0x8020008, 0x20208, 0x8000208, 0x20200, 0x20000, 0x8000208, 0x8, 0x8020208, 0x200, 0x8000000, 0x8020200, 0x8000000,\n    0x20008, 0x208, 0x20000, 0x8020200, 0x8000200, 0, 0x200, 0x20008, 0x8020208, 0x8000200, 0x8000008, 0x200, 0,\n    0x8020008, 0x8000208, 0x20000, 0x8000000, 0x8020208, 0x8, 0x20208, 0x20200, 0x8000008, 0x8020000, 0x8000208, 0x208,\n    0x8020000, 0x20208, 0x8, 0x8020008, 0x20200);\n  var spfunction4 = new Array(0x802001, 0x2081, 0x2081, 0x80, 0x802080, 0x800081, 0x800001, 0x2001, 0, 0x802000,\n    0x802000, 0x802081, 0x81, 0, 0x800080, 0x800001, 0x1, 0x2000, 0x800000, 0x802001, 0x80, 0x800000, 0x2001, 0x2080,\n    0x800081, 0x1, 0x2080, 0x800080, 0x2000, 0x802080, 0x802081, 0x81, 0x800080, 0x800001, 0x802000, 0x802081, 0x81, 0,\n    0, 0x802000, 0x2080, 0x800080, 0x800081, 0x1, 0x802001, 0x2081, 0x2081, 0x80, 0x802081, 0x81, 0x1, 0x2000, 0x800001,\n    0x2001, 0x802080, 0x800081, 0x2001, 0x2080, 0x800000, 0x802001, 0x80, 0x800000, 0x2000, 0x802080);\n  var spfunction5 = new Array(0x100, 0x2080100, 0x2080000, 0x42000100, 0x80000, 0x100, 0x40000000, 0x2080000,\n    0x40080100, 0x80000, 0x2000100, 0x40080100, 0x42000100, 0x42080000, 0x80100, 0x40000000, 0x2000000, 0x40080000,\n    0x40080000, 0, 0x40000100, 0x42080100, 0x42080100, 0x2000100, 0x42080000, 0x40000100, 0, 0x42000000, 0x2080100,\n    0x2000000, 0x42000000, 0x80100, 0x80000, 0x42000100, 0x100, 0x2000000, 0x40000000, 0x2080000, 0x42000100,\n    0x40080100, 0x2000100, 0x40000000, 0x42080000, 0x2080100, 0x40080100, 0x100, 0x2000000, 0x42080000, 0x42080100,\n    0x80100, 0x42000000, 0x42080100, 0x2080000, 0, 0x40080000, 0x42000000, 0x80100, 0x2000100, 0x40000100, 0x80000, 0,\n    0x40080000, 0x2080100, 0x40000100);\n  var spfunction6 = new Array(0x20000010, 0x20400000, 0x4000, 0x20404010, 0x20400000, 0x10, 0x20404010, 0x400000,\n    0x20004000, 0x404010, 0x400000, 0x20000010, 0x400010, 0x20004000, 0x20000000, 0x4010, 0, 0x400010, 0x20004010,\n    0x4000, 0x404000, 0x20004010, 0x10, 0x20400010, 0x20400010, 0, 0x404010, 0x20404000, 0x4010, 0x404000, 0x20404000,\n    0x20000000, 0x20004000, 0x10, 0x20400010, 0x404000, 0x20404010, 0x400000, 0x4010, 0x20000010, 0x400000, 0x20004000,\n    0x20000000, 0x4010, 0x20000010, 0x20404010, 0x404000, 0x20400000, 0x404010, 0x20404000, 0, 0x20400010, 0x10, 0x4000,\n    0x20400000, 0x404010, 0x4000, 0x400010, 0x20004010, 0, 0x20404000, 0x20000000, 0x400010, 0x20004010);\n  var spfunction7 = new Array(0x200000, 0x4200002, 0x4000802, 0, 0x800, 0x4000802, 0x200802, 0x4200800, 0x4200802,\n    0x200000, 0, 0x4000002, 0x2, 0x4000000, 0x4200002, 0x802, 0x4000800, 0x200802, 0x200002, 0x4000800, 0x4000002,\n    0x4200000, 0x4200800, 0x200002, 0x4200000, 0x800, 0x802, 0x4200802, 0x200800, 0x2, 0x4000000, 0x200800, 0x4000000,\n    0x200800, 0x200000, 0x4000802, 0x4000802, 0x4200002, 0x4200002, 0x2, 0x200002, 0x4000000, 0x4000800, 0x200000,\n    0x4200800, 0x802, 0x200802, 0x4200800, 0x802, 0x4000002, 0x4200802, 0x4200000, 0x200800, 0, 0x2, 0x4200802, 0,\n    0x200802, 0x4200000, 0x800, 0x4000002, 0x4000800, 0x800, 0x200002);\n  var spfunction8 = new Array(0x10001040, 0x1000, 0x40000, 0x10041040, 0x10000000, 0x10001040, 0x40, 0x10000000,\n    0x40040, 0x10040000, 0x10041040, 0x41000, 0x10041000, 0x41040, 0x1000, 0x40, 0x10040000, 0x10000040, 0x10001000,\n    0x1040, 0x41000, 0x40040, 0x10040040, 0x10041000, 0x1040, 0, 0, 0x10040040, 0x10000040, 0x10001000, 0x41040,\n    0x40000, 0x41040, 0x40000, 0x10041000, 0x1000, 0x40, 0x10040040, 0x1000, 0x41040, 0x10001000, 0x40, 0x10000040,\n    0x10040000, 0x10040040, 0x10000000, 0x40000, 0x10001040, 0, 0x10041040, 0x40040, 0x10000040, 0x10040000, 0x10001000,\n    0x10001040, 0, 0x10041040, 0x41000, 0x41000, 0x1040, 0x1040, 0x40040, 0x10000000, 0x10041000);\n\n  //create the 16 or 48 subkeys we will need\n  var m = 0,\n    i, j, temp, temp2, right1, right2, left, right, looping;\n  var cbcleft, cbcleft2, cbcright, cbcright2\n  var endloop, loopinc;\n  var len = message.length;\n  var chunk = 0;\n  //set up the loops for single and triple des\n  var iterations = keys.length == 32 ? 3 : 9; //single or triple des\n  if (iterations == 3) {\n    looping = encrypt ? new Array(0, 32, 2) : new Array(30, -2, -2);\n  } else {\n    looping = encrypt ? new Array(0, 32, 2, 62, 30, -2, 64, 96, 2) : new Array(94, 62, -2, 32, 64, 2, 30, -2, -2);\n  }\n\n  //pad the message depending on the padding parameter\n  //only add padding if encrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (encrypt) {\n    message = des_addPadding(message, padding);\n    len = message.length;\n  }\n\n  //store the result here\n  result = \"\";\n  tempresult = \"\";\n\n  if (mode == 1) { //CBC mode\n    cbcleft = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    cbcright = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    m = 0;\n  }\n\n  //loop through each 64 bit chunk of the message\n  while (m < len) {\n    left = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) | message\n      .charCodeAt(m++);\n    right = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) |\n      message.charCodeAt(m++);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        left ^= cbcleft;\n        right ^= cbcright;\n      } else {\n        cbcleft2 = cbcleft;\n        cbcright2 = cbcright;\n        cbcleft = left;\n        cbcright = right;\n      }\n    }\n\n    //first each 64 but chunk of the message must be permuted according to IP\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    left = ((left << 1) | (left >>> 31));\n    right = ((right << 1) | (right >>> 31));\n\n    //do this either 1 or 3 times for each chunk of the message\n    for (j = 0; j < iterations; j += 3) {\n      endloop = looping[j + 1];\n      loopinc = looping[j + 2];\n      //now go through and perform the encryption or decryption  \n      for (i = looping[j]; i != endloop; i += loopinc) { //for efficiency\n        right1 = right ^ keys[i];\n        right2 = ((right >>> 4) | (right << 28)) ^ keys[i + 1];\n        //the result is attained by passing these bytes through the S selection functions\n        temp = left;\n        left = right;\n        right = temp ^ (spfunction2[(right1 >>> 24) & 0x3f] | spfunction4[(right1 >>> 16) & 0x3f] | spfunction6[(right1 >>>\n          8) & 0x3f] | spfunction8[right1 & 0x3f] | spfunction1[(right2 >>> 24) & 0x3f] | spfunction3[(right2 >>> 16) &\n          0x3f] | spfunction5[(right2 >>> 8) & 0x3f] | spfunction7[right2 & 0x3f]);\n      }\n      temp = left;\n      left = right;\n      right = temp; //unreverse left and right\n    } //for either 1 or 3 iterations\n\n    //move then each one bit to the right\n    left = ((left >>> 1) | (left << 31));\n    right = ((right >>> 1) | (right << 31));\n\n    //now perform IP-1, which is IP in the opposite direction\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        cbcleft = left;\n        cbcright = right;\n      } else {\n        left ^= cbcleft2;\n        right ^= cbcright2;\n      }\n    }\n    tempresult += String.fromCharCode((left >>> 24), ((left >>> 16) & 0xff), ((left >>> 8) & 0xff), (left & 0xff), (\n      right >>> 24), ((right >>> 16) & 0xff), ((right >>> 8) & 0xff), (right & 0xff));\n\n    chunk += 8;\n    if (chunk == 512) {\n      result += tempresult;\n      tempresult = \"\";\n      chunk = 0;\n    }\n  } //for every 8 characters, or 64 bits in the message\n\n  //return the result as an array\n  result += tempresult;\n\n  //only remove padding if decrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (!encrypt) {\n    result = des_removePadding(result, padding);\n  }\n\n  return result;\n} //end of des\n\n\n\n//des_createKeys\n//this takes as input a 64 bit key (even though only 56 bits are used)\n//as an array of 2 integers, and returns 16 48 bit keys\n\nfunction des_createKeys(key) {\n  //declaring this locally speeds things up a bit\n  pc2bytes0 = new Array(0, 0x4, 0x20000000, 0x20000004, 0x10000, 0x10004, 0x20010000, 0x20010004, 0x200, 0x204,\n    0x20000200, 0x20000204, 0x10200, 0x10204, 0x20010200, 0x20010204);\n  pc2bytes1 = new Array(0, 0x1, 0x100000, 0x100001, 0x4000000, 0x4000001, 0x4100000, 0x4100001, 0x100, 0x101, 0x100100,\n    0x100101, 0x4000100, 0x4000101, 0x4100100, 0x4100101);\n  pc2bytes2 = new Array(0, 0x8, 0x800, 0x808, 0x1000000, 0x1000008, 0x1000800, 0x1000808, 0, 0x8, 0x800, 0x808,\n    0x1000000, 0x1000008, 0x1000800, 0x1000808);\n  pc2bytes3 = new Array(0, 0x200000, 0x8000000, 0x8200000, 0x2000, 0x202000, 0x8002000, 0x8202000, 0x20000, 0x220000,\n    0x8020000, 0x8220000, 0x22000, 0x222000, 0x8022000, 0x8222000);\n  pc2bytes4 = new Array(0, 0x40000, 0x10, 0x40010, 0, 0x40000, 0x10, 0x40010, 0x1000, 0x41000, 0x1010, 0x41010, 0x1000,\n    0x41000, 0x1010, 0x41010);\n  pc2bytes5 = new Array(0, 0x400, 0x20, 0x420, 0, 0x400, 0x20, 0x420, 0x2000000, 0x2000400, 0x2000020, 0x2000420,\n    0x2000000, 0x2000400, 0x2000020, 0x2000420);\n  pc2bytes6 = new Array(0, 0x10000000, 0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002, 0, 0x10000000,\n    0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002);\n  pc2bytes7 = new Array(0, 0x10000, 0x800, 0x10800, 0x20000000, 0x20010000, 0x20000800, 0x20010800, 0x20000, 0x30000,\n    0x20800, 0x30800, 0x20020000, 0x20030000, 0x20020800, 0x20030800);\n  pc2bytes8 = new Array(0, 0x40000, 0, 0x40000, 0x2, 0x40002, 0x2, 0x40002, 0x2000000, 0x2040000, 0x2000000, 0x2040000,\n    0x2000002, 0x2040002, 0x2000002, 0x2040002);\n  pc2bytes9 = new Array(0, 0x10000000, 0x8, 0x10000008, 0, 0x10000000, 0x8, 0x10000008, 0x400, 0x10000400, 0x408,\n    0x10000408, 0x400, 0x10000400, 0x408, 0x10000408);\n  pc2bytes10 = new Array(0, 0x20, 0, 0x20, 0x100000, 0x100020, 0x100000, 0x100020, 0x2000, 0x2020, 0x2000, 0x2020,\n    0x102000, 0x102020, 0x102000, 0x102020);\n  pc2bytes11 = new Array(0, 0x1000000, 0x200, 0x1000200, 0x200000, 0x1200000, 0x200200, 0x1200200, 0x4000000, 0x5000000,\n    0x4000200, 0x5000200, 0x4200000, 0x5200000, 0x4200200, 0x5200200);\n  pc2bytes12 = new Array(0, 0x1000, 0x8000000, 0x8001000, 0x80000, 0x81000, 0x8080000, 0x8081000, 0x10, 0x1010,\n    0x8000010, 0x8001010, 0x80010, 0x81010, 0x8080010, 0x8081010);\n  pc2bytes13 = new Array(0, 0x4, 0x100, 0x104, 0, 0x4, 0x100, 0x104, 0x1, 0x5, 0x101, 0x105, 0x1, 0x5, 0x101, 0x105);\n\n  //how many iterations (1 for des, 3 for triple des)\n  var iterations = key.length > 8 ? 3 : 1; //changed by Paul 16/6/2007 to use Triple DES for 9+ byte keys\n  //stores the return keys\n  var keys = new Array(32 * iterations);\n  //now define the left shifts which need to be done\n  var shifts = new Array(0, 0, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 0);\n  //other variables\n  var lefttemp, righttemp, m = 0,\n    n = 0,\n    temp;\n\n  for (var j = 0; j < iterations; j++) { //either 1 or 3 iterations\n    left = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n    right = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 2) ^ right) & 0x33333333;\n    right ^= temp;\n    left ^= (temp << 2);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    //the right side needs to be shifted and to get the last four bits of the left side\n    temp = (left << 8) | ((right >>> 20) & 0x000000f0);\n    //left needs to be put upside down\n    left = (right << 24) | ((right << 8) & 0xff0000) | ((right >>> 8) & 0xff00) | ((right >>> 24) & 0xf0);\n    right = temp;\n\n    //now go through and perform these shifts on the left and right keys\n    for (i = 0; i < shifts.length; i++) {\n      //shift the keys either one or two bits to the left\n      if (shifts[i]) {\n        left = (left << 2) | (left >>> 26);\n        right = (right << 2) | (right >>> 26);\n      } else {\n        left = (left << 1) | (left >>> 27);\n        right = (right << 1) | (right >>> 27);\n      }\n      left &= -0xf;\n      right &= -0xf;\n\n      //now apply PC-2, in such a way that E is easier when encrypting or decrypting\n      //this conversion will look like PC-2 except only the last 6 bits of each byte are used\n      //rather than 48 consecutive bits and the order of lines will be according to \n      //how the S selection functions will be applied: S2, S4, S6, S8, S1, S3, S5, S7\n      lefttemp = pc2bytes0[left >>> 28] | pc2bytes1[(left >>> 24) & 0xf] | pc2bytes2[(left >>> 20) & 0xf] | pc2bytes3[(\n        left >>> 16) & 0xf] | pc2bytes4[(left >>> 12) & 0xf] | pc2bytes5[(left >>> 8) & 0xf] | pc2bytes6[(left >>> 4) &\n        0xf];\n      righttemp = pc2bytes7[right >>> 28] | pc2bytes8[(right >>> 24) & 0xf] | pc2bytes9[(right >>> 20) & 0xf] |\n        pc2bytes10[(right >>> 16) & 0xf] | pc2bytes11[(right >>> 12) & 0xf] | pc2bytes12[(right >>> 8) & 0xf] |\n        pc2bytes13[(right >>> 4) & 0xf];\n      temp = ((righttemp >>> 16) ^ lefttemp) & 0x0000ffff;\n      keys[n++] = lefttemp ^ temp;\n      keys[n++] = righttemp ^ (temp << 16);\n    }\n  } //for each iterations\n  //return the keys we've created\n  return keys;\n} //end of des_createKeys\n\n\nfunction des_addPadding(message, padding) {\n  var padLength = 8 - (message.length % 8);\n  if ((padding == 2) && (padLength < 8)) { //pad the message with spaces\n    message += \"        \".substr(0, padLength);\n  } else if (padding == 1) { //PKCS7 padding\n    message += String.fromCharCode(padLength, padLength, padLength, padLength, padLength, padLength, padLength,\n      padLength).substr(0, padLength);\n  } else if (!padding && (padLength < 8)) { //pad the message out with null bytes\n    message += \"\\0\\0\\0\\0\\0\\0\\0\\0\".substr(0, padLength);\n  }\n  return message;\n}\n\nfunction des_removePadding(message, padding) {\n  if (padding == 2) { // space padded\n    message = message.replace(/ *$/g, \"\");\n  } else if (padding == 1) { // PKCS7\n    var padCount = message.charCodeAt(message.length - 1);\n    message = message.substr(0, message.length - padCount);\n  } else if (!padding) { // null padding\n    message = message.replace(/\\0*$/g, \"\");\n  }\n  return message;\n}\n\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction Des(key) {\n  this.key = [];\n\n  for (var i = 0; i < 3; i++) {\n    this.key.push(key.substr(i * 8, 8));\n  }\n\n  this.encrypt = function(block) {\n    return util.str2bin(des(des_createKeys(this.key[2]),\n      des(des_createKeys(this.key[1]),\n      des(des_createKeys(this.key[0]),\n      util.bin2str(block), true, 0, null, null),\n      false, 0, null, null), true, 0, null, null));\n  }\n}\n\nDes.keySize = Des.prototype.keySize = 24;\nDes.blockSize = Des.prototype.blockSize = 8;\n\n// This is \"original\" DES - Des is actually Triple DES.\n// This is only exported so we can unit test.\n\nfunction OriginalDes(key) {\n  this.key = key;\n\n  this.encrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), true, 0, null, padding));\n  }\n\n  this.decrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), false, 0, null, padding));\n  }\n}\n\nmodule.exports = {\n  des: Des,\n  originalDes: OriginalDes\n}\n","var desModule = require('./des.js');\n\nmodule.exports = {\n  des: desModule['des'],\n  originalDes: desModule['originalDes'],\n  cast5: require('./cast5.js'),\n  twofish: require('./twofish.js'),\n  blowfish: require('./blowfish.js')\n}\n\nvar aes = require('./aes.js');\n\nfor (var i in aes) {\n  module.exports['aes' + i] = aes[i];\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Cipher.js\n * A block-cipher algorithm implementation on JavaScript\n * See Cipher.readme.txt for further information.\n *\n * Copyright(c) 2009 Atsushi Oka [ http://oka.nu/ ]\n * This script file is distributed under the LGPL\n *\n * ACKNOWLEDGMENT\n *\n *     The main subroutines are written by Michiel van Everdingen.\n * \n *     Michiel van Everdingen\n *     http://home.versatel.nl/MAvanEverdingen/index.html\n * \n *     All rights for these routines are reserved to Michiel van Everdingen.\n *\n */\n\n\n\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n//Math\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nvar MAXINT = 0xFFFFFFFF;\n\nfunction rotb(b, n) {\n  return (b << n | b >>> (8 - n)) & 0xFF;\n}\n\nfunction rotw(w, n) {\n  return (w << n | w >>> (32 - n)) & MAXINT;\n}\n\nfunction getW(a, i) {\n  return a[i] | a[i + 1] << 8 | a[i + 2] << 16 | a[i + 3] << 24;\n}\n\nfunction setW(a, i, w) {\n  a.splice(i, 4, w & 0xFF, (w >>> 8) & 0xFF, (w >>> 16) & 0xFF, (w >>> 24) & 0xFF);\n}\n\nfunction setWInv(a, i, w) {\n  a.splice(i, 4, (w >>> 24) & 0xFF, (w >>> 16) & 0xFF, (w >>> 8) & 0xFF, w & 0xFF);\n}\n\nfunction getB(x, n) {\n  return (x >>> (n * 8)) & 0xFF;\n}\n\nfunction getNrBits(i) {\n  var n = 0;\n  while (i > 0) {\n    n++;\n    i >>>= 1;\n  }\n  return n;\n}\n\nfunction getMask(n) {\n  return (1 << n) - 1;\n}\n\n//added 2008/11/13 XXX MUST USE ONE-WAY HASH FUNCTION FOR SECURITY REASON\n\nfunction randByte() {\n  return Math.floor(Math.random() * 256);\n}\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n// Twofish\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nfunction createTwofish() {\n  //\n  var keyBytes = null;\n  var dataBytes = null;\n  var dataOffset = -1;\n  // var dataLength = -1;\n  var algorithmName = null;\n  // var idx2 = -1;\n  //\n\n  algorithmName = \"twofish\";\n\n  var tfsKey = [];\n  var tfsM = [\n    [],\n    [],\n    [],\n    []\n  ];\n\n  function tfsInit(key) {\n    keyBytes = key;\n    var i, a, b, c, d, meKey = [],\n      moKey = [],\n      inKey = [];\n    var kLen;\n    var sKey = [];\n    var f01, f5b, fef;\n\n    var q0 = [\n      [8, 1, 7, 13, 6, 15, 3, 2, 0, 11, 5, 9, 14, 12, 10, 4],\n      [2, 8, 11, 13, 15, 7, 6, 14, 3, 1, 9, 4, 0, 10, 12, 5]\n    ];\n    var q1 = [\n      [14, 12, 11, 8, 1, 2, 3, 5, 15, 4, 10, 6, 7, 0, 9, 13],\n      [1, 14, 2, 11, 4, 12, 3, 7, 6, 13, 10, 5, 15, 9, 0, 8]\n    ];\n    var q2 = [\n      [11, 10, 5, 14, 6, 13, 9, 0, 12, 8, 15, 3, 2, 4, 7, 1],\n      [4, 12, 7, 5, 1, 6, 9, 10, 0, 14, 13, 8, 2, 11, 3, 15]\n    ];\n    var q3 = [\n      [13, 7, 15, 4, 1, 2, 6, 14, 9, 11, 3, 0, 8, 5, 12, 10],\n      [11, 9, 5, 1, 12, 3, 13, 14, 6, 4, 7, 15, 2, 0, 8, 10]\n    ];\n    var ror4 = [0, 8, 1, 9, 2, 10, 3, 11, 4, 12, 5, 13, 6, 14, 7, 15];\n    var ashx = [0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 5, 14, 7];\n    var q = [\n      [],\n      []\n    ];\n    var m = [\n      [],\n      [],\n      [],\n      []\n    ];\n\n    function ffm5b(x) {\n      return x ^ (x >> 2) ^ [0, 90, 180, 238][x & 3];\n    }\n\n    function ffmEf(x) {\n      return x ^ (x >> 1) ^ (x >> 2) ^ [0, 238, 180, 90][x & 3];\n    }\n\n    function mdsRem(p, q) {\n      var i, t, u;\n      for (i = 0; i < 8; i++) {\n        t = q >>> 24;\n        q = ((q << 8) & MAXINT) | p >>> 24;\n        p = (p << 8) & MAXINT;\n        u = t << 1;\n        if (t & 128) {\n          u ^= 333;\n        }\n        q ^= t ^ (u << 16);\n        u ^= t >>> 1;\n        if (t & 1) {\n          u ^= 166;\n        }\n        q ^= u << 24 | u << 8;\n      }\n      return q;\n    }\n\n    function qp(n, x) {\n      var a, b, c, d;\n      a = x >> 4;\n      b = x & 15;\n      c = q0[n][a ^ b];\n      d = q1[n][ror4[b] ^ ashx[a]];\n      return q3[n][ror4[d] ^ ashx[c]] << 4 | q2[n][c ^ d];\n    }\n\n    function hFun(x, key) {\n      var a = getB(x, 0),\n        b = getB(x, 1),\n        c = getB(x, 2),\n        d = getB(x, 3);\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(key[3], 0);\n          b = q[0][b] ^ getB(key[3], 1);\n          c = q[0][c] ^ getB(key[3], 2);\n          d = q[1][d] ^ getB(key[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(key[2], 0);\n          b = q[1][b] ^ getB(key[2], 1);\n          c = q[0][c] ^ getB(key[2], 2);\n          d = q[0][d] ^ getB(key[2], 3);\n        case 2:\n          a = q[0][q[0][a] ^ getB(key[1], 0)] ^ getB(key[0], 0);\n          b = q[0][q[1][b] ^ getB(key[1], 1)] ^ getB(key[0], 1);\n          c = q[1][q[0][c] ^ getB(key[1], 2)] ^ getB(key[0], 2);\n          d = q[1][q[1][d] ^ getB(key[1], 3)] ^ getB(key[0], 3);\n      }\n      return m[0][a] ^ m[1][b] ^ m[2][c] ^ m[3][d];\n    }\n\n    keyBytes = keyBytes.slice(0, 32);\n    i = keyBytes.length;\n    while (i != 16 && i != 24 && i != 32)\n      keyBytes[i++] = 0;\n\n    for (i = 0; i < keyBytes.length; i += 4) {\n      inKey[i >> 2] = getW(keyBytes, i);\n    }\n    for (i = 0; i < 256; i++) {\n      q[0][i] = qp(0, i);\n      q[1][i] = qp(1, i);\n    }\n    for (i = 0; i < 256; i++) {\n      f01 = q[1][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[0][i] = f01 + (f5b << 8) + (fef << 16) + (fef << 24);\n      m[2][i] = f5b + (fef << 8) + (f01 << 16) + (fef << 24);\n      f01 = q[0][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[1][i] = fef + (fef << 8) + (f5b << 16) + (f01 << 24);\n      m[3][i] = f5b + (f01 << 8) + (fef << 16) + (f5b << 24);\n    }\n\n    kLen = inKey.length / 2;\n    for (i = 0; i < kLen; i++) {\n      a = inKey[i + i];\n      meKey[i] = a;\n      b = inKey[i + i + 1];\n      moKey[i] = b;\n      sKey[kLen - i - 1] = mdsRem(a, b);\n    }\n    for (i = 0; i < 40; i += 2) {\n      a = 0x1010101 * i;\n      b = a + 0x1010101;\n      a = hFun(a, meKey);\n      b = rotw(hFun(b, moKey), 8);\n      tfsKey[i] = (a + b) & MAXINT;\n      tfsKey[i + 1] = rotw(a + 2 * b, 9);\n    }\n    for (i = 0; i < 256; i++) {\n      a = b = c = d = i;\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(sKey[3], 0);\n          b = q[0][b] ^ getB(sKey[3], 1);\n          c = q[0][c] ^ getB(sKey[3], 2);\n          d = q[1][d] ^ getB(sKey[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(sKey[2], 0);\n          b = q[1][b] ^ getB(sKey[2], 1);\n          c = q[0][c] ^ getB(sKey[2], 2);\n          d = q[0][d] ^ getB(sKey[2], 3);\n        case 2:\n          tfsM[0][i] = m[0][q[0][q[0][a] ^ getB(sKey[1], 0)] ^ getB(sKey[0], 0)];\n          tfsM[1][i] = m[1][q[0][q[1][b] ^ getB(sKey[1], 1)] ^ getB(sKey[0], 1)];\n          tfsM[2][i] = m[2][q[1][q[0][c] ^ getB(sKey[1], 2)] ^ getB(sKey[0], 2)];\n          tfsM[3][i] = m[3][q[1][q[1][d] ^ getB(sKey[1], 3)] ^ getB(sKey[0], 3)];\n      }\n    }\n  }\n\n  function tfsG0(x) {\n    return tfsM[0][getB(x, 0)] ^ tfsM[1][getB(x, 1)] ^ tfsM[2][getB(x, 2)] ^ tfsM[3][getB(x, 3)];\n  }\n\n  function tfsG1(x) {\n    return tfsM[0][getB(x, 3)] ^ tfsM[1][getB(x, 0)] ^ tfsM[2][getB(x, 1)] ^ tfsM[3][getB(x, 2)];\n  }\n\n  function tfsFrnd(r, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2] ^ (a + b + tfsKey[4 * r + 8]) & MAXINT, 31);\n    blk[3] = rotw(blk[3], 1) ^ (a + 2 * b + tfsKey[4 * r + 9]) & MAXINT;\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0] ^ (a + b + tfsKey[4 * r + 10]) & MAXINT, 31);\n    blk[1] = rotw(blk[1], 1) ^ (a + 2 * b + tfsKey[4 * r + 11]) & MAXINT;\n  }\n\n  function tfsIrnd(i, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2], 1) ^ (a + b + tfsKey[4 * i + 10]) & MAXINT;\n    blk[3] = rotw(blk[3] ^ (a + 2 * b + tfsKey[4 * i + 11]) & MAXINT, 31);\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0], 1) ^ (a + b + tfsKey[4 * i + 8]) & MAXINT;\n    blk[1] = rotw(blk[1] ^ (a + 2 * b + tfsKey[4 * i + 9]) & MAXINT, 31);\n  }\n\n  function tfsClose() {\n    tfsKey = [];\n    tfsM = [\n      [],\n      [],\n      [],\n      []\n    ];\n  }\n\n  function tfsEncrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[0],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[1],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[2],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[3]\n    ];\n    for (var j = 0; j < 8; j++) {\n      tfsFrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[4]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[5]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[6]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[7]);\n    dataOffset += 16;\n    return dataBytes;\n  }\n\n  function tfsDecrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[4],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[5],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[6],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[7]\n    ];\n    for (var j = 7; j >= 0; j--) {\n      tfsIrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[0]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[1]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[2]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[3]);\n    dataOffset += 16;\n  }\n\n  // added by Recurity Labs\n\n  function tfsFinal() {\n    return dataBytes;\n  }\n\n  return {\n    name: \"twofish\",\n    blocksize: 128 / 8,\n    open: tfsInit,\n    close: tfsClose,\n    encrypt: tfsEncrypt,\n    decrypt: tfsDecrypt,\n    // added by Recurity Labs\n    finalize: tfsFinal\n  };\n}\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction TFencrypt(block, key) {\n  var block_copy = [].concat(block);\n  var tf = createTwofish();\n  tf.open(util.str2bin(key), 0);\n  var result = tf.encrypt(block_copy, 0);\n  tf.close();\n  return result;\n}\n\nfunction TF(key) {\n  this.tf = createTwofish();\n  this.tf.open(util.str2bin(key), 0);\n\n  this.encrypt = function(block) {\n    return this.tf.encrypt([].concat(block), 0);\n  }\n}\n\n\nmodule.exports = TF;\nmodule.exports.keySize = TF.prototype.keySize = 32;\nmodule.exports.blockSize = TF.prototype.blockSize = 16;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar random = require('./random.js'),\n  cipher = require('./cipher'),\n  publicKey = require('./public_key'),\n  type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Encrypts data using the specified public key multiprecision integers \n   * and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi[]} if RSA an openpgp_type_mpi; \n   * if elgamal encryption an array of two openpgp_type_mpi is returned; otherwise null\n   */\n  publicKeyEncrypt: function(algo, publicMPIs, data) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          var rsa = new publicKey.rsa();\n          var n = publicMPIs[0].toBigInteger();\n          var e = publicMPIs[1].toBigInteger();\n          var m = data.toBigInteger();\n          return [rsa.encrypt(m, e, n)];\n\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var p = publicMPIs[0].toBigInteger();\n          var g = publicMPIs[1].toBigInteger();\n          var y = publicMPIs[2].toBigInteger();\n          var m = data.toBigInteger();\n          return elgamal.encrypt(m, g, p, y);\n\n        default:\n          return [];\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n  /**\n   * Decrypts data using the specified public key multiprecision integers of the private key,\n   * the specified secretMPIs of the private key and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers \n   * of the public key part of the private key\n   * @param {openpgp_type_mpi[]} secretMPIs Algorithm dependent multiprecision integers \n   * of the private key used\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi} returns a big integer containing the decrypted data; otherwise null\n   */\n\n  publicKeyDecrypt: function(algo, keyIntegers, dataIntegers) {\n    var bn = (function() {\n      switch (algo) {\n        case 'rsa_encrypt_sign':\n        case 'rsa_encrypt':\n          var rsa = new publicKey.rsa();\n          // 0 and 1 are the public key.\n          var d = keyIntegers[2].toBigInteger();\n          var p = keyIntegers[3].toBigInteger();\n          var q = keyIntegers[4].toBigInteger();\n          var u = keyIntegers[5].toBigInteger();\n          var m = dataIntegers[0].toBigInteger();\n          return rsa.decrypt(m, d, p, q, u);\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var x = keyIntegers[3].toBigInteger();\n          var c1 = dataIntegers[0].toBigInteger();\n          var c2 = dataIntegers[1].toBigInteger();\n          var p = keyIntegers[0].toBigInteger();\n          return elgamal.decrypt(c1, c2, p, x);\n        default:\n          return null;\n      }\n    })();\n\n    var result = new type_mpi();\n    result.fromBigInteger(bn);\n    return result;\n  },\n\n  /** Returns the number of integers comprising the private key of an algorithm\n   * @param {openpgp.publickey} algo The public key algorithm\n   * @return {Integer} The number of integers.\n   */\n  getPrivateMpiCount: function(algo) {\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        //   Algorithm-Specific Fields for RSA secret keys:\n        //   - multiprecision integer (MPI) of RSA secret exponent d.\n        //   - MPI of RSA secret prime value p.\n        //   - MPI of RSA secret prime value q (p < q).\n        //   - MPI of u, the multiplicative inverse of p, mod q.\n        return 4;\n      case 'elgamal':\n        // Algorithm-Specific Fields for Elgamal secret keys:\n        //   - MPI of Elgamal secret exponent x.\n        return 1;\n      case 'dsa':\n        // Algorithm-Specific Fields for DSA secret keys:\n        //   - MPI of DSA secret exponent x.\n        return 1;\n      default:\n        throw new Error('Unknown algorithm');\n    }\n  },\n\n  getPublicMpiCount: function(algo) {\n    // - A series of multiprecision integers comprising the key material:\n    //   Algorithm-Specific Fields for RSA public keys:\n    //       - a multiprecision integer (MPI) of RSA public modulus n;\n    //       - an MPI of RSA public encryption exponent e.\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        return 2;\n\n        //   Algorithm-Specific Fields for Elgamal public keys:\n        //     - MPI of Elgamal prime p;\n        //     - MPI of Elgamal group generator g;\n        //     - MPI of Elgamal public key value y (= g**x mod p where x  is secret).\n      case 'elgamal':\n        return 3;\n\n        //   Algorithm-Specific Fields for DSA public keys:\n        //       - MPI of DSA prime p;\n        //       - MPI of DSA group order q (q is a prime divisor of p-1);\n        //       - MPI of DSA group generator g;\n        //       - MPI of DSA public-key value y (= g**x mod p where x  is secret).\n      case 'dsa':\n        return 4;\n\n      default:\n        throw new Error('Unknown algorithm.');\n    }\n  },\n\n  generateMpi: function(algo, bits) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n        case 'rsa_sign':\n          //remember \"publicKey\" refers to the crypto/public_key dir\n          var rsa = new publicKey.rsa();\n          var keyObject = rsa.generate(bits, \"10001\");\n          var output = [];\n          output.push(keyObject.n);\n          output.push(keyObject.ee);\n          output.push(keyObject.d);\n          output.push(keyObject.p);\n          output.push(keyObject.q);\n          output.push(keyObject.u);\n          return output;\n        default:\n          throw new Error('Unsupported algorithm for key generation.');\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n\n  /**\n   * generate random byte prefix as string for the specified algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes with length equal to the block\n   * size of the cipher\n   */\n  getPrefixRandom: function(algo) {\n    return random.getRandomBytes(cipher[algo].blockSize);\n  },\n\n  /**\n   * Generating a session key for the specified symmetric algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes as a string to be used as a key\n   */\n  generateSessionKey: function(algo) {\n    return random.getRandomBytes(cipher[algo].keySize);\n  },\n\n};\n","var sha = require('./sha.js');\n\nmodule.exports = {\n  md5: require('./md5.js'),\n  sha1: sha.sha1,\n  sha256: sha.sha256,\n  sha224: sha.sha224,\n  sha384: sha.sha384,\n  sha512: sha.sha512,\n  ripemd: require('./ripe-md.js'),\n\n  /**\n   * Create a hash on the specified data using the specified algorithm\n   * @param {Integer} algo Hash algorithm type (see RFC4880 9.4)\n   * @param {String} data Data to be hashed\n   * @return {String} hash value\n   */\n  digest: function(algo, data) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return this.md5(data);\n      case 2:\n        // - SHA-1 [FIPS180]\n        return this.sha1(data);\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return this.ripemd(data);\n      case 8:\n        // - SHA256 [FIPS180]\n        return this.sha256(data);\n      case 9:\n        // - SHA384 [FIPS180]\n        return this.sha384(data);\n      case 10:\n        // - SHA512 [FIPS180]\n        return this.sha512(data);\n      case 11:\n        // - SHA224 [FIPS180]\n        return this.sha224(data);\n      default:\n        throw new Error('Invalid hash function.');\n    }\n  },\n\n  /**\n   * Returns the hash size in bytes of the specified hash algorithm type\n   * @param {Integer} algo Hash algorithm type (See RFC4880 9.4)\n   * @return {Integer} Size in bytes of the resulting hash\n   */\n  getHashByteLength: function(algo) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return 16;\n      case 2:\n        // - SHA-1 [FIPS180]\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return 20;\n      case 8:\n        // - SHA256 [FIPS180]\n        return 32;\n      case 9:\n        // - SHA384 [FIPS180]\n        return 48\n      case 10:\n        // - SHA512 [FIPS180]\n        return 64;\n      case 11:\n        // - SHA224 [FIPS180]\n        return 28;\n      default:\n        throw new Error('Invalid hash algorithm.');\n    }\n  }\n\n}\n","/**\n * A fast MD5 JavaScript implementation\n * Copyright (c) 2012 Joseph Myers\n * http://www.myersdaily.org/joseph/javascript/md5-text.html\n *\n * Permission to use, copy, modify, and distribute this software\n * and its documentation for any purposes and without\n * fee is hereby granted provided that this copyright notice\n * appears in all copies.\n *\n * Of course, this soft is provided \"as is\" without express or implied\n * warranty of any kind.\n */\n\nvar util = require('../../util/util.js');\n\nfunction MD5(entree) {\n  var hex = md5(entree);\n  var bin = util.hex2bin(hex);\n  return bin;\n}\n\nfunction md5cycle(x, k) {\n  var a = x[0],\n    b = x[1],\n    c = x[2],\n    d = x[3];\n\n  a = ff(a, b, c, d, k[0], 7, -680876936);\n  d = ff(d, a, b, c, k[1], 12, -389564586);\n  c = ff(c, d, a, b, k[2], 17, 606105819);\n  b = ff(b, c, d, a, k[3], 22, -1044525330);\n  a = ff(a, b, c, d, k[4], 7, -176418897);\n  d = ff(d, a, b, c, k[5], 12, 1200080426);\n  c = ff(c, d, a, b, k[6], 17, -1473231341);\n  b = ff(b, c, d, a, k[7], 22, -45705983);\n  a = ff(a, b, c, d, k[8], 7, 1770035416);\n  d = ff(d, a, b, c, k[9], 12, -1958414417);\n  c = ff(c, d, a, b, k[10], 17, -42063);\n  b = ff(b, c, d, a, k[11], 22, -1990404162);\n  a = ff(a, b, c, d, k[12], 7, 1804603682);\n  d = ff(d, a, b, c, k[13], 12, -40341101);\n  c = ff(c, d, a, b, k[14], 17, -1502002290);\n  b = ff(b, c, d, a, k[15], 22, 1236535329);\n\n  a = gg(a, b, c, d, k[1], 5, -165796510);\n  d = gg(d, a, b, c, k[6], 9, -1069501632);\n  c = gg(c, d, a, b, k[11], 14, 643717713);\n  b = gg(b, c, d, a, k[0], 20, -373897302);\n  a = gg(a, b, c, d, k[5], 5, -701558691);\n  d = gg(d, a, b, c, k[10], 9, 38016083);\n  c = gg(c, d, a, b, k[15], 14, -660478335);\n  b = gg(b, c, d, a, k[4], 20, -405537848);\n  a = gg(a, b, c, d, k[9], 5, 568446438);\n  d = gg(d, a, b, c, k[14], 9, -1019803690);\n  c = gg(c, d, a, b, k[3], 14, -187363961);\n  b = gg(b, c, d, a, k[8], 20, 1163531501);\n  a = gg(a, b, c, d, k[13], 5, -1444681467);\n  d = gg(d, a, b, c, k[2], 9, -51403784);\n  c = gg(c, d, a, b, k[7], 14, 1735328473);\n  b = gg(b, c, d, a, k[12], 20, -1926607734);\n\n  a = hh(a, b, c, d, k[5], 4, -378558);\n  d = hh(d, a, b, c, k[8], 11, -2022574463);\n  c = hh(c, d, a, b, k[11], 16, 1839030562);\n  b = hh(b, c, d, a, k[14], 23, -35309556);\n  a = hh(a, b, c, d, k[1], 4, -1530992060);\n  d = hh(d, a, b, c, k[4], 11, 1272893353);\n  c = hh(c, d, a, b, k[7], 16, -155497632);\n  b = hh(b, c, d, a, k[10], 23, -1094730640);\n  a = hh(a, b, c, d, k[13], 4, 681279174);\n  d = hh(d, a, b, c, k[0], 11, -358537222);\n  c = hh(c, d, a, b, k[3], 16, -722521979);\n  b = hh(b, c, d, a, k[6], 23, 76029189);\n  a = hh(a, b, c, d, k[9], 4, -640364487);\n  d = hh(d, a, b, c, k[12], 11, -421815835);\n  c = hh(c, d, a, b, k[15], 16, 530742520);\n  b = hh(b, c, d, a, k[2], 23, -995338651);\n\n  a = ii(a, b, c, d, k[0], 6, -198630844);\n  d = ii(d, a, b, c, k[7], 10, 1126891415);\n  c = ii(c, d, a, b, k[14], 15, -1416354905);\n  b = ii(b, c, d, a, k[5], 21, -57434055);\n  a = ii(a, b, c, d, k[12], 6, 1700485571);\n  d = ii(d, a, b, c, k[3], 10, -1894986606);\n  c = ii(c, d, a, b, k[10], 15, -1051523);\n  b = ii(b, c, d, a, k[1], 21, -2054922799);\n  a = ii(a, b, c, d, k[8], 6, 1873313359);\n  d = ii(d, a, b, c, k[15], 10, -30611744);\n  c = ii(c, d, a, b, k[6], 15, -1560198380);\n  b = ii(b, c, d, a, k[13], 21, 1309151649);\n  a = ii(a, b, c, d, k[4], 6, -145523070);\n  d = ii(d, a, b, c, k[11], 10, -1120210379);\n  c = ii(c, d, a, b, k[2], 15, 718787259);\n  b = ii(b, c, d, a, k[9], 21, -343485551);\n\n  x[0] = add32(a, x[0]);\n  x[1] = add32(b, x[1]);\n  x[2] = add32(c, x[2]);\n  x[3] = add32(d, x[3]);\n\n}\n\nfunction cmn(q, a, b, x, s, t) {\n  a = add32(add32(a, q), add32(x, t));\n  return add32((a << s) | (a >>> (32 - s)), b);\n}\n\nfunction ff(a, b, c, d, x, s, t) {\n  return cmn((b & c) | ((~b) & d), a, b, x, s, t);\n}\n\nfunction gg(a, b, c, d, x, s, t) {\n  return cmn((b & d) | (c & (~d)), a, b, x, s, t);\n}\n\nfunction hh(a, b, c, d, x, s, t) {\n  return cmn(b ^ c ^ d, a, b, x, s, t);\n}\n\nfunction ii(a, b, c, d, x, s, t) {\n  return cmn(c ^ (b | (~d)), a, b, x, s, t);\n}\n\nfunction md51(s) {\n  txt = '';\n  var n = s.length,\n    state = [1732584193, -271733879, -1732584194, 271733878],\n    i;\n  for (i = 64; i <= s.length; i += 64) {\n    md5cycle(state, md5blk(s.substring(i - 64, i)));\n  }\n  s = s.substring(i - 64);\n  var tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];\n  for (i = 0; i < s.length; i++)\n    tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3);\n  tail[i >> 2] |= 0x80 << ((i % 4) << 3);\n  if (i > 55) {\n    md5cycle(state, tail);\n    for (i = 0; i < 16; i++) tail[i] = 0;\n  }\n  tail[14] = n * 8;\n  md5cycle(state, tail);\n  return state;\n}\n\n/* there needs to be support for Unicode here,\n * unless we pretend that we can redefine the MD-5\n * algorithm for multi-byte characters (perhaps\n * by adding every four 16-bit characters and\n * shortening the sum to 32 bits). Otherwise\n * I suggest performing MD-5 as if every character\n * was two bytes--e.g., 0040 0025 = @%--but then\n * how will an ordinary MD-5 sum be matched?\n * There is no way to standardize text to something\n * like UTF-8 before transformation; speed cost is\n * utterly prohibitive. The JavaScript standard\n * itself needs to look at this: it should start\n * providing access to strings as preformed UTF-8\n * 8-bit unsigned value arrays.\n */\nfunction md5blk(s) { /* I figured global was faster.   */\n  var md5blks = [],\n    i; /* Andy King said do it this way. */\n  for (i = 0; i < 64; i += 4) {\n    md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) <<\n      24);\n  }\n  return md5blks;\n}\n\nvar hex_chr = '0123456789abcdef'.split('');\n\nfunction rhex(n) {\n  var s = '',\n    j = 0;\n  for (; j < 4; j++)\n    s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F];\n  return s;\n}\n\nfunction hex(x) {\n  for (var i = 0; i < x.length; i++)\n    x[i] = rhex(x[i]);\n  return x.join('');\n}\n\nfunction md5(s) {\n  return hex(md51(s));\n}\n\n/* this function is much faster,\nso if possible we use it. Some IEs\nare the only ones I know of that\nneed the idiotic second function,\ngenerated by an if clause.  */\n\nfunction add32(a, b) {\n  return (a + b) & 0xFFFFFFFF;\n}\n\nif (md5('hello') != '5d41402abc4b2a76b9719d911017c592') {\n  function add32(x, y) {\n    var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n      msw = (x >> 16) + (y >> 16) + (lsw >> 16);\n    return (msw << 16) | (lsw & 0xFFFF);\n  }\n}\n\nmodule.exports = MD5\n","/*\n * CryptoMX Tools\n * Copyright (C) 2004 - 2006 Derek Buitenhuis\n *\n * This program is free software; you can redistribute it and/or\n * modify it under the terms of the GNU General Public License\n * as published by the Free Software Foundation; either version 2\n * of the License, or (at your option) any later version.\n *\n * This program is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n * GNU General Public License for more details.\n *\n * You should have received a copy of the GNU General Public License\n * along with this program; if not, write to the Free Software\n * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.\n */\n\n/* Modified by Recurity Labs GmbH\n */\n\nvar RMDsize = 160;\nvar X = new Array();\n\nfunction ROL(x, n) {\n  return new Number((x << n) | (x >>> (32 - n)));\n}\n\nfunction F(x, y, z) {\n  return new Number(x ^ y ^ z);\n}\n\nfunction G(x, y, z) {\n  return new Number((x & y) | (~x & z));\n}\n\nfunction H(x, y, z) {\n  return new Number((x | ~y) ^ z);\n}\n\nfunction I(x, y, z) {\n  return new Number((x & z) | (y & ~z));\n}\n\nfunction J(x, y, z) {\n  return new Number(x ^ (y | ~z));\n}\n\nfunction mixOneRound(a, b, c, d, e, x, s, roundNumber) {\n  switch (roundNumber) {\n    case 0:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n    case 1:\n      a += G(b, c, d) + x + 0x5a827999;\n      break;\n    case 2:\n      a += H(b, c, d) + x + 0x6ed9eba1;\n      break;\n    case 3:\n      a += I(b, c, d) + x + 0x8f1bbcdc;\n      break;\n    case 4:\n      a += J(b, c, d) + x + 0xa953fd4e;\n      break;\n    case 5:\n      a += J(b, c, d) + x + 0x50a28be6;\n      break;\n    case 6:\n      a += I(b, c, d) + x + 0x5c4dd124;\n      break;\n    case 7:\n      a += H(b, c, d) + x + 0x6d703ef3;\n      break;\n    case 8:\n      a += G(b, c, d) + x + 0x7a6d76e9;\n      break;\n    case 9:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n\n    default:\n      document.write(\"Bogus round number\");\n      break;\n  }\n\n  a = ROL(a, s) + e;\n  c = ROL(c, 10);\n\n  a &= 0xffffffff;\n  b &= 0xffffffff;\n  c &= 0xffffffff;\n  d &= 0xffffffff;\n  e &= 0xffffffff;\n\n  var retBlock = new Array();\n  retBlock[0] = a;\n  retBlock[1] = b;\n  retBlock[2] = c;\n  retBlock[3] = d;\n  retBlock[4] = e;\n  retBlock[5] = x;\n  retBlock[6] = s;\n\n  return retBlock;\n}\n\nfunction MDinit(MDbuf) {\n  MDbuf[0] = 0x67452301;\n  MDbuf[1] = 0xefcdab89;\n  MDbuf[2] = 0x98badcfe;\n  MDbuf[3] = 0x10325476;\n  MDbuf[4] = 0xc3d2e1f0;\n}\n\nvar ROLs = [\n  [11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8],\n  [7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12],\n  [11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5],\n  [11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12],\n  [9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6],\n  [8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6],\n  [9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11],\n  [9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5],\n  [15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8],\n  [8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11]\n];\n\nvar indexes = [\n  [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15],\n  [7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8],\n  [3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12],\n  [1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2],\n  [4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13],\n  [5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12],\n  [6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2],\n  [15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13],\n  [8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14],\n  [12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11]\n];\n\nfunction compress(MDbuf, X) {\n  blockA = new Array();\n  blockB = new Array();\n\n  var retBlock;\n\n  for (var i = 0; i < 5; i++) {\n    blockA[i] = new Number(MDbuf[i]);\n    blockB[i] = new Number(MDbuf[i]);\n  }\n\n  var step = 0;\n  for (var j = 0; j < 5; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockA[(step + 0) % 5],\n        blockA[(step + 1) % 5],\n        blockA[(step + 2) % 5],\n        blockA[(step + 3) % 5],\n        blockA[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockA[(step + 0) % 5] = retBlock[0];\n      blockA[(step + 1) % 5] = retBlock[1];\n      blockA[(step + 2) % 5] = retBlock[2];\n      blockA[(step + 3) % 5] = retBlock[3];\n      blockA[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  step = 0;\n  for (var j = 5; j < 10; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockB[(step + 0) % 5],\n        blockB[(step + 1) % 5],\n        blockB[(step + 2) % 5],\n        blockB[(step + 3) % 5],\n        blockB[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockB[(step + 0) % 5] = retBlock[0];\n      blockB[(step + 1) % 5] = retBlock[1];\n      blockB[(step + 2) % 5] = retBlock[2];\n      blockB[(step + 3) % 5] = retBlock[3];\n      blockB[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  blockB[3] += blockA[2] + MDbuf[1];\n  MDbuf[1] = MDbuf[2] + blockA[3] + blockB[4];\n  MDbuf[2] = MDbuf[3] + blockA[4] + blockB[0];\n  MDbuf[3] = MDbuf[4] + blockA[0] + blockB[1];\n  MDbuf[4] = MDbuf[0] + blockA[1] + blockB[2];\n  MDbuf[0] = blockB[3];\n}\n\nfunction zeroX(X) {\n  for (var i = 0; i < 16; i++) {\n    X[i] = 0;\n  }\n}\n\nfunction MDfinish(MDbuf, strptr, lswlen, mswlen) {\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var i = 0; i < (lswlen & 63); i++) {\n    X[i >>> 2] ^= (strptr.charCodeAt(j++) & 255) << (8 * (i & 3));\n  }\n\n  X[(lswlen >>> 2) & 15] ^= 1 << (8 * (lswlen & 3) + 7);\n\n  if ((lswlen & 63) > 55) {\n    compress(MDbuf, X);\n    var X = new Array(16);\n    zeroX(X);\n  }\n\n  X[14] = lswlen << 3;\n  X[15] = (lswlen >>> 29) | (mswlen << 3);\n\n  compress(MDbuf, X);\n}\n\nfunction BYTES_TO_DWORD(fourChars) {\n  var tmp = (fourChars.charCodeAt(3) & 255) << 24;\n  tmp |= (fourChars.charCodeAt(2) & 255) << 16;\n  tmp |= (fourChars.charCodeAt(1) & 255) << 8;\n  tmp |= (fourChars.charCodeAt(0) & 255);\n\n  return tmp;\n}\n\nfunction RMD(message) {\n  var MDbuf = new Array(RMDsize / 32);\n  var hashcode = new Array(RMDsize / 8);\n  var length;\n  var nbytes;\n\n  MDinit(MDbuf);\n  length = message.length;\n\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var nbytes = length; nbytes > 63; nbytes -= 64) {\n    for (var i = 0; i < 16; i++) {\n      X[i] = BYTES_TO_DWORD(message.substr(j, 4));\n      j += 4;\n    }\n    compress(MDbuf, X);\n  }\n\n  MDfinish(MDbuf, message.substr(j), length, 0);\n\n  for (var i = 0; i < RMDsize / 8; i += 4) {\n    hashcode[i] = MDbuf[i >>> 2] & 255;\n    hashcode[i + 1] = (MDbuf[i >>> 2] >>> 8) & 255;\n    hashcode[i + 2] = (MDbuf[i >>> 2] >>> 16) & 255;\n    hashcode[i + 3] = (MDbuf[i >>> 2] >>> 24) & 255;\n  }\n\n  return hashcode;\n}\n\n\nfunction RMDstring(message) {\n  var hashcode = RMD(message);\n  var retString = \"\";\n\n  for (var i = 0; i < RMDsize / 8; i++) {\n    retString += String.fromCharCode(hashcode[i]);\n  }\n\n  return retString;\n}\n\nmodule.exports = RMDstring;\n","/* A JavaScript implementation of the SHA family of hashes, as defined in FIPS \n * PUB 180-2 as well as the corresponding HMAC implementation as defined in\n * FIPS PUB 198a\n *\n * Version 1.3 Copyright Brian Turek 2008-2010\n * Distributed under the BSD License\n * See http://jssha.sourceforge.net/ for more information\n *\n * Several functions taken from Paul Johnson\n */\n\n/* Modified by Recurity Labs GmbH\n * \n * This code has been slightly modified direct string output:\n * - bin2bstr has been added\n * - following wrappers of this library have been added:\n *   - str_sha1\n *   - str_sha256\n *   - str_sha224\n *   - str_sha384\n *   - str_sha512\n */\n\nvar jsSHA = (function() {\n\n  /*\n   * Configurable variables. Defaults typically work\n   */\n  /* Number of Bits Per character (8 for ASCII, 16 for Unicode) */\n  var charSize = 8,\n    /* base-64 pad character. \"=\" for strict RFC compliance */\n    b64pad = \"\",\n    /* hex output format. 0 - lowercase; 1 - uppercase */\n    hexCase = 0,\n\n    /*\n     * Int_64 is a object for 2 32-bit numbers emulating a 64-bit number\n     *\n     * @constructor\n     * @param {Number} msint_32 The most significant 32-bits of a 64-bit number\n     * @param {Number} lsint_32 The least significant 32-bits of a 64-bit number\n     */\n    Int_64 = function(msint_32, lsint_32) {\n      this.highOrder = msint_32;\n      this.lowOrder = lsint_32;\n    },\n\n    /*\n     * Convert a string to an array of big-endian words\n     * If charSize is ASCII, characters >255 have their hi-byte silently\n     * ignored.\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    str2binb = function(str) {\n      var bin = [],\n        mask = (1 << charSize) - 1,\n        length = str.length * charSize,\n        i;\n\n      for (i = 0; i < length; i += charSize) {\n        bin[i >> 5] |= (str.charCodeAt(i / charSize) & mask) <<\n          (32 - charSize - (i % 32));\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert a hex string to an array of big-endian words\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    hex2binb = function(str) {\n      var bin = [],\n        length = str.length,\n        i, num;\n\n      for (i = 0; i < length; i += 2) {\n        num = parseInt(str.substr(i, 2), 16);\n        if (!isNaN(num)) {\n          bin[i >> 3] |= num << (24 - (4 * (i % 8)));\n        } else {\n          return \"INVALID HEX STRING\";\n        }\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert an array of big-endian words to a hex string.\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to hexidecimal\n     *\t representation\n     * @return Hexidecimal representation of the parameter in String form\n     */\n    binb2hex = function(binarray) {\n      var hex_tab = (hexCase) ? \"0123456789ABCDEF\" : \"0123456789abcdef\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, srcByte;\n\n      for (i = 0; i < length; i += 1) {\n        srcByte = binarray[i >> 2] >> ((3 - (i % 4)) * 8);\n        str += hex_tab.charAt((srcByte >> 4) & 0xF) +\n          hex_tab.charAt(srcByte & 0xF);\n      }\n\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a base-64 string\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to base-64\n     *\t representation\n     * @return Base-64 encoded representation of the parameter in String form\n     */\n    binb2b64 = function(binarray) {\n      var tab = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\" +\n        \"0123456789+/\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, j,\n        triplet;\n\n      for (i = 0; i < length; i += 3) {\n        triplet = (((binarray[i >> 2] >> 8 * (3 - i % 4)) & 0xFF) << 16) |\n          (((binarray[i + 1 >> 2] >> 8 * (3 - (i + 1) % 4)) & 0xFF) << 8) |\n          ((binarray[i + 2 >> 2] >> 8 * (3 - (i + 2) % 4)) & 0xFF);\n        for (j = 0; j < 4; j += 1) {\n          if (i * 8 + j * 6 <= binarray.length * 32) {\n            str += tab.charAt((triplet >> 6 * (3 - j)) & 0x3F);\n          } else {\n            str += b64pad;\n          }\n        }\n      }\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a string\n     */\n    binb2str = function(bin) {\n      var str = \"\";\n      var mask = (1 << 8) - 1;\n      for (var i = 0; i < bin.length * 32; i += 8)\n        str += String.fromCharCode((bin[i >> 5] >>> (24 - i % 32)) & mask);\n      return str;\n    },\n    /*\n     * The 32-bit implementation of circular rotate left\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotl_32 = function(x, n) {\n      return (x << n) | (x >>> (32 - n));\n    },\n\n    /*\n     * The 32-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_32 = function(x, n) {\n      return (x >>> n) | (x << (32 - n));\n    },\n\n    /*\n     * The 64-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n        (x.highOrder >>> n) | (x.lowOrder << (32 - n)), (x.lowOrder >>> n) | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n        (x.lowOrder >>> n) | (x.highOrder << (32 - n)), (x.highOrder >>> n) | (x.lowOrder << (32 - n)));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of shift right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_32 = function(x, n) {\n      return x >>> n;\n    },\n\n    /*\n     * The 64-bit implementation of shift right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n          x.highOrder >>> n,\n          x.lowOrder >>> n | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n          0,\n          x.highOrder << (32 - n));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Parity function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    parity_32 = function(x, y, z) {\n      return x ^ y ^ z;\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_32 = function(x, y, z) {\n      return (x & y) ^ (~x & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^ (~x.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^ (~x.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_32 = function(x, y, z) {\n      return (x & y) ^ (x & z) ^ (y & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^\n        (x.highOrder & z.highOrder) ^\n        (y.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^\n        (x.lowOrder & z.lowOrder) ^\n        (y.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_32 = function(x) {\n      return rotr_32(x, 2) ^ rotr_32(x, 13) ^ rotr_32(x, 22);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_64 = function(x) {\n      var rotr28 = rotr_64(x, 28),\n        rotr34 = rotr_64(x, 34),\n        rotr39 = rotr_64(x, 39);\n\n      return new Int_64(\n        rotr28.highOrder ^ rotr34.highOrder ^ rotr39.highOrder,\n        rotr28.lowOrder ^ rotr34.lowOrder ^ rotr39.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_32 = function(x) {\n      return rotr_32(x, 6) ^ rotr_32(x, 11) ^ rotr_32(x, 25);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_64 = function(x) {\n      var rotr14 = rotr_64(x, 14),\n        rotr18 = rotr_64(x, 18),\n        rotr41 = rotr_64(x, 41);\n\n      return new Int_64(\n        rotr14.highOrder ^ rotr18.highOrder ^ rotr41.highOrder,\n        rotr14.lowOrder ^ rotr18.lowOrder ^ rotr41.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_32 = function(x) {\n      return rotr_32(x, 7) ^ rotr_32(x, 18) ^ shr_32(x, 3);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_64 = function(x) {\n      var rotr1 = rotr_64(x, 1),\n        rotr8 = rotr_64(x, 8),\n        shr7 = shr_64(x, 7);\n\n      return new Int_64(\n        rotr1.highOrder ^ rotr8.highOrder ^ shr7.highOrder,\n        rotr1.lowOrder ^ rotr8.lowOrder ^ shr7.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_32 = function(x) {\n      return rotr_32(x, 17) ^ rotr_32(x, 19) ^ shr_32(x, 10);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_64 = function(x) {\n      var rotr19 = rotr_64(x, 19),\n        rotr61 = rotr_64(x, 61),\n        shr6 = shr_64(x, 6);\n\n      return new Int_64(\n        rotr19.highOrder ^ rotr61.highOrder ^ shr6.highOrder,\n        rotr19.lowOrder ^ rotr61.lowOrder ^ shr6.lowOrder);\n    },\n\n    /*\n     * Add two 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument to be added\n     * @param {Number} y The second 32-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_32_2 = function(x, y) {\n      var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n        msw = (x >>> 16) + (y >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add four 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_32_4 = function(a, b, c, d) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add five 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @param {Number} e The fifth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_32_5 = function(a, b, c, d, e) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF) +\n        (e & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (e >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add two 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument to be added\n     * @param {Int_64} y The second 64-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_64_2 = function(x, y) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (x.lowOrder & 0xFFFF) + (y.lowOrder & 0xFFFF);\n      msw = (x.lowOrder >>> 16) + (y.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (x.highOrder & 0xFFFF) + (y.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (x.highOrder >>> 16) + (y.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add four 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_64_4 = function(a, b, c, d) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add five 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @param {Int_64} e The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_64_5 = function(a, b, c, d, e) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF) +\n        (e.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (e.lowOrder >>> 16) +\n        (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) +\n        (e.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) +\n        (e.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Calculates the SHA-1 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} message The binary array representation of the string to\n     *\t hash\n     * @param {Number} messageLen The number of bits in the message\n     * @return The array of integers representing the SHA-1 hash of message\n     */\n    coreSHA1 = function(message, messageLen) {\n      var W = [],\n        a, b, c, d, e, T, ch = ch_32,\n        parity = parity_32,\n        maj = maj_32,\n        rotl = rotl_32,\n        safeAdd_2 = safeAdd_32_2,\n        i, t,\n        safeAdd_5 = safeAdd_32_5,\n        appendedMessageLength,\n        H = [\n            0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0\n        ],\n        K = [\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6\n        ];\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - (messageLen % 32));\n      /* Append length of binary string in the position such that the new\n\t\tlength is a multiple of 512.  Logic does not work for even multiples\n\t\tof 512 but there can never be even multiples of 512 */\n      message[(((messageLen + 65) >> 9) << 4) + 15] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += 16) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n\n        for (t = 0; t < 80; t += 1) {\n          if (t < 16) {\n            W[t] = message[t + i];\n          } else {\n            W[t] = rotl(W[t - 3] ^ W[t - 8] ^ W[t - 14] ^ W[t - 16], 1);\n          }\n\n          if (t < 20) {\n            T = safeAdd_5(rotl(a, 5), ch(b, c, d), e, K[t], W[t]);\n          } else if (t < 40) {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          } else if (t < 60) {\n            T = safeAdd_5(rotl(a, 5), maj(b, c, d), e, K[t], W[t]);\n          } else {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          }\n\n          e = d;\n          d = c;\n          c = rotl(b, 30);\n          b = a;\n          a = T;\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n      }\n\n      return H;\n    },\n\n    /*\n     * Calculates the desired SHA-2 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} The binary array representation of the string to hash\n     * @param {Number} The number of bits in message\n     * @param {String} variant The desired SHA-2 variant\n     * @return The array of integers representing the SHA-2 hash of message\n     */\n    coreSHA2 = function(message, messageLen, variant) {\n      var a, b, c, d, e, f, g, h, T1, T2, H, numRounds, lengthPosition, i, t,\n        binaryStringInc, binaryStringMult, safeAdd_2, safeAdd_4, safeAdd_5,\n        gamma0, gamma1, sigma0, sigma1, ch, maj, Int, K, W = [],\n        appendedMessageLength;\n\n      /* Set up the various function handles and variable for the specific \n       * variant */\n      if (variant === \"SHA-224\" || variant === \"SHA-256\") {\n        /* 32-bit variant */\n        numRounds = 64;\n        lengthPosition = (((messageLen + 65) >> 9) << 4) + 15;\n        binaryStringInc = 16;\n        binaryStringMult = 1;\n        Int = Number;\n        safeAdd_2 = safeAdd_32_2;\n        safeAdd_4 = safeAdd_32_4;\n        safeAdd_5 = safeAdd_32_5;\n        gamma0 = gamma0_32;\n        gamma1 = gamma1_32;\n        sigma0 = sigma0_32;\n        sigma1 = sigma1_32;\n        maj = maj_32;\n        ch = ch_32;\n        K = [\n            0x428A2F98, 0x71374491, 0xB5C0FBCF, 0xE9B5DBA5,\n            0x3956C25B, 0x59F111F1, 0x923F82A4, 0xAB1C5ED5,\n            0xD807AA98, 0x12835B01, 0x243185BE, 0x550C7DC3,\n            0x72BE5D74, 0x80DEB1FE, 0x9BDC06A7, 0xC19BF174,\n            0xE49B69C1, 0xEFBE4786, 0x0FC19DC6, 0x240CA1CC,\n            0x2DE92C6F, 0x4A7484AA, 0x5CB0A9DC, 0x76F988DA,\n            0x983E5152, 0xA831C66D, 0xB00327C8, 0xBF597FC7,\n            0xC6E00BF3, 0xD5A79147, 0x06CA6351, 0x14292967,\n            0x27B70A85, 0x2E1B2138, 0x4D2C6DFC, 0x53380D13,\n            0x650A7354, 0x766A0ABB, 0x81C2C92E, 0x92722C85,\n            0xA2BFE8A1, 0xA81A664B, 0xC24B8B70, 0xC76C51A3,\n            0xD192E819, 0xD6990624, 0xF40E3585, 0x106AA070,\n            0x19A4C116, 0x1E376C08, 0x2748774C, 0x34B0BCB5,\n            0x391C0CB3, 0x4ED8AA4A, 0x5B9CCA4F, 0x682E6FF3,\n            0x748F82EE, 0x78A5636F, 0x84C87814, 0x8CC70208,\n            0x90BEFFFA, 0xA4506CEB, 0xBEF9A3F7, 0xC67178F2\n        ];\n\n        if (variant === \"SHA-224\") {\n          H = [\n              0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939,\n              0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4\n          ];\n        } else {\n          H = [\n              0x6A09E667, 0xBB67AE85, 0x3C6EF372, 0xA54FF53A,\n              0x510E527F, 0x9B05688C, 0x1F83D9AB, 0x5BE0CD19\n          ];\n        }\n      } else if (variant === \"SHA-384\" || variant === \"SHA-512\") {\n        /* 64-bit variant */\n        numRounds = 80;\n        lengthPosition = (((messageLen + 128) >> 10) << 5) + 31;\n        binaryStringInc = 32;\n        binaryStringMult = 2;\n        Int = Int_64;\n        safeAdd_2 = safeAdd_64_2;\n        safeAdd_4 = safeAdd_64_4;\n        safeAdd_5 = safeAdd_64_5;\n        gamma0 = gamma0_64;\n        gamma1 = gamma1_64;\n        sigma0 = sigma0_64;\n        sigma1 = sigma1_64;\n        maj = maj_64;\n        ch = ch_64;\n\n        K = [\n            new Int(0x428a2f98, 0xd728ae22), new Int(0x71374491, 0x23ef65cd),\n            new Int(0xb5c0fbcf, 0xec4d3b2f), new Int(0xe9b5dba5, 0x8189dbbc),\n            new Int(0x3956c25b, 0xf348b538), new Int(0x59f111f1, 0xb605d019),\n            new Int(0x923f82a4, 0xaf194f9b), new Int(0xab1c5ed5, 0xda6d8118),\n            new Int(0xd807aa98, 0xa3030242), new Int(0x12835b01, 0x45706fbe),\n            new Int(0x243185be, 0x4ee4b28c), new Int(0x550c7dc3, 0xd5ffb4e2),\n            new Int(0x72be5d74, 0xf27b896f), new Int(0x80deb1fe, 0x3b1696b1),\n            new Int(0x9bdc06a7, 0x25c71235), new Int(0xc19bf174, 0xcf692694),\n            new Int(0xe49b69c1, 0x9ef14ad2), new Int(0xefbe4786, 0x384f25e3),\n            new Int(0x0fc19dc6, 0x8b8cd5b5), new Int(0x240ca1cc, 0x77ac9c65),\n            new Int(0x2de92c6f, 0x592b0275), new Int(0x4a7484aa, 0x6ea6e483),\n            new Int(0x5cb0a9dc, 0xbd41fbd4), new Int(0x76f988da, 0x831153b5),\n            new Int(0x983e5152, 0xee66dfab), new Int(0xa831c66d, 0x2db43210),\n            new Int(0xb00327c8, 0x98fb213f), new Int(0xbf597fc7, 0xbeef0ee4),\n            new Int(0xc6e00bf3, 0x3da88fc2), new Int(0xd5a79147, 0x930aa725),\n            new Int(0x06ca6351, 0xe003826f), new Int(0x14292967, 0x0a0e6e70),\n            new Int(0x27b70a85, 0x46d22ffc), new Int(0x2e1b2138, 0x5c26c926),\n            new Int(0x4d2c6dfc, 0x5ac42aed), new Int(0x53380d13, 0x9d95b3df),\n            new Int(0x650a7354, 0x8baf63de), new Int(0x766a0abb, 0x3c77b2a8),\n            new Int(0x81c2c92e, 0x47edaee6), new Int(0x92722c85, 0x1482353b),\n            new Int(0xa2bfe8a1, 0x4cf10364), new Int(0xa81a664b, 0xbc423001),\n            new Int(0xc24b8b70, 0xd0f89791), new Int(0xc76c51a3, 0x0654be30),\n            new Int(0xd192e819, 0xd6ef5218), new Int(0xd6990624, 0x5565a910),\n            new Int(0xf40e3585, 0x5771202a), new Int(0x106aa070, 0x32bbd1b8),\n            new Int(0x19a4c116, 0xb8d2d0c8), new Int(0x1e376c08, 0x5141ab53),\n            new Int(0x2748774c, 0xdf8eeb99), new Int(0x34b0bcb5, 0xe19b48a8),\n            new Int(0x391c0cb3, 0xc5c95a63), new Int(0x4ed8aa4a, 0xe3418acb),\n            new Int(0x5b9cca4f, 0x7763e373), new Int(0x682e6ff3, 0xd6b2b8a3),\n            new Int(0x748f82ee, 0x5defb2fc), new Int(0x78a5636f, 0x43172f60),\n            new Int(0x84c87814, 0xa1f0ab72), new Int(0x8cc70208, 0x1a6439ec),\n            new Int(0x90befffa, 0x23631e28), new Int(0xa4506ceb, 0xde82bde9),\n            new Int(0xbef9a3f7, 0xb2c67915), new Int(0xc67178f2, 0xe372532b),\n            new Int(0xca273ece, 0xea26619c), new Int(0xd186b8c7, 0x21c0c207),\n            new Int(0xeada7dd6, 0xcde0eb1e), new Int(0xf57d4f7f, 0xee6ed178),\n            new Int(0x06f067aa, 0x72176fba), new Int(0x0a637dc5, 0xa2c898a6),\n            new Int(0x113f9804, 0xbef90dae), new Int(0x1b710b35, 0x131c471b),\n            new Int(0x28db77f5, 0x23047d84), new Int(0x32caab7b, 0x40c72493),\n            new Int(0x3c9ebe0a, 0x15c9bebc), new Int(0x431d67c4, 0x9c100d4c),\n            new Int(0x4cc5d4be, 0xcb3e42b6), new Int(0x597f299c, 0xfc657e2a),\n            new Int(0x5fcb6fab, 0x3ad6faec), new Int(0x6c44198c, 0x4a475817)\n        ];\n\n        if (variant === \"SHA-384\") {\n          H = [\n              new Int(0xcbbb9d5d, 0xc1059ed8), new Int(0x0629a292a, 0x367cd507),\n              new Int(0x9159015a, 0x3070dd17), new Int(0x0152fecd8, 0xf70e5939),\n              new Int(0x67332667, 0xffc00b31), new Int(0x98eb44a87, 0x68581511),\n              new Int(0xdb0c2e0d, 0x64f98fa7), new Int(0x047b5481d, 0xbefa4fa4)\n          ];\n        } else {\n          H = [\n              new Int(0x6a09e667, 0xf3bcc908), new Int(0xbb67ae85, 0x84caa73b),\n              new Int(0x3c6ef372, 0xfe94f82b), new Int(0xa54ff53a, 0x5f1d36f1),\n              new Int(0x510e527f, 0xade682d1), new Int(0x9b05688c, 0x2b3e6c1f),\n              new Int(0x1f83d9ab, 0xfb41bd6b), new Int(0x5be0cd19, 0x137e2179)\n          ];\n        }\n      }\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - messageLen % 32);\n      /* Append length of binary string in the position such that the new\n       * length is correct */\n      message[lengthPosition] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += binaryStringInc) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n        f = H[5];\n        g = H[6];\n        h = H[7];\n\n        for (t = 0; t < numRounds; t += 1) {\n          if (t < 16) {\n            /* Bit of a hack - for 32-bit, the second term is ignored */\n            W[t] = new Int(message[t * binaryStringMult + i],\n              message[t * binaryStringMult + i + 1]);\n          } else {\n            W[t] = safeAdd_4(\n              gamma1(W[t - 2]), W[t - 7],\n              gamma0(W[t - 15]), W[t - 16]);\n          }\n\n          T1 = safeAdd_5(h, sigma1(e), ch(e, f, g), K[t], W[t]);\n          T2 = safeAdd_2(sigma0(a), maj(a, b, c));\n          h = g;\n          g = f;\n          f = e;\n          e = safeAdd_2(d, T1);\n          d = c;\n          c = b;\n          b = a;\n          a = safeAdd_2(T1, T2);\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n        H[5] = safeAdd_2(f, H[5]);\n        H[6] = safeAdd_2(g, H[6]);\n        H[7] = safeAdd_2(h, H[7]);\n      }\n\n      switch (variant) {\n        case \"SHA-224\":\n          return [\n            H[0], H[1], H[2], H[3],\n            H[4], H[5], H[6]];\n        case \"SHA-256\":\n          return H;\n        case \"SHA-384\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder];\n        case \"SHA-512\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder,\n            H[6].highOrder, H[6].lowOrder,\n            H[7].highOrder, H[7].lowOrder];\n        default:\n          /* This should never be reached */\n          return [];\n      }\n    },\n\n    /*\n     * jsSHA is the workhorse of the library.  Instantiate it with the string to\n     * be hashed as the parameter\n     *\n     * @constructor\n     * @param {String} srcString The string to be hashed\n     * @param {String} inputFormat The format of srcString, ASCII or HEX\n     */\n    jsSHA = function(srcString, inputFormat) {\n\n      this.sha1 = null;\n      this.sha224 = null;\n      this.sha256 = null;\n      this.sha384 = null;\n      this.sha512 = null;\n\n      this.strBinLen = null;\n      this.strToHash = null;\n\n      /* Convert the input string into the correct type */\n      if (\"HEX\" === inputFormat) {\n        if (0 !== (srcString.length % 2)) {\n          return \"TEXT MUST BE IN BYTE INCREMENTS\";\n        }\n        this.strBinLen = srcString.length * 4;\n        this.strToHash = hex2binb(srcString);\n      } else if ((\"ASCII\" === inputFormat) ||\n        ('undefined' === typeof(inputFormat))) {\n        this.strBinLen = srcString.length * charSize;\n        this.strToHash = str2binb(srcString);\n      } else {\n        return \"UNKNOWN TEXT INPUT TYPE\";\n      }\n    };\n\n  jsSHA.prototype = {\n    /*\n     * Returns the desired SHA hash of the string specified at instantiation\n     * using the specified parameters\n     *\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} format The desired output formatting (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHash: function(variant, format) {\n      var formatFunc = null,\n        message = this.strToHash.slice();\n\n      switch (format) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      switch (variant) {\n        case \"SHA-1\":\n          if (null === this.sha1) {\n            this.sha1 = coreSHA1(message, this.strBinLen);\n          }\n          return formatFunc(this.sha1);\n        case \"SHA-224\":\n          if (null === this.sha224) {\n            this.sha224 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha224);\n        case \"SHA-256\":\n          if (null === this.sha256) {\n            this.sha256 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha256);\n        case \"SHA-384\":\n          if (null === this.sha384) {\n            this.sha384 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha384);\n        case \"SHA-512\":\n          if (null === this.sha512) {\n            this.sha512 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha512);\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n    },\n\n    /*\n     * Returns the desired HMAC of the string specified at instantiation\n     * using the key and variant param.\n     *\n     * @param {String} key The key used to calculate the HMAC\n     * @param {String} inputFormat The format of key, ASCII or HEX\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} outputFormat The desired output formatting\n     *\t (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHMAC: function(key, inputFormat, variant, outputFormat) {\n      var formatFunc, keyToUse, blockByteSize, blockBitSize, i,\n        retVal, lastArrayIndex, keyBinLen, hashBitSize,\n        keyWithIPad = [],\n        keyWithOPad = [];\n\n      /* Validate the output format selection */\n      switch (outputFormat) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      /* Validate the hash variant selection and set needed variables */\n      switch (variant) {\n        case \"SHA-1\":\n          blockByteSize = 64;\n          hashBitSize = 160;\n          break;\n        case \"SHA-224\":\n          blockByteSize = 64;\n          hashBitSize = 224;\n          break;\n        case \"SHA-256\":\n          blockByteSize = 64;\n          hashBitSize = 256;\n          break;\n        case \"SHA-384\":\n          blockByteSize = 128;\n          hashBitSize = 384;\n          break;\n        case \"SHA-512\":\n          blockByteSize = 128;\n          hashBitSize = 512;\n          break;\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n\n      /* Validate input format selection */\n      if (\"HEX\" === inputFormat) {\n        /* Nibbles must come in pairs */\n        if (0 !== (key.length % 2)) {\n          return \"KEY MUST BE IN BYTE INCREMENTS\";\n        }\n        keyToUse = hex2binb(key);\n        keyBinLen = key.length * 4;\n      } else if (\"ASCII\" === inputFormat) {\n        keyToUse = str2binb(key);\n        keyBinLen = key.length * charSize;\n      } else {\n        return \"UNKNOWN KEY INPUT TYPE\";\n      }\n\n      /* These are used multiple times, calculate and store them */\n      blockBitSize = blockByteSize * 8;\n      lastArrayIndex = (blockByteSize / 4) - 1;\n\n      /* Figure out what to do with the key based on its size relative to\n       * the hash's block size */\n      if (blockByteSize < (keyBinLen / 8)) {\n        if (\"SHA-1\" === variant) {\n          keyToUse = coreSHA1(keyToUse, keyBinLen);\n        } else {\n          keyToUse = coreSHA2(keyToUse, keyBinLen, variant);\n        }\n        /* For all variants, the block size is bigger than the output\n         * size so there will never be a useful byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      } else if (blockByteSize > (keyBinLen / 8)) {\n        /* If the blockByteSize is greater than the key length, there\n         * will always be at LEAST one \"useless\" byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      }\n\n      /* Create ipad and opad */\n      for (i = 0; i <= lastArrayIndex; i += 1) {\n        keyWithIPad[i] = keyToUse[i] ^ 0x36363636;\n        keyWithOPad[i] = keyToUse[i] ^ 0x5C5C5C5C;\n      }\n\n      /* Calculate the HMAC */\n      if (\"SHA-1\" === variant) {\n        retVal = coreSHA1(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen);\n        retVal = coreSHA1(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize);\n      } else {\n        retVal = coreSHA2(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen, variant);\n        retVal = coreSHA2(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize, variant);\n      }\n\n      return (formatFunc(retVal));\n    }\n  };\n\n  return jsSHA;\n}());\n\nmodule.exports = {\n  sha1: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-1\", \"ASCII\");\n  },\n  sha224: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-224\", \"ASCII\");\n  },\n  sha256: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-256\", \"ASCII\");\n  },\n  sha384: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-384\", \"ASCII\");\n\n  },\n  sha512: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-512\", \"ASCII\");\n  }\n}\n","module.exports = {\n  cipher: require('./cipher'),\n  hash: require('./hash'),\n  cfb: require('./cfb.js'),\n  publicKey: require('./public_key'),\n  signature: require('./signature.js'),\n  random: require('./random.js'),\n  pkcs1: require('./pkcs1.js')\n\n}\n\nvar crypto = require('./crypto.js');\n\nfor (var i in crypto)\n  module.exports[i] = crypto[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * ASN1 object identifiers for hashes (See RFC4880 5.2.2)\n */\nhash_headers = new Array();\nhash_headers[1] = [0x30, 0x20, 0x30, 0x0c, 0x06, 0x08, 0x2a, 0x86, 0x48, 0x86, 0xf7, 0x0d, 0x02, 0x05, 0x05, 0x00, 0x04,\n    0x10\n];\nhash_headers[3] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2B, 0x24, 0x03, 0x02, 0x01, 0x05, 0x00, 0x04, 0x14];\nhash_headers[2] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2b, 0x0e, 0x03, 0x02, 0x1a, 0x05, 0x00, 0x04, 0x14];\nhash_headers[8] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01, 0x05, 0x00,\n    0x04, 0x20\n];\nhash_headers[9] = [0x30, 0x41, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x02, 0x05, 0x00,\n    0x04, 0x30\n];\nhash_headers[10] = [0x30, 0x51, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x03, 0x05,\n    0x00, 0x04, 0x40\n];\nhash_headers[11] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x04, 0x05,\n    0x00, 0x04, 0x1C\n];\n\n\nvar crypto = require('./crypto.js'),\n  random = require('./random.js'),\n  util = require('../util'),\n  BigInteger = require('./public_key/jsbn.js'),\n  hash = require('./hash');\n\nmodule.exports = {\n  eme: {\n    /**\n     * create a EME-PKCS1-v1_5 padding (See RFC4880 13.1.1)\n     * @param {String} message message to be padded\n     * @param {Integer} length Length to the resulting message\n     * @return {String} EME-PKCS1 padded message\n     */\n    encode: function(message, length) {\n      if (message.length > length - 11)\n        return -1;\n      var result = \"\";\n      result += String.fromCharCode(0);\n      result += String.fromCharCode(2);\n      for (var i = 0; i < length - message.length - 3; i++) {\n        result += String.fromCharCode(random.getPseudoRandom(1, 255));\n      }\n      result += String.fromCharCode(0);\n      result += message;\n      return result;\n    },\n\n    /**\n     * decodes a EME-PKCS1-v1_5 padding (See RFC4880 13.1.2)\n     * @param {String} message EME-PKCS1 padded message\n     * @return {String} decoded message \n     */\n    decode: function(message, len) {\n      if (message.length < len)\n        message = String.fromCharCode(0) + message;\n      if (message.length < 12 || message.charCodeAt(0) != 0 || message.charCodeAt(1) != 2)\n        return -1;\n      var i = 2;\n      while (message.charCodeAt(i) != 0 && message.length > i)\n        i++;\n      return message.substring(i + 1, message.length);\n    },\n  },\n\n  emsa: {\n\n    /**\n     * create a EMSA-PKCS1-v1_5 padding (See RFC4880 13.1.3)\n     * @param {Integer} algo Hash algorithm type used\n     * @param {String} data Data to be hashed\n     * @param {Integer} keylength Key size of the public mpi in bytes\n     * @returns {String} Hashcode with pkcs1padding as string\n     */\n    encode: function(algo, data, keylength) {\n      var data2 = \"\";\n      data2 += String.fromCharCode(0x00);\n      data2 += String.fromCharCode(0x01);\n      for (var i = 0; i < (keylength - hash_headers[algo].length - 3 -\n        hash.getHashByteLength(algo)); i++)\n\n        data2 += String.fromCharCode(0xff);\n\n      data2 += String.fromCharCode(0x00);\n\n      for (var i = 0; i < hash_headers[algo].length; i++)\n        data2 += String.fromCharCode(hash_headers[algo][i]);\n\n      data2 += hash.digest(algo, data);\n      return new BigInteger(util.hexstrdump(data2), 16);\n    },\n\n    /**\n     * extract the hash out of an EMSA-PKCS1-v1.5 padding (See RFC4880 13.1.3) \n     * @param {String} data Hash in pkcs1 encoding\n     * @returns {String} The hash as string\n     */\n    decode: function(algo, data) {\n      var i = 0;\n      if (data.charCodeAt(0) == 0) i++;\n      else if (data.charCodeAt(0) != 1) return -1;\n      else i++;\n\n      while (data.charCodeAt(i) == 0xFF) i++;\n      if (data.charCodeAt(i++) != 0) return -1;\n      var j = 0;\n      for (j = 0; j < hash_headers[algo].length && j + i < data.length; j++) {\n        if (data.charCodeAt(j + i) != hash_headers[algo][j]) return -1;\n      }\n      i += j;\n      if (data.substring(i).length < hash.getHashByteLength(algo)) return -1;\n      return data.substring(i);\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// A Digital signature algorithm implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  hashModule = require('../hash'),\n  util = require('../../util');\n\nfunction DSA() {\n  // s1 = ((g**s) mod p) mod q\n  // s1 = ((s**-1)*(sha-1(m)+(s1*x) mod q)\n  function sign(hashalgo, m, g, p, q, x) {\n    // If the output size of the chosen hash is larger than the number of\n    // bits of q, the hash result is truncated to fit by taking the number\n    // of leftmost bits equal to the number of bits of q.  This (possibly\n    // truncated) hash function result is treated as a number and used\n    // directly in the DSA signature algorithm.\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    var k = random.getRandomBigIntegerInRange(BigInteger.ONE.add(BigInteger.ONE), q.subtract(BigInteger.ONE));\n    var s1 = (g.modPow(k, p)).mod(q);\n    var s2 = (k.modInverse(q).multiply(hash.add(x.multiply(s1)))).mod(q);\n    var result = new Array();\n    result[0] = s1.toMPI();\n    result[1] = s2.toMPI();\n    return result;\n  }\n\n  function select_hash_algorithm(q) {\n    var usersetting = openpgp.config.config.prefer_hash_algorithm;\n    /*\n     * 1024-bit key, 160-bit q, SHA-1, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 224-bit q, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     * 3072-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     */\n    switch (Math.round(q.bitLength() / 8)) {\n      case 20:\n        // 1024 bit\n        if (usersetting != 2 &&\n          usersetting > 11 &&\n          usersetting != 10 &&\n          usersetting < 8)\n          return 2; // prefer sha1\n        return usersetting;\n      case 28:\n        // 2048 bit\n        if (usersetting > 11 &&\n          usersetting < 8)\n          return 11;\n        return usersetting;\n      case 32:\n        // 4096 bit // prefer sha224\n        if (usersetting > 10 &&\n          usersetting < 8)\n          return 8; // prefer sha256\n        return usersetting;\n      default:\n        util.print_debug(\"DSA select hash algorithm: returning null for an unknown length of q\");\n        return null;\n\n    }\n  }\n  this.select_hash_algorithm = select_hash_algorithm;\n\n  function verify(hashalgo, s1, s2, m, p, q, g, y) {\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    if (BigInteger.ZERO.compareTo(s1) > 0 ||\n      s1.compareTo(q) > 0 ||\n      BigInteger.ZERO.compareTo(s2) > 0 ||\n      s2.compareTo(q) > 0) {\n      util.print_error(\"invalid DSA Signature\");\n      return null;\n    }\n    var w = s2.modInverse(q);\n    var u1 = hash.multiply(w).mod(q);\n    var u2 = s1.multiply(w).mod(q);\n    return g.modPow(u1, p).multiply(y.modPow(u2, p)).mod(p).mod(q);\n  }\n\n  /*\n\t * unused code. This can be used as a start to write a key generator\n\t * function.\n\t\n\tfunction generateKey(bitcount) {\n\t    var qi = new BigInteger(bitcount, primeCenterie);\n\t    var pi = generateP(q, 512);\n\t    var gi = generateG(p, q, bitcount);\n\t    var xi;\n\t    do {\n\t        xi = new BigInteger(q.bitCount(), rand);\n\t    } while (x.compareTo(BigInteger.ZERO) != 1 && x.compareTo(q) != -1);\n\t    var yi = g.modPow(x, p);\n\t    return {x: xi, q: qi, p: pi, g: gi, y: yi};\n\t}\n\n\tfunction generateP(q, bitlength, randomfn) {\n\t    if (bitlength % 64 != 0) {\n\t    \treturn false;\n\t    }\n\t    var pTemp;\n\t    var pTemp2;\n\t    do {\n\t        pTemp = randomfn(bitcount, true);\n\t        pTemp2 = pTemp.subtract(BigInteger.ONE);\n\t        pTemp = pTemp.subtract(pTemp2.remainder(q));\n\t    } while (!pTemp.isProbablePrime(primeCenterie) || pTemp.bitLength() != l);\n\t    return pTemp;\n\t}\n\t\n\tfunction generateG(p, q, bitlength, randomfn) {\n\t    var aux = p.subtract(BigInteger.ONE);\n\t    var pow = aux.divide(q);\n\t    var gTemp;\n\t    do {\n\t        gTemp = randomfn(bitlength);\n\t    } while (gTemp.compareTo(aux) != -1 && gTemp.compareTo(BigInteger.ONE) != 1);\n\t    return gTemp.modPow(pow, p);\n\t}\n\n\tfunction generateK(q, bitlength, randomfn) {\n\t    var tempK;\n\t    do {\n\t        tempK = randomfn(bitlength, false);\n\t    } while (tempK.compareTo(q) != -1 && tempK.compareTo(BigInteger.ZERO) != 1);\n\t    return tempK;\n\t}\n\n\tfunction generateR(q,p) {\n\t    k = generateK(q);\n\t    var r = g.modPow(k, p).mod(q);\n\t    return r;\n\t}\n\n\tfunction generateS(hashfn,k,r,m,q,x) {\n        var hash = hashfn(m);\n        s = (k.modInverse(q).multiply(hash.add(x.multiply(r)))).mod(q);\n\t    return s;\n\t} */\n  this.sign = sign;\n  this.verify = verify;\n  // this.generate = generateKey;\n}\n\nmodule.exports = DSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// ElGamal implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  util = require('../../util');\n\nfunction Elgamal() {\n\n  function encrypt(m, g, p, y) {\n    //  choose k in {2,...,p-2}\n    var two = BigInteger.ONE.add(BigInteger.ONE);\n    var pMinus2 = p.subtract(two);\n    var k = random.getRandomBigIntegerInRange(two, pMinus2);\n    k = k.mod(pMinus2).add(BigInteger.ONE);\n    var c = [];\n    c[0] = g.modPow(k, p);\n    c[1] = y.modPow(k, p).multiply(m).mod(p);\n    return c;\n  }\n\n  function decrypt(c1, c2, p, x) {\n    util.print_debug(\"Elgamal Decrypt:\\nc1:\" + util.hexstrdump(c1.toMPI()) + \"\\n\" +\n      \"c2:\" + util.hexstrdump(c2.toMPI()) + \"\\n\" +\n      \"p:\" + util.hexstrdump(p.toMPI()) + \"\\n\" +\n      \"x:\" + util.hexstrdump(x.toMPI()));\n    return (c1.modPow(x, p).modInverse(p)).multiply(c2).mod(p);\n    //var c = c1.pow(x).modInverse(p); // c0^-a mod p\n    //return c.multiply(c2).mod(p);\n  }\n\n  // signing and signature verification using Elgamal is not required by OpenPGP.\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n}\n\nmodule.exports = Elgamal;\n","module.exports = {\n  rsa: require('./rsa.js'),\n  elgamal: require('./elgamal.js'),\n  dsa: require('./dsa.js')\n}\n","/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH \n * \n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\nvar util = require('../../util');\n\n// Basic JavaScript BN library - subset useful for RSA encryption.\n\n// Bits per digit\nvar dbits;\n\n// JavaScript engine analysis\nvar canary = 0xdeadbeefcafe;\nvar j_lm = ((canary & 0xffffff) == 0xefcafe);\n\n// (public) Constructor\n\nfunction BigInteger(a, b, c) {\n  if (a != null)\n    if (\"number\" == typeof a) this.fromNumber(a, b, c);\n    else if (b == null && \"string\" != typeof a) this.fromString(a, 256);\n  else this.fromString(a, b);\n}\n\n// return new, unset BigInteger\n\nfunction nbi() {\n  return new BigInteger(null);\n}\n\n// am: Compute w_j += (x*this_i), propagate carries,\n// c is initial carry, returns final carry.\n// c < 3*dvalue, x < 2*dvalue, this_i < dvalue\n// We need to select the fastest one that works in this environment.\n\n// am1: use a single mult and divide to get the high bits,\n// max digit bits should be 26 because\n// max internal value = 2*dvalue^2-2*dvalue (< 2^53)\n\nfunction am1(i, x, w, j, c, n) {\n  while (--n >= 0) {\n    var v = x * this[i++] + w[j] + c;\n    c = Math.floor(v / 0x4000000);\n    w[j++] = v & 0x3ffffff;\n  }\n  return c;\n}\n// am2 avoids a big mult-and-extract completely.\n// Max digit bits should be <= 30 because we do bitwise ops\n// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)\n\nfunction am2(i, x, w, j, c, n) {\n  var xl = x & 0x7fff,\n    xh = x >> 15;\n  while (--n >= 0) {\n    var l = this[i] & 0x7fff;\n    var h = this[i++] >> 15;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x7fff) << 15) + w[j] + (c & 0x3fffffff);\n    c = (l >>> 30) + (m >>> 15) + xh * h + (c >>> 30);\n    w[j++] = l & 0x3fffffff;\n  }\n  return c;\n}\n// Alternately, set max digit bits to 28 since some\n// browsers slow down when dealing with 32-bit numbers.\n\nfunction am3(i, x, w, j, c, n) {\n  var xl = x & 0x3fff,\n    xh = x >> 14;\n  while (--n >= 0) {\n    var l = this[i] & 0x3fff;\n    var h = this[i++] >> 14;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x3fff) << 14) + w[j] + c;\n    c = (l >> 28) + (m >> 14) + xh * h;\n    w[j++] = l & 0xfffffff;\n  }\n  return c;\n}\n/*if(j_lm && (navigator != undefined && \n\tnavigator.appName == \"Microsoft Internet Explorer\")) {\n  BigInteger.prototype.am = am2;\n  dbits = 30;\n}\nelse if(j_lm && (navigator != undefined && navigator.appName != \"Netscape\")) {*/\nBigInteger.prototype.am = am1;\ndbits = 26;\n/*}\nelse { // Mozilla/Netscape seems to prefer am3\n  BigInteger.prototype.am = am3;\n  dbits = 28;\n}*/\n\nBigInteger.prototype.DB = dbits;\nBigInteger.prototype.DM = ((1 << dbits) - 1);\nBigInteger.prototype.DV = (1 << dbits);\n\nvar BI_FP = 52;\nBigInteger.prototype.FV = Math.pow(2, BI_FP);\nBigInteger.prototype.F1 = BI_FP - dbits;\nBigInteger.prototype.F2 = 2 * dbits - BI_FP;\n\n// Digit conversions\nvar BI_RM = \"0123456789abcdefghijklmnopqrstuvwxyz\";\nvar BI_RC = new Array();\nvar rr, vv;\nrr = \"0\".charCodeAt(0);\nfor (vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;\nrr = \"a\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\nrr = \"A\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\n\nfunction int2char(n) {\n  return BI_RM.charAt(n);\n}\n\nfunction intAt(s, i) {\n  var c = BI_RC[s.charCodeAt(i)];\n  return (c == null) ? -1 : c;\n}\n\n// (protected) copy this to r\n\nfunction bnpCopyTo(r) {\n  for (var i = this.t - 1; i >= 0; --i) r[i] = this[i];\n  r.t = this.t;\n  r.s = this.s;\n}\n\n// (protected) set from integer value x, -DV <= x < DV\n\nfunction bnpFromInt(x) {\n  this.t = 1;\n  this.s = (x < 0) ? -1 : 0;\n  if (x > 0) this[0] = x;\n  else if (x < -1) this[0] = x + DV;\n  else this.t = 0;\n}\n\n// return bigint initialized to value\n\nfunction nbv(i) {\n  var r = nbi();\n  r.fromInt(i);\n  return r;\n}\n\n// (protected) set from string and radix\n\nfunction bnpFromString(s, b) {\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 256) k = 8; // byte array\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else {\n    this.fromRadix(s, b);\n    return;\n  }\n  this.t = 0;\n  this.s = 0;\n  var i = s.length,\n    mi = false,\n    sh = 0;\n  while (--i >= 0) {\n    var x = (k == 8) ? s[i] & 0xff : intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\") mi = true;\n      continue;\n    }\n    mi = false;\n    if (sh == 0)\n      this[this.t++] = x;\n    else if (sh + k > this.DB) {\n      this[this.t - 1] |= (x & ((1 << (this.DB - sh)) - 1)) << sh;\n      this[this.t++] = (x >> (this.DB - sh));\n    } else\n      this[this.t - 1] |= x << sh;\n    sh += k;\n    if (sh >= this.DB) sh -= this.DB;\n  }\n  if (k == 8 && (s[0] & 0x80) != 0) {\n    this.s = -1;\n    if (sh > 0) this[this.t - 1] |= ((1 << (this.DB - sh)) - 1) << sh;\n  }\n  this.clamp();\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) clamp off excess high words\n\nfunction bnpClamp() {\n  var c = this.s & this.DM;\n  while (this.t > 0 && this[this.t - 1] == c)--this.t;\n}\n\n// (public) return string representation in given radix\n\nfunction bnToString(b) {\n  if (this.s < 0) return \"-\" + this.negate().toString(b);\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else return this.toRadix(b);\n  var km = (1 << k) - 1,\n    d, m = false,\n    r = \"\",\n    i = this.t;\n  var p = this.DB - (i * this.DB) % k;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) > 0) {\n      m = true;\n      r = int2char(d);\n    }\n    while (i >= 0) {\n      if (p < k) {\n        d = (this[i] & ((1 << p) - 1)) << (k - p);\n        d |= this[--i] >> (p += this.DB - k);\n      } else {\n        d = (this[i] >> (p -= k)) & km;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      if (d > 0) m = true;\n      if (m) r += int2char(d);\n    }\n  }\n  return m ? r : \"0\";\n}\n\n// (public) -this\n\nfunction bnNegate() {\n  var r = nbi();\n  BigInteger.ZERO.subTo(this, r);\n  return r;\n}\n\n// (public) |this|\n\nfunction bnAbs() {\n  return (this.s < 0) ? this.negate() : this;\n}\n\n// (public) return + if this > a, - if this < a, 0 if equal\n\nfunction bnCompareTo(a) {\n  var r = this.s - a.s;\n  if (r != 0) return r;\n  var i = this.t;\n  r = i - a.t;\n  if (r != 0) return r;\n  while (--i >= 0) if ((r = this[i] - a[i]) != 0) return r;\n  return 0;\n}\n\n// returns bit length of the integer x\n\nfunction nbits(x) {\n  var r = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    r += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    r += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    r += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    r += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    r += 1;\n  }\n  return r;\n}\n\n// (public) return the number of bits in \"this\"\n\nfunction bnBitLength() {\n  if (this.t <= 0) return 0;\n  return this.DB * (this.t - 1) + nbits(this[this.t - 1] ^ (this.s & this.DM));\n}\n\n// (protected) r = this << n*DB\n\nfunction bnpDLShiftTo(n, r) {\n  var i;\n  for (i = this.t - 1; i >= 0; --i) r[i + n] = this[i];\n  for (i = n - 1; i >= 0; --i) r[i] = 0;\n  r.t = this.t + n;\n  r.s = this.s;\n}\n\n// (protected) r = this >> n*DB\n\nfunction bnpDRShiftTo(n, r) {\n  for (var i = n; i < this.t; ++i) r[i - n] = this[i];\n  r.t = Math.max(this.t - n, 0);\n  r.s = this.s;\n}\n\n// (protected) r = this << n\n\nfunction bnpLShiftTo(n, r) {\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << cbs) - 1;\n  var ds = Math.floor(n / this.DB),\n    c = (this.s << bs) & this.DM,\n    i;\n  for (i = this.t - 1; i >= 0; --i) {\n    r[i + ds + 1] = (this[i] >> cbs) | c;\n    c = (this[i] & bm) << bs;\n  }\n  for (i = ds - 1; i >= 0; --i) r[i] = 0;\n  r[ds] = c;\n  r.t = this.t + ds + 1;\n  r.s = this.s;\n  r.clamp();\n}\n\n// (protected) r = this >> n\n\nfunction bnpRShiftTo(n, r) {\n  r.s = this.s;\n  var ds = Math.floor(n / this.DB);\n  if (ds >= this.t) {\n    r.t = 0;\n    return;\n  }\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << bs) - 1;\n  r[0] = this[ds] >> bs;\n  for (var i = ds + 1; i < this.t; ++i) {\n    r[i - ds - 1] |= (this[i] & bm) << cbs;\n    r[i - ds] = this[i] >> bs;\n  }\n  if (bs > 0) r[this.t - ds - 1] |= (this.s & bm) << cbs;\n  r.t = this.t - ds;\n  r.clamp();\n}\n\n// (protected) r = this - a\n\nfunction bnpSubTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] - a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c -= a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c -= a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c -= a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c < -1) r[i++] = this.DV + c;\n  else if (c > 0) r[i++] = c;\n  r.t = i;\n  r.clamp();\n}\n\n// (protected) r = this * a, r != this,a (HAC 14.12)\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyTo(a, r) {\n  var x = this.abs(),\n    y = a.abs();\n  var i = x.t;\n  r.t = i + y.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < y.t; ++i) r[i + x.t] = x.am(0, y[i], r, i, 0, x.t);\n  r.s = 0;\n  r.clamp();\n  if (this.s != a.s) BigInteger.ZERO.subTo(r, r);\n}\n\n// (protected) r = this^2, r != this (HAC 14.16)\n\nfunction bnpSquareTo(r) {\n  var x = this.abs();\n  var i = r.t = 2 * x.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < x.t - 1; ++i) {\n    var c = x.am(i, x[i], r, 2 * i, 0, 1);\n    if ((r[i + x.t] += x.am(i + 1, 2 * x[i], r, 2 * i + 1, c, x.t - i - 1)) >= x.DV) {\n      r[i + x.t] -= x.DV;\n      r[i + x.t + 1] = 1;\n    }\n  }\n  if (r.t > 0) r[r.t - 1] += x.am(i, x[i], r, 2 * i, 0, 1);\n  r.s = 0;\n  r.clamp();\n}\n\n// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)\n// r != q, this != m.  q or r may be null.\n\nfunction bnpDivRemTo(m, q, r) {\n  var pm = m.abs();\n  if (pm.t <= 0) return;\n  var pt = this.abs();\n  if (pt.t < pm.t) {\n    if (q != null) q.fromInt(0);\n    if (r != null) this.copyTo(r);\n    return;\n  }\n  if (r == null) r = nbi();\n  var y = nbi(),\n    ts = this.s,\n    ms = m.s;\n  var nsh = this.DB - nbits(pm[pm.t - 1]); // normalize modulus\n  if (nsh > 0) {\n    pm.lShiftTo(nsh, y);\n    pt.lShiftTo(nsh, r);\n  } else {\n    pm.copyTo(y);\n    pt.copyTo(r);\n  }\n  var ys = y.t;\n  var y0 = y[ys - 1];\n  if (y0 == 0) return;\n  var yt = y0 * (1 << this.F1) + ((ys > 1) ? y[ys - 2] >> this.F2 : 0);\n  var d1 = this.FV / yt,\n    d2 = (1 << this.F1) / yt,\n    e = 1 << this.F2;\n  var i = r.t,\n    j = i - ys,\n    t = (q == null) ? nbi() : q;\n  y.dlShiftTo(j, t);\n  if (r.compareTo(t) >= 0) {\n    r[r.t++] = 1;\n    r.subTo(t, r);\n  }\n  BigInteger.ONE.dlShiftTo(ys, t);\n  t.subTo(y, y); // \"negative\" y so we can replace sub with am later\n  while (y.t < ys) y[y.t++] = 0;\n  while (--j >= 0) {\n    // Estimate quotient digit\n    var qd = (r[--i] == y0) ? this.DM : Math.floor(r[i] * d1 + (r[i - 1] + e) * d2);\n    if ((r[i] += y.am(0, qd, r, j, 0, ys)) < qd) { // Try it out\n      y.dlShiftTo(j, t);\n      r.subTo(t, r);\n      while (r[i] < --qd) r.subTo(t, r);\n    }\n  }\n  if (q != null) {\n    r.drShiftTo(ys, q);\n    if (ts != ms) BigInteger.ZERO.subTo(q, q);\n  }\n  r.t = ys;\n  r.clamp();\n  if (nsh > 0) r.rShiftTo(nsh, r); // Denormalize remainder\n  if (ts < 0) BigInteger.ZERO.subTo(r, r);\n}\n\n// (public) this mod a\n\nfunction bnMod(a) {\n  var r = nbi();\n  this.abs().divRemTo(a, null, r);\n  if (this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r, r);\n  return r;\n}\n\n// Modular reduction using \"classic\" algorithm\n\nfunction Classic(m) {\n  this.m = m;\n}\n\nfunction cConvert(x) {\n  if (x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);\n  else return x;\n}\n\nfunction cRevert(x) {\n  return x;\n}\n\nfunction cReduce(x) {\n  x.divRemTo(this.m, null, x);\n}\n\nfunction cMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nfunction cSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\nClassic.prototype.convert = cConvert;\nClassic.prototype.revert = cRevert;\nClassic.prototype.reduce = cReduce;\nClassic.prototype.mulTo = cMulTo;\nClassic.prototype.sqrTo = cSqrTo;\n\n// (protected) return \"-1/this % 2^DB\"; useful for Mont. reduction\n// justification:\n//         xy == 1 (mod m)\n//         xy =  1+km\n//   xy(2-xy) = (1+km)(1-km)\n// x[y(2-xy)] = 1-k^2m^2\n// x[y(2-xy)] == 1 (mod m^2)\n// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2\n// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.\n// JS multiply \"overflows\" differently from C/C++, so care is needed here.\n\nfunction bnpInvDigit() {\n  if (this.t < 1) return 0;\n  var x = this[0];\n  if ((x & 1) == 0) return 0;\n  var y = x & 3; // y == 1/x mod 2^2\n  y = (y * (2 - (x & 0xf) * y)) & 0xf; // y == 1/x mod 2^4\n  y = (y * (2 - (x & 0xff) * y)) & 0xff; // y == 1/x mod 2^8\n  y = (y * (2 - (((x & 0xffff) * y) & 0xffff))) & 0xffff; // y == 1/x mod 2^16\n  // last step - calculate inverse mod DV directly;\n  // assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints\n  y = (y * (2 - x * y % this.DV)) % this.DV; // y == 1/x mod 2^dbits\n  // we really want the negative inverse, and -DV < y < DV\n  return (y > 0) ? this.DV - y : -y;\n}\n\n// Montgomery reduction\n\nfunction Montgomery(m) {\n  this.m = m;\n  this.mp = m.invDigit();\n  this.mpl = this.mp & 0x7fff;\n  this.mph = this.mp >> 15;\n  this.um = (1 << (m.DB - 15)) - 1;\n  this.mt2 = 2 * m.t;\n}\n\n// xR mod m\n\nfunction montConvert(x) {\n  var r = nbi();\n  x.abs().dlShiftTo(this.m.t, r);\n  r.divRemTo(this.m, null, r);\n  if (x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r, r);\n  return r;\n}\n\n// x/R mod m\n\nfunction montRevert(x) {\n  var r = nbi();\n  x.copyTo(r);\n  this.reduce(r);\n  return r;\n}\n\n// x = x/R mod m (HAC 14.32)\n\nfunction montReduce(x) {\n  while (x.t <= this.mt2) // pad x so am has enough room later\n    x[x.t++] = 0;\n  for (var i = 0; i < this.m.t; ++i) {\n    // faster way of calculating u0 = x[i]*mp mod DV\n    var j = x[i] & 0x7fff;\n    var u0 = (j * this.mpl + (((j * this.mph + (x[i] >> 15) * this.mpl) & this.um) << 15)) & x.DM;\n    // use am to combine the multiply-shift-add into one call\n    j = i + this.m.t;\n    x[j] += this.m.am(0, u0, x, i, 0, this.m.t);\n    // propagate carry\n    while (x[j] >= x.DV) {\n      x[j] -= x.DV;\n      x[++j]++;\n    }\n  }\n  x.clamp();\n  x.drShiftTo(this.m.t, x);\n  if (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = \"x^2/R mod m\"; x != r\n\nfunction montSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = \"xy/R mod m\"; x,y != r\n\nfunction montMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nMontgomery.prototype.convert = montConvert;\nMontgomery.prototype.revert = montRevert;\nMontgomery.prototype.reduce = montReduce;\nMontgomery.prototype.mulTo = montMulTo;\nMontgomery.prototype.sqrTo = montSqrTo;\n\n// (protected) true iff this is even\n\nfunction bnpIsEven() {\n  return ((this.t > 0) ? (this[0] & 1) : this.s) == 0;\n}\n\n// (protected) this^e, e < 2^32, doing sqr and mul with \"r\" (HAC 14.79)\n\nfunction bnpExp(e, z) {\n  if (e > 0xffffffff || e < 1) return BigInteger.ONE;\n  var r = nbi(),\n    r2 = nbi(),\n    g = z.convert(this),\n    i = nbits(e) - 1;\n  g.copyTo(r);\n  while (--i >= 0) {\n    z.sqrTo(r, r2);\n    if ((e & (1 << i)) > 0) z.mulTo(r2, g, r);\n    else {\n      var t = r;\n      r = r2;\n      r2 = t;\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) this^e % m, 0 <= e < 2^32\n\nfunction bnModPowInt(e, m) {\n  var z;\n  if (e < 256 || m.isEven()) z = new Classic(m);\n  else z = new Montgomery(m);\n  return this.exp(e, z);\n}\n\n// protected\nBigInteger.prototype.copyTo = bnpCopyTo;\nBigInteger.prototype.fromInt = bnpFromInt;\nBigInteger.prototype.fromString = bnpFromString;\nBigInteger.prototype.clamp = bnpClamp;\nBigInteger.prototype.dlShiftTo = bnpDLShiftTo;\nBigInteger.prototype.drShiftTo = bnpDRShiftTo;\nBigInteger.prototype.lShiftTo = bnpLShiftTo;\nBigInteger.prototype.rShiftTo = bnpRShiftTo;\nBigInteger.prototype.subTo = bnpSubTo;\nBigInteger.prototype.multiplyTo = bnpMultiplyTo;\nBigInteger.prototype.squareTo = bnpSquareTo;\nBigInteger.prototype.divRemTo = bnpDivRemTo;\nBigInteger.prototype.invDigit = bnpInvDigit;\nBigInteger.prototype.isEven = bnpIsEven;\nBigInteger.prototype.exp = bnpExp;\n\n// public\nBigInteger.prototype.toString = bnToString;\nBigInteger.prototype.negate = bnNegate;\nBigInteger.prototype.abs = bnAbs;\nBigInteger.prototype.compareTo = bnCompareTo;\nBigInteger.prototype.bitLength = bnBitLength;\nBigInteger.prototype.mod = bnMod;\nBigInteger.prototype.modPowInt = bnModPowInt;\n\n// \"constants\"\nBigInteger.ZERO = nbv(0);\nBigInteger.ONE = nbv(1);\n\nmodule.exports = BigInteger;\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH\n *\n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\n// Extended JavaScript BN functions, required for RSA private ops.\n\n// Version 1.1: new BigInteger(\"0\", 10) returns \"proper\" zero\n// Version 1.2: square() API, isProbablePrime fix\n\n// (public)\nfunction bnClone() {\n  var r = nbi();\n  this.copyTo(r);\n  return r;\n}\n\n// (public) return value as integer\n\nfunction bnIntValue() {\n  if (this.s < 0) {\n    if (this.t == 1) return this[0] - this.DV;\n    else if (this.t == 0) return -1;\n  } else if (this.t == 1) return this[0];\n  else if (this.t == 0) return 0;\n  // assumes 16 < DB < 32\n  return ((this[1] & ((1 << (32 - this.DB)) - 1)) << this.DB) | this[0];\n}\n\n// (public) return value as byte\n\nfunction bnByteValue() {\n  return (this.t == 0) ? this.s : (this[0] << 24) >> 24;\n}\n\n// (public) return value as short (assumes DB>=16)\n\nfunction bnShortValue() {\n  return (this.t == 0) ? this.s : (this[0] << 16) >> 16;\n}\n\n// (protected) return x s.t. r^x < DV\n\nfunction bnpChunkSize(r) {\n  return Math.floor(Math.LN2 * this.DB / Math.log(r));\n}\n\n// (public) 0 if this == 0, 1 if this > 0\n\nfunction bnSigNum() {\n  if (this.s < 0) return -1;\n  else if (this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;\n  else return 1;\n}\n\n// (protected) convert to radix string\n\nfunction bnpToRadix(b) {\n  if (b == null) b = 10;\n  if (this.signum() == 0 || b < 2 || b > 36) return \"0\";\n  var cs = this.chunkSize(b);\n  var a = Math.pow(b, cs);\n  var d = nbv(a),\n    y = nbi(),\n    z = nbi(),\n    r = \"\";\n  this.divRemTo(d, y, z);\n  while (y.signum() > 0) {\n    r = (a + z.intValue()).toString(b).substr(1) + r;\n    y.divRemTo(d, y, z);\n  }\n  return z.intValue().toString(b) + r;\n}\n\n// (protected) convert from radix string\n\nfunction bnpFromRadix(s, b) {\n  this.fromInt(0);\n  if (b == null) b = 10;\n  var cs = this.chunkSize(b);\n  var d = Math.pow(b, cs),\n    mi = false,\n    j = 0,\n    w = 0;\n  for (var i = 0; i < s.length; ++i) {\n    var x = intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\" && this.signum() == 0) mi = true;\n      continue;\n    }\n    w = b * w + x;\n    if (++j >= cs) {\n      this.dMultiply(d);\n      this.dAddOffset(w, 0);\n      j = 0;\n      w = 0;\n    }\n  }\n  if (j > 0) {\n    this.dMultiply(Math.pow(b, j));\n    this.dAddOffset(w, 0);\n  }\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) alternate constructor\n\nfunction bnpFromNumber(a, b, c) {\n  if (\"number\" == typeof b) {\n    // new BigInteger(int,int,RNG)\n    if (a < 2) this.fromInt(1);\n    else {\n      this.fromNumber(a, c);\n      if (!this.testBit(a - 1)) // force MSB set\n        this.bitwiseTo(BigInteger.ONE.shiftLeft(a - 1), op_or, this);\n      if (this.isEven()) this.dAddOffset(1, 0); // force odd\n      while (!this.isProbablePrime(b)) {\n        this.dAddOffset(2, 0);\n        if (this.bitLength() > a) this.subTo(BigInteger.ONE.shiftLeft(a - 1), this);\n      }\n    }\n  } else {\n    // new BigInteger(int,RNG)\n    var x = new Array(),\n      t = a & 7;\n    x.length = (a >> 3) + 1;\n    b.nextBytes(x);\n    if (t > 0) x[0] &= ((1 << t) - 1);\n    else x[0] = 0;\n    this.fromString(x, 256);\n  }\n}\n\n// (public) convert to bigendian byte array\n\nfunction bnToByteArray() {\n  var i = this.t,\n    r = new Array();\n  r[0] = this.s;\n  var p = this.DB - (i * this.DB) % 8,\n    d, k = 0;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) != (this.s & this.DM) >> p)\n      r[k++] = d | (this.s << (this.DB - p));\n    while (i >= 0) {\n      if (p < 8) {\n        d = (this[i] & ((1 << p) - 1)) << (8 - p);\n        d |= this[--i] >> (p += this.DB - 8);\n      } else {\n        d = (this[i] >> (p -= 8)) & 0xff;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      //if((d&0x80) != 0) d |= -256;\n      //if(k == 0 && (this.s&0x80) != (d&0x80)) ++k;\n      if (k > 0 || d != this.s) r[k++] = d;\n    }\n  }\n  return r;\n}\n\nfunction bnEquals(a) {\n  return (this.compareTo(a) == 0);\n}\n\nfunction bnMin(a) {\n  return (this.compareTo(a) < 0) ? this : a;\n}\n\nfunction bnMax(a) {\n  return (this.compareTo(a) > 0) ? this : a;\n}\n\n// (protected) r = this op a (bitwise)\n\nfunction bnpBitwiseTo(a, op, r) {\n  var i, f, m = Math.min(a.t, this.t);\n  for (i = 0; i < m; ++i) r[i] = op(this[i], a[i]);\n  if (a.t < this.t) {\n    f = a.s & this.DM;\n    for (i = m; i < this.t; ++i) r[i] = op(this[i], f);\n    r.t = this.t;\n  } else {\n    f = this.s & this.DM;\n    for (i = m; i < a.t; ++i) r[i] = op(f, a[i]);\n    r.t = a.t;\n  }\n  r.s = op(this.s, a.s);\n  r.clamp();\n}\n\n// (public) this & a\n\nfunction op_and(x, y) {\n  return x & y;\n}\n\nfunction bnAnd(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_and, r);\n  return r;\n}\n\n// (public) this | a\n\nfunction op_or(x, y) {\n  return x | y;\n}\n\nfunction bnOr(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_or, r);\n  return r;\n}\n\n// (public) this ^ a\n\nfunction op_xor(x, y) {\n  return x ^ y;\n}\n\nfunction bnXor(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_xor, r);\n  return r;\n}\n\n// (public) this & ~a\n\nfunction op_andnot(x, y) {\n  return x & ~y;\n}\n\nfunction bnAndNot(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_andnot, r);\n  return r;\n}\n\n// (public) ~this\n\nfunction bnNot() {\n  var r = nbi();\n  for (var i = 0; i < this.t; ++i) r[i] = this.DM & ~this[i];\n  r.t = this.t;\n  r.s = ~this.s;\n  return r;\n}\n\n// (public) this << n\n\nfunction bnShiftLeft(n) {\n  var r = nbi();\n  if (n < 0) this.rShiftTo(-n, r);\n  else this.lShiftTo(n, r);\n  return r;\n}\n\n// (public) this >> n\n\nfunction bnShiftRight(n) {\n  var r = nbi();\n  if (n < 0) this.lShiftTo(-n, r);\n  else this.rShiftTo(n, r);\n  return r;\n}\n\n// return index of lowest 1-bit in x, x < 2^31\n\nfunction lbit(x) {\n  if (x == 0) return -1;\n  var r = 0;\n  if ((x & 0xffff) == 0) {\n    x >>= 16;\n    r += 16;\n  }\n  if ((x & 0xff) == 0) {\n    x >>= 8;\n    r += 8;\n  }\n  if ((x & 0xf) == 0) {\n    x >>= 4;\n    r += 4;\n  }\n  if ((x & 3) == 0) {\n    x >>= 2;\n    r += 2;\n  }\n  if ((x & 1) == 0)++r;\n  return r;\n}\n\n// (public) returns index of lowest 1-bit (or -1 if none)\n\nfunction bnGetLowestSetBit() {\n  for (var i = 0; i < this.t; ++i)\n    if (this[i] != 0) return i * this.DB + lbit(this[i]);\n  if (this.s < 0) return this.t * this.DB;\n  return -1;\n}\n\n// return number of 1 bits in x\n\nfunction cbit(x) {\n  var r = 0;\n  while (x != 0) {\n    x &= x - 1;\n    ++r;\n  }\n  return r;\n}\n\n// (public) return number of set bits\n\nfunction bnBitCount() {\n  var r = 0,\n    x = this.s & this.DM;\n  for (var i = 0; i < this.t; ++i) r += cbit(this[i] ^ x);\n  return r;\n}\n\n// (public) true iff nth bit is set\n\nfunction bnTestBit(n) {\n  var j = Math.floor(n / this.DB);\n  if (j >= this.t) return (this.s != 0);\n  return ((this[j] & (1 << (n % this.DB))) != 0);\n}\n\n// (protected) this op (1<<n)\n\nfunction bnpChangeBit(n, op) {\n  var r = BigInteger.ONE.shiftLeft(n);\n  this.bitwiseTo(r, op, r);\n  return r;\n}\n\n// (public) this | (1<<n)\n\nfunction bnSetBit(n) {\n  return this.changeBit(n, op_or);\n}\n\n// (public) this & ~(1<<n)\n\nfunction bnClearBit(n) {\n  return this.changeBit(n, op_andnot);\n}\n\n// (public) this ^ (1<<n)\n\nfunction bnFlipBit(n) {\n  return this.changeBit(n, op_xor);\n}\n\n// (protected) r = this + a\n\nfunction bnpAddTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] + a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c += a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c += a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c > 0) r[i++] = c;\n  else if (c < -1) r[i++] = this.DV + c;\n  r.t = i;\n  r.clamp();\n}\n\n// (public) this + a\n\nfunction bnAdd(a) {\n  var r = nbi();\n  this.addTo(a, r);\n  return r;\n}\n\n// (public) this - a\n\nfunction bnSubtract(a) {\n  var r = nbi();\n  this.subTo(a, r);\n  return r;\n}\n\n// (public) this * a\n\nfunction bnMultiply(a) {\n  var r = nbi();\n  this.multiplyTo(a, r);\n  return r;\n}\n\n// (public) this^2\n\nfunction bnSquare() {\n  var r = nbi();\n  this.squareTo(r);\n  return r;\n}\n\n// (public) this / a\n\nfunction bnDivide(a) {\n  var r = nbi();\n  this.divRemTo(a, r, null);\n  return r;\n}\n\n// (public) this % a\n\nfunction bnRemainder(a) {\n  var r = nbi();\n  this.divRemTo(a, null, r);\n  return r;\n}\n\n// (public) [this/a,this%a]\n\nfunction bnDivideAndRemainder(a) {\n  var q = nbi(),\n    r = nbi();\n  this.divRemTo(a, q, r);\n  return new Array(q, r);\n}\n\n// (protected) this *= n, this >= 0, 1 < n < DV\n\nfunction bnpDMultiply(n) {\n  this[this.t] = this.am(0, n - 1, this, 0, 0, this.t);\n  ++this.t;\n  this.clamp();\n}\n\n// (protected) this += n << w words, this >= 0\n\nfunction bnpDAddOffset(n, w) {\n  if (n == 0) return;\n  while (this.t <= w) this[this.t++] = 0;\n  this[w] += n;\n  while (this[w] >= this.DV) {\n    this[w] -= this.DV;\n    if (++w >= this.t) this[this.t++] = 0;\n    ++this[w];\n  }\n}\n\n// A \"null\" reducer\n\nfunction NullExp() {}\n\nfunction nNop(x) {\n  return x;\n}\n\nfunction nMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n}\n\nfunction nSqrTo(x, r) {\n  x.squareTo(r);\n}\n\nNullExp.prototype.convert = nNop;\nNullExp.prototype.revert = nNop;\nNullExp.prototype.mulTo = nMulTo;\nNullExp.prototype.sqrTo = nSqrTo;\n\n// (public) this^e\n\nfunction bnPow(e) {\n  return this.exp(e, new NullExp());\n}\n\n// (protected) r = lower n words of \"this * a\", a.t <= n\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyLowerTo(a, n, r) {\n  var i = Math.min(this.t + a.t, n);\n  r.s = 0; // assumes a,this >= 0\n  r.t = i;\n  while (i > 0) r[--i] = 0;\n  var j;\n  for (j = r.t - this.t; i < j; ++i) r[i + this.t] = this.am(0, a[i], r, i, 0, this.t);\n  for (j = Math.min(a.t, n); i < j; ++i) this.am(0, a[i], r, i, 0, n - i);\n  r.clamp();\n}\n\n// (protected) r = \"this * a\" without lower n words, n > 0\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyUpperTo(a, n, r) {\n  --n;\n  var i = r.t = this.t + a.t - n;\n  r.s = 0; // assumes a,this >= 0\n  while (--i >= 0) r[i] = 0;\n  for (i = Math.max(n - this.t, 0); i < a.t; ++i)\n    r[this.t + i - n] = this.am(n - i, a[i], r, 0, 0, this.t + i - n);\n  r.clamp();\n  r.drShiftTo(1, r);\n}\n\n// Barrett modular reduction\n\nfunction Barrett(m) {\n  // setup Barrett\n  this.r2 = nbi();\n  this.q3 = nbi();\n  BigInteger.ONE.dlShiftTo(2 * m.t, this.r2);\n  this.mu = this.r2.divide(m);\n  this.m = m;\n}\n\nfunction barrettConvert(x) {\n  if (x.s < 0 || x.t > 2 * this.m.t) return x.mod(this.m);\n  else if (x.compareTo(this.m) < 0) return x;\n  else {\n    var r = nbi();\n    x.copyTo(r);\n    this.reduce(r);\n    return r;\n  }\n}\n\nfunction barrettRevert(x) {\n  return x;\n}\n\n// x = x mod m (HAC 14.42)\n\nfunction barrettReduce(x) {\n  x.drShiftTo(this.m.t - 1, this.r2);\n  if (x.t > this.m.t + 1) {\n    x.t = this.m.t + 1;\n    x.clamp();\n  }\n  this.mu.multiplyUpperTo(this.r2, this.m.t + 1, this.q3);\n  this.m.multiplyLowerTo(this.q3, this.m.t + 1, this.r2);\n  while (x.compareTo(this.r2) < 0) x.dAddOffset(1, this.m.t + 1);\n  x.subTo(this.r2, x);\n  while (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = x^2 mod m; x != r\n\nfunction barrettSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = x*y mod m; x,y != r\n\nfunction barrettMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nBarrett.prototype.convert = barrettConvert;\nBarrett.prototype.revert = barrettRevert;\nBarrett.prototype.reduce = barrettReduce;\nBarrett.prototype.mulTo = barrettMulTo;\nBarrett.prototype.sqrTo = barrettSqrTo;\n\n// (public) this^e % m (HAC 14.85)\n\nfunction bnModPow(e, m) {\n  var i = e.bitLength(),\n    k, r = nbv(1),\n    z;\n  if (i <= 0) return r;\n  else if (i < 18) k = 1;\n  else if (i < 48) k = 3;\n  else if (i < 144) k = 4;\n  else if (i < 768) k = 5;\n  else k = 6;\n  if (i < 8)\n    z = new Classic(m);\n  else if (m.isEven())\n    z = new Barrett(m);\n  else\n    z = new Montgomery(m);\n\n  // precomputation\n  var g = new Array(),\n    n = 3,\n    k1 = k - 1,\n    km = (1 << k) - 1;\n  g[1] = z.convert(this);\n  if (k > 1) {\n    var g2 = nbi();\n    z.sqrTo(g[1], g2);\n    while (n <= km) {\n      g[n] = nbi();\n      z.mulTo(g2, g[n - 2], g[n]);\n      n += 2;\n    }\n  }\n\n  var j = e.t - 1,\n    w, is1 = true,\n    r2 = nbi(),\n    t;\n  i = nbits(e[j]) - 1;\n  while (j >= 0) {\n    if (i >= k1) w = (e[j] >> (i - k1)) & km;\n    else {\n      w = (e[j] & ((1 << (i + 1)) - 1)) << (k1 - i);\n      if (j > 0) w |= e[j - 1] >> (this.DB + i - k1);\n    }\n\n    n = k;\n    while ((w & 1) == 0) {\n      w >>= 1;\n      --n;\n    }\n    if ((i -= n) < 0) {\n      i += this.DB;\n      --j;\n    }\n    if (is1) { // ret == 1, don't bother squaring or multiplying it\n      g[w].copyTo(r);\n      is1 = false;\n    } else {\n      while (n > 1) {\n        z.sqrTo(r, r2);\n        z.sqrTo(r2, r);\n        n -= 2;\n      }\n      if (n > 0) z.sqrTo(r, r2);\n      else {\n        t = r;\n        r = r2;\n        r2 = t;\n      }\n      z.mulTo(r2, g[w], r);\n    }\n\n    while (j >= 0 && (e[j] & (1 << i)) == 0) {\n      z.sqrTo(r, r2);\n      t = r;\n      r = r2;\n      r2 = t;\n      if (--i < 0) {\n        i = this.DB - 1;\n        --j;\n      }\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) gcd(this,a) (HAC 14.54)\n\nfunction bnGCD(a) {\n  var x = (this.s < 0) ? this.negate() : this.clone();\n  var y = (a.s < 0) ? a.negate() : a.clone();\n  if (x.compareTo(y) < 0) {\n    var t = x;\n    x = y;\n    y = t;\n  }\n  var i = x.getLowestSetBit(),\n    g = y.getLowestSetBit();\n  if (g < 0) return x;\n  if (i < g) g = i;\n  if (g > 0) {\n    x.rShiftTo(g, x);\n    y.rShiftTo(g, y);\n  }\n  while (x.signum() > 0) {\n    if ((i = x.getLowestSetBit()) > 0) x.rShiftTo(i, x);\n    if ((i = y.getLowestSetBit()) > 0) y.rShiftTo(i, y);\n    if (x.compareTo(y) >= 0) {\n      x.subTo(y, x);\n      x.rShiftTo(1, x);\n    } else {\n      y.subTo(x, y);\n      y.rShiftTo(1, y);\n    }\n  }\n  if (g > 0) y.lShiftTo(g, y);\n  return y;\n}\n\n// (protected) this % n, n < 2^26\n\nfunction bnpModInt(n) {\n  if (n <= 0) return 0;\n  var d = this.DV % n,\n    r = (this.s < 0) ? n - 1 : 0;\n  if (this.t > 0)\n    if (d == 0) r = this[0] % n;\n    else for (var i = this.t - 1; i >= 0; --i) r = (d * r + this[i]) % n;\n  return r;\n}\n\n// (public) 1/this % m (HAC 14.61)\n\nfunction bnModInverse(m) {\n  var ac = m.isEven();\n  if ((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;\n  var u = m.clone(),\n    v = this.clone();\n  var a = nbv(1),\n    b = nbv(0),\n    c = nbv(0),\n    d = nbv(1);\n  while (u.signum() != 0) {\n    while (u.isEven()) {\n      u.rShiftTo(1, u);\n      if (ac) {\n        if (!a.isEven() || !b.isEven()) {\n          a.addTo(this, a);\n          b.subTo(m, b);\n        }\n        a.rShiftTo(1, a);\n      } else if (!b.isEven()) b.subTo(m, b);\n      b.rShiftTo(1, b);\n    }\n    while (v.isEven()) {\n      v.rShiftTo(1, v);\n      if (ac) {\n        if (!c.isEven() || !d.isEven()) {\n          c.addTo(this, c);\n          d.subTo(m, d);\n        }\n        c.rShiftTo(1, c);\n      } else if (!d.isEven()) d.subTo(m, d);\n      d.rShiftTo(1, d);\n    }\n    if (u.compareTo(v) >= 0) {\n      u.subTo(v, u);\n      if (ac) a.subTo(c, a);\n      b.subTo(d, b);\n    } else {\n      v.subTo(u, v);\n      if (ac) c.subTo(a, c);\n      d.subTo(b, d);\n    }\n  }\n  if (v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;\n  if (d.compareTo(m) >= 0) return d.subtract(m);\n  if (d.signum() < 0) d.addTo(m, d);\n  else return d;\n  if (d.signum() < 0) return d.add(m);\n  else return d;\n}\n\nvar lowprimes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97, 101,\n    103, 107, 109, 113, 127, 131, 137, 139, 149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223, 227,\n    229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293, 307, 311, 313, 317, 331, 337, 347, 349, 353, 359,\n    367, 373, 379, 383, 389, 397, 401, 409, 419, 421, 431, 433, 439, 443, 449, 457, 461, 463, 467, 479, 487, 491, 499,\n    503, 509, 521, 523, 541, 547, 557, 563, 569, 571, 577, 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, 643, 647,\n    653, 659, 661, 673, 677, 683, 691, 701, 709, 719, 727, 733, 739, 743, 751, 757, 761, 769, 773, 787, 797, 809, 811,\n    821, 823, 827, 829, 839, 853, 857, 859, 863, 877, 881, 883, 887, 907, 911, 919, 929, 937, 941, 947, 953, 967, 971,\n    977, 983, 991, 997\n];\nvar lplim = (1 << 26) / lowprimes[lowprimes.length - 1];\n\n// (public) test primality with certainty >= 1-.5^t\n\nfunction bnIsProbablePrime(t) {\n  var i, x = this.abs();\n  if (x.t == 1 && x[0] <= lowprimes[lowprimes.length - 1]) {\n    for (i = 0; i < lowprimes.length; ++i)\n      if (x[0] == lowprimes[i]) return true;\n    return false;\n  }\n  if (x.isEven()) return false;\n  i = 1;\n  while (i < lowprimes.length) {\n    var m = lowprimes[i],\n      j = i + 1;\n    while (j < lowprimes.length && m < lplim) m *= lowprimes[j++];\n    m = x.modInt(m);\n    while (i < j) if (m % lowprimes[i++] == 0) return false;\n  }\n  return x.millerRabin(t);\n}\n\n/* added by Recurity Labs */\n\nfunction nbits(x) {\n  var n = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    n += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    n += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    n += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    n += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    n += 1;\n  }\n  return n;\n}\n\nfunction bnToMPI() {\n  var ba = this.toByteArray();\n  var size = (ba.length - 1) * 8 + nbits(ba[0]);\n  var result = \"\";\n  result += String.fromCharCode((size & 0xFF00) >> 8);\n  result += String.fromCharCode(size & 0xFF);\n  result += util.bin2str(ba);\n  return result;\n}\n/* END of addition */\n\n// (protected) true if probably prime (HAC 4.24, Miller-Rabin)\nfunction bnpMillerRabin(t) {\n  var n1 = this.subtract(BigInteger.ONE);\n  var k = n1.getLowestSetBit();\n  if (k <= 0) return false;\n  var r = n1.shiftRight(k);\n  t = (t + 1) >> 1;\n  if (t > lowprimes.length) t = lowprimes.length;\n  var a = nbi();\n  var j, bases = [];\n  for (var i = 0; i < t; ++i) {\n    //Pick bases at random, instead of starting at 2\n    for (;;) {\n      j = lowprimes[Math.floor(Math.random() * lowprimes.length)];\n      if (bases.indexOf(j) == -1) break;\n    }\n    bases.push(j);\n    a.fromInt(j);\n    var y = a.modPow(r, this);\n    if (y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {\n      var j = 1;\n      while (j++ < k && y.compareTo(n1) != 0) {\n        y = y.modPowInt(2, this);\n        if (y.compareTo(BigInteger.ONE) == 0) return false;\n      }\n      if (y.compareTo(n1) != 0) return false;\n    }\n  }\n  return true;\n}\n\nvar BigInteger = require('./jsbn.js');\n\n// protected\nBigInteger.prototype.chunkSize = bnpChunkSize;\nBigInteger.prototype.toRadix = bnpToRadix;\nBigInteger.prototype.fromRadix = bnpFromRadix;\nBigInteger.prototype.fromNumber = bnpFromNumber;\nBigInteger.prototype.bitwiseTo = bnpBitwiseTo;\nBigInteger.prototype.changeBit = bnpChangeBit;\nBigInteger.prototype.addTo = bnpAddTo;\nBigInteger.prototype.dMultiply = bnpDMultiply;\nBigInteger.prototype.dAddOffset = bnpDAddOffset;\nBigInteger.prototype.multiplyLowerTo = bnpMultiplyLowerTo;\nBigInteger.prototype.multiplyUpperTo = bnpMultiplyUpperTo;\nBigInteger.prototype.modInt = bnpModInt;\nBigInteger.prototype.millerRabin = bnpMillerRabin;\n\n// public\nBigInteger.prototype.clone = bnClone;\nBigInteger.prototype.intValue = bnIntValue;\nBigInteger.prototype.byteValue = bnByteValue;\nBigInteger.prototype.shortValue = bnShortValue;\nBigInteger.prototype.signum = bnSigNum;\nBigInteger.prototype.toByteArray = bnToByteArray;\nBigInteger.prototype.equals = bnEquals;\nBigInteger.prototype.min = bnMin;\nBigInteger.prototype.max = bnMax;\nBigInteger.prototype.and = bnAnd;\nBigInteger.prototype.or = bnOr;\nBigInteger.prototype.xor = bnXor;\nBigInteger.prototype.andNot = bnAndNot;\nBigInteger.prototype.not = bnNot;\nBigInteger.prototype.shiftLeft = bnShiftLeft;\nBigInteger.prototype.shiftRight = bnShiftRight;\nBigInteger.prototype.getLowestSetBit = bnGetLowestSetBit;\nBigInteger.prototype.bitCount = bnBitCount;\nBigInteger.prototype.testBit = bnTestBit;\nBigInteger.prototype.setBit = bnSetBit;\nBigInteger.prototype.clearBit = bnClearBit;\nBigInteger.prototype.flipBit = bnFlipBit;\nBigInteger.prototype.add = bnAdd;\nBigInteger.prototype.subtract = bnSubtract;\nBigInteger.prototype.multiply = bnMultiply;\nBigInteger.prototype.divide = bnDivide;\nBigInteger.prototype.remainder = bnRemainder;\nBigInteger.prototype.divideAndRemainder = bnDivideAndRemainder;\nBigInteger.prototype.modPow = bnModPow;\nBigInteger.prototype.modInverse = bnModInverse;\nBigInteger.prototype.pow = bnPow;\nBigInteger.prototype.gcd = bnGCD;\nBigInteger.prototype.isProbablePrime = bnIsProbablePrime;\nBigInteger.prototype.toMPI = bnToMPI;\n\n// JSBN-specific extension\nBigInteger.prototype.square = bnSquare;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// RSA implementation\n\nvar BigInteger = require('./jsbn.js'),\n  util = require('../../util'),\n  random = require('../random.js');\n\nfunction SecureRandom() {\n  function nextBytes(byteArray) {\n    for (var n = 0; n < byteArray.length; n++) {\n      byteArray[n] = random.getSecureRandomOctet();\n    }\n  }\n  this.nextBytes = nextBytes;\n}\n\nfunction RSA() {\n  /**\n   * This function uses jsbn Big Num library to decrypt RSA\n   * @param m\n   *            message\n   * @param d\n   *            RSA d as BigInteger\n   * @param p\n   *            RSA p as BigInteger\n   * @param q\n   *            RSA q as BigInteger\n   * @param u\n   *            RSA u as BigInteger\n   * @return {BigInteger} The decrypted value of the message\n   */\n  function decrypt(m, d, p, q, u) {\n    var xp = m.mod(p).modPow(d.mod(p.subtract(BigInteger.ONE)), p);\n    var xq = m.mod(q).modPow(d.mod(q.subtract(BigInteger.ONE)), q);\n    util.print_debug(\"rsa.js decrypt\\nxpn:\" + util.hexstrdump(xp.toMPI()) + \"\\nxqn:\" + util.hexstrdump(xq.toMPI()));\n\n    var t = xq.subtract(xp);\n    if (t[0] == 0) {\n      t = xp.subtract(xq);\n      t = t.multiply(u).mod(q);\n      t = q.subtract(t);\n    } else {\n      t = t.multiply(u).mod(q);\n    }\n    return t.multiply(p).add(xp);\n  }\n\n  /**\n   * encrypt message\n   * @param m message as BigInteger\n   * @param e public MPI part as BigInteger\n   * @param n public MPI part as BigInteger\n   * @return BigInteger\n   */\n  function encrypt(m, e, n) {\n    return m.modPowInt(e, n);\n  }\n\n  /* Sign and Verify */\n  function sign(m, d, n) {\n    return m.modPow(d, n);\n  }\n\n  function verify(x, e, n) {\n    return x.modPowInt(e, n);\n  }\n\n  // \"empty\" RSA key constructor\n\n  function keyObject() {\n    this.n = null;\n    this.e = 0;\n    this.ee = null;\n    this.d = null;\n    this.p = null;\n    this.q = null;\n    this.dmp1 = null;\n    this.dmq1 = null;\n    this.u = null;\n  }\n\n  // Generate a new random private key B bits long, using public expt E\n\n  function generate(B, E) {\n    var key = new keyObject();\n    var rng = new SecureRandom();\n    var qs = B >> 1;\n    key.e = parseInt(E, 16);\n    key.ee = new BigInteger(E, 16);\n    for (;;) {\n      for (;;) {\n        key.p = new BigInteger(B - qs, 1, rng);\n        if (key.p.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.p.isProbablePrime(10))\n          break;\n      }\n      for (;;) {\n        key.q = new BigInteger(qs, 1, rng);\n        if (key.q.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.q.isProbablePrime(10))\n          break;\n      }\n      if (key.p.compareTo(key.q) <= 0) {\n        var t = key.p;\n        key.p = key.q;\n        key.q = t;\n      }\n      var p1 = key.p.subtract(BigInteger.ONE);\n      var q1 = key.q.subtract(BigInteger.ONE);\n      var phi = p1.multiply(q1);\n      if (phi.gcd(key.ee).compareTo(BigInteger.ONE) == 0) {\n        key.n = key.p.multiply(key.q);\n        key.d = key.ee.modInverse(phi);\n        key.dmp1 = key.d.mod(p1);\n        key.dmq1 = key.d.mod(q1);\n        key.u = key.p.modInverse(key.q);\n        break;\n      }\n    }\n    return key;\n  }\n\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n  this.verify = verify;\n  this.sign = sign;\n  this.generate = generate;\n  this.keyObject = keyObject;\n}\n\nmodule.exports = RSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Retrieve secure random byte string of the specified length\n   * @param {Integer} length Length in bytes to generate\n   * @return {String} Random byte string\n   */\n  getRandomBytes: function(length) {\n    var result = '';\n    for (var i = 0; i < length; i++) {\n      result += String.fromCharCode(this.getSecureRandomOctet());\n    }\n    return result;\n  },\n\n  /**\n   * Return a pseudo-random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A pseudo random number\n   */\n  getPseudoRandom: function(from, to) {\n    return Math.round(Math.random() * (to - from)) + from;\n  },\n\n  /**\n   * Return a secure random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A secure random number\n   */\n  getSecureRandom: function(from, to) {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    var bits = ((to - from)).toString(2).length;\n    while ((buf[0] & (Math.pow(2, bits) - 1)) > (to - from))\n      window.crypto.getRandomValues(buf);\n    return from + (Math.abs(buf[0] & (Math.pow(2, bits) - 1)));\n  },\n\n  getSecureRandomOctet: function() {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    return buf[0] & 0xFF;\n  },\n\n  /**\n   * Create a secure random big integer of bits length\n   * @param {Integer} bits Bit length of the MPI to create\n   * @return {BigInteger} Resulting big integer\n   */\n  getRandomBigInteger: function(bits) {\n    if (bits < 0) {\n      return null;\n    }\n    var numBytes = Math.floor((bits + 7) / 8);\n\n    var randomBits = this.getRandomBytes(numBytes);\n    if (bits % 8 > 0) {\n\n      randomBits = String.fromCharCode(\n      (Math.pow(2, bits % 8) - 1) &\n        randomBits.charCodeAt(0)) +\n        randomBits.substring(1);\n    }\n    var mpi = new type_mpi();\n    mpi.fromBytes(randomBits);\n    return mpi.toBigInteger();\n  },\n\n  getRandomBigIntegerInRange: function(min, max) {\n    if (max.compareTo(min) <= 0) {\n      return;\n    }\n\n    var range = max.subtract(min);\n    var r = this.getRandomBigInteger(range.bitLength());\n    while (r > range) {\n      r = this.getRandomBigInteger(range.bitLength());\n    }\n    return min.add(r);\n  }\n\n};\n","var publicKey = require('./public_key'),\n  pkcs1 = require('./pkcs1.js'),\n  hashModule = require('./hash');\n\nmodule.exports = {\n  /**\n   * \n   * @param {Integer} algo public Key algorithm\n   * @param {Integer} hash_algo Hash algorithm\n   * @param {openpgp_type_mpi[]} msg_MPIs Signature multiprecision integers\n   * @param {openpgp_type_mpi[]} publickey_MPIs Public key multiprecision integers \n   * @param {String} data Data on where the signature was computed on.\n   * @return {Boolean} true if signature (sig_data was equal to data over hash)\n   */\n  verify: function(algo, hash_algo, msg_MPIs, publickey_MPIs, data) {\n    var calc_hash = hashModule.digest(hash_algo, data);\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var n = publickey_MPIs[0].toBigInteger();\n        var e = publickey_MPIs[1].toBigInteger();\n        var x = msg_MPIs[0].toBigInteger();\n        var dopublic = rsa.verify(x, e, n);\n        var hash = pkcs1.emsa.decode(hash_algo, dopublic.toMPI().substring(2));\n        if (hash == -1) {\n          throw new Error('PKCS1 padding in message or key incorrect. Aborting...');\n        }\n        return hash == calc_hash;\n\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error(\"signing with Elgamal is not defined in the OpenPGP standard.\");\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n        var s1 = msg_MPIs[0].toBigInteger();\n        var s2 = msg_MPIs[1].toBigInteger();\n        var p = publickey_MPIs[0].toBigInteger();\n        var q = publickey_MPIs[1].toBigInteger();\n        var g = publickey_MPIs[2].toBigInteger();\n        var y = publickey_MPIs[3].toBigInteger();\n        var m = data;\n        var dopublic = dsa.verify(hash_algo, s1, s2, m, p, q, g, y);\n        return dopublic.compareTo(s1) == 0;\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n\n  },\n\n  /**\n   * Create a signature on data using the specified algorithm\n   * @param {Integer} hash_algo hash Algorithm to use (See RFC4880 9.4)\n   * @param {Integer} algo Asymmetric cipher algorithm to use (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Public key multiprecision integers \n   * of the private key \n   * @param {openpgp_type_mpi[]} secretMPIs Private key multiprecision \n   * integers which is used to sign the data\n   * @param {String} data Data to be signed\n   * @return {openpgp_type_mpi[]}\n   */\n  sign: function(hash_algo, algo, keyIntegers, data) {\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var d = keyIntegers[2].toBigInteger();\n        var n = keyIntegers[0].toBigInteger();\n        var m = pkcs1.emsa.encode(hash_algo,\n          data, keyIntegers[0].byteLength());\n\n        return rsa.sign(m, d, n).toMPI();\n\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n\n        var p = keyIntegers[0].toBigInteger();\n        var q = keyIntegers[1].toBigInteger();\n        var g = keyIntegers[2].toBigInteger();\n        var y = keyIntegers[3].toBigInteger();\n        var x = keyIntegers[4].toBigInteger();\n        var m = data;\n        var result = dsa.sign(hash_algo, m, g, p, q, x);\n\n        return result[0].toString() + result[1].toString();\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error('Signing with Elgamal is not defined in the OpenPGP standard.');\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar base64 = require('./base64.js');\nvar enums = require('../enums.js');\n\n/**\n * Finds out which Ascii Armoring type is used. This is an internal function\n * @param {String} text [String] ascii armored text\n * @returns {Integer} 0 = MESSAGE PART n of m\n *         1 = MESSAGE PART n\n *         2 = SIGNED MESSAGE\n *         3 = PGP MESSAGE\n *         4 = PUBLIC KEY BLOCK\n *         5 = PRIVATE KEY BLOCK\n *         null = unknown\n */\nfunction get_type(text) {\n  var splittedtext = text.split('-----');\n  // BEGIN PGP MESSAGE, PART X/Y\n  // Used for multi-part messages, where the armor is split amongst Y\n  // parts, and this is the Xth part out of Y.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+\\/\\d+/)) {\n    return enums.armor.multipart_section;\n  } else\n  // BEGIN PGP MESSAGE, PART X\n  // Used for multi-part messages, where this is the Xth part of an\n  // unspecified number of parts. Requires the MESSAGE-ID Armor\n  // Header to be used.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+/)) {\n    return enums.armor.multipart_last;\n\n  } else\n  // BEGIN PGP SIGNATURE\n  // Used for detached signatures, OpenPGP/MIME signatures, and\n  // cleartext signatures. Note that PGP 2.x uses BEGIN PGP MESSAGE\n  // for detached signatures.\n  if (splittedtext[1].match(/BEGIN PGP SIGNED MESSAGE/)) {\n    return enums.armor.signed;\n\n  } else\n  // BEGIN PGP MESSAGE\n  // Used for signed, encrypted, or compressed files.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE/)) {\n    return enums.armor.message;\n\n  } else\n  // BEGIN PGP PUBLIC KEY BLOCK\n  // Used for armoring public keys.\n  if (splittedtext[1].match(/BEGIN PGP PUBLIC KEY BLOCK/)) {\n    return enums.armor.public_key;\n\n  } else\n  // BEGIN PGP PRIVATE KEY BLOCK\n  // Used for armoring private keys.\n  if (splittedtext[1].match(/BEGIN PGP PRIVATE KEY BLOCK/)) {\n    return enums.armor.private_key;\n  }\n}\n\n/**\n * Add additional information to the armor version of an OpenPGP binary\n * packet block.\n * @author  Alex\n * @version 2011-12-16\n * @returns {String} The header information\n */\nfunction armor_addheader(options) {\n  var result = \"\";\n  if (options.show_version) {\n    result += \"Version: \" + options.versionstring + '\\r\\n';\n  }\n  if (options.show_comment) {\n    result += \"Comment: \" + options.commentstring + '\\r\\n';\n  }\n  result += '\\r\\n';\n  return result;\n}\n\n\n\n/**\n * Calculates a checksum over the given data and returns it base64 encoded\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {String} Base64 encoded checksum\n */\nfunction getCheckSum(data) {\n  var c = createcrc24(data);\n  var str = \"\" + String.fromCharCode(c >> 16) +\n    String.fromCharCode((c >> 8) & 0xFF) +\n    String.fromCharCode(c & 0xFF);\n  return base64.encode(str);\n}\n\n/**\n * Calculates the checksum over the given data and compares it with the \n * given base64 encoded checksum\n * @param {String} data Data to create a CRC-24 checksum for\n * @param {String} checksum Base64 encoded checksum\n * @return {Boolean} True if the given checksum is correct; otherwise false\n */\nfunction verifyCheckSum(data, checksum) {\n  var c = getCheckSum(data);\n  var d = checksum;\n  return c[0] == d[0] && c[1] == d[1] && c[2] == d[2];\n}\n/**\n * Internal function to calculate a CRC-24 checksum over a given string (data)\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {Integer} The CRC-24 checksum as number\n */\nvar crc_table = [\n    0x00000000, 0x00864cfb, 0x018ad50d, 0x010c99f6, 0x0393e6e1, 0x0315aa1a, 0x021933ec, 0x029f7f17, 0x07a18139,\n    0x0727cdc2, 0x062b5434, 0x06ad18cf, 0x043267d8, 0x04b42b23, 0x05b8b2d5, 0x053efe2e, 0x0fc54e89, 0x0f430272,\n    0x0e4f9b84, 0x0ec9d77f, 0x0c56a868, 0x0cd0e493, 0x0ddc7d65, 0x0d5a319e, 0x0864cfb0, 0x08e2834b, 0x09ee1abd,\n    0x09685646, 0x0bf72951, 0x0b7165aa, 0x0a7dfc5c, 0x0afbb0a7, 0x1f0cd1e9, 0x1f8a9d12, 0x1e8604e4, 0x1e00481f,\n    0x1c9f3708, 0x1c197bf3, 0x1d15e205, 0x1d93aefe, 0x18ad50d0, 0x182b1c2b, 0x192785dd, 0x19a1c926, 0x1b3eb631,\n    0x1bb8faca, 0x1ab4633c, 0x1a322fc7, 0x10c99f60, 0x104fd39b, 0x11434a6d, 0x11c50696, 0x135a7981, 0x13dc357a,\n    0x12d0ac8c, 0x1256e077, 0x17681e59, 0x17ee52a2, 0x16e2cb54, 0x166487af, 0x14fbf8b8, 0x147db443, 0x15712db5,\n    0x15f7614e, 0x3e19a3d2, 0x3e9fef29, 0x3f9376df, 0x3f153a24, 0x3d8a4533, 0x3d0c09c8, 0x3c00903e, 0x3c86dcc5,\n    0x39b822eb, 0x393e6e10, 0x3832f7e6, 0x38b4bb1d, 0x3a2bc40a, 0x3aad88f1, 0x3ba11107, 0x3b275dfc, 0x31dced5b,\n    0x315aa1a0,\n    0x30563856, 0x30d074ad, 0x324f0bba, 0x32c94741, 0x33c5deb7, 0x3343924c, 0x367d6c62, 0x36fb2099, 0x37f7b96f,\n    0x3771f594, 0x35ee8a83, 0x3568c678, 0x34645f8e, 0x34e21375, 0x2115723b, 0x21933ec0, 0x209fa736, 0x2019ebcd,\n    0x228694da, 0x2200d821, 0x230c41d7, 0x238a0d2c, 0x26b4f302, 0x2632bff9, 0x273e260f, 0x27b86af4, 0x252715e3,\n    0x25a15918, 0x24adc0ee, 0x242b8c15, 0x2ed03cb2, 0x2e567049, 0x2f5ae9bf, 0x2fdca544, 0x2d43da53, 0x2dc596a8,\n    0x2cc90f5e, 0x2c4f43a5, 0x2971bd8b, 0x29f7f170, 0x28fb6886, 0x287d247d, 0x2ae25b6a, 0x2a641791, 0x2b688e67,\n    0x2beec29c, 0x7c3347a4, 0x7cb50b5f, 0x7db992a9, 0x7d3fde52, 0x7fa0a145, 0x7f26edbe, 0x7e2a7448, 0x7eac38b3,\n    0x7b92c69d, 0x7b148a66, 0x7a181390, 0x7a9e5f6b, 0x7801207c, 0x78876c87, 0x798bf571, 0x790db98a, 0x73f6092d,\n    0x737045d6, 0x727cdc20, 0x72fa90db, 0x7065efcc, 0x70e3a337, 0x71ef3ac1, 0x7169763a, 0x74578814, 0x74d1c4ef,\n    0x75dd5d19, 0x755b11e2, 0x77c46ef5, 0x7742220e, 0x764ebbf8, 0x76c8f703, 0x633f964d, 0x63b9dab6, 0x62b54340,\n    0x62330fbb,\n    0x60ac70ac, 0x602a3c57, 0x6126a5a1, 0x61a0e95a, 0x649e1774, 0x64185b8f, 0x6514c279, 0x65928e82, 0x670df195,\n    0x678bbd6e, 0x66872498, 0x66016863, 0x6cfad8c4, 0x6c7c943f, 0x6d700dc9, 0x6df64132, 0x6f693e25, 0x6fef72de,\n    0x6ee3eb28, 0x6e65a7d3, 0x6b5b59fd, 0x6bdd1506, 0x6ad18cf0, 0x6a57c00b, 0x68c8bf1c, 0x684ef3e7, 0x69426a11,\n    0x69c426ea, 0x422ae476, 0x42aca88d, 0x43a0317b, 0x43267d80, 0x41b90297, 0x413f4e6c, 0x4033d79a, 0x40b59b61,\n    0x458b654f, 0x450d29b4, 0x4401b042, 0x4487fcb9, 0x461883ae, 0x469ecf55, 0x479256a3, 0x47141a58, 0x4defaaff,\n    0x4d69e604, 0x4c657ff2, 0x4ce33309, 0x4e7c4c1e, 0x4efa00e5, 0x4ff69913, 0x4f70d5e8, 0x4a4e2bc6, 0x4ac8673d,\n    0x4bc4fecb, 0x4b42b230, 0x49ddcd27, 0x495b81dc, 0x4857182a, 0x48d154d1, 0x5d26359f, 0x5da07964, 0x5cace092,\n    0x5c2aac69, 0x5eb5d37e, 0x5e339f85, 0x5f3f0673, 0x5fb94a88, 0x5a87b4a6, 0x5a01f85d, 0x5b0d61ab, 0x5b8b2d50,\n    0x59145247, 0x59921ebc, 0x589e874a, 0x5818cbb1, 0x52e37b16, 0x526537ed, 0x5369ae1b, 0x53efe2e0, 0x51709df7,\n    0x51f6d10c,\n    0x50fa48fa, 0x507c0401, 0x5542fa2f, 0x55c4b6d4, 0x54c82f22, 0x544e63d9, 0x56d11cce, 0x56575035, 0x575bc9c3,\n    0x57dd8538\n];\n\nfunction createcrc24(input) {\n  var crc = 0xB704CE;\n  var index = 0;\n\n  while ((input.length - index) > 16) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 1)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 2)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 3)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 4)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 5)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 6)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 7)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 8)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 9)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 10)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 11)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 12)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 13)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 14)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 15)) & 0xff];\n    index += 16;\n  }\n\n  for (var j = index; j < input.length; j++) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index++)) & 0xff];\n  }\n  return crc & 0xffffff;\n}\n\n/**\n * DeArmor an OpenPGP armored message; verify the checksum and return \n * the encoded bytes\n * @param {String} text OpenPGP armored message\n * @returns {(Boolean|Object)} Either false in case of an error \n * or an object with attribute \"text\" containing the message text\n * and an attribute \"openpgp\" containing the bytes.\n */\nfunction dearmor(text) {\n  text = text.replace(/\\r/g, '');\n\n  var type = get_type(text);\n\n  if (type != 2) {\n    var splittedtext = text.split('-----');\n\n    var data = {\n      openpgp: base64.decode(\n        splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[0]\n        .replace(/\\n- /g, \"\\n\")),\n      type: type\n    };\n\n    if (verifyCheckSum(data.openpgp,\n      splittedtext[2]\n      .split('\\n\\n')[1]\n      .split(\"\\n=\")[1]\n      .split('\\n')[0]))\n\n      return data;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed: '\" + splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[1]\n        .split('\\n')[0] + \"' should be '\" + getCheckSum(data)) + \"'\";\n      return false;\n    }\n  } else {\n    var splittedtext = text.split('-----');\n\n    var result = {\n      text: splittedtext[2]\n        .replace(/\\n- /g, \"\\n\")\n        .split(\"\\n\\n\")[1],\n      openpgp: base64_decode(splittedtext[4]\n        .split(\"\\n\\n\")[1]\n        .split(\"\\n=\")[0]),\n      type: type\n    };\n\n    if (verifyCheckSum(result.openpgp, splittedtext[4]\n      .split(\"\\n\\n\")[1]\n      .split(\"\\n=\")[1]))\n\n      return result;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed\");\n      return false;\n    }\n  }\n}\n\n\n/**\n * Armor an OpenPGP binary packet block\n * @param {Integer} messagetype type of the message\n * @param data\n * @param {Integer} partindex\n * @param {Integer} parttotal\n * @returns {String} Armored text\n */\nfunction armor(messagetype, data, options, partindex, parttotal) {\n  var result = \"\";\n  switch (messagetype) {\n    case enums.armor.multipart_section:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      break;\n    case enums.armor.mutlipart_last:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      break;\n    case enums.armor.signed:\n      result += \"\\r\\n-----BEGIN PGP SIGNED MESSAGE-----\\r\\nHash: \" + data.hash + \"\\r\\n\\r\\n\";\n      result += data.text.replace(/\\n-/g, \"\\n- -\");\n      result += \"\\r\\n-----BEGIN PGP SIGNATURE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data.openpgp);\n      result += \"\\r\\n=\" + getCheckSum(data.openpgp) + \"\\r\\n\";\n      result += \"-----END PGP SIGNATURE-----\\r\\n\";\n      break;\n    case enums.armor.message:\n      result += \"-----BEGIN PGP MESSAGE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE-----\\r\\n\";\n      break;\n    case enums.armor.public_key:\n      result += \"-----BEGIN PGP PUBLIC KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PUBLIC KEY BLOCK-----\\r\\n\\r\\n\";\n      break;\n    case enums.armor.private_key:\n      result += \"-----BEGIN PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      break;\n  }\n\n  return result;\n}\n\nmodule.exports = {\n  encode: armor,\n  decode: dearmor\n};\n","/* OpenPGP radix-64/base64 string encoding/decoding\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.0, check www.haneWIN.de for the latest version\n *\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other materials\n * provided with the application or distribution.\n */\n\nvar b64s = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\n\nfunction s2r(t) {\n  var a, c, n;\n  var r = '',\n    l = 0,\n    s = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = t.charCodeAt(n);\n    if (s == 0) {\n      r += b64s.charAt((c >> 2) & 63);\n      a = (c & 3) << 4;\n    } else if (s == 1) {\n      r += b64s.charAt((a | (c >> 4) & 15));\n      a = (c & 15) << 2;\n    } else if (s == 2) {\n      r += b64s.charAt(a | ((c >> 6) & 3));\n      l += 1;\n      if ((l % 60) == 0)\n        r += \"\\n\";\n      r += b64s.charAt(c & 63);\n    }\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n\n    s += 1;\n    if (s == 3)\n      s = 0;\n  }\n  if (s > 0) {\n    r += b64s.charAt(a);\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n    l += 1;\n  }\n  if (s == 1) {\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n  }\n\n  return r;\n}\n\nfunction r2s(t) {\n  var c, n;\n  var r = '',\n    s = 0,\n    a = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = b64s.indexOf(t.charAt(n));\n    if (c >= 0) {\n      if (s)\n        r += String.fromCharCode(a | (c >> (6 - s)) & 255);\n      s = (s + 2) & 7;\n      a = (c << s) & 255;\n    }\n  }\n  return r;\n}\n\nmodule.exports = {\n  encode: s2r,\n  decode: r2s\n}\n","var enums = {\n\n  /** A string to key specifier type\n   * @enum {Integer}\n   */\n  s2k: {\n    simple: 0,\n    salted: 1,\n    iterated: 3,\n    gnu: 101\n  },\n\n  /** RFC4880, section 9.1 \n   * @enum {String}\n   */\n  publicKey: {\n    rsa_encrypt_sign: 1,\n    rsa_encrypt: 2,\n    rsa_sign: 3,\n    elgamal: 16,\n    dsa: 17\n  },\n\n  /** RFC4880, section 9.2 \n   * @enum {String}\n   */\n  symmetric: {\n    plaintext: 0,\n    /** Not implemented! */\n    idea: 1,\n    tripledes: 2,\n    cast5: 3,\n    blowfish: 4,\n    aes128: 7,\n    aes192: 8,\n    aes256: 9,\n    twofish: 10\n  },\n\n  /** RFC4880, section 9.3\n   * @enum {String}\n   */\n  compression: {\n    uncompressed: 0,\n    /** RFC1951 */\n    zip: 1,\n    /** RFC1950 */\n    zlib: 2,\n    bzip2: 3\n  },\n\n  /** RFC4880, section 9.4\n   * @enum {String}\n   */\n  hash: {\n    md5: 1,\n    sha1: 2,\n    ripemd: 3,\n    sha256: 8,\n    sha384: 9,\n    sha512: 10,\n    sha224: 11\n  },\n\n\n  /**\n   * @enum {String}\n   * A list of packet types and numeric tags associated with them.\n   */\n  packet: {\n    public_key_encrypted_session_key: 1,\n    signature: 2,\n    sym_encrypted_session_key: 3,\n    one_pass_signature: 4,\n    secret_key: 5,\n    public_key: 6,\n    secret_subkey: 7,\n    compressed: 8,\n    symmetrically_encrypted: 9,\n    marker: 10,\n    literal: 11,\n    trust: 12,\n    userid: 13,\n    public_subkey: 14,\n    user_attribute: 17,\n    sym_encrypted_integrity_protected: 18,\n    modification_detection_code: 19\n  },\n\n\n  /**\n   * Data types in the literal packet\n   * @readonly\n   * @enum {String}\n   */\n  literal: {\n    /** Binary data */\n    binary: 'b'.charCodeAt(),\n    /** Text data */\n    text: 't'.charCodeAt(),\n    /** Utf8 data */\n    utf8: 'u'.charCodeAt()\n  },\n\n\n  /** One pass signature packet type\n   * @enum {String} */\n  signature: {\n    /** 0x00: Signature of a binary document. */\n    binary: 0,\n    /** 0x01: Signature of a canonical text document.\n     * Canonicalyzing the document by converting line endings. */\n    text: 1,\n    /** 0x02: Standalone signature.\n     * This signature is a signature of only its own subpacket contents.\n     * It is calculated identically to a signature over a zero-lengh\n     * binary document.  Note that it doesn't make sense to have a V3\n     * standalone signature. */\n    standalone: 2,\n    /** 0x10: Generic certification of a User ID and Public-Key packet.\n     * The issuer of this certification does not make any particular\n     * assertion as to how well the certifier has checked that the owner\n     * of the key is in fact the person described by the User ID. */\n    cert_generic: 16,\n    /** 0x11: Persona certification of a User ID and Public-Key packet.\n     * The issuer of this certification has not done any verification of\n     * the claim that the owner of this key is the User ID specified. */\n    cert_persona: 17,\n    /** 0x12: Casual certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done some casual\n     * verification of the claim of identity. */\n    cert_casual: 18,\n    /** 0x13: Positive certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done substantial\n     * verification of the claim of identity.\n     * \n     * Most OpenPGP implementations make their \"key signatures\" as 0x10\n     * certifications.  Some implementations can issue 0x11-0x13\n     * certifications, but few differentiate between the types. */\n    cert_positive: 19,\n    /** 0x30: Certification revocation signature\n     * This signature revokes an earlier User ID certification signature\n     * (signature class 0x10 through 0x13) or direct-key signature\n     * (0x1F).  It should be issued by the same key that issued the\n     * revoked signature or an authorized revocation key.  The signature\n     * is computed over the same data as the certificate that it\n     * revokes, and should have a later creation date than that\n     * certificate. */\n    cert_revocation: 48,\n    /** 0x18: Subkey Binding Signature\n     * This signature is a statement by the top-level signing key that\n     * indicates that it owns the subkey.  This signature is calculated\n     * directly on the primary key and subkey, and not on any User ID or\n     * other packets.  A signature that binds a signing subkey MUST have\n     * an Embedded Signature subpacket in this binding signature that\n     * contains a 0x19 signature made by the signing subkey on the\n     * primary key and subkey. */\n    subkey_binding: 24,\n    /** 0x19: Primary Key Binding Signature\n\t\t* This signature is a statement by a signing subkey, indicating\n\t\t* that it is owned by the primary key and subkey.  This signature\n\t\t* is calculated the same way as a 0x18 signature: directly on the\n\t\t* primary key and subkey, and not on any User ID or other packets.\n\t\t\n\t\t* When a signature is made over a key, the hash data starts with the\n\t\t* octet 0x99, followed by a two-octet length of the key, and then body\n\t\t* of the key packet.  (Note that this is an old-style packet header for\n\t\t* a key packet with two-octet length.)  A subkey binding signature\n\t\t* (type 0x18) or primary key binding signature (type 0x19) then hashes\n\t\t* the subkey using the same format as the main key (also using 0x99 as\n\t\t* the first octet). */\n    key_binding: 25,\n    /** 0x1F: Signature directly on a key\n     * This signature is calculated directly on a key.  It binds the\n     * information in the Signature subpackets to the key, and is\n     * appropriate to be used for subpackets that provide information\n     * about the key, such as the Revocation Key subpacket.  It is also\n     * appropriate for statements that non-self certifiers want to make\n     * about the key itself, rather than the binding between a key and a\n     * name. */\n    key: 31,\n    /** 0x20: Key revocation signature\n     * The signature is calculated directly on the key being revoked.  A\n     * revoked key is not to be used.  Only revocation signatures by the\n     * key being revoked, or by an authorized revocation key, should be\n     * considered valid revocation signatures.a */\n    key_revocation: 32,\n    /** 0x28: Subkey revocation signature\n     * The signature is calculated directly on the subkey being revoked.\n     * A revoked subkey is not to be used.  Only revocation signatures\n     * by the top-level signature key that is bound to this subkey, or\n     * by an authorized revocation key, should be considered valid\n     * revocation signatures.\n     * Key revocation signatures (types 0x20 and 0x28)\n     * hash only the key being revoked. */\n    subkey_revocation: 40,\n    /** 0x40: Timestamp signature.\n     * This signature is only meaningful for the timestamp contained in\n     * it. */\n    timestamp: 64,\n    /**    0x50: Third-Party Confirmation signature.\n     * This signature is a signature over some other OpenPGP Signature\n     * packet(s).  It is analogous to a notary seal on the signed data.\n     * A third-party signature SHOULD include Signature Target\n     * subpacket(s) to give easy identification.  Note that we really do\n     * mean SHOULD.  There are plausible uses for this (such as a blind\n     * party that only sees the signature, not the key or source\n     * document) that cannot include a target subpacket. */\n    third_party: 80\n  },\n\n  signatureSubpacket: {\n    signature_creation_time: 2,\n    signature_expiration_time: 3,\n    exportable_certification: 4,\n    trust_signature: 5,\n    regular_expression: 6,\n    revocable: 7,\n    reserved: 8,\n    key_expiration_time: 9,\n    placeholder_backwards_compatibility: 10,\n    preferred_symmetric_algorithms: 11,\n    revocation_key: 12,\n    issuer: 16,\n    notification_data: 20,\n    preferred_hash_algorithms: 21,\n    preferred_compression_algorithms: 22,\n    key_server_preferences: 23,\n    preferred_key_server: 24,\n    primary_user_id: 25,\n    policy_uri: 26,\n    key_flags: 27,\n    signers_user_id: 28,\n    reason_for_revocation: 29,\n    features: 30,\n    signature_target: 31,\n    embedded_signature: 32\n  },\n\n  armor: {\n    multipart_section: 0,\n    multipart_last: 1,\n    signed: 2,\n    message: 3,\n    public_key: 4,\n    private_key: 5\n  },\n\n  // Asserts validity and converts from string/integer to integer.\n  write: function(type, e) {\n    if (typeof e == 'number') {\n      e = this.read(type, e);\n    }\n\n    if (type[e] !== undefined) {\n      return type[e];\n    } else throw new Error('Invalid enum value.');\n  },\n  // Converts from an integer to string.\n  read: function(type, e) {\n    for (var i in type)\n      if (type[i] == e) return i;\n\n    throw new Error('Invalid enum value.');\n  }\n}\n\nmodule.exports = enums;\n","\nmodule.exports = require('./openpgp.js');\nmodule.exports.key = require('./key.js');\nmodule.exports.message = require('./message.js');\nmodule.exports.util = require('./util');\nmodule.exports.packet = require('./packet');\nmodule.exports.mpi = require('./type/mpi.js');\nmodule.exports.s2k = require('./type/s2k.js');\nmodule.exports.keyid = require('./type/keyid.js');\nmodule.exports.armor = require('./encoding/armor.js');\nmodule.exports.enums = require('./enums.js');\nmodule.exports.config = require('./config');\nmodule.exports.crypto = require('./crypto');\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc Class that represents an OpenPGP key. Must contain a master key. \n * @param  {packetlist} packetlist [description]\n * Can contain additional subkeys, signatures,\n * user ids, user attributes.\n */\n\n function key(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n\n  /** Returns the primary key (secret or public)\n   * @returns {packet_secret_key|packet_public_key|null} */\n  this.getKey = function() {\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_key ||\n        this.packets[i].tag == enums.packet.secret_key) {\n        return this.packets[i];\n      }\n    }\n    return null;\n  };\n\n  /** Returns all the private and public subkeys \n   * @returns {[public_subkey|secret_subkey]} */\n  this.getSubkeys = function() {\n\n    var subkeys = [];\n\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_subkey ||\n        this.packets[i].tag == enums.packet.secret_subkey) {\n        subkeys.push(this.packets[i]);\n      }\n    }\n\n    return subkeys;\n  };\n\n  this.getAllKeys = function() {\n    return [this.getKey()].concat(this.getSubkeys());\n  };\n\n  this.getKeyids = function() {\n    var keyids = [];\n    var keys = this.getAllKeys();\n    for (var i = 0; i < keys.length; i++) {\n      keyids.push(keys[i].getKeyId());\n    }\n    return keyids;\n  };\n\n  this.getKeyById = function(keyid) {\n    var keys = this.getAllKeys();\n    for (var i = 0; i < keys.length; i++) {\n      if (keys[i].getKeyId().equals(keyid)) {\n        return keys[i];\n      }\n    }\n  }\n\n  this.getSigningKey = function() {\n\n    var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa];\n\n    signing = signing.map(function(s) {\n      return enums.read(enums.publicKey, s);\n    });\n\n    var keys = this.getAllKeys();\n\n    for (var i = 0; i < keys.length; i++) {\n      if (signing.indexOf(keys[i].algorithm) !== -1) {\n        return keys[i];\n      }\n    }\n\n    return null;\n  };\n\n  function getPreferredSignatureHashAlgorithm() {\n    var pkey = this.getSigningKey();\n    if (pkey === null) {\n      util.print_error(\"private key is for encryption only! Cannot create a signature.\");\n      return null;\n    }\n    if (pkey.publicKey.publicKeyAlgorithm == 17) {\n      var dsa = new DSA();\n      return dsa.select_hash_algorithm(pkey.publicKey.MPIs[1].toBigInteger()); // q\n    }\n    //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8\n    //separate private key preference from digest preferences\n    return openpgp.config.config.prefer_hash_algorithm;\n  }\n\n  /**\n   * Finds an encryption key for this key\n   * @returns null if no encryption key has been found\n   */\n  this.getEncryptionKey = function() {\n    // V4: by convention subkeys are prefered for encryption service\n    // V3: keys MUST NOT have subkeys\n    var isValidEncryptionKey = function(key) {\n      return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey,\n        enums.publicKey.rsa_sign);\n      //TODO verify key\n      //&& keys.verifyKey()\n    };\n\n    var subkeys = this.getSubkeys();\n\n    for (var j = 0; j < subkeys.length; j++) {\n      if (isValidEncryptionKey(subkeys[j])) {\n        return subkeys[j];\n      }\n    }\n    // if no valid subkey for encryption, use primary key\n    var primaryKey = this.getKey();\n    if (isValidEncryptionKey(primaryKey)) {\n      return primaryKey;\n    }\n    return null;\n  };\n\n  this.decrypt = function(passphrase) {\n    var keys = this.getAllKeys();\n\n    for (var i in keys)\n      if (keys[i].tag == enums.packet.secret_subkey ||\n        keys[i].tag == enums.packet.secret_key) {\n        keys[i].decrypt(passphrase);\n      }\n  };\n\n\n  // TODO need to implement this\n\n  function revoke() {\n\n  }\n\n};\n\n/**\n * reads an OpenPGP armored text and returns a key object\n * @param {String} armoredText text to be parsed\n * @return {key} new key object\n */\nkey.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newKey = new key(packetlist);\n  return newKey;\n}\n\nmodule.exports = key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc A generic message containing one or more literal packets.\n */\n\nfunction message(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n  /**\n   * Returns the key IDs of the public keys to which the session key is encrypted\n   * @return {[keyId]} keyId provided as string of hex number (lowercase)\n   */\n  this.getKeyIds = function() {\n    var keyIds = [];\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    pkESKeyPacketlist.forEach(function(packet) {\n      keyIds.push(packet.publicKeyId.toHex());\n    });\n    return keyIds;\n  }\n\n  function generic_decrypt(packets, passphrase) {\n    var sessionkey;\n\n    for (var i = 0; i < packets.length; i++) {\n      if (packets[i].tag == openpgp_packet.tags.public_key_encrypted_session_key) {\n        var key = openpgp.keyring.getKeyById(packets[i].public_key_id);\n\n      }\n    }\n\n  }\n\n  /**\n   * Decrypts a message and generates user interface message out of the found.\n   * MDC will be verified as well as message signatures\n   * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key)\n   * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message\n   * @return {} plaintext of the message or null on error\n   */\n  this.decrypt = function(key) {\n    var decryptedMessages = [];\n    var keyids = key.getKeyids();\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    outer: for (var i = 0; i < pkESKeyPacketlist.length; i++) {\n      var pkESKeyPacket = pkESKeyPacketlist[i];\n      for (var j = 0; j < keyids.length; j++) {\n        if (pkESKeyPacket.publicKeyId.equals(keyids[j])) {\n          pkESKeyPacket.decrypt(key.getKeyById(pkESKeyPacket.publicKeyId));\n          var symEncryptedPacketlist = this.packets.filter(function(packet) {\n            return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected;\n          });\n          for (var k = 0; k < symEncryptedPacketlist.length; k++) {\n            var symEncryptedPacket = symEncryptedPacketlist[k];\n            symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey);\n            for (var l = 0; l < symEncryptedPacket.packets.length; l++) {\n              var dataPacket = symEncryptedPacket.packets[l];\n              switch (dataPacket.tag) {\n                case enums.packet.literal:\n                  decryptedMessages.push(dataPacket.getBytes());\n                  break;\n                case enums.packet.compressed:\n                  //TODO\n                  break;\n                default:\n                  //TODO\n              }\n            }\n          }\n          break outer;\n        }\n      }\n    }\n    return decryptedMessages;\n  }\n\n  /**\n   * Decrypts a message and generates user interface message out of the found.\n   * MDC will be verified as well as message signatures\n   * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key)\n   * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {String} plaintext of the message or null on error\n   */\n  function decryptAndVerifySignature(private_key, sessionkey, pubkey) {\n    if (private_key == null || sessionkey == null || sessionkey == \"\")\n      return null;\n    var decrypted = sessionkey.decrypt(this, private_key.keymaterial);\n    if (decrypted == null)\n      return null;\n    var packet;\n    var position = 0;\n    var len = decrypted.length;\n    var validSignatures = new Array();\n    util.print_debug_hexstr_dump(\"openpgp.msg.messge decrypt:\\n\", decrypted);\n\n    var messages = openpgp.read_messages_dearmored({\n      text: decrypted,\n      openpgp: decrypted\n    });\n    for (var m in messages) {\n      if (messages[m].data) {\n        this.text = messages[m].data;\n      }\n      if (messages[m].signature) {\n        validSignatures.push(messages[m].verifySignature(pubkey));\n      }\n    }\n    return {\n      text: this.text,\n      validSignatures: validSignatures\n    };\n  }\n\n  /**\n   * Verifies a message signature. This function can be called after read_message if the message was signed only.\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {boolean} true if the signature was correct; otherwise false\n   */\n  function verifySignature(pubkey) {\n    var result = false;\n    if (this.signature.tagType == 2) {\n      if (!pubkey || pubkey.length == 0) {\n        var pubkey;\n        if (this.signature.version == 4) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.issuerKeyId);\n        } else if (this.signature.version == 3) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.keyId);\n        } else {\n          util.print_error(\"unknown signature type on message!\");\n          return false;\n        }\n      }\n      if (pubkey.length == 0)\n        util.print_warning(\"Unable to verify signature of issuer: \" + util.hexstrdump(this.signature.issuerKeyId) +\n          \". Public key not found in keyring.\");\n      else {\n        for (var i = 0; i < pubkey.length; i++) {\n          var tohash = this.text.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g, \"\\r\\n\");\n          if (this.signature.verify(tohash, pubkey[i])) {\n            util.print_info(\"Found Good Signature from \" + pubkey[i].obj.userIds[0].text + \" (0x\" + util.hexstrdump(\n              pubkey[i].obj.getKeyId()).substring(8) + \")\");\n            result = true;\n          } else {\n            util.print_error(\"Signature verification failed: Bad Signature from \" + pubkey[i].obj.userIds[0].text +\n              \" (0x\" + util.hexstrdump(pubkey[0].obj.getKeyId()).substring(8) + \")\");\n          }\n        }\n      }\n    }\n    return result;\n  }\n}\n\n/**\n * reads an OpenPGP armored message and returns a message object\n * @param {String} armoredText text to be parsed\n * @return {key} new message object\n */\nmessage.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newMessage = new message(packetlist);\n  return newMessage;\n}\n\nmodule.exports = message;","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @fileoverview The openpgp base class should provide all of the functionality \n * to consume the openpgp.js library. All additional classes are documented \n * for extending and developing on top of the base library.\n */\n\nvar armor = require('./encoding/armor.js');\nvar packet = require('./packet');\nvar util = require('./util');\nvar enums = require('./enums.js');\nvar crypto = require('./crypto');\nvar config = require('./config');\n\n/**\n * GPG4Browsers Core interface. A single instance is hold\n * from the beginning. To use this library call \"openpgp.init()\"\n * @alias openpgp\n * @class\n * @classdesc Main Openpgp.js class. Use this to initiate and make all calls to this library.\n */\nfunction _openpgp() {\n  this.tostring = \"\";\n\n  /**\n   * encrypts message with keys\n   * @param  {[key]} keys    array of keys, used to encrypt the message\n   * @param  {String} message text message as native JavaScript string\n   * @return {String}         encrypted ASCII armored message \n   */\n  function encryptMessage(keys, message) {\n\n    var messagePacketlist = new packet.list();\n\n    //TODO get preferred algo from signature\n    var sessionKey = crypto.generateSessionKey(enums.read(enums.symmetric, config.encryption_cipher));\n\n    keys.forEach(function(key) {\n      var encryptionKey = key.getEncryptionKey();\n      if (encryptionKey) {\n        var pkESKeyPacket = new packet.public_key_encrypted_session_key();\n        pkESKeyPacket.publicKeyId = encryptionKey.getKeyId();\n        pkESKeyPacket.publicKeyAlgorithm = encryptionKey.algorithm;\n        pkESKeyPacket.sessionKey = sessionKey;\n        //TODO get preferred algo from signature\n        pkESKeyPacket.sessionKeyAlgorithm = enums.read(enums.symmetric, config.encryption_cipher);\n        pkESKeyPacket.encrypt(encryptionKey);\n        messagePacketlist.push(pkESKeyPacket);\n      }\n    });\n\n    var literalDataPacket = new packet.literal();\n    literalDataPacket.set(message, 'utf8');\n    var literalDataPacketlist = new packet.list();\n    literalDataPacketlist.push(literalDataPacket);\n\n    var symEncryptedPacket;\n    if (config.integrity_protect) {\n      symEncryptedPacket = new packet.sym_encrypted_integrity_protected();\n    } else {\n      symEncryptedPacket = new packet.symmetrically_encrypted();\n    }\n    symEncryptedPacket.packets = literalDataPacketlist;\n    //TODO get preferred algo from signature\n    symEncryptedPacket.encrypt(enums.read(enums.symmetric, config.encryption_cipher), sessionKey);\n    messagePacketlist.push(symEncryptedPacket);\n\n    var armored = armor.encode(enums.armor.message, messagePacketlist.write(), config);\n    return armored;\n  }\n\n  function encryptAndSignMessage(publicKeys, privateKey, message) {\n\n  }\n\n  /**\n   * decrypts message\n   * @param  {[key]} decrypted privateKey\n   * @param  {message} message the message object with the encrypted data\n   * @return {String}         decrypted message as as native JavaScript string\n   */\n  function decryptMessage(privateKey, message) {\n    return message.decrypt(privateKey);\n  }\n\n  function decryptAndVerifyMessage(privateKey, publicKeys, messagePacketlist) {\n\n  }\n\n  function verifyMessage(publicKeys, messagePacketlist) {\n\n  }\n\n  function signMessage(privateKey, messagePacketlist) {\n\n  }\n\n  /**\n   * TODO: update this doc\n   * generates a new key pair for openpgp. Beta stage. Currently only \n   * supports RSA keys, and no subkeys.\n   * @param {Integer} keyType to indicate what type of key to make. \n   * RSA is 1. Follows algorithms outlined in OpenPGP.\n   * @param {Integer} numBits number of bits for the key creation. (should \n   * be 1024+, generally)\n   * @param {String} userId assumes already in form of \"User Name \n   * <username@email.com>\"\n   * @param {String} passphrase The passphrase used to encrypt the resulting private key\n   * @return {Object} {privateKey: [openpgp_msg_privatekey], \n   * privateKeyArmored: [string], publicKeyArmored: [string]}\n   */\n  function generateKeyPair(keyType, numBits, userId, passphrase) {\n    var packetlist = new packet.list();\n\n    var secretKeyPacket = new packet.secret_key();\n    secretKeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretKeyPacket.generate(numBits);\n    secretKeyPacket.encrypt(passphrase);\n\n    var userIdPacket = new packet.userid();\n    userIdPacket.read(userId);\n\n    var dataToSign = {};\n    dataToSign.userid = userIdPacket;\n    dataToSign.key = secretKeyPacket;\n    var signaturePacket = new packet.signature();\n    signaturePacket.issuerKeyId = secretKeyPacket.getKeyId().write();\n    signaturePacket.signatureType = enums.signature.cert_generic;\n    signaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    signaturePacket.hashAlgorithm = enums.hash.sha256;\n    signaturePacket.sign(secretKeyPacket, dataToSign);\n\n    var secretSubkeyPacket = new packet.secret_subkey();\n    secretSubkeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretSubkeyPacket.generate(numBits);\n    secretSubkeyPacket.encrypt(passphrase);\n\n    dataToSign = {};\n    dataToSign.key = secretKeyPacket;\n    dataToSign.bind = secretSubkeyPacket;\n    var subkeySignaturePacket = new packet.signature();\n    subkeySignaturePacket.issuerKeyId = secretSubkeyPacket.getKeyId().write();\n    subkeySignaturePacket.signatureType = enums.signature.subkey_binding;\n    subkeySignaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    subkeySignaturePacket.hashAlgorithm = enums.hash.sha256;\n    subkeySignaturePacket.sign(secretSubkeyPacket, dataToSign);\n\n    packetlist.push(secretKeyPacket);\n    packetlist.push(userIdPacket);\n    packetlist.push(signaturePacket);\n    packetlist.push(secretSubkeyPacket);\n    packetlist.push(subkeySignaturePacket);\n\n    var armored = armor.encode(enums.armor.private_key, packetlist.write(), this.config);\n    return armored;\n  }\n\n  /**\n   * creates a binary string representation of an encrypted and signed message.\n   * The message will be encrypted with the public keys specified and signed\n   * with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]} Private key \n   * to be used to sign the message\n   * @param {Object[]} publickeys An arraf of {obj: [openpgp_msg_publickey]}\n   * - public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt and sign\n   * @return {String} a binary string representation of the message which \n   * can be OpenPGP armored\n   */\n  function write_signed_and_encrypted_message(privatekey, publickeys, messagetext) {\n    var result = \"\";\n    var i;\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    for (i = 0; i < publickeys.length; i++) {\n      var onepasssignature = new openpgp_packet_onepasssignature();\n      var onepasssigstr = \"\";\n      if (i === 0)\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      else\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      util.print_debug_hexstr_dump(\"onepasssigstr: |\" + onepasssigstr + \"|\\n\", onepasssigstr);\n      var datasignature = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(\n        /\\n/g, \"\\r\\n\"), privatekey);\n      util.print_debug_hexstr_dump(\"datasignature: |\" + datasignature.openpgp + \"|\\n\", datasignature.openpgp);\n      if (i === 0) {\n        result = onepasssigstr + literal + datasignature.openpgp;\n      } else {\n        result = onepasssigstr + result + datasignature.openpgp;\n      }\n    }\n\n    util.print_debug_hexstr_dump(\"signed packet: |\" + result + \"|\\n\", result);\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n  /**\n   * creates a binary string representation of an encrypted message.\n   * The message will be encrypted with the public keys specified \n   * @param {Object[]} publickeys An array of {obj: [openpgp_msg_publickey]}\n   * -public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt\n   * @return {String} a binary string representation of the message\n   * which can be OpenPGP armored\n   */\n  function write_encrypted_message(publickeys, messagetext) {\n    var result = \"\";\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    result = literal;\n\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (var i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n\n  /**\n   * creates a binary string representation a signed message.\n   * The message will be signed with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]}\n   * - the private key to be used to sign the message \n   * @param {String} messagetext message text to sign\n   * @return {Object} {Object: text [String]}, openpgp: {String} a binary\n   *  string representation of the message which can be OpenPGP\n   *   armored(openpgp) and a text representation of the message (text). \n   * This can be directly used to OpenPGP armor the message\n   */\n  function write_signed_message(privatekey, messagetext) {\n    var sig = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/,\n      \"\\r\\n\"), privatekey);\n    var result = {\n      text: messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/, \"\\r\\n\"),\n      openpgp: sig.openpgp,\n      hash: sig.hash\n    };\n    return armor.encode(2, result, null, null);\n  }\n\n  this.generateKeyPair = generateKeyPair;\n  this.write_signed_message = write_signed_message;\n  this.write_signed_and_encrypted_message = write_signed_and_encrypted_message;\n  this.encryptMessage = encryptMessage;\n  this.decryptMessage = decryptMessage;\n\n}\n\nmodule.exports = new _openpgp();\n","var enums = require('../enums.js');\n\n// This is pretty ugly, but browserify needs to have the requires explicitly written.\nmodule.exports = {\n  compressed: require('./compressed.js'),\n  sym_encrypted_integrity_protected: require('./sym_encrypted_integrity_protected.js'),\n  public_key_encrypted_session_key: require('./public_key_encrypted_session_key.js'),\n  sym_encrypted_session_key: require('./sym_encrypted_session_key.js'),\n  literal: require('./literal.js'),\n  public_key: require('./public_key.js'),\n  symmetrically_encrypted: require('./symmetrically_encrypted.js'),\n  marker: require('./marker.js'),\n  public_subkey: require('./public_subkey.js'),\n  user_attribute: require('./user_attribute.js'),\n  one_pass_signature: require('./one_pass_signature.js'),\n  secret_key: require('./secret_key.js'),\n  userid: require('./userid.js'),\n  secret_subkey: require('./secret_subkey.js'),\n  signature: require('./signature.js'),\n  trust: require('./trust.js')\n}\n\nfor (var i in enums.packet) {\n  var packetClass = module.exports[i];\n\n  if (packetClass != undefined)\n    packetClass.prototype.tag = enums.packet[i];\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  JXG = require('../compression/jxg.js'),\n  base64 = require('../encoding/base64.js');\n\n/**\n * @class\n * @classdesc Implementation of the Compressed Data Packet (Tag 8)\n * \n * RFC4880 5.6:\n * The Compressed Data packet contains compressed data.  Typically, this\n * packet is found as the contents of an encrypted packet, or following\n * a Signature or One-Pass Signature packet, and contains a literal data\n * packet.\n */\nmodule.exports = function packet_compressed() {\n  /** @type {packetlist} */\n  this.packets;\n  /** @type {compression} */\n  this.algorithm = 'uncompressed';\n\n  this.compressed = null;\n\n\n  /**\n   * Parsing function for the packet.\n   * @param {String} input Payload of a tag 8 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @parAM {iNTEGER} LEN lENGTH OF the packet or the remaining length of \n   * input at position\n   * @return {openpgp_packet_compressed} Object representation\n   */\n  this.read = function(bytes) {\n    // One octet that gives the algorithm used to compress the packet.\n    this.algorithm = enums.read(enums.compression, bytes.charCodeAt(0));\n\n    // Compressed data, which makes up the remainder of the packet.\n    this.compressed = bytes.substr(1);\n\n    this.decompress();\n  }\n\n\n\n  this.write = function() {\n    if (this.compressed == null)\n      this.compress();\n\n    return String.fromCharCode(enums.write(enums.compression, this.algorithm)) + this.compressed;\n  }\n\n\n  /**\n   * Decompression method for decompressing the compressed data\n   * read by read_packet\n   * @return {String} The decompressed data\n   */\n  this.decompress = function() {\n    var decompressed;\n\n    switch (this.algorithm) {\n      case 'uncompressed':\n        decompressed = this.compressed;\n        break;\n\n      case 'zip':\n        var compData = this.compressed;\n\n        var radix = base64.encode(compData).replace(/\\n/g, \"\");\n        // no header in this case, directly call deflate\n        var jxg_obj = new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(radix));\n\n        decompressed = unescape(jxg_obj.deflate()[0][0]);\n        break;\n\n      case 'zlib':\n        //RFC 1950. Bits 0-3 Compression Method\n        var compressionMethod = this.compressed.charCodeAt(0) % 0x10;\n\n        //Bits 4-7 RFC 1950 are LZ77 Window. Generally this value is 7 == 32k window size.\n        // 2nd Byte in RFC 1950 is for \"FLAGs\" Allows for a Dictionary \n        // (how is this defined). Basic checksum, and compression level.\n\n        if (compressionMethod == 8) { //CM 8 is for DEFLATE, RFC 1951\n          // remove 4 bytes ADLER32 checksum from the end\n          var compData = this.compressed.substring(0, this.compressed.length - 4);\n          var radix = base64.encode(compData).replace(/\\n/g, \"\");\n          //TODO check ADLER32 checksum\n          decompressed = JXG.decompress(radix);\n          break;\n\n        } else {\n          util.print_error(\"Compression algorithm ZLIB only supports \" +\n            \"DEFLATE compression method.\");\n        }\n        break;\n\n      case 'bzip2':\n        // TODO: need to implement this\n        throw new Error('Compression algorithm BZip2 [BZ2] is not implemented.');\n        break;\n\n      default:\n        throw new Error(\"Compression algorithm unknown :\" + this.alogrithm);\n        break;\n    }\n\n    this.packets.read(decompressed);\n  }\n\n  /**\n   * Compress the packet data (member decompressedData)\n   * @param {Integer} type Algorithm to be used // See RFC 4880 9.3\n   * @param {String} data Data to be compressed\n   * @return {String} The compressed data stored in attribute compressedData\n   */\n  this.compress = function() {\n    switch (this.algorithm) {\n\n      case 'uncompressed':\n        // - Uncompressed\n        this.compressed = this.packets.write();\n        break;\n\n      case 'zip':\n        // - ZIP [RFC1951]\n        util.print_error(\"Compression algorithm ZIP [RFC1951] is not implemented.\");\n        break;\n\n      case 'zlib':\n        // - ZLIB [RFC1950]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm ZLIB [RFC1950] is not implemented.\");\n        break;\n\n      case 'bzip2':\n        //  - BZip2 [BZ2]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm BZip2 [BZ2] is not implemented.\");\n        break;\n\n      default:\n        util.print_error(\"Compression algorithm unknown :\" + this.type);\n        break;\n    }\n  }\n};\n","var enums = require('../enums.js');\n\nmodule.exports = {\n  list: require('./packetlist.js'),\n};\n\nvar packets = require('./all_packets.js');\n\nfor (var i in packets)\n  module.exports[i] = packets[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc Implementation of the Literal Data Packet (Tag 11)\n * \n * RFC4880 5.9: A Literal Data packet contains the body of a message; data that\n * is not to be further interpreted.\n */\nmodule.exports = function packet_literal() {\n  this.format = 'utf8';\n  this.data = '';\n  this.date = new Date();\n\n\n  /**\n   * Set the packet data to a javascript native string or a squence of \n   * bytes. Conversion to a proper utf8 encoding takes place when the \n   * packet is written.\n   * @param {String} str Any native javascript string\n   * @param {openpgp_packet_literaldata.format} format \n   */\n  this.set = function(str, format) {\n    this.format = format;\n    this.data = str;\n  }\n\n  /**\n   * Set the packet data to value represented by the provided string\n   * of bytes together with the appropriate conversion format.\n   * @param {String} bytes The string of bytes\n   * @param {openpgp_packet_literaldata.format} format\n   */\n  this.setBytes = function(bytes, format) {\n    this.format = format;\n\n    if (format == 'utf8')\n      bytes = util.decode_utf8(bytes);\n\n    this.data = bytes;\n  }\n\n  /**\n   * Get the byte sequence representing the literal packet data\n   * @returns {String} A sequence of bytes\n   */\n  this.getBytes = function() {\n    if (this.format == 'utf8')\n      return util.encode_utf8(this.data);\n    else\n      return this.data;\n  }\n\n\n\n  /**\n   * Parsing function for a literal data packet (tag 11).\n   * \n   * @param {String} input Payload of a tag 11 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    // - A one-octet field that describes how the data is formatted.\n\n    var format = enums.read(enums.literal, bytes[0].charCodeAt());\n\n    var filename_len = bytes.charCodeAt(1);\n    this.filename = util.decode_utf8(bytes.substr(2, filename_len));\n\n    this.date = util.readDate(bytes.substr(2 + filename_len, 4));\n\n    var data = bytes.substring(6 + filename_len);\n\n    this.setBytes(data, format);\n  }\n\n  /**\n   * Creates a string representation of the packet\n   * \n   * @param {String} data The data to be inserted as body\n   * @return {String} string-representation of the packet\n   */\n  this.write = function() {\n    var filename = util.encode_utf8(\"msg.txt\");\n\n    var data = this.getBytes();\n\n    var result = '';\n    result += String.fromCharCode(enums.write(enums.literal, this.format));\n    result += String.fromCharCode(filename.length);\n    result += filename;\n    result += util.writeDate(this.date);\n    result += data;\n    return result;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the strange \"Marker packet\" (Tag 10)\n * \n * RFC4880 5.8: An experimental version of PGP used this packet as the Literal\n * packet, but no released version of PGP generated Literal packets with this\n * tag. With PGP 5.x, this packet has been reassigned and is reserved for use as\n * the Marker packet.\n * \n * Such a packet MUST be ignored when received.\n */\nfunction packet_marker() {\n  /**\n   * Parsing function for a literal data packet (tag 10).\n   * \n   * @param {String} input Payload of a tag 10 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    if (bytes[0].charCodeAt() == 0x50 && // P\n    bytes[1].charCodeAt() == 0x47 && // G\n    bytes[2].charCodeAt() == 0x50) // P\n      return true;\n    // marker packet does not contain \"PGP\"\n    return false;\n  }\n}\n\nmodule.exports = packet_marker;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the One-Pass Signature Packets (Tag 4)\n * \n * RFC4880 5.4:\n * The One-Pass Signature packet precedes the signed data and contains\n * enough information to allow the receiver to begin calculating any\n * hashes needed to verify the signature.  It allows the Signature\n * packet to be placed at the end of the message, so that the signer\n * can compute the entire signed message in one pass.\n */\n\nvar enums = require('../enums.js'),\n  type_keyid = require('../type/keyid.js');\n\nmodule.exports = function packet_one_pass_signature() {\n  this.version = null; // A one-octet version number.  The current version is 3.\n  this.type = null; // A one-octet signature type.  Signature types are described in RFC4880 Section 5.2.1.\n  this.hashAlgorithm = null; // A one-octet number describing the hash algorithm used. (See RFC4880 9.4)\n  this.publicKeyAlgorithm = null; // A one-octet number describing the public-key algorithm used. (See RFC4880 9.1)\n  this.signingKeyId = null; // An eight-octet number holding the Key ID of the signing key.\n  this.flags = null; //  A one-octet number holding a flag showing whether the signature is nested.  A zero value indicates that the next packet is another One-Pass Signature packet that describes another signature to be applied to the same message data.\n\n  /**\n   * parsing function for a one-pass signature packet (tag 4).\n   * @param {String} bytes payload of a tag 4 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var mypos = 0;\n    // A one-octet version number.  The current version is 3.\n    this.version = bytes.charCodeAt(mypos++);\n\n    // A one-octet signature type.  Signature types are described in\n    //   Section 5.2.1.\n    this.type = enums.read(enums.signature, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the hash algorithm used.\n    this.hashAlgorithm = enums.read(enums.hash, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the public-key algorithm used.\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes.charCodeAt(mypos++));\n\n    // An eight-octet number holding the Key ID of the signing key.\n    this.signingKeyId = new type_keyid();\n    this.signingKeyId.read(bytes.substr(mypos));\n    mypos += 8;\n\n    // A one-octet number holding a flag showing whether the signature\n    //   is nested.  A zero value indicates that the next packet is\n    //   another One-Pass Signature packet that describes another\n    //   signature to be applied to the same message data.\n    this.flags = bytes.charCodeAt(mypos++);\n    return this;\n  }\n\n  /**\n   * creates a string representation of a one-pass signature packet\n   * @param {Integer} type Signature types as described in RFC4880 Section 5.2.1.\n   * @param {Integer} hashalgorithm the hash algorithm used within the signature\n   * @param {openpgp_msg_privatekey} privatekey the private key used to generate the signature\n   * @param {Integer} length length of data to be signed\n   * @param {boolean} nested boolean showing whether the signature is nested. \n   *  \"true\" indicates that the next packet is another One-Pass Signature packet\n   *   that describes another signature to be applied to the same message data. \n   * @return {String} a string representation of a one-pass signature packet\n   */\n  this.write = function(type, hashalgorithm, privatekey, length, nested) {\n    var result = \"\";\n\n    result += String.fromCharCode(3);\n    result += String.fromCharCode(enums.write(enums.signature, type));\n    result += String.fromCharCode(enums.write(enums.hash, this.hashAlgorithm));\n    result += String.fromCharCode(enums.write(enums.publicKey, privatekey.algorithm));\n    result += privatekey.getKeyId().write();\n    if (nested)\n      result += String.fromCharCode(0);\n    else\n      result += String.fromCharCode(1);\n\n    return result;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util');\n\n\nmodule.exports = {\n  readSimpleLength: function(bytes) {\n    var len = 0,\n      offset,\n      type = bytes[0].charCodeAt();\n\n\n    if (type < 192) {\n      len = bytes[0].charCodeAt();\n      offset = 1;\n    } else if (type < 255) {\n      len = ((bytes[0].charCodeAt() - 192) << 8) + (bytes[1].charCodeAt()) + 192;\n      offset = 2;\n    } else if (type == 255) {\n      len = util.readNumber(bytes.substr(1, 4));\n      offset = 5;\n    }\n\n    return {\n      len: len,\n      offset: offset\n    };\n  },\n\n  /**\n   * Encodes a given integer of length to the openpgp length specifier to a\n   * string\n   * \n   * @param {Integer} length The length to encode\n   * @return {String} String with openpgp length representation\n   */\n  writeSimpleLength: function(length) {\n    var result = \"\";\n    if (length < 192) {\n      result += String.fromCharCode(length);\n    } else if (length > 191 && length < 8384) {\n      /*\n       * let a = (total data packet length) - 192 let bc = two octet\n       * representation of a let d = b + 192\n       */\n      result += String.fromCharCode(((length - 192) >> 8) + 192);\n      result += String.fromCharCode((length - 192) & 0xFF);\n    } else {\n      result += String.fromCharCode(255);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Writes a packet header version 4 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeHeader: function(tag_type, length) {\n    /* we're only generating v4 packet headers here */\n    var result = \"\";\n    result += String.fromCharCode(0xC0 | tag_type);\n    result += this.writeSimpleLength(length);\n    return result;\n  },\n\n  /**\n   * Writes a packet header Version 3 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeOldHeader: function(tag_type, length) {\n    var result = \"\";\n    if (length < 256) {\n      result += String.fromCharCode(0x80 | (tag_type << 2));\n      result += String.fromCharCode(length);\n    } else if (length < 65536) {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 1);\n      result += util.writeNumber(length, 2);\n    } else {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 2);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Generic static Packet Parser function\n   * \n   * @param {String} input Input stream as string\n   * @param {integer} position Position to start parsing\n   * @param {integer} len Length of the input from position on\n   * @return {Object} Returns a parsed openpgp_packet\n   */\n  read: function(input, position, len) {\n    // some sanity checks\n    if (input == null || input.length <= position || input.substring(position).length < 2 || (input[position].charCodeAt() &\n      0x80) == 0) {\n      util\n        .print_error(\"Error during parsing. This message / key is probably not containing a valid OpenPGP format.\");\n      return null;\n    }\n    var mypos = position;\n    var tag = -1;\n    var format = -1;\n    var packet_length;\n\n    format = 0; // 0 = old format; 1 = new format\n    if ((input[mypos].charCodeAt() & 0x40) != 0) {\n      format = 1;\n    }\n\n    var packet_length_type;\n    if (format) {\n      // new format header\n      tag = input[mypos].charCodeAt() & 0x3F; // bit 5-0\n    } else {\n      // old format header\n      tag = (input[mypos].charCodeAt() & 0x3F) >> 2; // bit 5-2\n      packet_length_type = input[mypos].charCodeAt() & 0x03; // bit 1-0\n    }\n\n    // header octet parsing done\n    mypos++;\n\n    // parsed length from length field\n    var bodydata = null;\n\n    // used for partial body lengths\n    var real_packet_length = -1;\n    if (!format) {\n      // 4.2.1. Old Format Packet Lengths\n      switch (packet_length_type) {\n        case 0:\n          // The packet has a one-octet length. The header is 2 octets\n          // long.\n          packet_length = input[mypos++].charCodeAt();\n          break;\n        case 1:\n          // The packet has a two-octet length. The header is 3 octets\n          // long.\n          packet_length = (input[mypos++].charCodeAt() << 8) | input[mypos++].charCodeAt();\n          break;\n        case 2:\n          // The packet has a four-octet length. The header is 5\n          // octets long.\n          packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n            8) | input[mypos++].charCodeAt();\n          break;\n        default:\n          // 3 - The packet is of indeterminate length. The header is 1\n          // octet long, and the implementation must determine how long\n          // the packet is. If the packet is in a file, this means that\n          // the packet extends until the end of the file. In general, \n          // an implementation SHOULD NOT use indeterminate-length \n          // packets except where the end of the data will be clear \n          // from the context, and even then it is better to use a \n          // definite length, or a new format header. The new format \n          // headers described below have a mechanism for precisely\n          // encoding data of indeterminate length.\n          packet_length = len;\n          break;\n      }\n\n    } else // 4.2.2. New Format Packet Lengths\n    {\n\n      // 4.2.2.1. One-Octet Lengths\n      if (input[mypos].charCodeAt() < 192) {\n        packet_length = input[mypos++].charCodeAt();\n        util.print_debug(\"1 byte length:\" + packet_length);\n        // 4.2.2.2. Two-Octet Lengths\n      } else if (input[mypos].charCodeAt() >= 192 && input[mypos].charCodeAt() < 224) {\n        packet_length = ((input[mypos++].charCodeAt() - 192) << 8) + (input[mypos++].charCodeAt()) + 192;\n        util.print_debug(\"2 byte length:\" + packet_length);\n        // 4.2.2.4. Partial Body Lengths\n      } else if (input[mypos].charCodeAt() > 223 && input[mypos].charCodeAt() < 255) {\n        packet_length = 1 << (input[mypos++].charCodeAt() & 0x1F);\n        util.print_debug(\"4 byte length:\" + packet_length);\n        // EEEK, we're reading the full data here...\n        var mypos2 = mypos + packet_length;\n        bodydata = input.substring(mypos, mypos + packet_length);\n        while (true) {\n          if (input[mypos2].charCodeAt() < 192) {\n            var tmplen = input[mypos2++].charCodeAt();\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() >= 192 && input[mypos2].charCodeAt() < 224) {\n            var tmplen = ((input[mypos2++].charCodeAt() - 192) << 8) + (input[mypos2++].charCodeAt()) + 192;\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() > 223 && input[mypos2].charCodeAt() < 255) {\n            var tmplen = 1 << (input[mypos2++].charCodeAt() & 0x1F);\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n          } else {\n            mypos2++;\n            var tmplen = (input[mypos2++].charCodeAt() << 24) | (input[mypos2++].charCodeAt() << 16) | (input[mypos2++]\n              .charCodeAt() << 8) | input[mypos2++].charCodeAt();\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            packet_length += tmplen;\n            mypos2 += tmplen;\n            break;\n          }\n        }\n        real_packet_length = mypos2;\n        // 4.2.2.3. Five-Octet Lengths\n      } else {\n        mypos++;\n        packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n          8) | input[mypos++].charCodeAt();\n      }\n    }\n\n    // if there was'nt a partial body length: use the specified\n    // packet_length\n    if (real_packet_length == -1) {\n      real_packet_length = packet_length;\n    }\n\n    if (bodydata == null) {\n      bodydata = input.substring(mypos, mypos + real_packet_length);\n    }\n\n    return {\n      tag: tag,\n      packet: bodydata,\n      offset: mypos + real_packet_length\n    };\n  }\n}\n","var packetParser = require('./packet.js'),\n  packets = require('./all_packets.js'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc This class represents a list of openpgp packets.\n * Take care when iterating over it - the packets themselves\n * are stored as numerical indices.\n */\nmodule.exports = function packetlist() {\n  /** The number of packets contained within the list.\n   * @readonly\n   * @type {Integer} */\n  this.length = 0;\n\n  /**\n   * Reads a stream of binary data and interprents it as a list of packets.\n   * @param {openpgp_bytearray} An array of bytes.\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    while (i < bytes.length) {\n      var parsed = packetParser.read(bytes, i, bytes.length - i);\n      i = parsed.offset;\n\n      var tag = enums.read(enums.packet, parsed.tag);\n      var packet = new packets[tag]();\n\n      this.push(packet);\n\n      packet.read(parsed.packet);\n    }\n  }\n\n  /**\n   * Creates a binary representation of openpgp objects contained within the\n   * class instance.\n   * @returns {openpgp_bytearray} An array of bytes containing valid openpgp packets.\n   */\n  this.write = function() {\n    var bytes = '';\n\n    for (var i = 0; i < this.length; i++) {\n      var packetbytes = this[i].write();\n      bytes += packetParser.writeHeader(this[i].tag, packetbytes.length);\n      bytes += packetbytes;\n    }\n\n    return bytes;\n  }\n\n  /**\n   * Adds a packet to the list. This is the only supported method of doing so;\n   * writing to packetlist[i] directly will result in an error.\n   */\n  this.push = function(packet) {\n    packet.packets = new packetlist();\n\n    this[this.length] = packet;\n    this.length++;\n  }\n\n    /**\n   * Creates a new packetList with all packets that pass the test implemented by the provided function.\n   */\n   this.filter = function(callback) {\n\n    var filtered = new packetlist();\n\n    for (var i = 0; i < this.length; i++) {\n      if (callback(this[i], i, this)) {\n        filtered.push(this[i]);\n      }\n    }\n\n    return filtered; \n   }\n\n   /**\n   * Creates a new packetList with all packets from the given type\n   */\n   this.filterByType = function(packetType) {\n     return this.filter(function(packet) {\n       return packet.tag == packetType;\n     });\n   } \n\n   /**\n    * Executes the provided callback once for each element\n    */\n   this.forEach = function(callback) {\n     for (var i = 0; i < this.length; i++) {\n      callback(this[i]);\n    }\n   }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  type_keyid = require('../type/keyid.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nmodule.exports = function packet_public_key() {\n  /** Key creation date.\n   * @type {Date} */\n  this.created = new Date();\n  /** A list of multiprecision integers\n   * @type {openpgp_type_mpi} */\n  this.mpi = [];\n  /** Public key algorithm\n   * @type {openpgp.publickey} */\n  this.algorithm = 'rsa_sign';\n\n\n  /**\n   * Internal Parser for public keys as specified in RFC 4880 section \n   * 5.5.2 Public-Key Packet Formats\n   * called by read_tag&lt;num&gt;\n   * @param {String} input Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of input\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.readPublicKey = this.read = function(bytes) {\n    // A one-octet version number (3 or 4).\n    var version = bytes[0].charCodeAt();\n\n    if (version == 4) {\n      // - A four-octet number denoting the time that the key was created.\n      this.created = util.readDate(bytes.substr(1, 4));\n\n      // - A one-octet number denoting the public-key algorithm of this key.\n      this.algorithm = enums.read(enums.publicKey, bytes[5].charCodeAt());\n\n      var mpicount = crypto.getPublicMpiCount(this.algorithm);\n      this.mpi = [];\n\n      var bmpi = bytes.substr(6);\n      var p = 0;\n\n      for (var i = 0; i < mpicount && p < bmpi.length; i++) {\n\n        this.mpi[i] = new type_mpi();\n\n        p += this.mpi[i].read(bmpi.substr(p))\n\n        if (p > bmpi.length)\n          util.print_error(\"openpgp.packet.keymaterial.js\\n\" + 'error reading MPI @:' + p);\n      }\n\n      return p + 6;\n    } else {\n      throw new Error('Version ' + version + ' of the key packet is unsupported.');\n    }\n  }\n\n  /*\n   * Same as write_private_key, but has less information because of \n   * public key.\n   * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n   * IE 1 corresponds to RSA.\n   * @param {RSA.keyObject} key\n   * @param timePacket\n   * @return {Object} {body: [string]OpenPGP packet body contents, \n   * header: [string] OpenPGP packet header, string: [string] header+body}\n   */\n  this.writePublicKey = this.write = function() {\n    // Version\n    var result = String.fromCharCode(4);\n    result += util.writeDate(this.created);\n    result += String.fromCharCode(enums.write(enums.publicKey, this.algorithm));\n\n    var mpicount = crypto.getPublicMpiCount(this.algorithm);\n\n    for (var i = 0; i < mpicount; i++) {\n      result += this.mpi[i].write();\n    }\n\n    return result;\n  }\n\n  // Write an old version packet - it's used by some of the internal routines.\n  this.writeOld = function() {\n    var bytes = this.writePublicKey();\n\n    return String.fromCharCode(0x99) +\n      util.writeNumber(bytes.length, 2) +\n      bytes;\n  }\n\n  /**\n   * Calculates the key id of the key \n   * @return {String} A 8 byte key id\n   */\n  this.getKeyId = function() {\n    var keyid = new type_keyid();\n    keyid.read(this.getFingerprint().substr(12, 8));\n    return keyid;\n  }\n\n  /**\n   * Calculates the fingerprint of the key\n   * @return {String} A string containing the fingerprint\n   */\n  this.getFingerprint = function() {\n    var toHash = this.writeOld();\n    return crypto.hash.sha1(toHash, toHash.length);\n  }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_keyid = require('../type/keyid.js'),\n  util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_public_key_encrypted_session_key() {\n  this.version = 3;\n\n  this.publicKeyId = new type_keyid();\n  this.publicKeyAlgorithm = 'rsa_encrypt';\n\n  this.sessionKey = null;\n  this.sessionKeyAlgorithm = 'aes256';\n\n  /** @type {openpgp_type_mpi[]} */\n  this.encrypted = [];\n\n  /**\n   * Parsing function for a publickey encrypted session key packet (tag 1).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n\n    this.version = bytes[0].charCodeAt();\n    this.publicKeyId.read(bytes.substr(1));\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes[9].charCodeAt());\n\n    var i = 10;\n\n    var integerCount = (function(algo) {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          return 1;\n\n        case 'elgamal':\n          return 2;\n\n        default:\n          throw new Error(\"Invalid algorithm.\");\n      }\n    })(this.publicKeyAlgorithm);\n\n    this.encrypted = [];\n\n    for (var j = 0; j < integerCount; j++) {\n      var mpi = new type_mpi();\n      i += mpi.read(bytes.substr(i));\n      this.encrypted.push(mpi);\n    }\n  }\n\n  /**\n   * Create a string representation of a tag 1 packet\n   * \n   * @param {String} publicKeyId\n   *             The public key id corresponding to publicMPIs key as string\n   * @param {openpgp_type_mpi[]} publicMPIs\n   *            Multiprecision integer objects describing the public key\n   * @param {Integer} pubalgo\n   *            The corresponding public key algorithm // See RFC4880 9.1\n   * @param {Integer} symmalgo\n   *            The symmetric cipher algorithm used to encrypt the data \n   *            within an encrypteddatapacket or encryptedintegrity-\n   *            protecteddatapacket \n   *            following this packet //See RFC4880 9.2\n   * @param {String} sessionkey\n   *            A string of randombytes representing the session key\n   * @return {String} The string representation\n   */\n  this.write = function() {\n\n    var result = String.fromCharCode(this.version);\n    result += this.publicKeyId.write();\n    result += String.fromCharCode(\n      enums.write(enums.publicKey, this.publicKeyAlgorithm));\n\n    for (var i = 0; i < this.encrypted.length; i++) {\n      result += this.encrypted[i].write()\n    }\n\n    return result;\n  }\n\n  this.encrypt = function(key) {\n    var data = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm));\n\n    data += this.sessionKey;\n    var checksum = util.calc_checksum(this.sessionKey);\n    data += util.writeNumber(checksum, 2);\n\n    var mpi = new type_mpi();\n    mpi.fromBytes(crypto.pkcs1.eme.encode(\n      data,\n      key.mpi[0].byteLength()));\n\n    this.encrypted = crypto.publicKeyEncrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      mpi);\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(key) {\n    var result = crypto.publicKeyDecrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      this.encrypted).toBytes();\n\n    var checksum = util.readNumber(result.substr(result.length - 2));\n\n    var decoded = crypto.pkcs1.eme.decode(\n      result,\n      key.mpi[0].byteLength());\n\n    var key = decoded.substring(1, decoded.length - 2);\n\n    if (checksum != util.calc_checksum(key)) {\n      throw new Error('Checksum mismatch');\n    } else {\n      this.sessionKey = key;\n      this.sessionKeyAlgorithm =\n        enums.read(enums.symmetric, decoded.charCodeAt(0));\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar public_key = require('./public_key.js');\n\nmodule.exports = function public_subkey() {\n  public_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar publicKey = require('./public_key.js'),\n  enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js'),\n  type_s2k = require('../type/s2k.js');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nfunction packet_secret_key() {\n  publicKey.call(this);\n\n  this.encrypted = null;\n\n\n  function get_hash_len(hash) {\n    if (hash == 'sha1')\n      return 20;\n    else\n      return 2;\n  }\n\n  function get_hash_fn(hash) {\n    if (hash == 'sha1')\n      return crypto.hash.sha1;\n    else\n      return function(c) {\n        return util.writeNumber(util.calc_checksum(c), 2);\n    };\n  }\n\n  // Helper function\n\n  function parse_cleartext_mpi(hash_algorithm, cleartext, algorithm) {\n    var hashlen = get_hash_len(hash_algorithm),\n      hashfn = get_hash_fn(hash_algorithm);\n\n    var hashtext = cleartext.substr(cleartext.length - hashlen);\n    cleartext = cleartext.substr(0, cleartext.length - hashlen);\n\n    var hash = hashfn(cleartext);\n\n    if (hash != hashtext)\n      throw new Error(\"Hash mismatch.\");\n\n    var mpis = crypto.getPrivateMpiCount(algorithm);\n\n    var j = 0;\n    var mpi = [];\n\n    for (var i = 0; i < mpis && j < cleartext.length; i++) {\n      mpi[i] = new type_mpi();\n      j += mpi[i].read(cleartext.substr(j));\n    }\n\n    return mpi;\n  }\n\n  function write_cleartext_mpi(hash_algorithm, algorithm, mpi) {\n    var bytes = '';\n    var discard = crypto.getPublicMpiCount(algorithm);\n\n    for (var i = discard; i < mpi.length; i++) {\n      bytes += mpi[i].write();\n    }\n\n\n    bytes += get_hash_fn(hash_algorithm)(bytes);\n\n    return bytes;\n  }\n\n\n  // 5.5.3.  Secret-Key Packet Formats\n\n  /**\n   * Internal parser for private keys as specified in RFC 4880 section 5.5.3\n   * @param {String} bytes Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of bytes\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.read = function(bytes) {\n    // - A Public-Key or Public-Subkey packet, as described above.\n    var len = this.readPublicKey(bytes);\n\n    bytes = bytes.substr(len);\n\n\n    // - One octet indicating string-to-key usage conventions.  Zero\n    //   indicates that the secret-key data is not encrypted.  255 or 254\n    //   indicates that a string-to-key specifier is being given.  Any\n    //   other value is a symmetric-key encryption algorithm identifier.\n    var isEncrypted = bytes[0].charCodeAt();\n\n    if (isEncrypted) {\n      this.encrypted = bytes;\n    } else {\n\n      // - Plain or encrypted multiprecision integers comprising the secret\n      //   key data.  These algorithm-specific fields are as described\n      //   below.\n\n      this.mpi = this.mpi.concat(parse_cleartext_mpi('mod', bytes.substr(1),\n        this.algorithm));\n    }\n\n  };\n\n  /*\n     * Creates an OpenPGP key packet for the given key. much \n\t * TODO in regards to s2k, subkeys.\n     * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n\t * IE 1 corresponds to RSA.\n     * @param {RSA.keyObject} key\n     * @param passphrase\n     * @param s2kHash\n     * @param symmetricEncryptionAlgorithm\n     * @param timePacket\n     * @return {Object} {body: [string]OpenPGP packet body contents, \n\t\theader: [string] OpenPGP packet header, string: [string] header+body}\n     */\n  this.write = function() {\n    var bytes = this.writePublicKey();\n\n    if (!this.encrypted) {\n      bytes += String.fromCharCode(0);\n\n      bytes += write_cleartext_mpi('mod', this.algorithm, this.mpi);\n    } else {\n      bytes += this.encrypted;\n    }\n\n    return bytes;\n  };\n\n\n\n\n  /** Encrypt the payload. By default, we use aes256 and iterated, salted string\n   * to key specifier\n   * @param {String} passphrase\n   */\n  this.encrypt = function(passphrase) {\n\n    var s2k = new type_s2k(),\n      symmetric = 'aes256',\n      cleartext = write_cleartext_mpi('sha1', this.algorithm, this.mpi),\n      key = produceEncryptionKey(s2k, passphrase, symmetric),\n      blockLen = crypto.cipher[symmetric].blockSize,\n      iv = crypto.random.getRandomBytes(blockLen);\n\n\n    this.encrypted = '';\n    this.encrypted += String.fromCharCode(254);\n    this.encrypted += String.fromCharCode(enums.write(enums.symmetric, symmetric));\n    this.encrypted += s2k.write();\n    this.encrypted += iv;\n\n    this.encrypted += crypto.cfb.normalEncrypt(symmetric, key, cleartext, iv);\n  };\n\n  function produceEncryptionKey(s2k, passphrase, algorithm) {\n    return s2k.produce_key(passphrase,\n      crypto.cipher[algorithm].keySize);\n  }\n\n  /**\n   * Decrypts the private key MPIs which are needed to use the key.\n   * openpgp_packet_keymaterial.hasUnencryptedSecretKeyData should be \n   * false otherwise\n   * a call to this function is not needed\n   * \n   * @param {String} str_passphrase The passphrase for this private key \n   * as string\n   * @return {Boolean} True if the passphrase was correct; false if not\n   */\n  this.decrypt = function(passphrase) {\n    if (!this.encrypted)\n      return;\n\n    var i = 0,\n      symmetric,\n      key;\n\n    var s2k_usage = this.encrypted[i++].charCodeAt();\n\n    // - [Optional] If string-to-key usage octet was 255 or 254, a one-\n    //   octet symmetric encryption algorithm.\n    if (s2k_usage == 255 || s2k_usage == 254) {\n      symmetric = this.encrypted[i++].charCodeAt();\n      symmetric = enums.read(enums.symmetric, symmetric);\n\n      // - [Optional] If string-to-key usage octet was 255 or 254, a\n      //   string-to-key specifier.  The length of the string-to-key\n      //   specifier is implied by its type, as described above.\n      var s2k = new type_s2k();\n      i += s2k.read(this.encrypted.substr(i));\n\n      key = produceEncryptionKey(s2k, passphrase, symmetric);\n    } else {\n      symmetric = s2k_usage;\n      symmetric = enums.read(enums.symmetric, symmetric);\n      key = crypto.hash.md5(passphrase);\n    }\n\n\n    // - [Optional] If secret data is encrypted (string-to-key usage octet\n    //   not zero), an Initial Vector (IV) of the same length as the\n    //   cipher's block size.\n    var iv = this.encrypted.substr(i,\n      crypto.cipher[symmetric].blockSize);\n\n    i += iv.length;\n\n    var cleartext,\n      ciphertext = this.encrypted.substr(i);\n\n    cleartext = crypto.cfb.normalDecrypt(symmetric, key, ciphertext, iv);\n\n    var hash = s2k_usage == 254 ?\n      'sha1' :\n      'mod';\n\n\n    this.mpi = this.mpi.concat(parse_cleartext_mpi(hash, cleartext,\n      this.algorithm));\n  };\n\n  this.generate = function(bits, passphrase) {\n    this.mpi = crypto.generateMpi(this.algorithm, bits);\n  };\n\n}\n\npacket_secret_key.prototype = new publicKey;\n\nmodule.exports = packet_secret_key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar secret_key = require('./secret_key.js');\n\nmodule.exports = function secret_subkey() {\n  secret_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  packet = require('./packet.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js');\n\n/**\n * @class\n * @classdesc Implementation of the Signature Packet (Tag 2)\n * \n * RFC4480 5.2:\n * A Signature packet describes a binding between some public key and\n * some data.  The most common signatures are a signature of a file or a\n * block of text, and a signature that is a certification of a User ID.\n */\nmodule.exports = function packet_signature() {\n\n  this.signatureType = null;\n  this.hashAlgorithm = null;\n  this.publicKeyAlgorithm = null;\n\n  this.signatureData = null;\n  this.signedHashValue = null;\n  this.mpi = null;\n\n  this.created = null;\n  this.signatureExpirationTime = null;\n  this.signatureNeverExpires = null;\n  this.exportable = null;\n  this.trustLevel = null;\n  this.trustAmount = null;\n  this.regularExpression = null;\n  this.revocable = null;\n  this.keyExpirationTime = null;\n  this.keyNeverExpires = null;\n  this.preferredSymmetricAlgorithms = null;\n  this.revocationKeyClass = null;\n  this.revocationKeyAlgorithm = null;\n  this.revocationKeyFingerprint = null;\n  this.issuerKeyId = null;\n  this.notation = {};\n  this.preferredHashAlgorithms = null;\n  this.preferredCompressionAlgorithms = null;\n  this.keyServerPreferences = null;\n  this.preferredKeyServer = null;\n  this.isPrimaryUserID = null;\n  this.policyURI = null;\n  this.keyFlags = null;\n  this.signersUserId = null;\n  this.reasonForRevocationFlag = null;\n  this.reasonForRevocationString = null;\n  this.signatureTargetPublicKeyAlgorithm = null;\n  this.signatureTargetHashAlgorithm = null;\n  this.signatureTargetHash = null;\n  this.embeddedSignature = null;\n\n  this.verified = false;\n\n  /**\n   * parsing function for a signature packet (tag 2).\n   * @param {String} bytes payload of a tag 2 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    var version = bytes[i++].charCodeAt();\n    // switch on version (3 and 4)\n    switch (version) {\n      case 3:\n        // One-octet length of following hashed material. MUST be 5.\n        if (bytes[i++].charCodeAt() != 5)\n          util.print_debug(\"openpgp.packet.signature.js\\n\" +\n            'invalid One-octet length of following hashed material.' +\n            'MUST be 5. @:' + (i - 1));\n\n        var sigpos = i;\n        // One-octet signature type.\n        this.signatureType = bytes[i++].charCodeAt();\n\n        // Four-octet creation time.\n        this.created = util.readDate(bytes.substr(i, 4));\n        i += 4;\n\n        // storing data appended to data which gets verified\n        this.signatureData = bytes.substring(position, i);\n\n        // Eight-octet Key ID of signer.\n        this.issuerKeyId = bytes.substring(i, i + 8);\n        i += 8;\n\n        // One-octet public-key algorithm.\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n\n        // One-octet hash algorithm.\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n        break;\n      case 4:\n        this.signatureType = bytes[i++].charCodeAt();\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n\n\n        function subpackets(bytes, signed) {\n          // Two-octet scalar octet count for following hashed subpacket\n          // data.\n          var subpacket_length = util.readNumber(\n            bytes.substr(0, 2));\n\n          var i = 2;\n\n          // Hashed subpacket data set (zero or more subpackets)\n          var subpacked_read = 0;\n          while (i < 2 + subpacket_length) {\n\n            var len = packet.readSimpleLength(bytes.substr(i));\n            i += len.offset;\n\n            // Since it is trivial to add data to the unhashed portion of \n            // the packet we simply ignore all unauthenticated data.\n            if (signed)\n              this.read_sub_packet(bytes.substr(i, len.len));\n\n            i += len.len;\n          }\n\n          return i;\n        }\n\n        i += subpackets.call(this, bytes.substr(i), true);\n\n        // A V4 signature hashes the packet body\n        // starting from its first field, the version number, through the end\n        // of the hashed subpacket data.  Thus, the fields hashed are the\n        // signature version, the signature type, the public-key algorithm, the\n        // hash algorithm, the hashed subpacket length, and the hashed\n        // subpacket body.\n        this.signatureData = bytes.substr(0, i);\n\n        i += subpackets.call(this, bytes.substr(i), false);\n\n        break;\n      default:\n        throw new Error('Version ' + version + ' of the signature is unsupported.');\n        break;\n    }\n\n    // Two-octet field holding left 16 bits of signed hash value.\n    this.signedHashValue = bytes.substr(i, 2);\n    i += 2;\n\n    this.signature = bytes.substr(i);\n  };\n\n  this.write = function() {\n    return this.signatureData +\n      util.writeNumber(0, 2) + // Number of unsigned subpackets.\n    this.signedHashValue +\n      this.signature;\n  };\n\n  /**\n   * Signs provided data. This needs to be done prior to serialization.\n   * @param {Object} data Contains packets to be signed.\n   * @param {openpgp_msg_privatekey} privatekey private key used to sign the message. \n   */\n  this.sign = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var result = String.fromCharCode(4);\n    result += String.fromCharCode(signatureType);\n    result += String.fromCharCode(publicKeyAlgorithm);\n    result += String.fromCharCode(hashAlgorithm);\n\n    //Calculate subpackets\n    var creationTimeSubpacket = write_sub_packet(enums.signatureSubpacket.signature_creation_time,\n      util.writeDate(new Date()));\n\n    var issuerSubpacket = write_sub_packet(enums.signatureSubpacket.issuer, key.getKeyId().write());\n\n    // Add subpackets here\n    result += util.writeNumber(creationTimeSubpacket.length + issuerSubpacket.length, 2);\n    result += creationTimeSubpacket + issuerSubpacket;\n\n    this.signatureData = result;\n\n    var trailer = this.calculateTrailer();\n\n    var toHash = this.toSign(signatureType, data) +\n      this.signatureData + trailer;\n\n    var hash = crypto.hash.digest(hashAlgorithm, toHash);\n\n    this.signedHashValue = hash.substr(0, 2);\n\n    this.signature = crypto.signature.sign(hashAlgorithm,\n      publicKeyAlgorithm, key.mpi, toHash);\n  };\n\n  /**\n   * creates a string representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   * @param {Integer} type subpacket signature type. Signature types as described \n   * in RFC4880 Section 5.2.3.2\n   * @param {String} data data to be included\n   * @return {String} a string-representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   */\n  function write_sub_packet(type, data) {\n    var result = \"\";\n    result += packet.writeSimpleLength(data.length + 1);\n    result += String.fromCharCode(type);\n    result += data;\n    return result;\n  }\n\n  // V4 signature sub packets\n\n  this.read_sub_packet = function(bytes) {\n    var mypos = 0;\n\n    function read_array(prop, bytes) {\n      this[prop] = [];\n\n      for (var i = 0; i < bytes.length; i++) {\n        this[prop].push(bytes[i].charCodeAt());\n      }\n    }\n\n    // The leftwost bit denotes a \"critical\" packet, but we ignore it.\n    var type = bytes[mypos++].charCodeAt() & 0x7F;\n\n    // subpacket type\n    switch (type) {\n      case 2:\n        // Signature Creation Time\n        this.created = util.readDate(bytes.substr(mypos));\n        break;\n      case 3:\n        // Signature Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.signatureNeverExpires = time.getTime() == 0;\n        this.signatureExpirationTime = time;\n\n        break;\n      case 4:\n        // Exportable Certification\n        this.exportable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 5:\n        // Trust Signature\n        this.trustLevel = bytes[mypos++].charCodeAt();\n        this.trustAmount = bytes[mypos++].charCodeAt();\n        break;\n      case 6:\n        // Regular Expression\n        this.regularExpression = bytes.substr(mypos);\n        break;\n      case 7:\n        // Revocable\n        this.revocable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 9:\n        // Key Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.keyExpirationTime = time;\n        this.keyNeverExpires = time.getTime() == 0;\n\n        break;\n      case 11:\n        // Preferred Symmetric Algorithms\n        this.preferredSymmetricAlgorithms = [];\n\n        while (mypos != bytes.length) {\n          this.preferredSymmetricAlgorithms.push(bytes[mypos++].charCodeAt());\n        }\n\n        break;\n      case 12:\n        // Revocation Key\n        // (1 octet of class, 1 octet of public-key algorithm ID, 20\n        // octets of\n        // fingerprint)\n        this.revocationKeyClass = bytes[mypos++].charCodeAt();\n        this.revocationKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.revocationKeyFingerprint = bytes.substr(mypos, 20);\n        break;\n\n      case 16:\n        // Issuer\n        this.issuerKeyId = bytes.substr(mypos, 8);\n        break;\n\n      case 20:\n        // Notation Data\n        // We don't know how to handle anything but a text flagged data.\n        if (bytes[mypos].charCodeAt() == 0x80) {\n\n          // We extract key/value tuple from the byte stream.\n          mypos += 4;\n          var m = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n          var n = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n\n          var name = bytes.substr(mypos, m),\n            value = bytes.substr(mypos + m, n);\n\n          this.notation[name] = value;\n        } else throw new Error(\"Unsupported notation flag.\");\n        break;\n      case 21:\n        // Preferred Hash Algorithms\n        read_array.call(this, 'preferredHashAlgorithms', bytes.substr(mypos));\n        break;\n      case 22:\n        // Preferred Compression Algorithms\n        read_array.call(this, 'preferredCompressionAlgorithms ', bytes.substr(mypos));\n        break;\n      case 23:\n        // Key Server Preferences\n        read_array.call(this, 'keyServerPreferencess', bytes.substr(mypos));\n        break;\n      case 24:\n        // Preferred Key Server\n        this.preferredKeyServer = bytes.substr(mypos);\n        break;\n      case 25:\n        // Primary User ID\n        this.isPrimaryUserID = bytes[mypos++] != 0;\n        break;\n      case 26:\n        // Policy URI\n        this.policyURI = bytes.substr(mypos);\n        break;\n      case 27:\n        // Key Flags\n        read_array.call(this, 'keyFlags', bytes.substr(mypos));\n        break;\n      case 28:\n        // Signer's User ID\n        this.signersUserId += bytes.substr(mypos);\n        break;\n      case 29:\n        // Reason for Revocation\n        this.reasonForRevocationFlag = bytes[mypos++].charCodeAt();\n        this.reasonForRevocationString = bytes.substr(mypos);\n        break;\n      case 30:\n        // Features\n        read_array.call(this, 'features', bytes.substr(mypos));\n        break;\n      case 31:\n        // Signature Target\n        // (1 octet public-key algorithm, 1 octet hash algorithm, N octets hash)\n        this.signatureTargetPublicKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.signatureTargetHashAlgorithm = bytes[mypos++].charCodeAt();\n\n        var len = crypto.getHashByteLength(this.signatureTargetHashAlgorithm);\n\n        this.signatureTargetHash = bytes.substr(mypos, len);\n        break;\n      case 32:\n        // Embedded Signature\n        this.embeddedSignature = new packet_signature();\n        this.embeddedSignature.read(bytes.substr(mypos));\n        break;\n      default:\n        util.print_error(\"openpgp.packet.signature.js\\n\" +\n          'unknown signature subpacket type ' + type + \" @:\" + mypos +\n          \" subplen:\" + subplen + \" len:\" + len);\n        break;\n    }\n  };\n\n  // Produces data to produce signature on\n  this.toSign = function(type, data) {\n    var t = enums.signature;\n\n    switch (type) {\n      case t.binary:\n        return data.getBytes();\n\n      case t.text:\n        return this.toSign(t.binary, data)\n          .replace(/\\r\\n/g, '\\n')\n          .replace(/\\n/g, '\\r\\n');\n\n      case t.standalone:\n        return '';\n\n      case t.cert_generic:\n      case t.cert_persona:\n      case t.cert_casual:\n      case t.cert_positive:\n      case t.cert_revocation:\n        {\n          var packet, tag;\n\n          if (data.userid !== undefined) {\n            tag = 0xB4;\n            packet = data.userid;\n          } else if (data.userattribute !== undefined) {\n            tag = 0xD1;\n            packet = data.userattribute;\n          } else throw new Error('Either a userid or userattribute packet needs to be ' +\n              'supplied for certification.');\n\n          var bytes = packet.write();\n\n          return this.toSign(t.key, data) +\n            String.fromCharCode(tag) +\n            util.writeNumber(bytes.length, 4) +\n            bytes;\n        }\n      case t.subkey_binding:\n      case t.key_binding:\n        {\n          return this.toSign(t.key, data) + this.toSign(t.key, {\n            key: data.bind\n          });\n        }\n      case t.key:\n        {\n          if (data.key == undefined)\n            throw new Error('Key packet is required for this sigtature.');\n\n          return data.key.writeOld();\n        }\n      case t.key_revocation:\n      case t.subkey_revocation:\n        return this.toSign(t.key, data);\n      case t.timestamp:\n        return '';\n      case t.thrid_party:\n        throw new Error('Not implemented');\n        break;\n      default:\n        throw new Error('Unknown signature type.')\n    }\n  }\n\n\n  this.calculateTrailer = function() {\n    // calculating the trailer\n    var trailer = '';\n    trailer += String.fromCharCode(4); // Version\n    trailer += String.fromCharCode(0xFF);\n    trailer += util.writeNumber(this.signatureData.length, 4);\n    return trailer\n  }\n\n\n  /**\n   * verifys the signature packet. Note: not signature types are implemented\n   * @param {String} data data which on the signature applies\n   * @param {openpgp_msg_privatekey} key the public key to verify the signature\n   * @return {boolean} True if message is verified, else false.\n   */\n  this.verify = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var bytes = this.toSign(signatureType, data),\n      trailer = this.calculateTrailer();\n\n\n    var mpicount = 0;\n    // Algorithm-Specific Fields for RSA signatures:\n    // \t    - multiprecision number (MPI) of RSA signature value m**d mod n.\n    if (publicKeyAlgorithm > 0 && publicKeyAlgorithm < 4)\n      mpicount = 1;\n    //    Algorithm-Specific Fields for DSA signatures:\n    //      - MPI of DSA value r.\n    //      - MPI of DSA value s.\n    else if (publicKeyAlgorithm == 17)\n      mpicount = 2;\n\n    var mpi = [],\n      i = 0;\n    for (var j = 0; j < mpicount; j++) {\n      mpi[j] = new type_mpi();\n      i += mpi[j].read(this.signature.substr(i));\n    }\n\n    this.verified = crypto.signature.verify(publicKeyAlgorithm,\n      hashAlgorithm, mpi, key.mpi,\n      bytes + this.signatureData + trailer);\n\n    return this.verified;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Sym. Encrypted Integrity Protected Data \n * Packet (Tag 18)\n * \n * RFC4880 5.13: The Symmetrically Encrypted Integrity Protected Data packet is\n * a variant of the Symmetrically Encrypted Data packet. It is a new feature\n * created for OpenPGP that addresses the problem of detecting a modification to\n * encrypted data. It is used in combination with a Modification Detection Code\n * packet.\n */\n\nmodule.exports = function packet_sym_encrypted_integrity_protected() {\n  /** The encrypted payload. */\n  this.encrypted = null; // string\n  /** @type {Boolean}\n   * If after decrypting the packet this is set to true,\n   * a modification has been detected and thus the contents\n   * should be discarded.\n   */\n  this.modification = false;\n  this.packets;\n\n\n  this.read = function(bytes) {\n    // - A one-octet version number. The only currently defined value is\n    // 1.\n    var version = bytes[0].charCodeAt();\n\n    if (version != 1) {\n      throw new Error('Invalid packet version.');\n    }\n\n    // - Encrypted data, the output of the selected symmetric-key cipher\n    //   operating in Cipher Feedback mode with shift amount equal to the\n    //   block size of the cipher (CFB-n where n is the block size).\n    this.encrypted = bytes.substr(1);\n  }\n\n  this.write = function() {\n\n    return String.fromCharCode(1) // Version\n    + this.encrypted;\n  }\n\n  this.encrypt = function(sessionKeyAlgorithm, key) {\n    var bytes = this.packets.write()\n\n    var prefixrandom = crypto.getPrefixRandom(sessionKeyAlgorithm);\n    var prefix = prefixrandom + prefixrandom.charAt(prefixrandom.length - 2) + prefixrandom.charAt(prefixrandom.length -\n      1)\n\n    var tohash = bytes;\n\n\n    // Modification detection code packet.\n    tohash += String.fromCharCode(0xD3);\n    tohash += String.fromCharCode(0x14);\n\n\n    tohash += crypto.hash.sha1(prefix + tohash);\n\n\n    this.encrypted = crypto.cfb.encrypt(prefixrandom,\n      sessionKeyAlgorithm, tohash, key, false).substring(0,\n      prefix.length + tohash.length);\n  }\n\n  /**\n   * Decrypts the encrypted data contained in this object read_packet must\n   * have been called before\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *            The selected symmetric encryption algorithm to be used\n   * @param {String} key The key of cipher blocksize length to be used\n   * @return {String} The decrypted data of this packet\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, false);\n\n\n    // there must be a modification detection code packet as the\n    // last packet and everything gets hashed except the hash itself\n    this.hash = crypto.hash.sha1(\n      crypto.cfb.mdc(sessionKeyAlgorithm, key, this.encrypted) + decrypted.substring(0, decrypted.length - 20));\n\n\n    var mdc = decrypted.substr(decrypted.length - 20, 20);\n\n    if (this.hash != mdc) {\n      throw new Error('Modification detected.');\n    } else\n      this.packets.read(decrypted.substr(0, decrypted.length - 22));\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_s2k = require('../type/s2k.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_sym_encrypted_session_key() {\n  this.tag = 3;\n  this.sessionKeyEncryptionAlgorithm = null;\n  this.sessionKeyAlgorithm = 'aes256';\n  this.encrypted = null;\n  this.s2k = new type_s2k();\n\n  /**\n   * Parsing function for a symmetric encrypted session key packet (tag 3).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    // A one-octet version number. The only currently defined version is 4.\n    this.version = bytes[0].charCodeAt();\n\n    // A one-octet number describing the symmetric algorithm used.\n    var algo = enums.read(enums.symmetric, bytes[1].charCodeAt());\n\n    // A string-to-key (S2K) specifier, length as defined above.\n    var s2klength = this.s2k.read(bytes.substr(2));\n\n    // Optionally, the encrypted session key itself, which is decrypted\n    // with the string-to-key object.\n    var done = s2klength + 2;\n\n    if (done < bytes.length) {\n      this.encrypted = bytes.substr(done);\n      this.sessionKeyEncryptionAlgorithm = algo\n    } else\n      this.sessionKeyAlgorithm = algo;\n  }\n\n  this.write = function() {\n    var algo = this.encrypted == null ?\n      this.sessionKeyAlgorithm :\n      this.sessionKeyEncryptionAlgorithm;\n\n    var bytes = String.fromCharCode(this.version) +\n      String.fromCharCode(enums.write(enums.symmetric, algo)) +\n      this.s2k.write();\n\n    if (this.encrypted != null)\n      bytes += this.encrypted;\n    return bytes;\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(passphrase) {\n    var algo = this.sessionKeyEncryptionAlgorithm != null ?\n      this.sessionKeyEncryptionAlgorithm :\n      this.sessionKeyAlgorithm;\n\n\n    var length = crypto.cipher[algo].keySize;\n    var key = this.s2k.produce_key(passphrase, length);\n\n    if (this.encrypted == null) {\n      this.sessionKey = key;\n\n    } else {\n      var decrypted = crypto.cfb.decrypt(\n        this.sessionKeyEncryptionAlgorithm, key, this.encrypted, true);\n\n      this.sessionKeyAlgorithm = enums.read(enums.symmetric,\n        decrypted[0].keyCodeAt());\n\n      this.sessionKey = decrypted.substr(1);\n    }\n  }\n\n  this.encrypt = function(passphrase) {\n    var length = crypto.getKeyLength(this.sessionKeyEncryptionAlgorithm);\n    var key = this.s2k.produce_key(passphrase, length);\n\n    var private_key = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm)) +\n\n    crypto.getRandomBytes(\n      crypto.getKeyLength(this.sessionKeyAlgorithm));\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(this.sessionKeyEncryptionAlgorithm),\n      this.sessionKeyEncryptionAlgorithm, key, private_key, true);\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Symmetrically Encrypted Data Packet (Tag 9)\n * \n * RFC4880 5.7: The Symmetrically Encrypted Data packet contains data encrypted\n * with a symmetric-key algorithm. When it has been decrypted, it contains other\n * packets (usually a literal data packet or compressed data packet, but in\n * theory other Symmetrically Encrypted Data packets or sequences of packets\n * that form whole OpenPGP messages).\n */\n\nmodule.exports = function packet_symmetrically_encrypted() {\n  this.encrypted = null;\n  /** Decrypted packets contained within. \n   * @type {openpgp_packetlist} */\n  this.packets;\n\n  this.read = function(bytes) {\n    this.encrypted = bytes;\n  }\n\n  this.write = function() {\n    return this.encrypted;\n  }\n\n  /**\n   * Symmetrically decrypt the packet data\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *             Symmetric key algorithm to use // See RFC4880 9.2\n   * @param {String} key\n   *             Key as string with the corresponding length to the\n   *            algorithm\n   * @return The decrypted data;\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, true);\n\n    this.packets.read(decrypted);\n  }\n\n  this.encrypt = function(algo, key) {\n    var data = this.packets.write();\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(algo), algo, data, key, true);\n  }\n};\n","module.exports = function packet_trust() {\n\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/** \n * @class\n * @classdesc Implementation of the User Attribute Packet (Tag 17)\n *  The User Attribute packet is a variation of the User ID packet.  It\n *  is capable of storing more types of data than the User ID packet,\n *  which is limited to text.  Like the User ID packet, a User Attribute\n *  packet may be certified by the key owner (\"self-signed\") or any other\n *  key owner who cares to certify it.  Except as noted, a User Attribute\n *  packet may be used anywhere that a User ID packet may be used.\n *\n *  While User Attribute packets are not a required part of the OpenPGP\n *  standard, implementations SHOULD provide at least enough\n *  compatibility to properly handle a certification signature on the\n *  User Attribute packet.  A simple way to do this is by treating the\n *  User Attribute packet as a User ID packet with opaque contents, but\n *  an implementation may use any method desired.\n */\nmodule.exports = function packet_user_attribute() {\n  this.tag = 17;\n  this.attributes = [];\n\n  /**\n   * parsing function for a user attribute packet (tag 17).\n   * @param {String} input payload of a tag 17 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    while (i < bytes.length) {\n      var len = openpgp_packet.read_simple_length(bytes);\n\n      i += len.offset;\n      this.attributes.push(bytes.substr(i, len.len));\n      i += len.len;\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of the User ID Packet (Tag 13)\n * A User ID packet consists of UTF-8 text that is intended to represent\n * the name and email address of the key holder.  By convention, it\n * includes an RFC 2822 [RFC2822] mail name-addr, but there are no\n * restrictions on its content.  The packet length in the header\n * specifies the length of the User ID. \n */\nmodule.exports = function packet_userid() {\n  /** @type {String} A string containing the user id. Usually in the form\n   * John Doe <john@example.com> \n   */\n  this.userid = '';\n\n\n  /**\n   * Parsing function for a user id packet (tag 13).\n   * @param {String} input payload of a tag 13 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input \n   * at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    this.userid = util.decode_utf8(bytes);\n  }\n\n  /**\n   * Creates a string representation of the user id packet\n   * @param {String} user_id the user id as string (\"John Doe <john.doe@mail.us\")\n   * @return {String} string representation\n   */\n  this.write = function() {\n    return util.encode_utf8(this.userid);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of type key id (RFC4880 3.3)\n *  A Key ID is an eight-octet scalar that identifies a key.\n   Implementations SHOULD NOT assume that Key IDs are unique.  The\n   section \"Enhanced Key Formats\" below describes how Key IDs are\n   formed.\n */\nmodule.exports = function keyid() {\n\n  this.bytes = '';\n\n  for (var i = 0; i < 8; i++) {\n    this.bytes += String.fromCharCode(0);\n  }\n  /**\n   * Parsing method for a key id\n   * @param {String} input Input to read the key id from \n   * @param {integer} position Position where to start reading the key \n   * id from input\n   * @return {openpgp_type_keyid} This object\n   */\n  this.read = function(bytes) {\n    this.bytes = bytes.substr(0, 8);\n  }\n\n  this.write = function() {\n    return this.bytes;\n  }\n\n  this.toHex = function() {\n    return util.hexstrdump(this.bytes);\n  }\n\n  this.equals = function(keyid) {\n    return this.bytes == keyid.bytes;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n// Hint: We hold our MPIs as an array of octets in big endian format preceeding a two\n// octet scalar: MPI: [a,b,c,d,e,f]\n// - MPI size: (a << 8) | b \n// - MPI = c | d << 8 | e << ((MPI.length -2)*8) | f ((MPI.length -2)*8)\n\nvar BigInteger = require('../crypto/public_key/jsbn.js'),\n  util = require('../util');\n\n/**\n * @class\n * @classdescImplementation of type MPI (RFC4880 3.2)\n * Multiprecision integers (also called MPIs) are unsigned integers used\n * to hold large integers such as the ones used in cryptographic\n * calculations.\n * An MPI consists of two pieces: a two-octet scalar that is the length\n * of the MPI in bits followed by a string of octets that contain the\n * actual integer.\n */\nmodule.exports = function mpi() {\n  /** An implementation dependent integer */\n  this.data = null;\n\n  /**\n   * Parsing function for a mpi (RFC 4880 3.2).\n   * @param {String} input Payload of mpi data\n   * @param {Integer} position Position to start reading from the input \n   * string\n   * @param {Integer} len Length of the packet or the remaining length of \n   * input at position\n   * @return {openpgp_type_mpi} Object representation\n   */\n  this.read = function(bytes) {\n    var bits = (bytes[0].charCodeAt() << 8) | bytes[1].charCodeAt();\n\n    // Additional rules:\n    //\n    //    The size of an MPI is ((MPI.length + 7) / 8) + 2 octets.\n    //\n    //    The length field of an MPI describes the length starting from its\n    //\t  most significant non-zero bit.  Thus, the MPI [00 02 01] is not\n    //    formed correctly.  It should be [00 01 01].\n\n    // TODO: Verification of this size method! This size calculation as\n    // \t\t specified above is not applicable in JavaScript\n    var bytelen = Math.ceil(bits / 8);\n\n    var raw = bytes.substr(2, bytelen);\n    this.fromBytes(raw);\n\n    return 2 + bytelen;\n  }\n\n  this.fromBytes = function(bytes) {\n    this.data = new BigInteger(util.hexstrdump(bytes), 16);\n  }\n\n  this.toBytes = function() {\n    return this.write().substr(2);\n  }\n\n  this.byteLength = function() {\n    return this.toBytes().length;\n  }\n\n  /**\n   * Converts the mpi object to a string as specified in RFC4880 3.2\n   * @return {String} mpi Byte representation\n   */\n  this.write = function() {\n    return this.data.toMPI();\n  }\n\n  this.toBigInteger = function() {\n    return this.data.clone();\n  }\n\n  this.fromBigInteger = function(bn) {\n    this.data = bn.clone();\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the String-to-key specifier (RFC4880 3.7)\n * String-to-key (S2K) specifiers are used to convert passphrase strings\n   into symmetric-key encryption/decryption keys.  They are used in two\n   places, currently: to encrypt the secret part of private keys in the\n   private keyring, and to convert passphrases to encryption keys for\n   symmetrically encrypted messages.\n */\nmodule.exports = function s2k() {\n  /** @type {openpgp.hash} */\n  this.algorithm = 'sha256';\n  /** @type {openpgp_type_s2k.type} */\n  this.type = 'iterated';\n  this.c = 96;\n  /** @type {openpgp_bytearray} \n   * Eight bytes of salt. */\n  this.salt = crypto.random.getRandomBytes(8);\n\n\n  // Exponen bias, defined in RFC4880\n  var expbias = 6;\n\n  this.get_count = function() {\n    return (16 + (this.c & 15)) << ((this.c >> 4) + expbias);\n  }\n\n  /**\n   * Parsing function for a string-to-key specifier (RFC 4880 3.7).\n   * @param {String} input Payload of string-to-key specifier\n   * @return {Integer} Actual length of the object\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    this.type = enums.read(enums.s2k, bytes[i++].charCodeAt());\n    this.algorithm = enums.read(enums.hash, bytes[i++].charCodeAt());\n\n    switch (this.type) {\n      case 'simple':\n        break;\n\n      case 'salted':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n        break;\n\n      case 'iterated':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n\n        // Octet 10: count, a one-octet, coded value\n        this.c = bytes[i++].charCodeAt();\n        break;\n\n      case 'gnu':\n        if (bytes.substr(i, 3) == \"GNU\") {\n          i += 3; // GNU\n          var gnuExtType = 1000 + bytes[i++].charCodeAt();\n          if (gnuExtType == 1001) {\n            this.type = gnuExtType;\n            // GnuPG extension mode 1001 -- don't write secret key at all\n          } else {\n            throw new Error(\"Unknown s2k gnu protection mode.\");\n          }\n        } else {\n          throw new Error(\"Unknown s2k type.\");\n        }\n        break;\n\n      default:\n        throw new Error(\"Unknown s2k type.\");\n        break;\n    }\n\n    return i;\n  }\n\n\n  /**\n   * writes an s2k hash based on the inputs.\n   * @return {String} Produced key of hashAlgorithm hash length\n   */\n  this.write = function() {\n    var bytes = String.fromCharCode(enums.write(enums.s2k, this.type));\n    bytes += String.fromCharCode(enums.write(enums.hash, this.algorithm));\n\n    switch (this.type) {\n      case 'simple':\n        break;\n      case 'salted':\n        bytes += this.salt;\n        break;\n      case 'iterated':\n        bytes += this.salt;\n        bytes += String.fromCharCode(this.c);\n        break;\n    };\n\n    return bytes;\n  }\n\n  /**\n   * Produces a key using the specified passphrase and the defined \n   * hashAlgorithm \n   * @param {String} passphrase Passphrase containing user input\n   * @return {String} Produced key with a length corresponding to \n   * hashAlgorithm hash length\n   */\n  this.produce_key = function(passphrase, numBytes) {\n    passphrase = util.encode_utf8(passphrase);\n\n    function round(prefix, s2k) {\n      var algorithm = enums.write(enums.hash, s2k.algorithm);\n\n      switch (s2k.type) {\n        case 'simple':\n          return crypto.hash.digest(algorithm, prefix + passphrase);\n\n        case 'salted':\n          return crypto.hash.digest(algorithm,\n            prefix + s2k.salt + passphrase);\n\n        case 'iterated':\n          var isp = [],\n            count = s2k.get_count();\n          data = s2k.salt + passphrase;\n\n          while (isp.length * data.length < count)\n            isp.push(data);\n\n          isp = isp.join('');\n\n          if (isp.length > count)\n            isp = isp.substr(0, count);\n\n          return crypto.hash.digest(algorithm, prefix + isp);\n      };\n    }\n\n    var result = '',\n      prefix = '';\n\n    while (result.length <= numBytes) {\n      result += round(prefix, this);\n      prefix += String.fromCharCode(0);\n    }\n\n    return result.substr(0, numBytes);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar Util = function() {\n\n  this.readNumber = function(bytes) {\n    var n = 0;\n\n    for (var i = 0; i < bytes.length; i++) {\n      n <<= 8;\n      n += bytes[i].charCodeAt();\n    }\n\n    return n;\n  };\n\n  this.writeNumber = function(n, bytes) {\n    var b = '';\n    for (var i = 0; i < bytes; i++) {\n      b += String.fromCharCode((n >> (8 * (bytes - i - 1))) & 0xFF);\n    }\n\n    return b;\n  };\n\n\n\n  this.readDate = function(bytes) {\n    var n = this.readNumber(bytes);\n    var d = new Date();\n    d.setTime(n * 1000);\n    return d;\n  };\n\n  this.writeDate = function(time) {\n    var numeric = Math.round(time.getTime() / 1000);\n\n    return this.writeNumber(numeric, 4);\n  };\n\n  this.emailRegEx =\n    /[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?/;\n\n  this.debug = false;\n\n  this.hexdump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    var i = 0;\n    while (c < e) {\n      h = str.charCodeAt(c++).toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\" \" + h);\n      i++;\n      if (i % 32 == 0)\n        r.push(\"\\n           \");\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create hexstring from a binary\n   * @param {String} str String to convert\n   * @return {String} String containing the hexadecimal values\n   */\n  this.hexstrdump = function(str) {\n    if (str == null)\n      return \"\";\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].charCodeAt().toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create binary string from a hex encoded string\n   * @param {String} str Hex string to convert\n   * @return {String} String containing the binary values\n   */\n  this.hex2bin = function(hex) {\n    var str = '';\n    for (var i = 0; i < hex.length; i += 2)\n      str += String.fromCharCode(parseInt(hex.substr(i, 2), 16));\n    return str;\n  };\n\n  /**\n   * Creating a hex string from an binary array of integers (0..255)\n   * @param {String} str Array of bytes to convert\n   * @return {String} Hexadecimal representation of the array\n   */\n  this.hexidump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n\n  /**\n   * Convert a native javascript string to a string of utf8 bytes\n   * @param {String} str The string to convert\n   * @return {String} A valid squence of utf8 bytes\n   */\n  this.encode_utf8 = function(str) {\n    return unescape(encodeURIComponent(str));\n  };\n\n  /**\n   * Convert a string of utf8 bytes to a native javascript string\n   * @param {String} utf8 A valid squence of utf8 bytes\n   * @return {String} A native javascript string\n   */\n  this.decode_utf8 = function(utf8) {\n    return decodeURIComponent(escape(utf8));\n  };\n\n  var str2bin = function(str, result) {\n    for (var i = 0; i < str.length; i++) {\n      result[i] = str.charCodeAt(i);\n    }\n\n    return result;\n  };\n\n  var bin2str = function(bin) {\n    var result = [];\n\n    for (var i = 0; i < bin.length; i++) {\n      result.push(String.fromCharCode(bin[i]));\n    }\n\n    return result.join('');\n  };\n\n  /**\n   * Convert a string to an array of integers(0.255)\n   * @param {String} str String to convert\n   * @return {Integer[]} An array of (binary) integers\n   */\n  this.str2bin = function(str) {\n    return str2bin(str, new Array(str.length));\n  };\n\n\n  /**\n   * Convert an array of integers(0.255) to a string \n   * @param {Integer[]} bin An array of (binary) integers to convert\n   * @return {String} The string representation of the array\n   */\n  this.bin2str = bin2str;\n\n  /**\n   * Convert a string to a Uint8Array\n   * @param {String} str String to convert\n   * @return {Uint8Array} The array of (binary) integers\n   */\n  this.str2Uint8Array = function(str) {\n    return str2bin(str, new Uint8Array(new ArrayBuffer(str.length)));\n  };\n\n  /**\n   * Convert a Uint8Array to a string. This currently functions \n   * the same as bin2str. \n   * @param {Uint8Array} bin An array of (binary) integers to convert\n   * @return {String} String representation of the array\n   */\n  this.Uint8Array2str = bin2str;\n\n  /**\n   * Calculates a 16bit sum of a string by adding each character \n   * codes modulus 65535\n   * @param {String} text String to create a sum of\n   * @return {Integer} An integer containing the sum of all character \n   * codes % 65535\n   */\n  this.calc_checksum = function(text) {\n    var checksum = {\n      s: 0,\n      add: function(sadd) {\n        this.s = (this.s + sadd) % 65536;\n      }\n    };\n    for (var i = 0; i < text.length; i++) {\n      checksum.add(text.charCodeAt(i));\n    }\n    return checksum.s;\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug = function(str) {\n    if (this.debug) {\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * Different than print_debug because will call hexstrdump iff necessary.\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug_hexstr_dump = function(str, strToHex) {\n    if (this.debug) {\n      str = str + this.hexstrdump(strToHex);\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print an error message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the error message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded error message\n   */\n  this.print_error = function(str) {\n    if (this.debug)\n      throw str;\n    console.log(str);\n  };\n\n  /**\n   * Helper function to print an info message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'.\n   * @param {String} str String of the info message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded info message\n   */\n  this.print_info = function(str) {\n    if (this.debug)\n      console.log(str);\n  };\n\n  this.print_warning = function(str) {\n    console.log(str);\n  };\n\n  this.getLeftNBits = function(string, bitcount) {\n    var rest = bitcount % 8;\n    if (rest == 0)\n      return string.substring(0, bitcount / 8);\n    var bytes = (bitcount - rest) / 8 + 1;\n    var result = string.substring(0, bytes);\n    return this.shiftRight(result, 8 - rest); // +String.fromCharCode(string.charCodeAt(bytes -1) << (8-rest) & 0xFF);\n  };\n\n  /**\n   * Shifting a string to n bits right\n   * @param {String} value The string to shift\n   * @param {Integer} bitcount Amount of bits to shift (MUST be smaller \n   * than 9)\n   * @return {String} Resulting string. \n   */\n  this.shiftRight = function(value, bitcount) {\n    var temp = util.str2bin(value);\n    if (bitcount % 8 != 0) {\n      for (var i = temp.length - 1; i >= 0; i--) {\n        temp[i] >>= bitcount % 8;\n        if (i > 0)\n          temp[i] |= (temp[i - 1] << (8 - (bitcount % 8))) & 0xFF;\n      }\n    } else {\n      return value;\n    }\n    return util.bin2str(temp);\n  };\n\n  /**\n   * Return the algorithm type as string\n   * @return {String} String representing the message type\n   */\n  this.get_hashAlgorithmString = function(algo) {\n    switch (algo) {\n      case 1:\n        return \"MD5\";\n      case 2:\n        return \"SHA1\";\n      case 3:\n        return \"RIPEMD160\";\n      case 8:\n        return \"SHA256\";\n      case 9:\n        return \"SHA384\";\n      case 10:\n        return \"SHA512\";\n      case 11:\n        return \"SHA224\";\n    }\n    return \"unknown\";\n  };\n\n};\n\n/**\n * an instance that should be used. \n */\nmodule.exports = new Util();\n"]} +//@ sourceMappingURL=data:application/json;base64,{"version":3,"file":"generated.js","sources":["/home/toberndo/dev/openpgpjs-devel/src/compression/jxg.js","/home/toberndo/dev/openpgpjs-devel/src/config/config.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cfb.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/aes.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/blowfish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/cast5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/des.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/twofish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/crypto.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/md5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/ripe-md.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/sha.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/pkcs1.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/dsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/elgamal.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/jsbn.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/rsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/random.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/signature.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/armor.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/base64.js","/home/toberndo/dev/openpgpjs-devel/src/enums.js","/home/toberndo/dev/openpgpjs-devel/src/index.js","/home/toberndo/dev/openpgpjs-devel/src/key.js","/home/toberndo/dev/openpgpjs-devel/src/keyring.js","/home/toberndo/dev/openpgpjs-devel/src/message.js","/home/toberndo/dev/openpgpjs-devel/src/openpgp.js","/home/toberndo/dev/openpgpjs-devel/src/packet/all_packets.js","/home/toberndo/dev/openpgpjs-devel/src/packet/compressed.js","/home/toberndo/dev/openpgpjs-devel/src/packet/index.js","/home/toberndo/dev/openpgpjs-devel/src/packet/literal.js","/home/toberndo/dev/openpgpjs-devel/src/packet/marker.js","/home/toberndo/dev/openpgpjs-devel/src/packet/one_pass_signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packet.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packetlist.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_integrity_protected.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/symmetrically_encrypted.js","/home/toberndo/dev/openpgpjs-devel/src/packet/trust.js","/home/toberndo/dev/openpgpjs-devel/src/packet/user_attribute.js","/home/toberndo/dev/openpgpjs-devel/src/packet/userid.js","/home/toberndo/dev/openpgpjs-devel/src/type/keyid.js","/home/toberndo/dev/openpgpjs-devel/src/type/mpi.js","/home/toberndo/dev/openpgpjs-devel/src/type/s2k.js","/home/toberndo/dev/openpgpjs-devel/src/util/util.js"],"names":[],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3fA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzqDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5QA;;;;;;;;;;;;;;;;;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpEA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourcesContent":["JXG = {\n  exists: (function(undefined) {\n    return function(v) {\n      return !(v === undefined || v === null);\n    }\n  })()\n};\nJXG.decompress = function(str) {\n  return unescape((new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(str))).unzip()[0][0]);\n};\n/*\n    Copyright 2008-2012\n        Matthias Ehmann,\n        Michael Gerhaeuser,\n        Carsten Miller,\n        Bianca Valentin,\n        Alfred Wassermann,\n        Peter Wilfahrt\n\n    This file is part of JSXGraph.\n    \n    Dual licensed under the Apache License Version 2.0, or LGPL Version 3 licenses.\n\n    You should have received a copy of the GNU Lesser General Public License\n    along with JSXCompressor.  If not, see <http://www.gnu.org/licenses/>.\n    \n    You should have received a copy of the Apache License along with JSXCompressor.  \n    If not, see <http://www.apache.org/licenses/>.\n\n*/\n\n/**\n * @class Util class\n * @classdesc Utilities for uncompressing and base64 decoding\n * Class for gunzipping, unzipping and base64 decoding of files.\n * It is used for reading GEONExT, Geogebra and Intergeo files.\n *\n * Only Huffman codes are decoded in gunzip.\n * The code is based on the source code for gunzip.c by Pasi Ojala \n * {@link http://www.cs.tut.fi/~albert/Dev/gunzip/gunzip.c}\n * {@link http://www.cs.tut.fi/~albert}\n */\nJXG.Util = {};\n\n/**\n * Unzip zip files\n */\nJXG.Util.Unzip = function(barray) {\n  var outputArr = [],\n    output = \"\",\n    debug = false,\n    gpflags,\n    files = 0,\n    unzipped = [],\n    crc,\n    buf32k = new Array(32768),\n    bIdx = 0,\n    modeZIP = false,\n\n    CRC, SIZE,\n\n    bitReverse = [\n        0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0,\n        0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0,\n        0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,\n        0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8,\n        0x04, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4,\n        0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,\n        0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec,\n        0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc,\n        0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,\n        0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2,\n        0x0a, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea,\n        0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,\n        0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6,\n        0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6,\n        0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,\n        0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe,\n        0x01, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1,\n        0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,\n        0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9,\n        0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9,\n        0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,\n        0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5,\n        0x0d, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed,\n        0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,\n        0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3,\n        0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3,\n        0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,\n        0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb,\n        0x07, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7,\n        0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,\n        0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef,\n        0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff\n    ],\n\n    cplens = [\n        3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31,\n        35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0\n    ],\n\n    cplext = [\n        0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2,\n        3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 99, 99\n    ],\n    /* 99==invalid */\n\n    cpdist = [\n        0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0007, 0x0009, 0x000d,\n        0x0011, 0x0019, 0x0021, 0x0031, 0x0041, 0x0061, 0x0081, 0x00c1,\n        0x0101, 0x0181, 0x0201, 0x0301, 0x0401, 0x0601, 0x0801, 0x0c01,\n        0x1001, 0x1801, 0x2001, 0x3001, 0x4001, 0x6001\n    ],\n\n    cpdext = [\n        0, 0, 0, 0, 1, 1, 2, 2,\n        3, 3, 4, 4, 5, 5, 6, 6,\n        7, 7, 8, 8, 9, 9, 10, 10,\n        11, 11, 12, 12, 13, 13\n    ],\n\n    border = [16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15],\n\n    bA = barray,\n\n    bytepos = 0,\n    bitpos = 0,\n    bb = 1,\n    bits = 0,\n\n    NAMEMAX = 256,\n\n    nameBuf = [],\n\n    fileout;\n\n  function readByte() {\n    bits += 8;\n    if (bytepos < bA.length) {\n      //if (debug)\n      //    document.write(bytepos+\": \"+bA[bytepos]+\"<br>\");\n      return bA[bytepos++];\n    } else\n      return -1;\n  };\n\n  function byteAlign() {\n    bb = 1;\n  };\n\n  function readBit() {\n    var carry;\n    bits++;\n    carry = (bb & 1);\n    bb >>= 1;\n    if (bb == 0) {\n      bb = readByte();\n      carry = (bb & 1);\n      bb = (bb >> 1) | 0x80;\n    }\n    return carry;\n  };\n\n  function readBits(a) {\n    var res = 0,\n      i = a;\n\n    while (i--) {\n      res = (res << 1) | readBit();\n    }\n    if (a) {\n      res = bitReverse[res] >> (8 - a);\n    }\n    return res;\n  };\n\n  function flushBuffer() {\n    //document.write('FLUSHBUFFER:'+buf32k);\n    bIdx = 0;\n  };\n\n  function addBuffer(a) {\n    SIZE++;\n    //CRC=updcrc(a,crc);\n    buf32k[bIdx++] = a;\n    outputArr.push(String.fromCharCode(a));\n    //output+=String.fromCharCode(a);\n    if (bIdx == 0x8000) {\n      //document.write('ADDBUFFER:'+buf32k);\n      bIdx = 0;\n    }\n  };\n\n  function HufNode() {\n    this.b0 = 0;\n    this.b1 = 0;\n    this.jump = null;\n    this.jumppos = -1;\n  };\n\n  var LITERALS = 288;\n\n  var literalTree = new Array(LITERALS);\n  var distanceTree = new Array(32);\n  var treepos = 0;\n  var Places = null;\n  var Places2 = null;\n\n  var impDistanceTree = new Array(64);\n  var impLengthTree = new Array(64);\n\n  var len = 0;\n  var fpos = new Array(17);\n  fpos[0] = 0;\n  var flens;\n  var fmax;\n\n  function IsPat() {\n    while (1) {\n      if (fpos[len] >= fmax)\n        return -1;\n      if (flens[fpos[len]] == len)\n        return fpos[len]++;\n      fpos[len]++;\n    }\n  };\n\n  function Rec() {\n    var curplace = Places[treepos];\n    var tmp;\n    if (debug)\n      document.write(\"<br>len:\" + len + \" treepos:\" + treepos);\n    if (len == 17) { //war 17\n      return -1;\n    }\n    treepos++;\n    len++;\n\n    tmp = IsPat();\n    if (debug)\n      document.write(\"<br>IsPat \" + tmp);\n    if (tmp >= 0) {\n      curplace.b0 = tmp; /* leaf cell for 0-bit */\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n    } else {\n      /* Not a Leaf cell */\n      curplace.b0 = 0x8000;\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n      if (Rec())\n        return -1;\n    }\n    tmp = IsPat();\n    if (tmp >= 0) {\n      curplace.b1 = tmp; /* leaf cell for 1-bit */\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = null; /* Just for the display routine */\n    } else {\n      /* Not a Leaf cell */\n      curplace.b1 = 0x8000;\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = Places[treepos];\n      curplace.jumppos = treepos;\n      if (Rec())\n        return -1;\n    }\n    len--;\n    return 0;\n  };\n\n  function CreateTree(currentTree, numval, lengths, show) {\n    var i;\n    /* Create the Huffman decode tree/table */\n    //document.write(\"<br>createtree<br>\");\n    if (debug)\n      document.write(\"currentTree \" + currentTree + \" numval \" + numval + \" lengths \" + lengths + \" show \" + show);\n    Places = currentTree;\n    treepos = 0;\n    flens = lengths;\n    fmax = numval;\n    for (i = 0; i < 17; i++)\n      fpos[i] = 0;\n    len = 0;\n    if (Rec()) {\n      //fprintf(stderr, \"invalid huffman tree\\n\");\n      if (debug)\n        alert(\"invalid huffman tree\\n\");\n      return -1;\n    }\n    if (debug) {\n      document.write('<br>Tree: ' + Places.length);\n      for (var a = 0; a < 32; a++) {\n        document.write(\"Places[\" + a + \"].b0=\" + Places[a].b0 + \"<br>\");\n        document.write(\"Places[\" + a + \"].b1=\" + Places[a].b1 + \"<br>\");\n      }\n    }\n\n    /*if(show) {\n            var tmp;\n            for(tmp=currentTree;tmp<Places;tmp++) {\n                fprintf(stdout, \"0x%03x  0x%03x (0x%04x)\",tmp-currentTree, tmp->jump?tmp->jump-currentTree:0,(tmp->jump?tmp->jump-currentTree:0)*6+0xcf0);\n                if(!(tmp.b0 & 0x8000)) {\n                    //fprintf(stdout, \"  0x%03x (%c)\", tmp->b0,(tmp->b0<256 && isprint(tmp->b0))?tmp->b0:'�');\n                }\n                if(!(tmp.b1 & 0x8000)) {\n                    if((tmp.b0 & 0x8000))\n                        fprintf(stdout, \"           \");\n                    fprintf(stdout, \"  0x%03x (%c)\", tmp->b1,(tmp->b1<256 && isprint(tmp->b1))?tmp->b1:'�');\n                }\n                fprintf(stdout, \"\\n\");\n            }\n        }*/\n    return 0;\n  };\n\n  function DecodeValue(currentTree) {\n    var len, i,\n      xtreepos = 0,\n      X = currentTree[xtreepos],\n      b;\n\n    /* decode one symbol of the data */\n    while (1) {\n      b = readBit();\n      if (debug)\n        document.write(\"b=\" + b);\n      if (b) {\n        if (!(X.b1 & 0x8000)) {\n          if (debug)\n            document.write(\"ret1\");\n          return X.b1; /* If leaf node, return data */\n        }\n        X = X.jump;\n        len = currentTree.length;\n        for (i = 0; i < len; i++) {\n          if (currentTree[i] === X) {\n            xtreepos = i;\n            break;\n          }\n        }\n        //xtreepos++;\n      } else {\n        if (!(X.b0 & 0x8000)) {\n          if (debug)\n            document.write(\"ret2\");\n          return X.b0; /* If leaf node, return data */\n        }\n        //X++; //??????????????????\n        xtreepos++;\n        X = currentTree[xtreepos];\n      }\n    }\n    if (debug)\n      document.write(\"ret3\");\n    return -1;\n  };\n\n  function DeflateLoop() {\n    var last, c, type, i, len;\n\n    do {\n      /*if((last = readBit())){\n            fprintf(errfp, \"Last Block: \");\n        } else {\n            fprintf(errfp, \"Not Last Block: \");\n        }*/\n      last = readBit();\n      type = readBits(2);\n      switch (type) {\n        case 0:\n          if (debug)\n            alert(\"Stored\\n\");\n          break;\n        case 1:\n          if (debug)\n            alert(\"Fixed Huffman codes\\n\");\n          break;\n        case 2:\n          if (debug)\n            alert(\"Dynamic Huffman codes\\n\");\n          break;\n        case 3:\n          if (debug)\n            alert(\"Reserved block type!!\\n\");\n          break;\n        default:\n          if (debug)\n            alert(\"Unexpected value %d!\\n\", type);\n          break;\n      }\n\n      if (type == 0) {\n        var blockLen, cSum;\n\n        // Stored \n        byteAlign();\n        blockLen = readByte();\n        blockLen |= (readByte() << 8);\n\n        cSum = readByte();\n        cSum |= (readByte() << 8);\n\n        if (((blockLen ^ ~cSum) & 0xffff)) {\n          document.write(\"BlockLen checksum mismatch\\n\");\n        }\n        while (blockLen--) {\n          c = readByte();\n          addBuffer(c);\n        }\n      } else if (type == 1) {\n        var j;\n\n        /* Fixed Huffman tables -- fixed decode routine */\n        while (1) {\n          /*\n                256    0000000        0\n                :   :     :\n                279    0010111        23\n                0   00110000    48\n                :    :      :\n                143    10111111    191\n                280 11000000    192\n                :    :      :\n                287 11000111    199\n                144    110010000    400\n                :    :       :\n                255    111111111    511\n    \n                Note the bit order!\n                */\n\n          j = (bitReverse[readBits(7)] >> 1);\n          if (j > 23) {\n            j = (j << 1) | readBit(); /* 48..255 */\n\n            if (j > 199) { /* 200..255 */\n              j -= 128; /*  72..127 */\n              j = (j << 1) | readBit(); /* 144..255 << */\n            } else { /*  48..199 */\n              j -= 48; /*   0..151 */\n              if (j > 143) {\n                j = j + 136; /* 280..287 << */\n                /*   0..143 << */\n              }\n            }\n          } else { /*   0..23 */\n            j += 256; /* 256..279 << */\n          }\n          if (j < 256) {\n            addBuffer(j);\n            //document.write(\"out:\"+String.fromCharCode(j));\n            /*fprintf(errfp, \"@%d %02x\\n\", SIZE, j);*/\n          } else if (j == 256) {\n            /* EOF */\n            break;\n          } else {\n            var len, dist;\n\n            j -= 256 + 1; /* bytes + EOF */\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = bitReverse[readBits(5)] >> 3;\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n\n            /*fprintf(errfp, \"@%d (l%02x,d%04x)\\n\", SIZE, len, dist);*/\n            for (j = 0; j < len; j++) {\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          }\n        } // while\n      } else if (type == 2) {\n        var j, n, literalCodes, distCodes, lenCodes;\n        var ll = new Array(288 + 32); // \"static\" just to preserve stack\n\n        // Dynamic Huffman tables \n\n        literalCodes = 257 + readBits(5);\n        distCodes = 1 + readBits(5);\n        lenCodes = 4 + readBits(4);\n        //document.write(\"<br>param: \"+literalCodes+\" \"+distCodes+\" \"+lenCodes+\"<br>\");\n        for (j = 0; j < 19; j++) {\n          ll[j] = 0;\n        }\n\n        // Get the decode tree code lengths\n\n        //document.write(\"<br>\");\n        for (j = 0; j < lenCodes; j++) {\n          ll[border[j]] = readBits(3);\n          //document.write(ll[border[j]]+\" \");\n        }\n        //fprintf(errfp, \"\\n\");\n        //document.write('<br>ll:'+ll);\n        len = distanceTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        if (CreateTree(distanceTree, 19, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug) {\n          document.write(\"<br>distanceTree\");\n          for (var a = 0; a < distanceTree.length; a++) {\n            document.write(\"<br>\" + distanceTree[a].b0 + \" \" + distanceTree[a].b1 + \" \" + distanceTree[a].jump + \" \" +\n              distanceTree[a].jumppos);\n            /*if (distanceTree[a].jumppos!=-1)\n                    \tdocument.write(\" \"+distanceTree[a].jump.b0+\" \"+distanceTree[a].jump.b1);\n                \t*/\n          }\n        }\n        //document.write('<BR>tree created');\n\n        //read in literal and distance code lengths\n        n = literalCodes + distCodes;\n        i = 0;\n        var z = -1;\n        if (debug)\n          document.write(\"<br>n=\" + n + \" bits: \" + bits + \"<br>\");\n        while (i < n) {\n          z++;\n          j = DecodeValue(distanceTree);\n          if (debug)\n            document.write(\"<br>\" + z + \" i:\" + i + \" decode: \" + j + \"    bits \" + bits + \"<br>\");\n          if (j < 16) { // length of code in bits (0..15)\n            ll[i++] = j;\n          } else if (j == 16) { // repeat last length 3 to 6 times \n            var l;\n            j = 3 + readBits(2);\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            l = i ? ll[i - 1] : 0;\n            while (j--) {\n              ll[i++] = l;\n            }\n          } else {\n            if (j == 17) { // 3 to 10 zero length codes\n              j = 3 + readBits(3);\n            } else { // j == 18: 11 to 138 zero length codes \n              j = 11 + readBits(7);\n            }\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            while (j--) {\n              ll[i++] = 0;\n            }\n          }\n        }\n        /*for(j=0; j<literalCodes+distCodes; j++) {\n                //fprintf(errfp, \"%d \", ll[j]);\n                if ((j&7)==7)\n                    fprintf(errfp, \"\\n\");\n            }\n            fprintf(errfp, \"\\n\");*/\n        // Can overwrite tree decode tree as it is not used anymore\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          literalTree[i] = new HufNode();\n        if (CreateTree(literalTree, literalCodes, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        var ll2 = new Array();\n        for (i = literalCodes; i < ll.length; i++) {\n          ll2[i - literalCodes] = ll[i];\n        }\n        if (CreateTree(distanceTree, distCodes, ll2, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug)\n          document.write(\"<br>literalTree\");\n        outer: while (1) {\n          j = DecodeValue(literalTree);\n          if (j >= 256) { // In C64: if carry set\n            var len, dist;\n            j -= 256;\n            if (j == 0) {\n              // EOF\n              break;\n            }\n            j--;\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = DecodeValue(distanceTree);\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n            while (len--) {\n              if (bIdx - dist < 0) {\n                break outer;\n              }\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          } else {\n            addBuffer(j);\n          }\n        }\n      }\n    } while (!last);\n    flushBuffer();\n\n    byteAlign();\n    return 0;\n  };\n\n  JXG.Util.Unzip.prototype.unzipFile = function(name) {\n    var i;\n    this.unzip();\n    //alert(unzipped[0][1]);\n    for (i = 0; i < unzipped.length; i++) {\n      if (unzipped[i][1] == name) {\n        return unzipped[i][0];\n      }\n    }\n\n  };\n\n  JXG.Util.Unzip.prototype.deflate = function() {\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    DeflateLoop();\n    if (debug)\n      alert(outputArr.join(''));\n    unzipped[files] = new Array(2);\n    unzipped[files][0] = outputArr.join('');\n    unzipped[files][1] = \"DEFLATE\";\n    files++;\n    return unzipped;\n  }\n\n  JXG.Util.Unzip.prototype.unzip = function() {\n    //convertToByteArray(input);\n    if (debug)\n      alert(bA);\n    /*for (i=0;i<bA.length*8;i++){\n\t\tdocument.write(readBit());\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}*/\n    /*for (i=0;i<bA.length;i++){\n\t\tdocument.write(readByte()+\" \");\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}\n\tfor (i=0;i<bA.length;i++){\n\t\tdocument.write(bA[i]+\" \");\n\t\tif ((i+1)%16==0)\n\t\t\tdocument.write(\"<br>\");\n\t}\t\n\t*/\n    //alert(bA);\n    nextFile();\n    return unzipped;\n  };\n\n  function nextFile() {\n    if (debug)\n      alert(\"NEXTFILE\");\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    tmp[0] = readByte();\n    tmp[1] = readByte();\n    if (debug)\n      alert(\"type: \" + tmp[0] + \" \" + tmp[1]);\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"da\", 16)) { //GZIP\n      if (debug)\n        alert(\"GEONExT-GZIP\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"geonext.gxt\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"9c\", 16)) { //ZLIB\n      if (debug)\n        alert(\"ZLIB\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"ZLIB\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"1f\", 16) && tmp[1] == parseInt(\"8b\", 16)) { //GZIP\n      if (debug)\n        alert(\"GZIP\");\n      //DeflateLoop();\n      skipdir();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"file\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"50\", 16) && tmp[1] == parseInt(\"4b\", 16)) { //ZIP\n      modeZIP = true;\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n      if (tmp[2] == parseInt(\"3\", 16) && tmp[3] == parseInt(\"4\", 16)) {\n        //MODE_ZIP\n        tmp[0] = readByte();\n        tmp[1] = readByte();\n        if (debug)\n          alert(\"ZIP-Version: \" + tmp[1] + \" \" + tmp[0] / 10 + \".\" + tmp[0] % 10);\n\n        gpflags = readByte();\n        gpflags |= (readByte() << 8);\n        if (debug)\n          alert(\"gpflags: \" + gpflags);\n\n        var method = readByte();\n        method |= (readByte() << 8);\n        if (debug)\n          alert(\"method: \" + method);\n\n        readByte();\n        readByte();\n        readByte();\n        readByte();\n\n        var crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n\n        var compSize = readByte();\n        compSize |= (readByte() << 8);\n        compSize |= (readByte() << 16);\n        compSize |= (readByte() << 24);\n\n        var size = readByte();\n        size |= (readByte() << 8);\n        size |= (readByte() << 16);\n        size |= (readByte() << 24);\n\n        if (debug)\n          alert(\"local CRC: \" + crc + \"\\nlocal Size: \" + size + \"\\nlocal CompSize: \" + compSize);\n\n        var filelen = readByte();\n        filelen |= (readByte() << 8);\n\n        var extralen = readByte();\n        extralen |= (readByte() << 8);\n\n        if (debug)\n          alert(\"filelen \" + filelen);\n        i = 0;\n        nameBuf = [];\n        while (filelen--) {\n          var c = readByte();\n          if (c == \"/\" | c == \":\") {\n            i = 0;\n          } else if (i < NAMEMAX - 1)\n            nameBuf[i++] = String.fromCharCode(c);\n        }\n        if (debug)\n          alert(\"nameBuf: \" + nameBuf);\n\n        //nameBuf[i] = \"\\0\";\n        if (!fileout)\n          fileout = nameBuf;\n\n        var i = 0;\n        while (i < extralen) {\n          c = readByte();\n          i++;\n        }\n\n        CRC = 0xffffffff;\n        SIZE = 0;\n\n        if (size = 0 && fileOut.charAt(fileout.length - 1) == \"/\") {\n          //skipdir\n          if (debug)\n            alert(\"skipdir\");\n        }\n        if (method == 8) {\n          DeflateLoop();\n          if (debug)\n            alert(outputArr.join(''));\n          unzipped[files] = new Array(2);\n          unzipped[files][0] = outputArr.join('');\n          unzipped[files][1] = nameBuf.join('');\n          files++;\n          //return outputArr.join('');\n        }\n        skipdir();\n      }\n    }\n  };\n\n  function skipdir() {\n    var crc,\n      tmp = [],\n      compSize, size, os, i, c;\n\n    if ((gpflags & 8)) {\n      tmp[0] = readByte();\n      tmp[1] = readByte();\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n\n      if (tmp[0] == parseInt(\"50\", 16) &&\n        tmp[1] == parseInt(\"4b\", 16) &&\n        tmp[2] == parseInt(\"07\", 16) &&\n        tmp[3] == parseInt(\"08\", 16)) {\n        crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n      } else {\n        crc = tmp[0] | (tmp[1] << 8) | (tmp[2] << 16) | (tmp[3] << 24);\n      }\n\n      compSize = readByte();\n      compSize |= (readByte() << 8);\n      compSize |= (readByte() << 16);\n      compSize |= (readByte() << 24);\n\n      size = readByte();\n      size |= (readByte() << 8);\n      size |= (readByte() << 16);\n      size |= (readByte() << 24);\n\n      if (debug)\n        alert(\"CRC:\");\n    }\n\n    if (modeZIP)\n      nextFile();\n\n    tmp[0] = readByte();\n    if (tmp[0] != 8) {\n      if (debug)\n        alert(\"Unknown compression method!\");\n      return 0;\n    }\n\n    gpflags = readByte();\n    if (debug) {\n      if ((gpflags & ~(parseInt(\"1f\", 16))))\n        alert(\"Unknown flags set!\");\n    }\n\n    readByte();\n    readByte();\n    readByte();\n    readByte();\n\n    readByte();\n    os = readByte();\n\n    if ((gpflags & 4)) {\n      tmp[0] = readByte();\n      tmp[2] = readByte();\n      len = tmp[0] + 256 * tmp[1];\n      if (debug)\n        alert(\"Extra field size: \" + len);\n      for (i = 0; i < len; i++)\n        readByte();\n    }\n\n    if ((gpflags & 8)) {\n      i = 0;\n      nameBuf = [];\n      while (c = readByte()) {\n        if (c == \"7\" || c == \":\")\n          i = 0;\n        if (i < NAMEMAX - 1)\n          nameBuf[i++] = c;\n      }\n      //nameBuf[i] = \"\\0\";\n      if (debug)\n        alert(\"original file name: \" + nameBuf);\n    }\n\n    if ((gpflags & 16)) {\n      while (c = readByte()) {\n        //FILE COMMENT\n      }\n    }\n\n    if ((gpflags & 2)) {\n      readByte();\n      readByte();\n    }\n\n    DeflateLoop();\n\n    crc = readByte();\n    crc |= (readByte() << 8);\n    crc |= (readByte() << 16);\n    crc |= (readByte() << 24);\n\n    size = readByte();\n    size |= (readByte() << 8);\n    size |= (readByte() << 16);\n    size |= (readByte() << 24);\n\n    if (modeZIP)\n      nextFile();\n\n  };\n\n};\n\n/**\n *  Base64 encoding / decoding\n *  {@link http://www.webtoolkit.info/}\n */\nJXG.Util.Base64 = {\n\n  // private property\n  _keyStr: \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=\",\n\n  // public method for encoding\n  encode: function(input) {\n    var output = [],\n      chr1, chr2, chr3, enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = JXG.Util.Base64._utf8_encode(input);\n\n    while (i < input.length) {\n\n      chr1 = input.charCodeAt(i++);\n      chr2 = input.charCodeAt(i++);\n      chr3 = input.charCodeAt(i++);\n\n      enc1 = chr1 >> 2;\n      enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);\n      enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);\n      enc4 = chr3 & 63;\n\n      if (isNaN(chr2)) {\n        enc3 = enc4 = 64;\n      } else if (isNaN(chr3)) {\n        enc4 = 64;\n      }\n\n      output.push([this._keyStr.charAt(enc1),\n          this._keyStr.charAt(enc2),\n          this._keyStr.charAt(enc3),\n          this._keyStr.charAt(enc4)\n      ].join(''));\n    }\n\n    return output.join('');\n  },\n\n  // public method for decoding\n  decode: function(input, utf8) {\n    var output = [],\n      chr1, chr2, chr3,\n      enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = input.replace(/[^A-Za-z0-9\\+\\/\\=]/g, \"\");\n\n    while (i < input.length) {\n\n      enc1 = this._keyStr.indexOf(input.charAt(i++));\n      enc2 = this._keyStr.indexOf(input.charAt(i++));\n      enc3 = this._keyStr.indexOf(input.charAt(i++));\n      enc4 = this._keyStr.indexOf(input.charAt(i++));\n\n      chr1 = (enc1 << 2) | (enc2 >> 4);\n      chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);\n      chr3 = ((enc3 & 3) << 6) | enc4;\n\n      output.push(String.fromCharCode(chr1));\n\n      if (enc3 != 64) {\n        output.push(String.fromCharCode(chr2));\n      }\n      if (enc4 != 64) {\n        output.push(String.fromCharCode(chr3));\n      }\n    }\n\n    output = output.join('');\n\n    if (utf8) {\n      output = JXG.Util.Base64._utf8_decode(output);\n    }\n    return output;\n\n  },\n\n  // private method for UTF-8 encoding\n  _utf8_encode: function(string) {\n    string = string.replace(/\\r\\n/g, \"\\n\");\n    var utftext = \"\";\n\n    for (var n = 0; n < string.length; n++) {\n\n      var c = string.charCodeAt(n);\n\n      if (c < 128) {\n        utftext += String.fromCharCode(c);\n      } else if ((c > 127) && (c < 2048)) {\n        utftext += String.fromCharCode((c >> 6) | 192);\n        utftext += String.fromCharCode((c & 63) | 128);\n      } else {\n        utftext += String.fromCharCode((c >> 12) | 224);\n        utftext += String.fromCharCode(((c >> 6) & 63) | 128);\n        utftext += String.fromCharCode((c & 63) | 128);\n      }\n\n    }\n\n    return utftext;\n  },\n\n  // private method for UTF-8 decoding\n  _utf8_decode: function(utftext) {\n    var string = [],\n      i = 0,\n      c = 0,\n      c2 = 0,\n      c3 = 0;\n\n    while (i < utftext.length) {\n      c = utftext.charCodeAt(i);\n      if (c < 128) {\n        string.push(String.fromCharCode(c));\n        i++;\n      } else if ((c > 191) && (c < 224)) {\n        c2 = utftext.charCodeAt(i + 1);\n        string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n        i += 2;\n      } else {\n        c2 = utftext.charCodeAt(i + 1);\n        c3 = utftext.charCodeAt(i + 2);\n        string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n        i += 3;\n      }\n    }\n    return string.join('');\n  },\n\n  _destrip: function(stripped, wrap) {\n    var lines = [],\n      lineno, i,\n      destripped = [];\n\n    if (wrap == null)\n      wrap = 76;\n\n    stripped.replace(/ /g, \"\");\n    lineno = stripped.length / wrap;\n    for (i = 0; i < lineno; i++)\n      lines[i] = stripped.substr(i * wrap, wrap);\n    if (lineno != stripped.length / wrap)\n      lines[lines.length] = stripped.substr(lineno * wrap, stripped.length - (lineno * wrap));\n\n    for (i = 0; i < lines.length; i++)\n      destripped.push(lines[i]);\n    return destripped.join('\\n');\n  },\n\n  decodeAsArray: function(input) {\n    var dec = this.decode(input),\n      ar = [],\n      i;\n    for (i = 0; i < dec.length; i++) {\n      ar[i] = dec.charCodeAt(i);\n    }\n    return ar;\n  },\n\n  decodeGEONExT: function(input) {\n    return decodeAsArray(destrip(input), false);\n  }\n};\n\n/**\n * @private\n */\nJXG.Util.asciiCharCodeAt = function(str, i) {\n  var c = str.charCodeAt(i);\n  if (c > 255) {\n    switch (c) {\n      case 8364:\n        c = 128;\n        break;\n      case 8218:\n        c = 130;\n        break;\n      case 402:\n        c = 131;\n        break;\n      case 8222:\n        c = 132;\n        break;\n      case 8230:\n        c = 133;\n        break;\n      case 8224:\n        c = 134;\n        break;\n      case 8225:\n        c = 135;\n        break;\n      case 710:\n        c = 136;\n        break;\n      case 8240:\n        c = 137;\n        break;\n      case 352:\n        c = 138;\n        break;\n      case 8249:\n        c = 139;\n        break;\n      case 338:\n        c = 140;\n        break;\n      case 381:\n        c = 142;\n        break;\n      case 8216:\n        c = 145;\n        break;\n      case 8217:\n        c = 146;\n        break;\n      case 8220:\n        c = 147;\n        break;\n      case 8221:\n        c = 148;\n        break;\n      case 8226:\n        c = 149;\n        break;\n      case 8211:\n        c = 150;\n        break;\n      case 8212:\n        c = 151;\n        break;\n      case 732:\n        c = 152;\n        break;\n      case 8482:\n        c = 153;\n        break;\n      case 353:\n        c = 154;\n        break;\n      case 8250:\n        c = 155;\n        break;\n      case 339:\n        c = 156;\n        break;\n      case 382:\n        c = 158;\n        break;\n      case 376:\n        c = 159;\n        break;\n      default:\n        break;\n    }\n  }\n  return c;\n};\n\n/**\n * Decoding string into utf-8\n * @param {String} string to decode\n * @return {String} utf8 decoded string\n */\nJXG.Util.utf8Decode = function(utftext) {\n  var string = [];\n  var i = 0;\n  var c = 0,\n    c1 = 0,\n    c2 = 0,\n    c3;\n  if (!JXG.exists(utftext)) return '';\n\n  while (i < utftext.length) {\n    c = utftext.charCodeAt(i);\n\n    if (c < 128) {\n      string.push(String.fromCharCode(c));\n      i++;\n    } else if ((c > 191) && (c < 224)) {\n      c2 = utftext.charCodeAt(i + 1);\n      string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n      i += 2;\n    } else {\n      c2 = utftext.charCodeAt(i + 1);\n      c3 = utftext.charCodeAt(i + 2);\n      string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n      i += 3;\n    }\n  };\n  return string.join('');\n};\n\n/**\n * Generate a random uuid.\n * http://www.broofa.com\n * mailto:robert@broofa.com\n *\n * Copyright (c) 2010 Robert Kieffer\n * Dual licensed under the MIT and GPL licenses.\n *\n * EXAMPLES:\n *   >>> Math.uuid()\n *   \"92329D39-6F5C-4520-ABFC-AAB64544E172\"\n */\nJXG.Util.genUUID = function() {\n  // Private array of chars to use\n  var chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''),\n    uuid = new Array(36),\n    rnd = 0,\n    r;\n\n  for (var i = 0; i < 36; i++) {\n    if (i == 8 || i == 13 || i == 18 || i == 23) {\n      uuid[i] = '-';\n    } else if (i == 14) {\n      uuid[i] = '4';\n    } else {\n      if (rnd <= 0x02) rnd = 0x2000000 + (Math.random() * 0x1000000) | 0;\n      r = rnd & 0xf;\n      rnd = rnd >> 4;\n      uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r];\n    }\n  }\n\n  return uuid.join('');\n};\n\n\nmodule.exports = JXG;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js');\n\n/**\n *\n * This object contains configuration values and implements\n * storing and retrieving configuration them from HTML5 local storage.\n *\n * This object can be accessed after calling openpgp.init()\n * using openpgp.config\n * Stored config parameters can be accessed using\n * openpgp.config.config\n * @class\n * @classdesc Implementation of the GPG4Browsers config object\n */\nvar config = function() {\n  /**\n   * @property {Integer} prefer_hash_algorithm\n   * @property {Integer} encryption_cipher\n   * @property {Integer} compression\n   * @property {Boolean} show_version\n   * @property {Boolean} show_comment\n   * @property {Boolean} integrity_protect\n   * @property {String} keyserver\n   */\n  this.prefer_hash_algorithm = enums.hash.sha256;\n  this.encryption_cipher = enums.symmetric.aes256;\n  this.compression = enums.compression.zip;\n  this.show_version = true;\n  this.show_comment = true;\n  this.integrity_protect = true;\n  this.keyserver = \"keyserver.linux.it\"; // \"pgp.mit.edu:11371\"\n\n  this.versionstring = \"OpenPGP.js VERSION\";\n  this.commentstring = \"http://openpgpjs.org\";\n\n  /**\n   * If enabled, debug messages will be printed\n   */\n  this.debug = false;\n\n};\n\nmodule.exports = new config();\n","// Modified by Recurity Labs GmbH \n\n// modified version of http://www.hanewin.net/encrypt/PGdecode.js:\n\n/* OpenPGP encryption using RSA/AES\n * Copyright 2005-2006 Herbert Hanewinkel, www.haneWIN.de\n * version 2.0, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../util'),\n  cipher = require('./cipher');\n\nmodule.exports = {\n\n  /**\n   * An array of bytes, that is integers with values from 0 to 255\n   * @typedef {(Array|Uint8Array)} openpgp_byte_array\n   */\n\n  /**\n   * Block cipher function\n   * @callback openpgp_cipher_block_fn\n   * @param {openpgp_byte_array} block A block to perform operations on\n   * @param {openpgp_byte_array} key to use in encryption/decryption\n   * @return {openpgp_byte_array} Encrypted/decrypted block\n   */\n\n\n  // --------------------------------------\n  /**\n   * This function encrypts a given with the specified prefixrandom \n   * using the specified blockcipher to encrypt a message\n   * @param {String} prefixrandom random bytes of block_size length provided \n   *  as a string to be used in prefixing the data\n   * @param {openpgp_cipher_block_fn} blockcipherfn the algorithm encrypt function to encrypt\n   *  data in one block_size encryption. \n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext data to be encrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to encrypt the data. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Encryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the encrypted data\n   */\n  encrypt: function(prefixrandom, cipherfn, plaintext, key, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var FR = new Array(block_size);\n    var FRE = new Array(block_size);\n\n    prefixrandom = prefixrandom + prefixrandom.charAt(block_size - 2) + prefixrandom.charAt(block_size - 1);\n    util.print_debug(\"prefixrandom:\" + util.hexstrdump(prefixrandom));\n    var ciphertext = \"\";\n    // 1.  The feedback register (FR) is set to the IV, which is all zeros.\n    for (var i = 0; i < block_size; i++) FR[i] = 0;\n\n    // 2.  FR is encrypted to produce FRE (FR Encrypted).  This is the\n    //     encryption of an all-zero value.\n    FRE = cipherfn.encrypt(FR);\n    // 3.  FRE is xored with the first BS octets of random data prefixed to\n    //     the plaintext to produce C[1] through C[BS], the first BS octets\n    //     of ciphertext.\n    for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ prefixrandom.charCodeAt(i));\n\n    // 4.  FR is loaded with C[1] through C[BS].\n    for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n\n    // 5.  FR is encrypted to produce FRE, the encryption of the first BS\n    // \t   octets of ciphertext.\n    FRE = cipherfn.encrypt(FR);\n\n    // 6.  The left two octets of FRE get xored with the next two octets of\n    //     data that were prefixed to the plaintext.  This produces C[BS+1]\n    //     and C[BS+2], the next two octets of ciphertext.\n    ciphertext += String.fromCharCode(FRE[0] ^ prefixrandom.charCodeAt(block_size));\n    ciphertext += String.fromCharCode(FRE[1] ^ prefixrandom.charCodeAt(block_size + 1));\n\n    if (resync) {\n      // 7.  (The resync step) FR is loaded with C3-C10.\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i + 2);\n    } else {\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n    }\n    // 8.  FR is encrypted to produce FRE.\n    FRE = cipherfn.encrypt(FR, key);\n\n    if (resync) {\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 0; i < block_size; i++)\n        ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      for (n = block_size + 2; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(n + i);\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        // next 8 octets of ciphertext.  These are loaded into FR and the\n        // process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt((n - 2) +\n            i));\n      }\n    } else {\n      plaintext = \"  \" + plaintext;\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 2; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      var tempCiphertext = ciphertext.substring(0, 2 * block_size).split('');\n      var tempCiphertextString = ciphertext.substring(block_size);\n      for (n = block_size; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = tempCiphertextString.charCodeAt(i);\n        tempCiphertextString = '';\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        //     next 8 octets of ciphertext.  These are loaded into FR and the\n        //     process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) {\n          tempCiphertext.push(String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i)));\n          tempCiphertextString += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i));\n        }\n      }\n      ciphertext = tempCiphertext.join('');\n\n    }\n\n    ciphertext = ciphertext.substring(0, plaintext.length + 2 + block_size);\n\n    return ciphertext;\n  },\n\n  /**\n   * Decrypts the prefixed data for the Modification Detection Code (MDC) computation\n   * @param {openpgp_block_cipher_fn} cipherfn.encrypt Cipher function to use\n   * @param {Integer} block_size Blocksize of the algorithm\n   * @param {openpgp_byte_array} key The key for encryption\n   * @param {String} ciphertext The encrypted data\n   * @return {String} plaintext Data of D(ciphertext) with blocksize length +2\n   */\n  mdc: function(cipherfn, key, ciphertext) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i;\n\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock);\n\n    return util.bin2str(iblock) +\n      String.fromCharCode(ablock[0] ^ ciphertext.charCodeAt(block_size)) +\n      String.fromCharCode(ablock[1] ^ ciphertext.charCodeAt(block_size + 1));\n  },\n  /**\n   * This function decrypts a given plaintext using the specified\n   * blockcipher to decrypt a message\n   * @param {openpgp_cipher_block_fn} blockcipherfn The algorithm _encrypt_ function to encrypt\n   *  data in one block_size encryption.\n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext ciphertext to be decrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to decrypt the ciphertext. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Decryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the plaintext data\n   */\n\n  decrypt: function(cipherfn, key, ciphertext, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i, n = '';\n    var text = [];\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock, key);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock, key);\n\n    // test check octets\n    if (iblock[block_size - 2] != (ablock[0] ^ ciphertext.charCodeAt(block_size)) || iblock[block_size - 1] != (ablock[\n      1] ^ ciphertext.charCodeAt(block_size + 1))) {\n      throw new Error('Invalid data.');\n    }\n\n    /*  RFC4880: Tag 18 and Resync:\n\t\t *  [...] Unlike the Symmetrically Encrypted Data Packet, no\n\t\t *  special CFB resynchronization is done after encrypting this prefix\n\t\t *  data.  See \"OpenPGP CFB Mode\" below for more details.\n\n\t\t */\n\n    if (resync) {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i + 2);\n      for (n = block_size + 2; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    } else {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i);\n      for (n = block_size; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    }\n\n    var n = resync ? 0 : 2;\n\n    text = text.join('');\n\n    text = text.substring(n, ciphertext.length - block_size - 2 + n);\n\n\n    return text;\n  },\n\n\n  normalEncrypt: function(cipherfn, key, plaintext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blocki = \"\";\n    var blockc = \"\";\n    var pos = 0;\n    var cyphertext = [];\n    var tempBlock = [];\n    blockc = iv.substring(0, block_size);\n    while (plaintext.length > block_size * pos) {\n      var encblock = cipherfn.encrypt(util.str2bin(blockc));\n      blocki = plaintext.substring((pos * block_size), (pos * block_size) + block_size);\n      for (var i = 0; i < blocki.length; i++)\n        tempBlock.push(String.fromCharCode(blocki.charCodeAt(i) ^ encblock[i]));\n      blockc = tempBlock.join('');\n      tempBlock = [];\n      cyphertext.push(blockc);\n      pos++;\n    }\n    return cyphertext.join('');\n  },\n\n  normalDecrypt: function(cipherfn, key, ciphertext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blockp = \"\";\n    var pos = 0;\n    var plaintext = [];\n    var offset = 0;\n    if (iv == null)\n      for (var i = 0; i < block_size; i++) blockp += String.fromCharCode(0);\n    else\n      blockp = iv.substring(0, block_size);\n    while (ciphertext.length > (block_size * pos)) {\n      var decblock = cipherfn.encrypt(util.str2bin(blockp));\n      blockp = ciphertext.substring((pos * (block_size)) + offset, (pos * (block_size)) + (block_size) + offset);\n      for (var i = 0; i < blockp.length; i++) {\n        plaintext.push(String.fromCharCode(blockp.charCodeAt(i) ^ decblock[i]));\n      }\n      pos++;\n    }\n\n    return plaintext.join('');\n  }\n}\n","/* Rijndael (AES) Encryption\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.1, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../../util');\n\n// The round constants used in subkey expansion\nvar Rcon = [\n    0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36, 0x6c, 0xd8,\n    0xab, 0x4d, 0x9a, 0x2f, 0x5e, 0xbc, 0x63, 0xc6, 0x97, 0x35, 0x6a, 0xd4,\n    0xb3, 0x7d, 0xfa, 0xef, 0xc5, 0x91\n];\n\n// Precomputed lookup table for the SBox\nvar S = [\n    99, 124, 119, 123, 242, 107, 111, 197, 48, 1, 103, 43, 254, 215, 171,\n    118, 202, 130, 201, 125, 250, 89, 71, 240, 173, 212, 162, 175, 156, 164,\n    114, 192, 183, 253, 147, 38, 54, 63, 247, 204, 52, 165, 229, 241, 113,\n    216, 49, 21, 4, 199, 35, 195, 24, 150, 5, 154, 7, 18, 128, 226,\n    235, 39, 178, 117, 9, 131, 44, 26, 27, 110, 90, 160, 82, 59, 214,\n    179, 41, 227, 47, 132, 83, 209, 0, 237, 32, 252, 177, 91, 106, 203,\n    190, 57, 74, 76, 88, 207, 208, 239, 170, 251, 67, 77, 51, 133, 69,\n    249, 2, 127, 80, 60, 159, 168, 81, 163, 64, 143, 146, 157, 56, 245,\n    188, 182, 218, 33, 16, 255, 243, 210, 205, 12, 19, 236, 95, 151, 68,\n    23, 196, 167, 126, 61, 100, 93, 25, 115, 96, 129, 79, 220, 34, 42,\n    144, 136, 70, 238, 184, 20, 222, 94, 11, 219, 224, 50, 58, 10, 73,\n    6, 36, 92, 194, 211, 172, 98, 145, 149, 228, 121, 231, 200, 55, 109,\n    141, 213, 78, 169, 108, 86, 244, 234, 101, 122, 174, 8, 186, 120, 37,\n    46, 28, 166, 180, 198, 232, 221, 116, 31, 75, 189, 139, 138, 112, 62,\n    181, 102, 72, 3, 246, 14, 97, 53, 87, 185, 134, 193, 29, 158, 225,\n    248, 152, 17, 105, 217, 142, 148, 155, 30, 135, 233, 206, 85, 40, 223,\n    140, 161, 137, 13, 191, 230, 66, 104, 65, 153, 45, 15, 176, 84, 187,\n    22\n];\n\nvar T1 = [\n    0xa56363c6, 0x847c7cf8, 0x997777ee, 0x8d7b7bf6,\n    0x0df2f2ff, 0xbd6b6bd6, 0xb16f6fde, 0x54c5c591,\n    0x50303060, 0x03010102, 0xa96767ce, 0x7d2b2b56,\n    0x19fefee7, 0x62d7d7b5, 0xe6abab4d, 0x9a7676ec,\n    0x45caca8f, 0x9d82821f, 0x40c9c989, 0x877d7dfa,\n    0x15fafaef, 0xeb5959b2, 0xc947478e, 0x0bf0f0fb,\n    0xecadad41, 0x67d4d4b3, 0xfda2a25f, 0xeaafaf45,\n    0xbf9c9c23, 0xf7a4a453, 0x967272e4, 0x5bc0c09b,\n    0xc2b7b775, 0x1cfdfde1, 0xae93933d, 0x6a26264c,\n    0x5a36366c, 0x413f3f7e, 0x02f7f7f5, 0x4fcccc83,\n    0x5c343468, 0xf4a5a551, 0x34e5e5d1, 0x08f1f1f9,\n    0x937171e2, 0x73d8d8ab, 0x53313162, 0x3f15152a,\n    0x0c040408, 0x52c7c795, 0x65232346, 0x5ec3c39d,\n    0x28181830, 0xa1969637, 0x0f05050a, 0xb59a9a2f,\n    0x0907070e, 0x36121224, 0x9b80801b, 0x3de2e2df,\n    0x26ebebcd, 0x6927274e, 0xcdb2b27f, 0x9f7575ea,\n    0x1b090912, 0x9e83831d, 0x742c2c58, 0x2e1a1a34,\n    0x2d1b1b36, 0xb26e6edc, 0xee5a5ab4, 0xfba0a05b,\n    0xf65252a4, 0x4d3b3b76, 0x61d6d6b7, 0xceb3b37d,\n    0x7b292952, 0x3ee3e3dd, 0x712f2f5e, 0x97848413,\n    0xf55353a6, 0x68d1d1b9, 0x00000000, 0x2cededc1,\n    0x60202040, 0x1ffcfce3, 0xc8b1b179, 0xed5b5bb6,\n    0xbe6a6ad4, 0x46cbcb8d, 0xd9bebe67, 0x4b393972,\n    0xde4a4a94, 0xd44c4c98, 0xe85858b0, 0x4acfcf85,\n    0x6bd0d0bb, 0x2aefefc5, 0xe5aaaa4f, 0x16fbfbed,\n    0xc5434386, 0xd74d4d9a, 0x55333366, 0x94858511,\n    0xcf45458a, 0x10f9f9e9, 0x06020204, 0x817f7ffe,\n    0xf05050a0, 0x443c3c78, 0xba9f9f25, 0xe3a8a84b,\n    0xf35151a2, 0xfea3a35d, 0xc0404080, 0x8a8f8f05,\n    0xad92923f, 0xbc9d9d21, 0x48383870, 0x04f5f5f1,\n    0xdfbcbc63, 0xc1b6b677, 0x75dadaaf, 0x63212142,\n    0x30101020, 0x1affffe5, 0x0ef3f3fd, 0x6dd2d2bf,\n    0x4ccdcd81, 0x140c0c18, 0x35131326, 0x2fececc3,\n    0xe15f5fbe, 0xa2979735, 0xcc444488, 0x3917172e,\n    0x57c4c493, 0xf2a7a755, 0x827e7efc, 0x473d3d7a,\n    0xac6464c8, 0xe75d5dba, 0x2b191932, 0x957373e6,\n    0xa06060c0, 0x98818119, 0xd14f4f9e, 0x7fdcdca3,\n    0x66222244, 0x7e2a2a54, 0xab90903b, 0x8388880b,\n    0xca46468c, 0x29eeeec7, 0xd3b8b86b, 0x3c141428,\n    0x79dedea7, 0xe25e5ebc, 0x1d0b0b16, 0x76dbdbad,\n    0x3be0e0db, 0x56323264, 0x4e3a3a74, 0x1e0a0a14,\n    0xdb494992, 0x0a06060c, 0x6c242448, 0xe45c5cb8,\n    0x5dc2c29f, 0x6ed3d3bd, 0xefacac43, 0xa66262c4,\n    0xa8919139, 0xa4959531, 0x37e4e4d3, 0x8b7979f2,\n    0x32e7e7d5, 0x43c8c88b, 0x5937376e, 0xb76d6dda,\n    0x8c8d8d01, 0x64d5d5b1, 0xd24e4e9c, 0xe0a9a949,\n    0xb46c6cd8, 0xfa5656ac, 0x07f4f4f3, 0x25eaeacf,\n    0xaf6565ca, 0x8e7a7af4, 0xe9aeae47, 0x18080810,\n    0xd5baba6f, 0x887878f0, 0x6f25254a, 0x722e2e5c,\n    0x241c1c38, 0xf1a6a657, 0xc7b4b473, 0x51c6c697,\n    0x23e8e8cb, 0x7cdddda1, 0x9c7474e8, 0x211f1f3e,\n    0xdd4b4b96, 0xdcbdbd61, 0x868b8b0d, 0x858a8a0f,\n    0x907070e0, 0x423e3e7c, 0xc4b5b571, 0xaa6666cc,\n    0xd8484890, 0x05030306, 0x01f6f6f7, 0x120e0e1c,\n    0xa36161c2, 0x5f35356a, 0xf95757ae, 0xd0b9b969,\n    0x91868617, 0x58c1c199, 0x271d1d3a, 0xb99e9e27,\n    0x38e1e1d9, 0x13f8f8eb, 0xb398982b, 0x33111122,\n    0xbb6969d2, 0x70d9d9a9, 0x898e8e07, 0xa7949433,\n    0xb69b9b2d, 0x221e1e3c, 0x92878715, 0x20e9e9c9,\n    0x49cece87, 0xff5555aa, 0x78282850, 0x7adfdfa5,\n    0x8f8c8c03, 0xf8a1a159, 0x80898909, 0x170d0d1a,\n    0xdabfbf65, 0x31e6e6d7, 0xc6424284, 0xb86868d0,\n    0xc3414182, 0xb0999929, 0x772d2d5a, 0x110f0f1e,\n    0xcbb0b07b, 0xfc5454a8, 0xd6bbbb6d, 0x3a16162c\n];\n\nvar T2 = [\n    0x6363c6a5, 0x7c7cf884, 0x7777ee99, 0x7b7bf68d,\n    0xf2f2ff0d, 0x6b6bd6bd, 0x6f6fdeb1, 0xc5c59154,\n    0x30306050, 0x01010203, 0x6767cea9, 0x2b2b567d,\n    0xfefee719, 0xd7d7b562, 0xabab4de6, 0x7676ec9a,\n    0xcaca8f45, 0x82821f9d, 0xc9c98940, 0x7d7dfa87,\n    0xfafaef15, 0x5959b2eb, 0x47478ec9, 0xf0f0fb0b,\n    0xadad41ec, 0xd4d4b367, 0xa2a25ffd, 0xafaf45ea,\n    0x9c9c23bf, 0xa4a453f7, 0x7272e496, 0xc0c09b5b,\n    0xb7b775c2, 0xfdfde11c, 0x93933dae, 0x26264c6a,\n    0x36366c5a, 0x3f3f7e41, 0xf7f7f502, 0xcccc834f,\n    0x3434685c, 0xa5a551f4, 0xe5e5d134, 0xf1f1f908,\n    0x7171e293, 0xd8d8ab73, 0x31316253, 0x15152a3f,\n    0x0404080c, 0xc7c79552, 0x23234665, 0xc3c39d5e,\n    0x18183028, 0x969637a1, 0x05050a0f, 0x9a9a2fb5,\n    0x07070e09, 0x12122436, 0x80801b9b, 0xe2e2df3d,\n    0xebebcd26, 0x27274e69, 0xb2b27fcd, 0x7575ea9f,\n    0x0909121b, 0x83831d9e, 0x2c2c5874, 0x1a1a342e,\n    0x1b1b362d, 0x6e6edcb2, 0x5a5ab4ee, 0xa0a05bfb,\n    0x5252a4f6, 0x3b3b764d, 0xd6d6b761, 0xb3b37dce,\n    0x2929527b, 0xe3e3dd3e, 0x2f2f5e71, 0x84841397,\n    0x5353a6f5, 0xd1d1b968, 0x00000000, 0xededc12c,\n    0x20204060, 0xfcfce31f, 0xb1b179c8, 0x5b5bb6ed,\n    0x6a6ad4be, 0xcbcb8d46, 0xbebe67d9, 0x3939724b,\n    0x4a4a94de, 0x4c4c98d4, 0x5858b0e8, 0xcfcf854a,\n    0xd0d0bb6b, 0xefefc52a, 0xaaaa4fe5, 0xfbfbed16,\n    0x434386c5, 0x4d4d9ad7, 0x33336655, 0x85851194,\n    0x45458acf, 0xf9f9e910, 0x02020406, 0x7f7ffe81,\n    0x5050a0f0, 0x3c3c7844, 0x9f9f25ba, 0xa8a84be3,\n    0x5151a2f3, 0xa3a35dfe, 0x404080c0, 0x8f8f058a,\n    0x92923fad, 0x9d9d21bc, 0x38387048, 0xf5f5f104,\n    0xbcbc63df, 0xb6b677c1, 0xdadaaf75, 0x21214263,\n    0x10102030, 0xffffe51a, 0xf3f3fd0e, 0xd2d2bf6d,\n    0xcdcd814c, 0x0c0c1814, 0x13132635, 0xececc32f,\n    0x5f5fbee1, 0x979735a2, 0x444488cc, 0x17172e39,\n    0xc4c49357, 0xa7a755f2, 0x7e7efc82, 0x3d3d7a47,\n    0x6464c8ac, 0x5d5dbae7, 0x1919322b, 0x7373e695,\n    0x6060c0a0, 0x81811998, 0x4f4f9ed1, 0xdcdca37f,\n    0x22224466, 0x2a2a547e, 0x90903bab, 0x88880b83,\n    0x46468cca, 0xeeeec729, 0xb8b86bd3, 0x1414283c,\n    0xdedea779, 0x5e5ebce2, 0x0b0b161d, 0xdbdbad76,\n    0xe0e0db3b, 0x32326456, 0x3a3a744e, 0x0a0a141e,\n    0x494992db, 0x06060c0a, 0x2424486c, 0x5c5cb8e4,\n    0xc2c29f5d, 0xd3d3bd6e, 0xacac43ef, 0x6262c4a6,\n    0x919139a8, 0x959531a4, 0xe4e4d337, 0x7979f28b,\n    0xe7e7d532, 0xc8c88b43, 0x37376e59, 0x6d6ddab7,\n    0x8d8d018c, 0xd5d5b164, 0x4e4e9cd2, 0xa9a949e0,\n    0x6c6cd8b4, 0x5656acfa, 0xf4f4f307, 0xeaeacf25,\n    0x6565caaf, 0x7a7af48e, 0xaeae47e9, 0x08081018,\n    0xbaba6fd5, 0x7878f088, 0x25254a6f, 0x2e2e5c72,\n    0x1c1c3824, 0xa6a657f1, 0xb4b473c7, 0xc6c69751,\n    0xe8e8cb23, 0xdddda17c, 0x7474e89c, 0x1f1f3e21,\n    0x4b4b96dd, 0xbdbd61dc, 0x8b8b0d86, 0x8a8a0f85,\n    0x7070e090, 0x3e3e7c42, 0xb5b571c4, 0x6666ccaa,\n    0x484890d8, 0x03030605, 0xf6f6f701, 0x0e0e1c12,\n    0x6161c2a3, 0x35356a5f, 0x5757aef9, 0xb9b969d0,\n    0x86861791, 0xc1c19958, 0x1d1d3a27, 0x9e9e27b9,\n    0xe1e1d938, 0xf8f8eb13, 0x98982bb3, 0x11112233,\n    0x6969d2bb, 0xd9d9a970, 0x8e8e0789, 0x949433a7,\n    0x9b9b2db6, 0x1e1e3c22, 0x87871592, 0xe9e9c920,\n    0xcece8749, 0x5555aaff, 0x28285078, 0xdfdfa57a,\n    0x8c8c038f, 0xa1a159f8, 0x89890980, 0x0d0d1a17,\n    0xbfbf65da, 0xe6e6d731, 0x424284c6, 0x6868d0b8,\n    0x414182c3, 0x999929b0, 0x2d2d5a77, 0x0f0f1e11,\n    0xb0b07bcb, 0x5454a8fc, 0xbbbb6dd6, 0x16162c3a\n];\n\nvar T3 = [\n    0x63c6a563, 0x7cf8847c, 0x77ee9977, 0x7bf68d7b,\n    0xf2ff0df2, 0x6bd6bd6b, 0x6fdeb16f, 0xc59154c5,\n    0x30605030, 0x01020301, 0x67cea967, 0x2b567d2b,\n    0xfee719fe, 0xd7b562d7, 0xab4de6ab, 0x76ec9a76,\n    0xca8f45ca, 0x821f9d82, 0xc98940c9, 0x7dfa877d,\n    0xfaef15fa, 0x59b2eb59, 0x478ec947, 0xf0fb0bf0,\n    0xad41ecad, 0xd4b367d4, 0xa25ffda2, 0xaf45eaaf,\n    0x9c23bf9c, 0xa453f7a4, 0x72e49672, 0xc09b5bc0,\n    0xb775c2b7, 0xfde11cfd, 0x933dae93, 0x264c6a26,\n    0x366c5a36, 0x3f7e413f, 0xf7f502f7, 0xcc834fcc,\n    0x34685c34, 0xa551f4a5, 0xe5d134e5, 0xf1f908f1,\n    0x71e29371, 0xd8ab73d8, 0x31625331, 0x152a3f15,\n    0x04080c04, 0xc79552c7, 0x23466523, 0xc39d5ec3,\n    0x18302818, 0x9637a196, 0x050a0f05, 0x9a2fb59a,\n    0x070e0907, 0x12243612, 0x801b9b80, 0xe2df3de2,\n    0xebcd26eb, 0x274e6927, 0xb27fcdb2, 0x75ea9f75,\n    0x09121b09, 0x831d9e83, 0x2c58742c, 0x1a342e1a,\n    0x1b362d1b, 0x6edcb26e, 0x5ab4ee5a, 0xa05bfba0,\n    0x52a4f652, 0x3b764d3b, 0xd6b761d6, 0xb37dceb3,\n    0x29527b29, 0xe3dd3ee3, 0x2f5e712f, 0x84139784,\n    0x53a6f553, 0xd1b968d1, 0x00000000, 0xedc12ced,\n    0x20406020, 0xfce31ffc, 0xb179c8b1, 0x5bb6ed5b,\n    0x6ad4be6a, 0xcb8d46cb, 0xbe67d9be, 0x39724b39,\n    0x4a94de4a, 0x4c98d44c, 0x58b0e858, 0xcf854acf,\n    0xd0bb6bd0, 0xefc52aef, 0xaa4fe5aa, 0xfbed16fb,\n    0x4386c543, 0x4d9ad74d, 0x33665533, 0x85119485,\n    0x458acf45, 0xf9e910f9, 0x02040602, 0x7ffe817f,\n    0x50a0f050, 0x3c78443c, 0x9f25ba9f, 0xa84be3a8,\n    0x51a2f351, 0xa35dfea3, 0x4080c040, 0x8f058a8f,\n    0x923fad92, 0x9d21bc9d, 0x38704838, 0xf5f104f5,\n    0xbc63dfbc, 0xb677c1b6, 0xdaaf75da, 0x21426321,\n    0x10203010, 0xffe51aff, 0xf3fd0ef3, 0xd2bf6dd2,\n    0xcd814ccd, 0x0c18140c, 0x13263513, 0xecc32fec,\n    0x5fbee15f, 0x9735a297, 0x4488cc44, 0x172e3917,\n    0xc49357c4, 0xa755f2a7, 0x7efc827e, 0x3d7a473d,\n    0x64c8ac64, 0x5dbae75d, 0x19322b19, 0x73e69573,\n    0x60c0a060, 0x81199881, 0x4f9ed14f, 0xdca37fdc,\n    0x22446622, 0x2a547e2a, 0x903bab90, 0x880b8388,\n    0x468cca46, 0xeec729ee, 0xb86bd3b8, 0x14283c14,\n    0xdea779de, 0x5ebce25e, 0x0b161d0b, 0xdbad76db,\n    0xe0db3be0, 0x32645632, 0x3a744e3a, 0x0a141e0a,\n    0x4992db49, 0x060c0a06, 0x24486c24, 0x5cb8e45c,\n    0xc29f5dc2, 0xd3bd6ed3, 0xac43efac, 0x62c4a662,\n    0x9139a891, 0x9531a495, 0xe4d337e4, 0x79f28b79,\n    0xe7d532e7, 0xc88b43c8, 0x376e5937, 0x6ddab76d,\n    0x8d018c8d, 0xd5b164d5, 0x4e9cd24e, 0xa949e0a9,\n    0x6cd8b46c, 0x56acfa56, 0xf4f307f4, 0xeacf25ea,\n    0x65caaf65, 0x7af48e7a, 0xae47e9ae, 0x08101808,\n    0xba6fd5ba, 0x78f08878, 0x254a6f25, 0x2e5c722e,\n    0x1c38241c, 0xa657f1a6, 0xb473c7b4, 0xc69751c6,\n    0xe8cb23e8, 0xdda17cdd, 0x74e89c74, 0x1f3e211f,\n    0x4b96dd4b, 0xbd61dcbd, 0x8b0d868b, 0x8a0f858a,\n    0x70e09070, 0x3e7c423e, 0xb571c4b5, 0x66ccaa66,\n    0x4890d848, 0x03060503, 0xf6f701f6, 0x0e1c120e,\n    0x61c2a361, 0x356a5f35, 0x57aef957, 0xb969d0b9,\n    0x86179186, 0xc19958c1, 0x1d3a271d, 0x9e27b99e,\n    0xe1d938e1, 0xf8eb13f8, 0x982bb398, 0x11223311,\n    0x69d2bb69, 0xd9a970d9, 0x8e07898e, 0x9433a794,\n    0x9b2db69b, 0x1e3c221e, 0x87159287, 0xe9c920e9,\n    0xce8749ce, 0x55aaff55, 0x28507828, 0xdfa57adf,\n    0x8c038f8c, 0xa159f8a1, 0x89098089, 0x0d1a170d,\n    0xbf65dabf, 0xe6d731e6, 0x4284c642, 0x68d0b868,\n    0x4182c341, 0x9929b099, 0x2d5a772d, 0x0f1e110f,\n    0xb07bcbb0, 0x54a8fc54, 0xbb6dd6bb, 0x162c3a16\n];\n\nvar T4 = [\n    0xc6a56363, 0xf8847c7c, 0xee997777, 0xf68d7b7b,\n    0xff0df2f2, 0xd6bd6b6b, 0xdeb16f6f, 0x9154c5c5,\n    0x60503030, 0x02030101, 0xcea96767, 0x567d2b2b,\n    0xe719fefe, 0xb562d7d7, 0x4de6abab, 0xec9a7676,\n    0x8f45caca, 0x1f9d8282, 0x8940c9c9, 0xfa877d7d,\n    0xef15fafa, 0xb2eb5959, 0x8ec94747, 0xfb0bf0f0,\n    0x41ecadad, 0xb367d4d4, 0x5ffda2a2, 0x45eaafaf,\n    0x23bf9c9c, 0x53f7a4a4, 0xe4967272, 0x9b5bc0c0,\n    0x75c2b7b7, 0xe11cfdfd, 0x3dae9393, 0x4c6a2626,\n    0x6c5a3636, 0x7e413f3f, 0xf502f7f7, 0x834fcccc,\n    0x685c3434, 0x51f4a5a5, 0xd134e5e5, 0xf908f1f1,\n    0xe2937171, 0xab73d8d8, 0x62533131, 0x2a3f1515,\n    0x080c0404, 0x9552c7c7, 0x46652323, 0x9d5ec3c3,\n    0x30281818, 0x37a19696, 0x0a0f0505, 0x2fb59a9a,\n    0x0e090707, 0x24361212, 0x1b9b8080, 0xdf3de2e2,\n    0xcd26ebeb, 0x4e692727, 0x7fcdb2b2, 0xea9f7575,\n    0x121b0909, 0x1d9e8383, 0x58742c2c, 0x342e1a1a,\n    0x362d1b1b, 0xdcb26e6e, 0xb4ee5a5a, 0x5bfba0a0,\n    0xa4f65252, 0x764d3b3b, 0xb761d6d6, 0x7dceb3b3,\n    0x527b2929, 0xdd3ee3e3, 0x5e712f2f, 0x13978484,\n    0xa6f55353, 0xb968d1d1, 0x00000000, 0xc12ceded,\n    0x40602020, 0xe31ffcfc, 0x79c8b1b1, 0xb6ed5b5b,\n    0xd4be6a6a, 0x8d46cbcb, 0x67d9bebe, 0x724b3939,\n    0x94de4a4a, 0x98d44c4c, 0xb0e85858, 0x854acfcf,\n    0xbb6bd0d0, 0xc52aefef, 0x4fe5aaaa, 0xed16fbfb,\n    0x86c54343, 0x9ad74d4d, 0x66553333, 0x11948585,\n    0x8acf4545, 0xe910f9f9, 0x04060202, 0xfe817f7f,\n    0xa0f05050, 0x78443c3c, 0x25ba9f9f, 0x4be3a8a8,\n    0xa2f35151, 0x5dfea3a3, 0x80c04040, 0x058a8f8f,\n    0x3fad9292, 0x21bc9d9d, 0x70483838, 0xf104f5f5,\n    0x63dfbcbc, 0x77c1b6b6, 0xaf75dada, 0x42632121,\n    0x20301010, 0xe51affff, 0xfd0ef3f3, 0xbf6dd2d2,\n    0x814ccdcd, 0x18140c0c, 0x26351313, 0xc32fecec,\n    0xbee15f5f, 0x35a29797, 0x88cc4444, 0x2e391717,\n    0x9357c4c4, 0x55f2a7a7, 0xfc827e7e, 0x7a473d3d,\n    0xc8ac6464, 0xbae75d5d, 0x322b1919, 0xe6957373,\n    0xc0a06060, 0x19988181, 0x9ed14f4f, 0xa37fdcdc,\n    0x44662222, 0x547e2a2a, 0x3bab9090, 0x0b838888,\n    0x8cca4646, 0xc729eeee, 0x6bd3b8b8, 0x283c1414,\n    0xa779dede, 0xbce25e5e, 0x161d0b0b, 0xad76dbdb,\n    0xdb3be0e0, 0x64563232, 0x744e3a3a, 0x141e0a0a,\n    0x92db4949, 0x0c0a0606, 0x486c2424, 0xb8e45c5c,\n    0x9f5dc2c2, 0xbd6ed3d3, 0x43efacac, 0xc4a66262,\n    0x39a89191, 0x31a49595, 0xd337e4e4, 0xf28b7979,\n    0xd532e7e7, 0x8b43c8c8, 0x6e593737, 0xdab76d6d,\n    0x018c8d8d, 0xb164d5d5, 0x9cd24e4e, 0x49e0a9a9,\n    0xd8b46c6c, 0xacfa5656, 0xf307f4f4, 0xcf25eaea,\n    0xcaaf6565, 0xf48e7a7a, 0x47e9aeae, 0x10180808,\n    0x6fd5baba, 0xf0887878, 0x4a6f2525, 0x5c722e2e,\n    0x38241c1c, 0x57f1a6a6, 0x73c7b4b4, 0x9751c6c6,\n    0xcb23e8e8, 0xa17cdddd, 0xe89c7474, 0x3e211f1f,\n    0x96dd4b4b, 0x61dcbdbd, 0x0d868b8b, 0x0f858a8a,\n    0xe0907070, 0x7c423e3e, 0x71c4b5b5, 0xccaa6666,\n    0x90d84848, 0x06050303, 0xf701f6f6, 0x1c120e0e,\n    0xc2a36161, 0x6a5f3535, 0xaef95757, 0x69d0b9b9,\n    0x17918686, 0x9958c1c1, 0x3a271d1d, 0x27b99e9e,\n    0xd938e1e1, 0xeb13f8f8, 0x2bb39898, 0x22331111,\n    0xd2bb6969, 0xa970d9d9, 0x07898e8e, 0x33a79494,\n    0x2db69b9b, 0x3c221e1e, 0x15928787, 0xc920e9e9,\n    0x8749cece, 0xaaff5555, 0x50782828, 0xa57adfdf,\n    0x038f8c8c, 0x59f8a1a1, 0x09808989, 0x1a170d0d,\n    0x65dabfbf, 0xd731e6e6, 0x84c64242, 0xd0b86868,\n    0x82c34141, 0x29b09999, 0x5a772d2d, 0x1e110f0f,\n    0x7bcbb0b0, 0xa8fc5454, 0x6dd6bbbb, 0x2c3a1616\n];\n\nfunction B0(x) {\n  return (x & 255);\n}\n\nfunction B1(x) {\n  return ((x >> 8) & 255);\n}\n\nfunction B2(x) {\n  return ((x >> 16) & 255);\n}\n\nfunction B3(x) {\n  return ((x >> 24) & 255);\n}\n\nfunction F1(x0, x1, x2, x3) {\n  return B1(T1[x0 & 255]) | (B1(T1[(x1 >> 8) & 255]) << 8) | (B1(T1[(x2 >> 16) & 255]) << 16) | (B1(T1[x3 >>> 24]) <<\n    24);\n}\n\nfunction packBytes(octets) {\n  var i, j;\n  var len = octets.length;\n  var b = new Array(len / 4);\n\n  if (!octets || len % 4) return;\n\n  for (i = 0, j = 0; j < len; j += 4)\n    b[i++] = octets[j] | (octets[j + 1] << 8) | (octets[j + 2] << 16) | (octets[j + 3] << 24);\n\n  return b;\n}\n\nfunction unpackBytes(packed) {\n  var j;\n  var i = 0,\n    l = packed.length;\n  var r = new Array(l * 4);\n\n  for (j = 0; j < l; j++) {\n    r[i++] = B0(packed[j]);\n    r[i++] = B1(packed[j]);\n    r[i++] = B2(packed[j]);\n    r[i++] = B3(packed[j]);\n  }\n  return r;\n}\n\n// ------------------------------------------------\n\nvar maxkc = 8;\nvar maxrk = 14;\n\nfunction keyExpansion(key) {\n  var kc, i, j, r, t;\n  var rounds;\n  var keySched = new Array(maxrk + 1);\n  var keylen = key.length;\n  var k = new Array(maxkc);\n  var tk = new Array(maxkc);\n  var rconpointer = 0;\n\n  if (keylen == 16) {\n    rounds = 10;\n    kc = 4;\n  } else if (keylen == 24) {\n    rounds = 12;\n    kc = 6;\n  } else if (keylen == 32) {\n    rounds = 14;\n    kc = 8;\n  } else {\n    util.print_error('aes.js: Invalid key-length for AES key:' + keylen);\n    return;\n  }\n\n  for (i = 0; i < maxrk + 1; i++) keySched[i] = new Array(4);\n\n  for (i = 0, j = 0; j < keylen; j++, i += 4)\n    k[j] = key.charCodeAt(i) | (key.charCodeAt(i + 1) << 8) | (key.charCodeAt(i + 2) << 16) | (key.charCodeAt(i + 3) <<\n      24);\n\n  for (j = kc - 1; j >= 0; j--) tk[j] = k[j];\n\n  r = 0;\n  t = 0;\n  for (j = 0;\n  (j < kc) && (r < rounds + 1);) {\n    for (;\n    (j < kc) && (t < 4); j++, t++) {\n      keySched[r][t] = tk[j];\n    }\n    if (t == 4) {\n      r++;\n      t = 0;\n    }\n  }\n\n  while (r < rounds + 1) {\n    var temp = tk[kc - 1];\n\n    tk[0] ^= S[B1(temp)] | (S[B2(temp)] << 8) | (S[B3(temp)] << 16) | (S[B0(temp)] << 24);\n    tk[0] ^= Rcon[rconpointer++];\n\n    if (kc != 8) {\n      for (j = 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    } else {\n      for (j = 1; j < kc / 2; j++) tk[j] ^= tk[j - 1];\n\n      temp = tk[kc / 2 - 1];\n      tk[kc / 2] ^= S[B0(temp)] | (S[B1(temp)] << 8) | (S[B2(temp)] << 16) | (S[B3(temp)] << 24);\n\n      for (j = kc / 2 + 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    }\n\n    for (j = 0;\n    (j < kc) && (r < rounds + 1);) {\n      for (;\n      (j < kc) && (t < 4); j++, t++) {\n        keySched[r][t] = tk[j];\n      }\n      if (t == 4) {\n        r++;\n        t = 0;\n      }\n    }\n  }\n  this.rounds = rounds;\n  this.rk = keySched;\n  return this;\n}\n\nfunction AESencrypt(block, ctx) {\n  var r;\n  var t0, t1, t2, t3;\n\n  var b = packBytes(block);\n  var rounds = ctx.rounds;\n  var b0 = b[0];\n  var b1 = b[1];\n  var b2 = b[2];\n  var b3 = b[3];\n\n  for (r = 0; r < rounds - 1; r++) {\n    t0 = b0 ^ ctx.rk[r][0];\n    t1 = b1 ^ ctx.rk[r][1];\n    t2 = b2 ^ ctx.rk[r][2];\n    t3 = b3 ^ ctx.rk[r][3];\n\n    b0 = T1[t0 & 255] ^ T2[(t1 >> 8) & 255] ^ T3[(t2 >> 16) & 255] ^ T4[t3 >>> 24];\n    b1 = T1[t1 & 255] ^ T2[(t2 >> 8) & 255] ^ T3[(t3 >> 16) & 255] ^ T4[t0 >>> 24];\n    b2 = T1[t2 & 255] ^ T2[(t3 >> 8) & 255] ^ T3[(t0 >> 16) & 255] ^ T4[t1 >>> 24];\n    b3 = T1[t3 & 255] ^ T2[(t0 >> 8) & 255] ^ T3[(t1 >> 16) & 255] ^ T4[t2 >>> 24];\n  }\n\n  // last round is special\n  r = rounds - 1;\n\n  t0 = b0 ^ ctx.rk[r][0];\n  t1 = b1 ^ ctx.rk[r][1];\n  t2 = b2 ^ ctx.rk[r][2];\n  t3 = b3 ^ ctx.rk[r][3];\n\n  b[0] = F1(t0, t1, t2, t3) ^ ctx.rk[rounds][0];\n  b[1] = F1(t1, t2, t3, t0) ^ ctx.rk[rounds][1];\n  b[2] = F1(t2, t3, t0, t1) ^ ctx.rk[rounds][2];\n  b[3] = F1(t3, t0, t1, t2) ^ ctx.rk[rounds][3];\n\n  return unpackBytes(b);\n}\n\nfunction makeClass(length) {\n\n  var c = function(key) {\n    this.key = keyExpansion(key);\n\n    this.encrypt = function(block) {\n      return AESencrypt(block, this.key);\n    }\n  }\n\n  c.blockSize = c.prototype.blockSize = 16;\n  c.keySize = c.prototype.keySize = length / 8;\n\n  return c;\n}\n\nmodule.exports = {}\n\nvar types = [128, 192, 256];\n\nfor (var i in types) {\n  module.exports[types[i]] = makeClass(types[i]);\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Originally written by nklein software (nklein.com)\n */\n\n/* \n * Javascript implementation based on Bruce Schneier's reference implementation.\n *\n *\n * The constructor doesn't do much of anything.  It's just here\n * so we can start defining properties and methods and such.\n */\nfunction Blowfish() {};\n\n/*\n * Declare the block size so that protocols know what size\n * Initialization Vector (IV) they will need.\n */\nBlowfish.prototype.BLOCKSIZE = 8;\n\n/*\n * These are the default SBOXES.\n */\nBlowfish.prototype.SBOXES = [\n  [\n      0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96,\n      0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16,\n      0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658,\n      0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013,\n      0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e,\n      0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60,\n      0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6,\n      0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a,\n      0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c,\n      0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193,\n      0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1,\n      0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239,\n      0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a,\n      0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3,\n      0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176,\n      0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe,\n      0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706,\n      0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b,\n      0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b,\n      0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463,\n      0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c,\n      0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3,\n      0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a,\n      0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8,\n      0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760,\n      0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db,\n      0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8,\n      0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b,\n      0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33,\n      0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4,\n      0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0,\n      0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c,\n      0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777,\n      0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299,\n      0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705,\n      0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf,\n      0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e,\n      0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa,\n      0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9,\n      0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915,\n      0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f,\n      0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664,\n      0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a\n  ],\n  [\n      0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d,\n      0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1,\n      0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65,\n      0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1,\n      0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9,\n      0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737,\n      0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d,\n      0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd,\n      0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc,\n      0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41,\n      0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908,\n      0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af,\n      0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124,\n      0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c,\n      0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908,\n      0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd,\n      0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b,\n      0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e,\n      0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa,\n      0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a,\n      0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d,\n      0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66,\n      0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5,\n      0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84,\n      0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96,\n      0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14,\n      0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca,\n      0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7,\n      0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77,\n      0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99,\n      0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054,\n      0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73,\n      0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea,\n      0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105,\n      0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646,\n      0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285,\n      0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea,\n      0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb,\n      0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e,\n      0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc,\n      0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd,\n      0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20,\n      0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7\n  ],\n  [\n      0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7,\n      0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af,\n      0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af,\n      0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504,\n      0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4,\n      0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee,\n      0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec,\n      0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b,\n      0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332,\n      0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527,\n      0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58,\n      0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c,\n      0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22,\n      0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17,\n      0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60,\n      0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115,\n      0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99,\n      0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0,\n      0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74,\n      0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d,\n      0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3,\n      0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3,\n      0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979,\n      0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c,\n      0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa,\n      0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a,\n      0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086,\n      0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc,\n      0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24,\n      0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2,\n      0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84,\n      0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c,\n      0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09,\n      0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10,\n      0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe,\n      0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027,\n      0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0,\n      0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634,\n      0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188,\n      0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc,\n      0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8,\n      0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837,\n      0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0\n  ],\n  [\n      0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742,\n      0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b,\n      0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79,\n      0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6,\n      0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a,\n      0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4,\n      0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1,\n      0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59,\n      0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797,\n      0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28,\n      0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6,\n      0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28,\n      0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba,\n      0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a,\n      0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5,\n      0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f,\n      0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce,\n      0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680,\n      0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd,\n      0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb,\n      0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb,\n      0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370,\n      0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc,\n      0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048,\n      0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc,\n      0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9,\n      0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a,\n      0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f,\n      0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a,\n      0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1,\n      0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b,\n      0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e,\n      0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e,\n      0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f,\n      0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623,\n      0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc,\n      0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a,\n      0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6,\n      0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3,\n      0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060,\n      0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c,\n      0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f,\n      0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6\n  ]\n];\n\n//*\n//* This is the default PARRAY\n//*\nBlowfish.prototype.PARRAY = [\n    0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0,\n    0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c,\n    0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b\n];\n\n//*\n//* This is the number of rounds the cipher will go\n//*\nBlowfish.prototype.NN = 16;\n\n//*\n//* This function is needed to get rid of problems\n//* with the high-bit getting set.  If we don't do\n//* this, then sometimes ( aa & 0x00FFFFFFFF ) is not\n//* equal to ( bb & 0x00FFFFFFFF ) even when they\n//* agree bit-for-bit for the first 32 bits.\n//*\nBlowfish.prototype._clean = function(xx) {\n  if (xx < 0) {\n    var yy = xx & 0x7FFFFFFF;\n    xx = yy + 0x80000000;\n  }\n  return xx;\n};\n\n//*\n//* This is the mixing function that uses the sboxes\n//*\nBlowfish.prototype._F = function(xx) {\n  var aa;\n  var bb;\n  var cc;\n  var dd;\n  var yy;\n\n  dd = xx & 0x00FF;\n  xx >>>= 8;\n  cc = xx & 0x00FF;\n  xx >>>= 8;\n  bb = xx & 0x00FF;\n  xx >>>= 8;\n  aa = xx & 0x00FF;\n\n  yy = this.sboxes[0][aa] + this.sboxes[1][bb];\n  yy = yy ^ this.sboxes[2][cc];\n  yy = yy + this.sboxes[3][dd];\n\n  return yy;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and does NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._encrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = 0; ii < this.NN; ++ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[this.NN + 0];\n  dataR = dataR ^ this.parray[this.NN + 1];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a vector of numbers and turns them\n//* into long words so that they can be processed by the\n//* real algorithm.\n//*\n//* Maybe I should make the real algorithm above take a vector\n//* instead.  That will involve more looping, but it won't require\n//* the F() method to deconstruct the vector.\n//*\nBlowfish.prototype.encrypt_block = function(vector) {\n  var ii;\n  var vals = [0, 0];\n  var off = this.BLOCKSIZE / 2;\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    vals[0] = (vals[0] << 8) | (vector[ii + 0] & 0x00FF);\n    vals[1] = (vals[1] << 8) | (vector[ii + off] & 0x00FF);\n  }\n\n  this._encrypt_block(vals);\n\n  var ret = [];\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    ret[ii + 0] = (vals[0] >>> (24 - 8 * (ii)) & 0x00FF);\n    ret[ii + off] = (vals[1] >>> (24 - 8 * (ii)) & 0x00FF);\n    // vals[ 0 ] = ( vals[ 0 ] >>> 8 );\n    // vals[ 1 ] = ( vals[ 1 ] >>> 8 );\n  }\n\n  return ret;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and undoes NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._decrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = this.NN + 1; ii > 1; --ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[1];\n  dataR = dataR ^ this.parray[0];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a key array and initializes the\n//* sboxes and parray for this encryption.\n//*\nBlowfish.prototype.init = function(key) {\n  var ii;\n  var jj = 0;\n\n  this.parray = [];\n  for (ii = 0; ii < this.NN + 2; ++ii) {\n    var data = 0x00000000;\n    var kk;\n    for (kk = 0; kk < 4; ++kk) {\n      data = (data << 8) | (key[jj] & 0x00FF);\n      if (++jj >= key.length) {\n        jj = 0;\n      }\n    }\n    this.parray[ii] = this.PARRAY[ii] ^ data;\n  }\n\n  this.sboxes = [];\n  for (ii = 0; ii < 4; ++ii) {\n    this.sboxes[ii] = [];\n    for (jj = 0; jj < 256; ++jj) {\n      this.sboxes[ii][jj] = this.SBOXES[ii][jj];\n    }\n  }\n\n  var vals = [0x00000000, 0x00000000];\n\n  for (ii = 0; ii < this.NN + 2; ii += 2) {\n    this._encrypt_block(vals);\n    this.parray[ii + 0] = vals[0];\n    this.parray[ii + 1] = vals[1];\n  }\n\n  for (ii = 0; ii < 4; ++ii) {\n    for (jj = 0; jj < 256; jj += 2) {\n      this._encrypt_block(vals);\n      this.sboxes[ii][jj + 0] = vals[0];\n      this.sboxes[ii][jj + 1] = vals[1];\n    }\n  }\n};\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction BFencrypt(block, key) {\n  var bf = new Blowfish();\n  bf.init(util.str2bin(key));\n  return bf.encrypt_block(block);\n}\n\nfunction BF(key) {\n  this.bf = new Blowfish();\n  this.bf.init(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.bf.encrypt_block(block);\n  }\n}\n\n\nmodule.exports = BF;\nmodule.exports.keySize = BF.prototype.keySize = 16;\nmodule.exports.blockSize = BF.prototype.blockSize = 16;\n","// Use of this source code is governed by a BSD-style\n// license that can be found in the LICENSE file.\n\n// Copyright 2010 pjacobs@xeekr.com . All rights reserved.\n\n// Modified by Recurity Labs GmbH\n\n// fixed/modified by Herbert Hanewinkel, www.haneWIN.de\n// check www.haneWIN.de for the latest version\n\n// cast5.js is a Javascript implementation of CAST-128, as defined in RFC 2144.\n// CAST-128 is a common OpenPGP cipher.\n\n\n// CAST5 constructor\n\n\n\nfunction openpgp_symenc_cast5() {\n  this.BlockSize = 8;\n  this.KeySize = 16;\n\n  this.setKey = function(key) {\n    this.masking = new Array(16);\n    this.rotate = new Array(16);\n\n    this.reset();\n\n    if (key.length == this.KeySize) {\n      this.keySchedule(key);\n    } else {\n      util.print_error('cast5.js: CAST-128: keys must be 16 bytes');\n      return false;\n    }\n    return true;\n  };\n\n  this.reset = function() {\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = 0;\n      this.rotate[i] = 0;\n    }\n  };\n\n  this.getBlockSize = function() {\n    return BlockSize;\n  };\n\n  this.encrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >>> 16) & 255;\n      dst[i + 6] = (l >>> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n\n  this.decrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >> 16) & 255;\n      dst[i + 6] = (l >> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n  var scheduleA = new Array(4);\n\n  scheduleA[0] = new Array(4);\n  scheduleA[0][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 0x8);\n  scheduleA[0][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[0][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[0][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n  scheduleA[1] = new Array(4);\n  scheduleA[1][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[1][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[1][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[1][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  scheduleA[2] = new Array(4);\n  scheduleA[2][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 8);\n  scheduleA[2][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[2][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[2][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n\n  scheduleA[3] = new Array(4);\n  scheduleA[3][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[3][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[3][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[3][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  var scheduleB = new Array(4);\n\n  scheduleB[0] = new Array(4);\n  scheduleB[0][0] = new Array(16 + 8, 16 + 9, 16 + 7, 16 + 6, 16 + 2);\n  scheduleB[0][1] = new Array(16 + 0xa, 16 + 0xb, 16 + 5, 16 + 4, 16 + 6);\n  scheduleB[0][2] = new Array(16 + 0xc, 16 + 0xd, 16 + 3, 16 + 2, 16 + 9);\n  scheduleB[0][3] = new Array(16 + 0xe, 16 + 0xf, 16 + 1, 16 + 0, 16 + 0xc);\n\n  scheduleB[1] = new Array(4);\n  scheduleB[1][0] = new Array(3, 2, 0xc, 0xd, 8);\n  scheduleB[1][1] = new Array(1, 0, 0xe, 0xf, 0xd);\n  scheduleB[1][2] = new Array(7, 6, 8, 9, 3);\n  scheduleB[1][3] = new Array(5, 4, 0xa, 0xb, 7);\n\n\n  scheduleB[2] = new Array(4);\n  scheduleB[2][0] = new Array(16 + 3, 16 + 2, 16 + 0xc, 16 + 0xd, 16 + 9);\n  scheduleB[2][1] = new Array(16 + 1, 16 + 0, 16 + 0xe, 16 + 0xf, 16 + 0xc);\n  scheduleB[2][2] = new Array(16 + 7, 16 + 6, 16 + 8, 16 + 9, 16 + 2);\n  scheduleB[2][3] = new Array(16 + 5, 16 + 4, 16 + 0xa, 16 + 0xb, 16 + 6);\n\n\n  scheduleB[3] = new Array(4);\n  scheduleB[3][0] = new Array(8, 9, 7, 6, 3);\n  scheduleB[3][1] = new Array(0xa, 0xb, 5, 4, 7);\n  scheduleB[3][2] = new Array(0xc, 0xd, 3, 2, 8);\n  scheduleB[3][3] = new Array(0xe, 0xf, 1, 0, 0xd);\n\n  // changed 'in' to 'inn' (in javascript 'in' is a reserved word)\n  this.keySchedule = function(inn) {\n    var t = new Array(8);\n    var k = new Array(32);\n\n    for (var i = 0; i < 4; i++) {\n      var j = i * 4;\n      t[i] = inn[j] << 24 | inn[j + 1] << 16 | inn[j + 2] << 8 | inn[j + 3];\n    }\n\n    var x = [6, 7, 4, 5];\n    var ki = 0;\n\n    for (var half = 0; half < 2; half++) {\n      for (var round = 0; round < 4; round++) {\n        for (var j = 0; j < 4; j++) {\n          var a = scheduleA[round][j];\n          var w = t[a[1]];\n\n          w ^= sBox[4][(t[a[2] >>> 2] >>> (24 - 8 * (a[2] & 3))) & 0xff];\n          w ^= sBox[5][(t[a[3] >>> 2] >>> (24 - 8 * (a[3] & 3))) & 0xff];\n          w ^= sBox[6][(t[a[4] >>> 2] >>> (24 - 8 * (a[4] & 3))) & 0xff];\n          w ^= sBox[7][(t[a[5] >>> 2] >>> (24 - 8 * (a[5] & 3))) & 0xff];\n          w ^= sBox[x[j]][(t[a[6] >>> 2] >>> (24 - 8 * (a[6] & 3))) & 0xff];\n          t[a[0]] = w;\n        }\n\n        for (var j = 0; j < 4; j++) {\n          var b = scheduleB[round][j];\n          var w = sBox[4][(t[b[0] >>> 2] >>> (24 - 8 * (b[0] & 3))) & 0xff];\n\n          w ^= sBox[5][(t[b[1] >>> 2] >>> (24 - 8 * (b[1] & 3))) & 0xff];\n          w ^= sBox[6][(t[b[2] >>> 2] >>> (24 - 8 * (b[2] & 3))) & 0xff];\n          w ^= sBox[7][(t[b[3] >>> 2] >>> (24 - 8 * (b[3] & 3))) & 0xff];\n          w ^= sBox[4 + j][(t[b[4] >>> 2] >>> (24 - 8 * (b[4] & 3))) & 0xff];\n          k[ki] = w;\n          ki++;\n        }\n      }\n    }\n\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = k[i];\n      this.rotate[i] = k[16 + i] & 0x1f;\n    }\n  };\n\n  // These are the three 'f' functions. See RFC 2144, section 2.2.\n\n  function f1(d, m, r) {\n    var t = m + d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] ^ sBox[1][(I >>> 16) & 255]) - sBox[2][(I >>> 8) & 255]) + sBox[3][I & 255];\n  }\n\n  function f2(d, m, r) {\n    var t = m ^ d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] - sBox[1][(I >>> 16) & 255]) + sBox[2][(I >>> 8) & 255]) ^ sBox[3][I & 255];\n  }\n\n  function f3(d, m, r) {\n    var t = m - d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] + sBox[1][(I >>> 16) & 255]) ^ sBox[2][(I >>> 8) & 255]) - sBox[3][I & 255];\n  }\n\n  var sBox = new Array(8);\n  sBox[0] = new Array(\n    0x30fb40d4, 0x9fa0ff0b, 0x6beccd2f, 0x3f258c7a, 0x1e213f2f, 0x9c004dd3, 0x6003e540, 0xcf9fc949,\n    0xbfd4af27, 0x88bbbdb5, 0xe2034090, 0x98d09675, 0x6e63a0e0, 0x15c361d2, 0xc2e7661d, 0x22d4ff8e,\n    0x28683b6f, 0xc07fd059, 0xff2379c8, 0x775f50e2, 0x43c340d3, 0xdf2f8656, 0x887ca41a, 0xa2d2bd2d,\n    0xa1c9e0d6, 0x346c4819, 0x61b76d87, 0x22540f2f, 0x2abe32e1, 0xaa54166b, 0x22568e3a, 0xa2d341d0,\n    0x66db40c8, 0xa784392f, 0x004dff2f, 0x2db9d2de, 0x97943fac, 0x4a97c1d8, 0x527644b7, 0xb5f437a7,\n    0xb82cbaef, 0xd751d159, 0x6ff7f0ed, 0x5a097a1f, 0x827b68d0, 0x90ecf52e, 0x22b0c054, 0xbc8e5935,\n    0x4b6d2f7f, 0x50bb64a2, 0xd2664910, 0xbee5812d, 0xb7332290, 0xe93b159f, 0xb48ee411, 0x4bff345d,\n    0xfd45c240, 0xad31973f, 0xc4f6d02e, 0x55fc8165, 0xd5b1caad, 0xa1ac2dae, 0xa2d4b76d, 0xc19b0c50,\n    0x882240f2, 0x0c6e4f38, 0xa4e4bfd7, 0x4f5ba272, 0x564c1d2f, 0xc59c5319, 0xb949e354, 0xb04669fe,\n    0xb1b6ab8a, 0xc71358dd, 0x6385c545, 0x110f935d, 0x57538ad5, 0x6a390493, 0xe63d37e0, 0x2a54f6b3,\n    0x3a787d5f, 0x6276a0b5, 0x19a6fcdf, 0x7a42206a, 0x29f9d4d5, 0xf61b1891, 0xbb72275e, 0xaa508167,\n    0x38901091, 0xc6b505eb, 0x84c7cb8c, 0x2ad75a0f, 0x874a1427, 0xa2d1936b, 0x2ad286af, 0xaa56d291,\n    0xd7894360, 0x425c750d, 0x93b39e26, 0x187184c9, 0x6c00b32d, 0x73e2bb14, 0xa0bebc3c, 0x54623779,\n    0x64459eab, 0x3f328b82, 0x7718cf82, 0x59a2cea6, 0x04ee002e, 0x89fe78e6, 0x3fab0950, 0x325ff6c2,\n    0x81383f05, 0x6963c5c8, 0x76cb5ad6, 0xd49974c9, 0xca180dcf, 0x380782d5, 0xc7fa5cf6, 0x8ac31511,\n    0x35e79e13, 0x47da91d0, 0xf40f9086, 0xa7e2419e, 0x31366241, 0x051ef495, 0xaa573b04, 0x4a805d8d,\n    0x548300d0, 0x00322a3c, 0xbf64cddf, 0xba57a68e, 0x75c6372b, 0x50afd341, 0xa7c13275, 0x915a0bf5,\n    0x6b54bfab, 0x2b0b1426, 0xab4cc9d7, 0x449ccd82, 0xf7fbf265, 0xab85c5f3, 0x1b55db94, 0xaad4e324,\n    0xcfa4bd3f, 0x2deaa3e2, 0x9e204d02, 0xc8bd25ac, 0xeadf55b3, 0xd5bd9e98, 0xe31231b2, 0x2ad5ad6c,\n    0x954329de, 0xadbe4528, 0xd8710f69, 0xaa51c90f, 0xaa786bf6, 0x22513f1e, 0xaa51a79b, 0x2ad344cc,\n    0x7b5a41f0, 0xd37cfbad, 0x1b069505, 0x41ece491, 0xb4c332e6, 0x032268d4, 0xc9600acc, 0xce387e6d,\n    0xbf6bb16c, 0x6a70fb78, 0x0d03d9c9, 0xd4df39de, 0xe01063da, 0x4736f464, 0x5ad328d8, 0xb347cc96,\n    0x75bb0fc3, 0x98511bfb, 0x4ffbcc35, 0xb58bcf6a, 0xe11f0abc, 0xbfc5fe4a, 0xa70aec10, 0xac39570a,\n    0x3f04442f, 0x6188b153, 0xe0397a2e, 0x5727cb79, 0x9ceb418f, 0x1cacd68d, 0x2ad37c96, 0x0175cb9d,\n    0xc69dff09, 0xc75b65f0, 0xd9db40d8, 0xec0e7779, 0x4744ead4, 0xb11c3274, 0xdd24cb9e, 0x7e1c54bd,\n    0xf01144f9, 0xd2240eb1, 0x9675b3fd, 0xa3ac3755, 0xd47c27af, 0x51c85f4d, 0x56907596, 0xa5bb15e6,\n    0x580304f0, 0xca042cf1, 0x011a37ea, 0x8dbfaadb, 0x35ba3e4a, 0x3526ffa0, 0xc37b4d09, 0xbc306ed9,\n    0x98a52666, 0x5648f725, 0xff5e569d, 0x0ced63d0, 0x7c63b2cf, 0x700b45e1, 0xd5ea50f1, 0x85a92872,\n    0xaf1fbda7, 0xd4234870, 0xa7870bf3, 0x2d3b4d79, 0x42e04198, 0x0cd0ede7, 0x26470db8, 0xf881814c,\n    0x474d6ad7, 0x7c0c5e5c, 0xd1231959, 0x381b7298, 0xf5d2f4db, 0xab838653, 0x6e2f1e23, 0x83719c9e,\n    0xbd91e046, 0x9a56456e, 0xdc39200c, 0x20c8c571, 0x962bda1c, 0xe1e696ff, 0xb141ab08, 0x7cca89b9,\n    0x1a69e783, 0x02cc4843, 0xa2f7c579, 0x429ef47d, 0x427b169c, 0x5ac9f049, 0xdd8f0f00, 0x5c8165bf);\n\n  sBox[1] = new Array(\n    0x1f201094, 0xef0ba75b, 0x69e3cf7e, 0x393f4380, 0xfe61cf7a, 0xeec5207a, 0x55889c94, 0x72fc0651,\n    0xada7ef79, 0x4e1d7235, 0xd55a63ce, 0xde0436ba, 0x99c430ef, 0x5f0c0794, 0x18dcdb7d, 0xa1d6eff3,\n    0xa0b52f7b, 0x59e83605, 0xee15b094, 0xe9ffd909, 0xdc440086, 0xef944459, 0xba83ccb3, 0xe0c3cdfb,\n    0xd1da4181, 0x3b092ab1, 0xf997f1c1, 0xa5e6cf7b, 0x01420ddb, 0xe4e7ef5b, 0x25a1ff41, 0xe180f806,\n    0x1fc41080, 0x179bee7a, 0xd37ac6a9, 0xfe5830a4, 0x98de8b7f, 0x77e83f4e, 0x79929269, 0x24fa9f7b,\n    0xe113c85b, 0xacc40083, 0xd7503525, 0xf7ea615f, 0x62143154, 0x0d554b63, 0x5d681121, 0xc866c359,\n    0x3d63cf73, 0xcee234c0, 0xd4d87e87, 0x5c672b21, 0x071f6181, 0x39f7627f, 0x361e3084, 0xe4eb573b,\n    0x602f64a4, 0xd63acd9c, 0x1bbc4635, 0x9e81032d, 0x2701f50c, 0x99847ab4, 0xa0e3df79, 0xba6cf38c,\n    0x10843094, 0x2537a95e, 0xf46f6ffe, 0xa1ff3b1f, 0x208cfb6a, 0x8f458c74, 0xd9e0a227, 0x4ec73a34,\n    0xfc884f69, 0x3e4de8df, 0xef0e0088, 0x3559648d, 0x8a45388c, 0x1d804366, 0x721d9bfd, 0xa58684bb,\n    0xe8256333, 0x844e8212, 0x128d8098, 0xfed33fb4, 0xce280ae1, 0x27e19ba5, 0xd5a6c252, 0xe49754bd,\n    0xc5d655dd, 0xeb667064, 0x77840b4d, 0xa1b6a801, 0x84db26a9, 0xe0b56714, 0x21f043b7, 0xe5d05860,\n    0x54f03084, 0x066ff472, 0xa31aa153, 0xdadc4755, 0xb5625dbf, 0x68561be6, 0x83ca6b94, 0x2d6ed23b,\n    0xeccf01db, 0xa6d3d0ba, 0xb6803d5c, 0xaf77a709, 0x33b4a34c, 0x397bc8d6, 0x5ee22b95, 0x5f0e5304,\n    0x81ed6f61, 0x20e74364, 0xb45e1378, 0xde18639b, 0x881ca122, 0xb96726d1, 0x8049a7e8, 0x22b7da7b,\n    0x5e552d25, 0x5272d237, 0x79d2951c, 0xc60d894c, 0x488cb402, 0x1ba4fe5b, 0xa4b09f6b, 0x1ca815cf,\n    0xa20c3005, 0x8871df63, 0xb9de2fcb, 0x0cc6c9e9, 0x0beeff53, 0xe3214517, 0xb4542835, 0x9f63293c,\n    0xee41e729, 0x6e1d2d7c, 0x50045286, 0x1e6685f3, 0xf33401c6, 0x30a22c95, 0x31a70850, 0x60930f13,\n    0x73f98417, 0xa1269859, 0xec645c44, 0x52c877a9, 0xcdff33a6, 0xa02b1741, 0x7cbad9a2, 0x2180036f,\n    0x50d99c08, 0xcb3f4861, 0xc26bd765, 0x64a3f6ab, 0x80342676, 0x25a75e7b, 0xe4e6d1fc, 0x20c710e6,\n    0xcdf0b680, 0x17844d3b, 0x31eef84d, 0x7e0824e4, 0x2ccb49eb, 0x846a3bae, 0x8ff77888, 0xee5d60f6,\n    0x7af75673, 0x2fdd5cdb, 0xa11631c1, 0x30f66f43, 0xb3faec54, 0x157fd7fa, 0xef8579cc, 0xd152de58,\n    0xdb2ffd5e, 0x8f32ce19, 0x306af97a, 0x02f03ef8, 0x99319ad5, 0xc242fa0f, 0xa7e3ebb0, 0xc68e4906,\n    0xb8da230c, 0x80823028, 0xdcdef3c8, 0xd35fb171, 0x088a1bc8, 0xbec0c560, 0x61a3c9e8, 0xbca8f54d,\n    0xc72feffa, 0x22822e99, 0x82c570b4, 0xd8d94e89, 0x8b1c34bc, 0x301e16e6, 0x273be979, 0xb0ffeaa6,\n    0x61d9b8c6, 0x00b24869, 0xb7ffce3f, 0x08dc283b, 0x43daf65a, 0xf7e19798, 0x7619b72f, 0x8f1c9ba4,\n    0xdc8637a0, 0x16a7d3b1, 0x9fc393b7, 0xa7136eeb, 0xc6bcc63e, 0x1a513742, 0xef6828bc, 0x520365d6,\n    0x2d6a77ab, 0x3527ed4b, 0x821fd216, 0x095c6e2e, 0xdb92f2fb, 0x5eea29cb, 0x145892f5, 0x91584f7f,\n    0x5483697b, 0x2667a8cc, 0x85196048, 0x8c4bacea, 0x833860d4, 0x0d23e0f9, 0x6c387e8a, 0x0ae6d249,\n    0xb284600c, 0xd835731d, 0xdcb1c647, 0xac4c56ea, 0x3ebd81b3, 0x230eabb0, 0x6438bc87, 0xf0b5b1fa,\n    0x8f5ea2b3, 0xfc184642, 0x0a036b7a, 0x4fb089bd, 0x649da589, 0xa345415e, 0x5c038323, 0x3e5d3bb9,\n    0x43d79572, 0x7e6dd07c, 0x06dfdf1e, 0x6c6cc4ef, 0x7160a539, 0x73bfbe70, 0x83877605, 0x4523ecf1);\n\n  sBox[2] = new Array(\n    0x8defc240, 0x25fa5d9f, 0xeb903dbf, 0xe810c907, 0x47607fff, 0x369fe44b, 0x8c1fc644, 0xaececa90,\n    0xbeb1f9bf, 0xeefbcaea, 0xe8cf1950, 0x51df07ae, 0x920e8806, 0xf0ad0548, 0xe13c8d83, 0x927010d5,\n    0x11107d9f, 0x07647db9, 0xb2e3e4d4, 0x3d4f285e, 0xb9afa820, 0xfade82e0, 0xa067268b, 0x8272792e,\n    0x553fb2c0, 0x489ae22b, 0xd4ef9794, 0x125e3fbc, 0x21fffcee, 0x825b1bfd, 0x9255c5ed, 0x1257a240,\n    0x4e1a8302, 0xbae07fff, 0x528246e7, 0x8e57140e, 0x3373f7bf, 0x8c9f8188, 0xa6fc4ee8, 0xc982b5a5,\n    0xa8c01db7, 0x579fc264, 0x67094f31, 0xf2bd3f5f, 0x40fff7c1, 0x1fb78dfc, 0x8e6bd2c1, 0x437be59b,\n    0x99b03dbf, 0xb5dbc64b, 0x638dc0e6, 0x55819d99, 0xa197c81c, 0x4a012d6e, 0xc5884a28, 0xccc36f71,\n    0xb843c213, 0x6c0743f1, 0x8309893c, 0x0feddd5f, 0x2f7fe850, 0xd7c07f7e, 0x02507fbf, 0x5afb9a04,\n    0xa747d2d0, 0x1651192e, 0xaf70bf3e, 0x58c31380, 0x5f98302e, 0x727cc3c4, 0x0a0fb402, 0x0f7fef82,\n    0x8c96fdad, 0x5d2c2aae, 0x8ee99a49, 0x50da88b8, 0x8427f4a0, 0x1eac5790, 0x796fb449, 0x8252dc15,\n    0xefbd7d9b, 0xa672597d, 0xada840d8, 0x45f54504, 0xfa5d7403, 0xe83ec305, 0x4f91751a, 0x925669c2,\n    0x23efe941, 0xa903f12e, 0x60270df2, 0x0276e4b6, 0x94fd6574, 0x927985b2, 0x8276dbcb, 0x02778176,\n    0xf8af918d, 0x4e48f79e, 0x8f616ddf, 0xe29d840e, 0x842f7d83, 0x340ce5c8, 0x96bbb682, 0x93b4b148,\n    0xef303cab, 0x984faf28, 0x779faf9b, 0x92dc560d, 0x224d1e20, 0x8437aa88, 0x7d29dc96, 0x2756d3dc,\n    0x8b907cee, 0xb51fd240, 0xe7c07ce3, 0xe566b4a1, 0xc3e9615e, 0x3cf8209d, 0x6094d1e3, 0xcd9ca341,\n    0x5c76460e, 0x00ea983b, 0xd4d67881, 0xfd47572c, 0xf76cedd9, 0xbda8229c, 0x127dadaa, 0x438a074e,\n    0x1f97c090, 0x081bdb8a, 0x93a07ebe, 0xb938ca15, 0x97b03cff, 0x3dc2c0f8, 0x8d1ab2ec, 0x64380e51,\n    0x68cc7bfb, 0xd90f2788, 0x12490181, 0x5de5ffd4, 0xdd7ef86a, 0x76a2e214, 0xb9a40368, 0x925d958f,\n    0x4b39fffa, 0xba39aee9, 0xa4ffd30b, 0xfaf7933b, 0x6d498623, 0x193cbcfa, 0x27627545, 0x825cf47a,\n    0x61bd8ba0, 0xd11e42d1, 0xcead04f4, 0x127ea392, 0x10428db7, 0x8272a972, 0x9270c4a8, 0x127de50b,\n    0x285ba1c8, 0x3c62f44f, 0x35c0eaa5, 0xe805d231, 0x428929fb, 0xb4fcdf82, 0x4fb66a53, 0x0e7dc15b,\n    0x1f081fab, 0x108618ae, 0xfcfd086d, 0xf9ff2889, 0x694bcc11, 0x236a5cae, 0x12deca4d, 0x2c3f8cc5,\n    0xd2d02dfe, 0xf8ef5896, 0xe4cf52da, 0x95155b67, 0x494a488c, 0xb9b6a80c, 0x5c8f82bc, 0x89d36b45,\n    0x3a609437, 0xec00c9a9, 0x44715253, 0x0a874b49, 0xd773bc40, 0x7c34671c, 0x02717ef6, 0x4feb5536,\n    0xa2d02fff, 0xd2bf60c4, 0xd43f03c0, 0x50b4ef6d, 0x07478cd1, 0x006e1888, 0xa2e53f55, 0xb9e6d4bc,\n    0xa2048016, 0x97573833, 0xd7207d67, 0xde0f8f3d, 0x72f87b33, 0xabcc4f33, 0x7688c55d, 0x7b00a6b0,\n    0x947b0001, 0x570075d2, 0xf9bb88f8, 0x8942019e, 0x4264a5ff, 0x856302e0, 0x72dbd92b, 0xee971b69,\n    0x6ea22fde, 0x5f08ae2b, 0xaf7a616d, 0xe5c98767, 0xcf1febd2, 0x61efc8c2, 0xf1ac2571, 0xcc8239c2,\n    0x67214cb8, 0xb1e583d1, 0xb7dc3e62, 0x7f10bdce, 0xf90a5c38, 0x0ff0443d, 0x606e6dc6, 0x60543a49,\n    0x5727c148, 0x2be98a1d, 0x8ab41738, 0x20e1be24, 0xaf96da0f, 0x68458425, 0x99833be5, 0x600d457d,\n    0x282f9350, 0x8334b362, 0xd91d1120, 0x2b6d8da0, 0x642b1e31, 0x9c305a00, 0x52bce688, 0x1b03588a,\n    0xf7baefd5, 0x4142ed9c, 0xa4315c11, 0x83323ec5, 0xdfef4636, 0xa133c501, 0xe9d3531c, 0xee353783);\n\n  sBox[3] = new Array(\n    0x9db30420, 0x1fb6e9de, 0xa7be7bef, 0xd273a298, 0x4a4f7bdb, 0x64ad8c57, 0x85510443, 0xfa020ed1,\n    0x7e287aff, 0xe60fb663, 0x095f35a1, 0x79ebf120, 0xfd059d43, 0x6497b7b1, 0xf3641f63, 0x241e4adf,\n    0x28147f5f, 0x4fa2b8cd, 0xc9430040, 0x0cc32220, 0xfdd30b30, 0xc0a5374f, 0x1d2d00d9, 0x24147b15,\n    0xee4d111a, 0x0fca5167, 0x71ff904c, 0x2d195ffe, 0x1a05645f, 0x0c13fefe, 0x081b08ca, 0x05170121,\n    0x80530100, 0xe83e5efe, 0xac9af4f8, 0x7fe72701, 0xd2b8ee5f, 0x06df4261, 0xbb9e9b8a, 0x7293ea25,\n    0xce84ffdf, 0xf5718801, 0x3dd64b04, 0xa26f263b, 0x7ed48400, 0x547eebe6, 0x446d4ca0, 0x6cf3d6f5,\n    0x2649abdf, 0xaea0c7f5, 0x36338cc1, 0x503f7e93, 0xd3772061, 0x11b638e1, 0x72500e03, 0xf80eb2bb,\n    0xabe0502e, 0xec8d77de, 0x57971e81, 0xe14f6746, 0xc9335400, 0x6920318f, 0x081dbb99, 0xffc304a5,\n    0x4d351805, 0x7f3d5ce3, 0xa6c866c6, 0x5d5bcca9, 0xdaec6fea, 0x9f926f91, 0x9f46222f, 0x3991467d,\n    0xa5bf6d8e, 0x1143c44f, 0x43958302, 0xd0214eeb, 0x022083b8, 0x3fb6180c, 0x18f8931e, 0x281658e6,\n    0x26486e3e, 0x8bd78a70, 0x7477e4c1, 0xb506e07c, 0xf32d0a25, 0x79098b02, 0xe4eabb81, 0x28123b23,\n    0x69dead38, 0x1574ca16, 0xdf871b62, 0x211c40b7, 0xa51a9ef9, 0x0014377b, 0x041e8ac8, 0x09114003,\n    0xbd59e4d2, 0xe3d156d5, 0x4fe876d5, 0x2f91a340, 0x557be8de, 0x00eae4a7, 0x0ce5c2ec, 0x4db4bba6,\n    0xe756bdff, 0xdd3369ac, 0xec17b035, 0x06572327, 0x99afc8b0, 0x56c8c391, 0x6b65811c, 0x5e146119,\n    0x6e85cb75, 0xbe07c002, 0xc2325577, 0x893ff4ec, 0x5bbfc92d, 0xd0ec3b25, 0xb7801ab7, 0x8d6d3b24,\n    0x20c763ef, 0xc366a5fc, 0x9c382880, 0x0ace3205, 0xaac9548a, 0xeca1d7c7, 0x041afa32, 0x1d16625a,\n    0x6701902c, 0x9b757a54, 0x31d477f7, 0x9126b031, 0x36cc6fdb, 0xc70b8b46, 0xd9e66a48, 0x56e55a79,\n    0x026a4ceb, 0x52437eff, 0x2f8f76b4, 0x0df980a5, 0x8674cde3, 0xedda04eb, 0x17a9be04, 0x2c18f4df,\n    0xb7747f9d, 0xab2af7b4, 0xefc34d20, 0x2e096b7c, 0x1741a254, 0xe5b6a035, 0x213d42f6, 0x2c1c7c26,\n    0x61c2f50f, 0x6552daf9, 0xd2c231f8, 0x25130f69, 0xd8167fa2, 0x0418f2c8, 0x001a96a6, 0x0d1526ab,\n    0x63315c21, 0x5e0a72ec, 0x49bafefd, 0x187908d9, 0x8d0dbd86, 0x311170a7, 0x3e9b640c, 0xcc3e10d7,\n    0xd5cad3b6, 0x0caec388, 0xf73001e1, 0x6c728aff, 0x71eae2a1, 0x1f9af36e, 0xcfcbd12f, 0xc1de8417,\n    0xac07be6b, 0xcb44a1d8, 0x8b9b0f56, 0x013988c3, 0xb1c52fca, 0xb4be31cd, 0xd8782806, 0x12a3a4e2,\n    0x6f7de532, 0x58fd7eb6, 0xd01ee900, 0x24adffc2, 0xf4990fc5, 0x9711aac5, 0x001d7b95, 0x82e5e7d2,\n    0x109873f6, 0x00613096, 0xc32d9521, 0xada121ff, 0x29908415, 0x7fbb977f, 0xaf9eb3db, 0x29c9ed2a,\n    0x5ce2a465, 0xa730f32c, 0xd0aa3fe8, 0x8a5cc091, 0xd49e2ce7, 0x0ce454a9, 0xd60acd86, 0x015f1919,\n    0x77079103, 0xdea03af6, 0x78a8565e, 0xdee356df, 0x21f05cbe, 0x8b75e387, 0xb3c50651, 0xb8a5c3ef,\n    0xd8eeb6d2, 0xe523be77, 0xc2154529, 0x2f69efdf, 0xafe67afb, 0xf470c4b2, 0xf3e0eb5b, 0xd6cc9876,\n    0x39e4460c, 0x1fda8538, 0x1987832f, 0xca007367, 0xa99144f8, 0x296b299e, 0x492fc295, 0x9266beab,\n    0xb5676e69, 0x9bd3ddda, 0xdf7e052f, 0xdb25701c, 0x1b5e51ee, 0xf65324e6, 0x6afce36c, 0x0316cc04,\n    0x8644213e, 0xb7dc59d0, 0x7965291f, 0xccd6fd43, 0x41823979, 0x932bcdf6, 0xb657c34d, 0x4edfd282,\n    0x7ae5290c, 0x3cb9536b, 0x851e20fe, 0x9833557e, 0x13ecf0b0, 0xd3ffb372, 0x3f85c5c1, 0x0aef7ed2);\n\n  sBox[4] = new Array(\n    0x7ec90c04, 0x2c6e74b9, 0x9b0e66df, 0xa6337911, 0xb86a7fff, 0x1dd358f5, 0x44dd9d44, 0x1731167f,\n    0x08fbf1fa, 0xe7f511cc, 0xd2051b00, 0x735aba00, 0x2ab722d8, 0x386381cb, 0xacf6243a, 0x69befd7a,\n    0xe6a2e77f, 0xf0c720cd, 0xc4494816, 0xccf5c180, 0x38851640, 0x15b0a848, 0xe68b18cb, 0x4caadeff,\n    0x5f480a01, 0x0412b2aa, 0x259814fc, 0x41d0efe2, 0x4e40b48d, 0x248eb6fb, 0x8dba1cfe, 0x41a99b02,\n    0x1a550a04, 0xba8f65cb, 0x7251f4e7, 0x95a51725, 0xc106ecd7, 0x97a5980a, 0xc539b9aa, 0x4d79fe6a,\n    0xf2f3f763, 0x68af8040, 0xed0c9e56, 0x11b4958b, 0xe1eb5a88, 0x8709e6b0, 0xd7e07156, 0x4e29fea7,\n    0x6366e52d, 0x02d1c000, 0xc4ac8e05, 0x9377f571, 0x0c05372a, 0x578535f2, 0x2261be02, 0xd642a0c9,\n    0xdf13a280, 0x74b55bd2, 0x682199c0, 0xd421e5ec, 0x53fb3ce8, 0xc8adedb3, 0x28a87fc9, 0x3d959981,\n    0x5c1ff900, 0xfe38d399, 0x0c4eff0b, 0x062407ea, 0xaa2f4fb1, 0x4fb96976, 0x90c79505, 0xb0a8a774,\n    0xef55a1ff, 0xe59ca2c2, 0xa6b62d27, 0xe66a4263, 0xdf65001f, 0x0ec50966, 0xdfdd55bc, 0x29de0655,\n    0x911e739a, 0x17af8975, 0x32c7911c, 0x89f89468, 0x0d01e980, 0x524755f4, 0x03b63cc9, 0x0cc844b2,\n    0xbcf3f0aa, 0x87ac36e9, 0xe53a7426, 0x01b3d82b, 0x1a9e7449, 0x64ee2d7e, 0xcddbb1da, 0x01c94910,\n    0xb868bf80, 0x0d26f3fd, 0x9342ede7, 0x04a5c284, 0x636737b6, 0x50f5b616, 0xf24766e3, 0x8eca36c1,\n    0x136e05db, 0xfef18391, 0xfb887a37, 0xd6e7f7d4, 0xc7fb7dc9, 0x3063fcdf, 0xb6f589de, 0xec2941da,\n    0x26e46695, 0xb7566419, 0xf654efc5, 0xd08d58b7, 0x48925401, 0xc1bacb7f, 0xe5ff550f, 0xb6083049,\n    0x5bb5d0e8, 0x87d72e5a, 0xab6a6ee1, 0x223a66ce, 0xc62bf3cd, 0x9e0885f9, 0x68cb3e47, 0x086c010f,\n    0xa21de820, 0xd18b69de, 0xf3f65777, 0xfa02c3f6, 0x407edac3, 0xcbb3d550, 0x1793084d, 0xb0d70eba,\n    0x0ab378d5, 0xd951fb0c, 0xded7da56, 0x4124bbe4, 0x94ca0b56, 0x0f5755d1, 0xe0e1e56e, 0x6184b5be,\n    0x580a249f, 0x94f74bc0, 0xe327888e, 0x9f7b5561, 0xc3dc0280, 0x05687715, 0x646c6bd7, 0x44904db3,\n    0x66b4f0a3, 0xc0f1648a, 0x697ed5af, 0x49e92ff6, 0x309e374f, 0x2cb6356a, 0x85808573, 0x4991f840,\n    0x76f0ae02, 0x083be84d, 0x28421c9a, 0x44489406, 0x736e4cb8, 0xc1092910, 0x8bc95fc6, 0x7d869cf4,\n    0x134f616f, 0x2e77118d, 0xb31b2be1, 0xaa90b472, 0x3ca5d717, 0x7d161bba, 0x9cad9010, 0xaf462ba2,\n    0x9fe459d2, 0x45d34559, 0xd9f2da13, 0xdbc65487, 0xf3e4f94e, 0x176d486f, 0x097c13ea, 0x631da5c7,\n    0x445f7382, 0x175683f4, 0xcdc66a97, 0x70be0288, 0xb3cdcf72, 0x6e5dd2f3, 0x20936079, 0x459b80a5,\n    0xbe60e2db, 0xa9c23101, 0xeba5315c, 0x224e42f2, 0x1c5c1572, 0xf6721b2c, 0x1ad2fff3, 0x8c25404e,\n    0x324ed72f, 0x4067b7fd, 0x0523138e, 0x5ca3bc78, 0xdc0fd66e, 0x75922283, 0x784d6b17, 0x58ebb16e,\n    0x44094f85, 0x3f481d87, 0xfcfeae7b, 0x77b5ff76, 0x8c2302bf, 0xaaf47556, 0x5f46b02a, 0x2b092801,\n    0x3d38f5f7, 0x0ca81f36, 0x52af4a8a, 0x66d5e7c0, 0xdf3b0874, 0x95055110, 0x1b5ad7a8, 0xf61ed5ad,\n    0x6cf6e479, 0x20758184, 0xd0cefa65, 0x88f7be58, 0x4a046826, 0x0ff6f8f3, 0xa09c7f70, 0x5346aba0,\n    0x5ce96c28, 0xe176eda3, 0x6bac307f, 0x376829d2, 0x85360fa9, 0x17e3fe2a, 0x24b79767, 0xf5a96b20,\n    0xd6cd2595, 0x68ff1ebf, 0x7555442c, 0xf19f06be, 0xf9e0659a, 0xeeb9491d, 0x34010718, 0xbb30cab8,\n    0xe822fe15, 0x88570983, 0x750e6249, 0xda627e55, 0x5e76ffa8, 0xb1534546, 0x6d47de08, 0xefe9e7d4);\n\n  sBox[5] = new Array(\n    0xf6fa8f9d, 0x2cac6ce1, 0x4ca34867, 0xe2337f7c, 0x95db08e7, 0x016843b4, 0xeced5cbc, 0x325553ac,\n    0xbf9f0960, 0xdfa1e2ed, 0x83f0579d, 0x63ed86b9, 0x1ab6a6b8, 0xde5ebe39, 0xf38ff732, 0x8989b138,\n    0x33f14961, 0xc01937bd, 0xf506c6da, 0xe4625e7e, 0xa308ea99, 0x4e23e33c, 0x79cbd7cc, 0x48a14367,\n    0xa3149619, 0xfec94bd5, 0xa114174a, 0xeaa01866, 0xa084db2d, 0x09a8486f, 0xa888614a, 0x2900af98,\n    0x01665991, 0xe1992863, 0xc8f30c60, 0x2e78ef3c, 0xd0d51932, 0xcf0fec14, 0xf7ca07d2, 0xd0a82072,\n    0xfd41197e, 0x9305a6b0, 0xe86be3da, 0x74bed3cd, 0x372da53c, 0x4c7f4448, 0xdab5d440, 0x6dba0ec3,\n    0x083919a7, 0x9fbaeed9, 0x49dbcfb0, 0x4e670c53, 0x5c3d9c01, 0x64bdb941, 0x2c0e636a, 0xba7dd9cd,\n    0xea6f7388, 0xe70bc762, 0x35f29adb, 0x5c4cdd8d, 0xf0d48d8c, 0xb88153e2, 0x08a19866, 0x1ae2eac8,\n    0x284caf89, 0xaa928223, 0x9334be53, 0x3b3a21bf, 0x16434be3, 0x9aea3906, 0xefe8c36e, 0xf890cdd9,\n    0x80226dae, 0xc340a4a3, 0xdf7e9c09, 0xa694a807, 0x5b7c5ecc, 0x221db3a6, 0x9a69a02f, 0x68818a54,\n    0xceb2296f, 0x53c0843a, 0xfe893655, 0x25bfe68a, 0xb4628abc, 0xcf222ebf, 0x25ac6f48, 0xa9a99387,\n    0x53bddb65, 0xe76ffbe7, 0xe967fd78, 0x0ba93563, 0x8e342bc1, 0xe8a11be9, 0x4980740d, 0xc8087dfc,\n    0x8de4bf99, 0xa11101a0, 0x7fd37975, 0xda5a26c0, 0xe81f994f, 0x9528cd89, 0xfd339fed, 0xb87834bf,\n    0x5f04456d, 0x22258698, 0xc9c4c83b, 0x2dc156be, 0x4f628daa, 0x57f55ec5, 0xe2220abe, 0xd2916ebf,\n    0x4ec75b95, 0x24f2c3c0, 0x42d15d99, 0xcd0d7fa0, 0x7b6e27ff, 0xa8dc8af0, 0x7345c106, 0xf41e232f,\n    0x35162386, 0xe6ea8926, 0x3333b094, 0x157ec6f2, 0x372b74af, 0x692573e4, 0xe9a9d848, 0xf3160289,\n    0x3a62ef1d, 0xa787e238, 0xf3a5f676, 0x74364853, 0x20951063, 0x4576698d, 0xb6fad407, 0x592af950,\n    0x36f73523, 0x4cfb6e87, 0x7da4cec0, 0x6c152daa, 0xcb0396a8, 0xc50dfe5d, 0xfcd707ab, 0x0921c42f,\n    0x89dff0bb, 0x5fe2be78, 0x448f4f33, 0x754613c9, 0x2b05d08d, 0x48b9d585, 0xdc049441, 0xc8098f9b,\n    0x7dede786, 0xc39a3373, 0x42410005, 0x6a091751, 0x0ef3c8a6, 0x890072d6, 0x28207682, 0xa9a9f7be,\n    0xbf32679d, 0xd45b5b75, 0xb353fd00, 0xcbb0e358, 0x830f220a, 0x1f8fb214, 0xd372cf08, 0xcc3c4a13,\n    0x8cf63166, 0x061c87be, 0x88c98f88, 0x6062e397, 0x47cf8e7a, 0xb6c85283, 0x3cc2acfb, 0x3fc06976,\n    0x4e8f0252, 0x64d8314d, 0xda3870e3, 0x1e665459, 0xc10908f0, 0x513021a5, 0x6c5b68b7, 0x822f8aa0,\n    0x3007cd3e, 0x74719eef, 0xdc872681, 0x073340d4, 0x7e432fd9, 0x0c5ec241, 0x8809286c, 0xf592d891,\n    0x08a930f6, 0x957ef305, 0xb7fbffbd, 0xc266e96f, 0x6fe4ac98, 0xb173ecc0, 0xbc60b42a, 0x953498da,\n    0xfba1ae12, 0x2d4bd736, 0x0f25faab, 0xa4f3fceb, 0xe2969123, 0x257f0c3d, 0x9348af49, 0x361400bc,\n    0xe8816f4a, 0x3814f200, 0xa3f94043, 0x9c7a54c2, 0xbc704f57, 0xda41e7f9, 0xc25ad33a, 0x54f4a084,\n    0xb17f5505, 0x59357cbe, 0xedbd15c8, 0x7f97c5ab, 0xba5ac7b5, 0xb6f6deaf, 0x3a479c3a, 0x5302da25,\n    0x653d7e6a, 0x54268d49, 0x51a477ea, 0x5017d55b, 0xd7d25d88, 0x44136c76, 0x0404a8c8, 0xb8e5a121,\n    0xb81a928a, 0x60ed5869, 0x97c55b96, 0xeaec991b, 0x29935913, 0x01fdb7f1, 0x088e8dfa, 0x9ab6f6f5,\n    0x3b4cbf9f, 0x4a5de3ab, 0xe6051d35, 0xa0e1d855, 0xd36b4cf1, 0xf544edeb, 0xb0e93524, 0xbebb8fbd,\n    0xa2d762cf, 0x49c92f54, 0x38b5f331, 0x7128a454, 0x48392905, 0xa65b1db8, 0x851c97bd, 0xd675cf2f);\n\n  sBox[6] = new Array(\n    0x85e04019, 0x332bf567, 0x662dbfff, 0xcfc65693, 0x2a8d7f6f, 0xab9bc912, 0xde6008a1, 0x2028da1f,\n    0x0227bce7, 0x4d642916, 0x18fac300, 0x50f18b82, 0x2cb2cb11, 0xb232e75c, 0x4b3695f2, 0xb28707de,\n    0xa05fbcf6, 0xcd4181e9, 0xe150210c, 0xe24ef1bd, 0xb168c381, 0xfde4e789, 0x5c79b0d8, 0x1e8bfd43,\n    0x4d495001, 0x38be4341, 0x913cee1d, 0x92a79c3f, 0x089766be, 0xbaeeadf4, 0x1286becf, 0xb6eacb19,\n    0x2660c200, 0x7565bde4, 0x64241f7a, 0x8248dca9, 0xc3b3ad66, 0x28136086, 0x0bd8dfa8, 0x356d1cf2,\n    0x107789be, 0xb3b2e9ce, 0x0502aa8f, 0x0bc0351e, 0x166bf52a, 0xeb12ff82, 0xe3486911, 0xd34d7516,\n    0x4e7b3aff, 0x5f43671b, 0x9cf6e037, 0x4981ac83, 0x334266ce, 0x8c9341b7, 0xd0d854c0, 0xcb3a6c88,\n    0x47bc2829, 0x4725ba37, 0xa66ad22b, 0x7ad61f1e, 0x0c5cbafa, 0x4437f107, 0xb6e79962, 0x42d2d816,\n    0x0a961288, 0xe1a5c06e, 0x13749e67, 0x72fc081a, 0xb1d139f7, 0xf9583745, 0xcf19df58, 0xbec3f756,\n    0xc06eba30, 0x07211b24, 0x45c28829, 0xc95e317f, 0xbc8ec511, 0x38bc46e9, 0xc6e6fa14, 0xbae8584a,\n    0xad4ebc46, 0x468f508b, 0x7829435f, 0xf124183b, 0x821dba9f, 0xaff60ff4, 0xea2c4e6d, 0x16e39264,\n    0x92544a8b, 0x009b4fc3, 0xaba68ced, 0x9ac96f78, 0x06a5b79a, 0xb2856e6e, 0x1aec3ca9, 0xbe838688,\n    0x0e0804e9, 0x55f1be56, 0xe7e5363b, 0xb3a1f25d, 0xf7debb85, 0x61fe033c, 0x16746233, 0x3c034c28,\n    0xda6d0c74, 0x79aac56c, 0x3ce4e1ad, 0x51f0c802, 0x98f8f35a, 0x1626a49f, 0xeed82b29, 0x1d382fe3,\n    0x0c4fb99a, 0xbb325778, 0x3ec6d97b, 0x6e77a6a9, 0xcb658b5c, 0xd45230c7, 0x2bd1408b, 0x60c03eb7,\n    0xb9068d78, 0xa33754f4, 0xf430c87d, 0xc8a71302, 0xb96d8c32, 0xebd4e7be, 0xbe8b9d2d, 0x7979fb06,\n    0xe7225308, 0x8b75cf77, 0x11ef8da4, 0xe083c858, 0x8d6b786f, 0x5a6317a6, 0xfa5cf7a0, 0x5dda0033,\n    0xf28ebfb0, 0xf5b9c310, 0xa0eac280, 0x08b9767a, 0xa3d9d2b0, 0x79d34217, 0x021a718d, 0x9ac6336a,\n    0x2711fd60, 0x438050e3, 0x069908a8, 0x3d7fedc4, 0x826d2bef, 0x4eeb8476, 0x488dcf25, 0x36c9d566,\n    0x28e74e41, 0xc2610aca, 0x3d49a9cf, 0xbae3b9df, 0xb65f8de6, 0x92aeaf64, 0x3ac7d5e6, 0x9ea80509,\n    0xf22b017d, 0xa4173f70, 0xdd1e16c3, 0x15e0d7f9, 0x50b1b887, 0x2b9f4fd5, 0x625aba82, 0x6a017962,\n    0x2ec01b9c, 0x15488aa9, 0xd716e740, 0x40055a2c, 0x93d29a22, 0xe32dbf9a, 0x058745b9, 0x3453dc1e,\n    0xd699296e, 0x496cff6f, 0x1c9f4986, 0xdfe2ed07, 0xb87242d1, 0x19de7eae, 0x053e561a, 0x15ad6f8c,\n    0x66626c1c, 0x7154c24c, 0xea082b2a, 0x93eb2939, 0x17dcb0f0, 0x58d4f2ae, 0x9ea294fb, 0x52cf564c,\n    0x9883fe66, 0x2ec40581, 0x763953c3, 0x01d6692e, 0xd3a0c108, 0xa1e7160e, 0xe4f2dfa6, 0x693ed285,\n    0x74904698, 0x4c2b0edd, 0x4f757656, 0x5d393378, 0xa132234f, 0x3d321c5d, 0xc3f5e194, 0x4b269301,\n    0xc79f022f, 0x3c997e7e, 0x5e4f9504, 0x3ffafbbd, 0x76f7ad0e, 0x296693f4, 0x3d1fce6f, 0xc61e45be,\n    0xd3b5ab34, 0xf72bf9b7, 0x1b0434c0, 0x4e72b567, 0x5592a33d, 0xb5229301, 0xcfd2a87f, 0x60aeb767,\n    0x1814386b, 0x30bcc33d, 0x38a0c07d, 0xfd1606f2, 0xc363519b, 0x589dd390, 0x5479f8e6, 0x1cb8d647,\n    0x97fd61a9, 0xea7759f4, 0x2d57539d, 0x569a58cf, 0xe84e63ad, 0x462e1b78, 0x6580f87e, 0xf3817914,\n    0x91da55f4, 0x40a230f3, 0xd1988f35, 0xb6e318d2, 0x3ffa50bc, 0x3d40f021, 0xc3c0bdae, 0x4958c24c,\n    0x518f36b2, 0x84b1d370, 0x0fedce83, 0x878ddada, 0xf2a279c7, 0x94e01be8, 0x90716f4b, 0x954b8aa3);\n\n  sBox[7] = new Array(\n    0xe216300d, 0xbbddfffc, 0xa7ebdabd, 0x35648095, 0x7789f8b7, 0xe6c1121b, 0x0e241600, 0x052ce8b5,\n    0x11a9cfb0, 0xe5952f11, 0xece7990a, 0x9386d174, 0x2a42931c, 0x76e38111, 0xb12def3a, 0x37ddddfc,\n    0xde9adeb1, 0x0a0cc32c, 0xbe197029, 0x84a00940, 0xbb243a0f, 0xb4d137cf, 0xb44e79f0, 0x049eedfd,\n    0x0b15a15d, 0x480d3168, 0x8bbbde5a, 0x669ded42, 0xc7ece831, 0x3f8f95e7, 0x72df191b, 0x7580330d,\n    0x94074251, 0x5c7dcdfa, 0xabbe6d63, 0xaa402164, 0xb301d40a, 0x02e7d1ca, 0x53571dae, 0x7a3182a2,\n    0x12a8ddec, 0xfdaa335d, 0x176f43e8, 0x71fb46d4, 0x38129022, 0xce949ad4, 0xb84769ad, 0x965bd862,\n    0x82f3d055, 0x66fb9767, 0x15b80b4e, 0x1d5b47a0, 0x4cfde06f, 0xc28ec4b8, 0x57e8726e, 0x647a78fc,\n    0x99865d44, 0x608bd593, 0x6c200e03, 0x39dc5ff6, 0x5d0b00a3, 0xae63aff2, 0x7e8bd632, 0x70108c0c,\n    0xbbd35049, 0x2998df04, 0x980cf42a, 0x9b6df491, 0x9e7edd53, 0x06918548, 0x58cb7e07, 0x3b74ef2e,\n    0x522fffb1, 0xd24708cc, 0x1c7e27cd, 0xa4eb215b, 0x3cf1d2e2, 0x19b47a38, 0x424f7618, 0x35856039,\n    0x9d17dee7, 0x27eb35e6, 0xc9aff67b, 0x36baf5b8, 0x09c467cd, 0xc18910b1, 0xe11dbf7b, 0x06cd1af8,\n    0x7170c608, 0x2d5e3354, 0xd4de495a, 0x64c6d006, 0xbcc0c62c, 0x3dd00db3, 0x708f8f34, 0x77d51b42,\n    0x264f620f, 0x24b8d2bf, 0x15c1b79e, 0x46a52564, 0xf8d7e54e, 0x3e378160, 0x7895cda5, 0x859c15a5,\n    0xe6459788, 0xc37bc75f, 0xdb07ba0c, 0x0676a3ab, 0x7f229b1e, 0x31842e7b, 0x24259fd7, 0xf8bef472,\n    0x835ffcb8, 0x6df4c1f2, 0x96f5b195, 0xfd0af0fc, 0xb0fe134c, 0xe2506d3d, 0x4f9b12ea, 0xf215f225,\n    0xa223736f, 0x9fb4c428, 0x25d04979, 0x34c713f8, 0xc4618187, 0xea7a6e98, 0x7cd16efc, 0x1436876c,\n    0xf1544107, 0xbedeee14, 0x56e9af27, 0xa04aa441, 0x3cf7c899, 0x92ecbae6, 0xdd67016d, 0x151682eb,\n    0xa842eedf, 0xfdba60b4, 0xf1907b75, 0x20e3030f, 0x24d8c29e, 0xe139673b, 0xefa63fb8, 0x71873054,\n    0xb6f2cf3b, 0x9f326442, 0xcb15a4cc, 0xb01a4504, 0xf1e47d8d, 0x844a1be5, 0xbae7dfdc, 0x42cbda70,\n    0xcd7dae0a, 0x57e85b7a, 0xd53f5af6, 0x20cf4d8c, 0xcea4d428, 0x79d130a4, 0x3486ebfb, 0x33d3cddc,\n    0x77853b53, 0x37effcb5, 0xc5068778, 0xe580b3e6, 0x4e68b8f4, 0xc5c8b37e, 0x0d809ea2, 0x398feb7c,\n    0x132a4f94, 0x43b7950e, 0x2fee7d1c, 0x223613bd, 0xdd06caa2, 0x37df932b, 0xc4248289, 0xacf3ebc3,\n    0x5715f6b7, 0xef3478dd, 0xf267616f, 0xc148cbe4, 0x9052815e, 0x5e410fab, 0xb48a2465, 0x2eda7fa4,\n    0xe87b40e4, 0xe98ea084, 0x5889e9e1, 0xefd390fc, 0xdd07d35b, 0xdb485694, 0x38d7e5b2, 0x57720101,\n    0x730edebc, 0x5b643113, 0x94917e4f, 0x503c2fba, 0x646f1282, 0x7523d24a, 0xe0779695, 0xf9c17a8f,\n    0x7a5b2121, 0xd187b896, 0x29263a4d, 0xba510cdf, 0x81f47c9f, 0xad1163ed, 0xea7b5965, 0x1a00726e,\n    0x11403092, 0x00da6d77, 0x4a0cdd61, 0xad1f4603, 0x605bdfb0, 0x9eedc364, 0x22ebe6a8, 0xcee7d28a,\n    0xa0e736a0, 0x5564a6b9, 0x10853209, 0xc7eb8f37, 0x2de705ca, 0x8951570f, 0xdf09822b, 0xbd691a6c,\n    0xaa12e4f2, 0x87451c0f, 0xe0f6a27a, 0x3ada4819, 0x4cf1764f, 0x0d771c2b, 0x67cdb156, 0x350d8384,\n    0x5938fa0f, 0x42399ef3, 0x36997b07, 0x0e84093d, 0x4aa93e61, 0x8360d87b, 0x1fa98b0c, 0x1149382c,\n    0xe97625a5, 0x0614d1b7, 0x0e25244b, 0x0c768347, 0x589e8d82, 0x0d2059d1, 0xa466bb1e, 0xf8da0a82,\n    0x04f19130, 0xba6e4ec0, 0x99265164, 0x1ee7230d, 0x50b2ad80, 0xeaee6801, 0x8db2a283, 0xea8bf59e);\n\n};\n\nvar util = require('../../util');\n\nfunction cast5(key) {\n  this.cast5 = new openpgp_symenc_cast5();\n  this.cast5.setKey(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.cast5.encrypt(block);\n  }\n}\n\nmodule.exports = cast5;\nmodule.exports.blockSize = cast5.prototype.blockSize = 8;\nmodule.exports.keySize = cast5.prototype.keySize = 16;\n","//Paul Tero, July 2001\n//http://www.tero.co.uk/des/\n//\n//Optimised for performance with large blocks by Michael Hayworth, November 2001\n//http://www.netdealing.com\n//\n// Modified by Recurity Labs GmbH\n\n//THIS SOFTWARE IS PROVIDED \"AS IS\" AND\n//ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE\n//IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE\n//ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE\n//FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL\n//DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS\n//OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)\n//HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT\n//LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY\n//OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF\n//SUCH DAMAGE.\n\n//des\n//this takes the key, the message, and whether to encrypt or decrypt\n\n\n\n\nfunction des(keys, message, encrypt, mode, iv, padding) {\n  //declaring this locally speeds things up a bit\n  var spfunction1 = new Array(0x1010400, 0, 0x10000, 0x1010404, 0x1010004, 0x10404, 0x4, 0x10000, 0x400, 0x1010400,\n    0x1010404, 0x400, 0x1000404, 0x1010004, 0x1000000, 0x4, 0x404, 0x1000400, 0x1000400, 0x10400, 0x10400, 0x1010000,\n    0x1010000, 0x1000404, 0x10004, 0x1000004, 0x1000004, 0x10004, 0, 0x404, 0x10404, 0x1000000, 0x10000, 0x1010404, 0x4,\n    0x1010000, 0x1010400, 0x1000000, 0x1000000, 0x400, 0x1010004, 0x10000, 0x10400, 0x1000004, 0x400, 0x4, 0x1000404,\n    0x10404, 0x1010404, 0x10004, 0x1010000, 0x1000404, 0x1000004, 0x404, 0x10404, 0x1010400, 0x404, 0x1000400,\n    0x1000400, 0, 0x10004, 0x10400, 0, 0x1010004);\n  var spfunction2 = new Array(-0x7fef7fe0, -0x7fff8000, 0x8000, 0x108020, 0x100000, 0x20, -0x7fefffe0, -0x7fff7fe0, -\n    0x7fffffe0, -0x7fef7fe0, -0x7fef8000, -0x80000000, -0x7fff8000, 0x100000, 0x20, -0x7fefffe0, 0x108000, 0x100020, -\n    0x7fff7fe0, 0, -0x80000000, 0x8000, 0x108020, -0x7ff00000, 0x100020, -0x7fffffe0, 0, 0x108000, 0x8020, -0x7fef8000, -\n    0x7ff00000, 0x8020, 0, 0x108020, -0x7fefffe0, 0x100000, -0x7fff7fe0, -0x7ff00000, -0x7fef8000, 0x8000, -0x7ff00000, -\n    0x7fff8000, 0x20, -0x7fef7fe0, 0x108020, 0x20, 0x8000, -0x80000000, 0x8020, -0x7fef8000, 0x100000, -0x7fffffe0,\n    0x100020, -0x7fff7fe0, -0x7fffffe0, 0x100020, 0x108000, 0, -0x7fff8000, 0x8020, -0x80000000, -0x7fefffe0, -\n    0x7fef7fe0, 0x108000);\n  var spfunction3 = new Array(0x208, 0x8020200, 0, 0x8020008, 0x8000200, 0, 0x20208, 0x8000200, 0x20008, 0x8000008,\n    0x8000008, 0x20000, 0x8020208, 0x20008, 0x8020000, 0x208, 0x8000000, 0x8, 0x8020200, 0x200, 0x20200, 0x8020000,\n    0x8020008, 0x20208, 0x8000208, 0x20200, 0x20000, 0x8000208, 0x8, 0x8020208, 0x200, 0x8000000, 0x8020200, 0x8000000,\n    0x20008, 0x208, 0x20000, 0x8020200, 0x8000200, 0, 0x200, 0x20008, 0x8020208, 0x8000200, 0x8000008, 0x200, 0,\n    0x8020008, 0x8000208, 0x20000, 0x8000000, 0x8020208, 0x8, 0x20208, 0x20200, 0x8000008, 0x8020000, 0x8000208, 0x208,\n    0x8020000, 0x20208, 0x8, 0x8020008, 0x20200);\n  var spfunction4 = new Array(0x802001, 0x2081, 0x2081, 0x80, 0x802080, 0x800081, 0x800001, 0x2001, 0, 0x802000,\n    0x802000, 0x802081, 0x81, 0, 0x800080, 0x800001, 0x1, 0x2000, 0x800000, 0x802001, 0x80, 0x800000, 0x2001, 0x2080,\n    0x800081, 0x1, 0x2080, 0x800080, 0x2000, 0x802080, 0x802081, 0x81, 0x800080, 0x800001, 0x802000, 0x802081, 0x81, 0,\n    0, 0x802000, 0x2080, 0x800080, 0x800081, 0x1, 0x802001, 0x2081, 0x2081, 0x80, 0x802081, 0x81, 0x1, 0x2000, 0x800001,\n    0x2001, 0x802080, 0x800081, 0x2001, 0x2080, 0x800000, 0x802001, 0x80, 0x800000, 0x2000, 0x802080);\n  var spfunction5 = new Array(0x100, 0x2080100, 0x2080000, 0x42000100, 0x80000, 0x100, 0x40000000, 0x2080000,\n    0x40080100, 0x80000, 0x2000100, 0x40080100, 0x42000100, 0x42080000, 0x80100, 0x40000000, 0x2000000, 0x40080000,\n    0x40080000, 0, 0x40000100, 0x42080100, 0x42080100, 0x2000100, 0x42080000, 0x40000100, 0, 0x42000000, 0x2080100,\n    0x2000000, 0x42000000, 0x80100, 0x80000, 0x42000100, 0x100, 0x2000000, 0x40000000, 0x2080000, 0x42000100,\n    0x40080100, 0x2000100, 0x40000000, 0x42080000, 0x2080100, 0x40080100, 0x100, 0x2000000, 0x42080000, 0x42080100,\n    0x80100, 0x42000000, 0x42080100, 0x2080000, 0, 0x40080000, 0x42000000, 0x80100, 0x2000100, 0x40000100, 0x80000, 0,\n    0x40080000, 0x2080100, 0x40000100);\n  var spfunction6 = new Array(0x20000010, 0x20400000, 0x4000, 0x20404010, 0x20400000, 0x10, 0x20404010, 0x400000,\n    0x20004000, 0x404010, 0x400000, 0x20000010, 0x400010, 0x20004000, 0x20000000, 0x4010, 0, 0x400010, 0x20004010,\n    0x4000, 0x404000, 0x20004010, 0x10, 0x20400010, 0x20400010, 0, 0x404010, 0x20404000, 0x4010, 0x404000, 0x20404000,\n    0x20000000, 0x20004000, 0x10, 0x20400010, 0x404000, 0x20404010, 0x400000, 0x4010, 0x20000010, 0x400000, 0x20004000,\n    0x20000000, 0x4010, 0x20000010, 0x20404010, 0x404000, 0x20400000, 0x404010, 0x20404000, 0, 0x20400010, 0x10, 0x4000,\n    0x20400000, 0x404010, 0x4000, 0x400010, 0x20004010, 0, 0x20404000, 0x20000000, 0x400010, 0x20004010);\n  var spfunction7 = new Array(0x200000, 0x4200002, 0x4000802, 0, 0x800, 0x4000802, 0x200802, 0x4200800, 0x4200802,\n    0x200000, 0, 0x4000002, 0x2, 0x4000000, 0x4200002, 0x802, 0x4000800, 0x200802, 0x200002, 0x4000800, 0x4000002,\n    0x4200000, 0x4200800, 0x200002, 0x4200000, 0x800, 0x802, 0x4200802, 0x200800, 0x2, 0x4000000, 0x200800, 0x4000000,\n    0x200800, 0x200000, 0x4000802, 0x4000802, 0x4200002, 0x4200002, 0x2, 0x200002, 0x4000000, 0x4000800, 0x200000,\n    0x4200800, 0x802, 0x200802, 0x4200800, 0x802, 0x4000002, 0x4200802, 0x4200000, 0x200800, 0, 0x2, 0x4200802, 0,\n    0x200802, 0x4200000, 0x800, 0x4000002, 0x4000800, 0x800, 0x200002);\n  var spfunction8 = new Array(0x10001040, 0x1000, 0x40000, 0x10041040, 0x10000000, 0x10001040, 0x40, 0x10000000,\n    0x40040, 0x10040000, 0x10041040, 0x41000, 0x10041000, 0x41040, 0x1000, 0x40, 0x10040000, 0x10000040, 0x10001000,\n    0x1040, 0x41000, 0x40040, 0x10040040, 0x10041000, 0x1040, 0, 0, 0x10040040, 0x10000040, 0x10001000, 0x41040,\n    0x40000, 0x41040, 0x40000, 0x10041000, 0x1000, 0x40, 0x10040040, 0x1000, 0x41040, 0x10001000, 0x40, 0x10000040,\n    0x10040000, 0x10040040, 0x10000000, 0x40000, 0x10001040, 0, 0x10041040, 0x40040, 0x10000040, 0x10040000, 0x10001000,\n    0x10001040, 0, 0x10041040, 0x41000, 0x41000, 0x1040, 0x1040, 0x40040, 0x10000000, 0x10041000);\n\n  //create the 16 or 48 subkeys we will need\n  var m = 0,\n    i, j, temp, temp2, right1, right2, left, right, looping;\n  var cbcleft, cbcleft2, cbcright, cbcright2\n  var endloop, loopinc;\n  var len = message.length;\n  var chunk = 0;\n  //set up the loops for single and triple des\n  var iterations = keys.length == 32 ? 3 : 9; //single or triple des\n  if (iterations == 3) {\n    looping = encrypt ? new Array(0, 32, 2) : new Array(30, -2, -2);\n  } else {\n    looping = encrypt ? new Array(0, 32, 2, 62, 30, -2, 64, 96, 2) : new Array(94, 62, -2, 32, 64, 2, 30, -2, -2);\n  }\n\n  //pad the message depending on the padding parameter\n  //only add padding if encrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (encrypt) {\n    message = des_addPadding(message, padding);\n    len = message.length;\n  }\n\n  //store the result here\n  result = \"\";\n  tempresult = \"\";\n\n  if (mode == 1) { //CBC mode\n    cbcleft = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    cbcright = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    m = 0;\n  }\n\n  //loop through each 64 bit chunk of the message\n  while (m < len) {\n    left = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) | message\n      .charCodeAt(m++);\n    right = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) |\n      message.charCodeAt(m++);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        left ^= cbcleft;\n        right ^= cbcright;\n      } else {\n        cbcleft2 = cbcleft;\n        cbcright2 = cbcright;\n        cbcleft = left;\n        cbcright = right;\n      }\n    }\n\n    //first each 64 but chunk of the message must be permuted according to IP\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    left = ((left << 1) | (left >>> 31));\n    right = ((right << 1) | (right >>> 31));\n\n    //do this either 1 or 3 times for each chunk of the message\n    for (j = 0; j < iterations; j += 3) {\n      endloop = looping[j + 1];\n      loopinc = looping[j + 2];\n      //now go through and perform the encryption or decryption  \n      for (i = looping[j]; i != endloop; i += loopinc) { //for efficiency\n        right1 = right ^ keys[i];\n        right2 = ((right >>> 4) | (right << 28)) ^ keys[i + 1];\n        //the result is attained by passing these bytes through the S selection functions\n        temp = left;\n        left = right;\n        right = temp ^ (spfunction2[(right1 >>> 24) & 0x3f] | spfunction4[(right1 >>> 16) & 0x3f] | spfunction6[(right1 >>>\n          8) & 0x3f] | spfunction8[right1 & 0x3f] | spfunction1[(right2 >>> 24) & 0x3f] | spfunction3[(right2 >>> 16) &\n          0x3f] | spfunction5[(right2 >>> 8) & 0x3f] | spfunction7[right2 & 0x3f]);\n      }\n      temp = left;\n      left = right;\n      right = temp; //unreverse left and right\n    } //for either 1 or 3 iterations\n\n    //move then each one bit to the right\n    left = ((left >>> 1) | (left << 31));\n    right = ((right >>> 1) | (right << 31));\n\n    //now perform IP-1, which is IP in the opposite direction\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        cbcleft = left;\n        cbcright = right;\n      } else {\n        left ^= cbcleft2;\n        right ^= cbcright2;\n      }\n    }\n    tempresult += String.fromCharCode((left >>> 24), ((left >>> 16) & 0xff), ((left >>> 8) & 0xff), (left & 0xff), (\n      right >>> 24), ((right >>> 16) & 0xff), ((right >>> 8) & 0xff), (right & 0xff));\n\n    chunk += 8;\n    if (chunk == 512) {\n      result += tempresult;\n      tempresult = \"\";\n      chunk = 0;\n    }\n  } //for every 8 characters, or 64 bits in the message\n\n  //return the result as an array\n  result += tempresult;\n\n  //only remove padding if decrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (!encrypt) {\n    result = des_removePadding(result, padding);\n  }\n\n  return result;\n} //end of des\n\n\n\n//des_createKeys\n//this takes as input a 64 bit key (even though only 56 bits are used)\n//as an array of 2 integers, and returns 16 48 bit keys\n\nfunction des_createKeys(key) {\n  //declaring this locally speeds things up a bit\n  pc2bytes0 = new Array(0, 0x4, 0x20000000, 0x20000004, 0x10000, 0x10004, 0x20010000, 0x20010004, 0x200, 0x204,\n    0x20000200, 0x20000204, 0x10200, 0x10204, 0x20010200, 0x20010204);\n  pc2bytes1 = new Array(0, 0x1, 0x100000, 0x100001, 0x4000000, 0x4000001, 0x4100000, 0x4100001, 0x100, 0x101, 0x100100,\n    0x100101, 0x4000100, 0x4000101, 0x4100100, 0x4100101);\n  pc2bytes2 = new Array(0, 0x8, 0x800, 0x808, 0x1000000, 0x1000008, 0x1000800, 0x1000808, 0, 0x8, 0x800, 0x808,\n    0x1000000, 0x1000008, 0x1000800, 0x1000808);\n  pc2bytes3 = new Array(0, 0x200000, 0x8000000, 0x8200000, 0x2000, 0x202000, 0x8002000, 0x8202000, 0x20000, 0x220000,\n    0x8020000, 0x8220000, 0x22000, 0x222000, 0x8022000, 0x8222000);\n  pc2bytes4 = new Array(0, 0x40000, 0x10, 0x40010, 0, 0x40000, 0x10, 0x40010, 0x1000, 0x41000, 0x1010, 0x41010, 0x1000,\n    0x41000, 0x1010, 0x41010);\n  pc2bytes5 = new Array(0, 0x400, 0x20, 0x420, 0, 0x400, 0x20, 0x420, 0x2000000, 0x2000400, 0x2000020, 0x2000420,\n    0x2000000, 0x2000400, 0x2000020, 0x2000420);\n  pc2bytes6 = new Array(0, 0x10000000, 0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002, 0, 0x10000000,\n    0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002);\n  pc2bytes7 = new Array(0, 0x10000, 0x800, 0x10800, 0x20000000, 0x20010000, 0x20000800, 0x20010800, 0x20000, 0x30000,\n    0x20800, 0x30800, 0x20020000, 0x20030000, 0x20020800, 0x20030800);\n  pc2bytes8 = new Array(0, 0x40000, 0, 0x40000, 0x2, 0x40002, 0x2, 0x40002, 0x2000000, 0x2040000, 0x2000000, 0x2040000,\n    0x2000002, 0x2040002, 0x2000002, 0x2040002);\n  pc2bytes9 = new Array(0, 0x10000000, 0x8, 0x10000008, 0, 0x10000000, 0x8, 0x10000008, 0x400, 0x10000400, 0x408,\n    0x10000408, 0x400, 0x10000400, 0x408, 0x10000408);\n  pc2bytes10 = new Array(0, 0x20, 0, 0x20, 0x100000, 0x100020, 0x100000, 0x100020, 0x2000, 0x2020, 0x2000, 0x2020,\n    0x102000, 0x102020, 0x102000, 0x102020);\n  pc2bytes11 = new Array(0, 0x1000000, 0x200, 0x1000200, 0x200000, 0x1200000, 0x200200, 0x1200200, 0x4000000, 0x5000000,\n    0x4000200, 0x5000200, 0x4200000, 0x5200000, 0x4200200, 0x5200200);\n  pc2bytes12 = new Array(0, 0x1000, 0x8000000, 0x8001000, 0x80000, 0x81000, 0x8080000, 0x8081000, 0x10, 0x1010,\n    0x8000010, 0x8001010, 0x80010, 0x81010, 0x8080010, 0x8081010);\n  pc2bytes13 = new Array(0, 0x4, 0x100, 0x104, 0, 0x4, 0x100, 0x104, 0x1, 0x5, 0x101, 0x105, 0x1, 0x5, 0x101, 0x105);\n\n  //how many iterations (1 for des, 3 for triple des)\n  var iterations = key.length > 8 ? 3 : 1; //changed by Paul 16/6/2007 to use Triple DES for 9+ byte keys\n  //stores the return keys\n  var keys = new Array(32 * iterations);\n  //now define the left shifts which need to be done\n  var shifts = new Array(0, 0, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 0);\n  //other variables\n  var lefttemp, righttemp, m = 0,\n    n = 0,\n    temp;\n\n  for (var j = 0; j < iterations; j++) { //either 1 or 3 iterations\n    left = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n    right = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 2) ^ right) & 0x33333333;\n    right ^= temp;\n    left ^= (temp << 2);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    //the right side needs to be shifted and to get the last four bits of the left side\n    temp = (left << 8) | ((right >>> 20) & 0x000000f0);\n    //left needs to be put upside down\n    left = (right << 24) | ((right << 8) & 0xff0000) | ((right >>> 8) & 0xff00) | ((right >>> 24) & 0xf0);\n    right = temp;\n\n    //now go through and perform these shifts on the left and right keys\n    for (i = 0; i < shifts.length; i++) {\n      //shift the keys either one or two bits to the left\n      if (shifts[i]) {\n        left = (left << 2) | (left >>> 26);\n        right = (right << 2) | (right >>> 26);\n      } else {\n        left = (left << 1) | (left >>> 27);\n        right = (right << 1) | (right >>> 27);\n      }\n      left &= -0xf;\n      right &= -0xf;\n\n      //now apply PC-2, in such a way that E is easier when encrypting or decrypting\n      //this conversion will look like PC-2 except only the last 6 bits of each byte are used\n      //rather than 48 consecutive bits and the order of lines will be according to \n      //how the S selection functions will be applied: S2, S4, S6, S8, S1, S3, S5, S7\n      lefttemp = pc2bytes0[left >>> 28] | pc2bytes1[(left >>> 24) & 0xf] | pc2bytes2[(left >>> 20) & 0xf] | pc2bytes3[(\n        left >>> 16) & 0xf] | pc2bytes4[(left >>> 12) & 0xf] | pc2bytes5[(left >>> 8) & 0xf] | pc2bytes6[(left >>> 4) &\n        0xf];\n      righttemp = pc2bytes7[right >>> 28] | pc2bytes8[(right >>> 24) & 0xf] | pc2bytes9[(right >>> 20) & 0xf] |\n        pc2bytes10[(right >>> 16) & 0xf] | pc2bytes11[(right >>> 12) & 0xf] | pc2bytes12[(right >>> 8) & 0xf] |\n        pc2bytes13[(right >>> 4) & 0xf];\n      temp = ((righttemp >>> 16) ^ lefttemp) & 0x0000ffff;\n      keys[n++] = lefttemp ^ temp;\n      keys[n++] = righttemp ^ (temp << 16);\n    }\n  } //for each iterations\n  //return the keys we've created\n  return keys;\n} //end of des_createKeys\n\n\nfunction des_addPadding(message, padding) {\n  var padLength = 8 - (message.length % 8);\n  if ((padding == 2) && (padLength < 8)) { //pad the message with spaces\n    message += \"        \".substr(0, padLength);\n  } else if (padding == 1) { //PKCS7 padding\n    message += String.fromCharCode(padLength, padLength, padLength, padLength, padLength, padLength, padLength,\n      padLength).substr(0, padLength);\n  } else if (!padding && (padLength < 8)) { //pad the message out with null bytes\n    message += \"\\0\\0\\0\\0\\0\\0\\0\\0\".substr(0, padLength);\n  }\n  return message;\n}\n\nfunction des_removePadding(message, padding) {\n  if (padding == 2) { // space padded\n    message = message.replace(/ *$/g, \"\");\n  } else if (padding == 1) { // PKCS7\n    var padCount = message.charCodeAt(message.length - 1);\n    message = message.substr(0, message.length - padCount);\n  } else if (!padding) { // null padding\n    message = message.replace(/\\0*$/g, \"\");\n  }\n  return message;\n}\n\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction Des(key) {\n  this.key = [];\n\n  for (var i = 0; i < 3; i++) {\n    this.key.push(key.substr(i * 8, 8));\n  }\n\n  this.encrypt = function(block) {\n    return util.str2bin(des(des_createKeys(this.key[2]),\n      des(des_createKeys(this.key[1]),\n      des(des_createKeys(this.key[0]),\n      util.bin2str(block), true, 0, null, null),\n      false, 0, null, null), true, 0, null, null));\n  }\n}\n\nDes.keySize = Des.prototype.keySize = 24;\nDes.blockSize = Des.prototype.blockSize = 8;\n\n// This is \"original\" DES - Des is actually Triple DES.\n// This is only exported so we can unit test.\n\nfunction OriginalDes(key) {\n  this.key = key;\n\n  this.encrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), true, 0, null, padding));\n  }\n\n  this.decrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), false, 0, null, padding));\n  }\n}\n\nmodule.exports = {\n  des: Des,\n  originalDes: OriginalDes\n}\n","var desModule = require('./des.js');\n\nmodule.exports = {\n  des: desModule['des'],\n  originalDes: desModule['originalDes'],\n  cast5: require('./cast5.js'),\n  twofish: require('./twofish.js'),\n  blowfish: require('./blowfish.js')\n}\n\nvar aes = require('./aes.js');\n\nfor (var i in aes) {\n  module.exports['aes' + i] = aes[i];\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Cipher.js\n * A block-cipher algorithm implementation on JavaScript\n * See Cipher.readme.txt for further information.\n *\n * Copyright(c) 2009 Atsushi Oka [ http://oka.nu/ ]\n * This script file is distributed under the LGPL\n *\n * ACKNOWLEDGMENT\n *\n *     The main subroutines are written by Michiel van Everdingen.\n * \n *     Michiel van Everdingen\n *     http://home.versatel.nl/MAvanEverdingen/index.html\n * \n *     All rights for these routines are reserved to Michiel van Everdingen.\n *\n */\n\n\n\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n//Math\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nvar MAXINT = 0xFFFFFFFF;\n\nfunction rotb(b, n) {\n  return (b << n | b >>> (8 - n)) & 0xFF;\n}\n\nfunction rotw(w, n) {\n  return (w << n | w >>> (32 - n)) & MAXINT;\n}\n\nfunction getW(a, i) {\n  return a[i] | a[i + 1] << 8 | a[i + 2] << 16 | a[i + 3] << 24;\n}\n\nfunction setW(a, i, w) {\n  a.splice(i, 4, w & 0xFF, (w >>> 8) & 0xFF, (w >>> 16) & 0xFF, (w >>> 24) & 0xFF);\n}\n\nfunction setWInv(a, i, w) {\n  a.splice(i, 4, (w >>> 24) & 0xFF, (w >>> 16) & 0xFF, (w >>> 8) & 0xFF, w & 0xFF);\n}\n\nfunction getB(x, n) {\n  return (x >>> (n * 8)) & 0xFF;\n}\n\nfunction getNrBits(i) {\n  var n = 0;\n  while (i > 0) {\n    n++;\n    i >>>= 1;\n  }\n  return n;\n}\n\nfunction getMask(n) {\n  return (1 << n) - 1;\n}\n\n//added 2008/11/13 XXX MUST USE ONE-WAY HASH FUNCTION FOR SECURITY REASON\n\nfunction randByte() {\n  return Math.floor(Math.random() * 256);\n}\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n// Twofish\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nfunction createTwofish() {\n  //\n  var keyBytes = null;\n  var dataBytes = null;\n  var dataOffset = -1;\n  // var dataLength = -1;\n  var algorithmName = null;\n  // var idx2 = -1;\n  //\n\n  algorithmName = \"twofish\";\n\n  var tfsKey = [];\n  var tfsM = [\n    [],\n    [],\n    [],\n    []\n  ];\n\n  function tfsInit(key) {\n    keyBytes = key;\n    var i, a, b, c, d, meKey = [],\n      moKey = [],\n      inKey = [];\n    var kLen;\n    var sKey = [];\n    var f01, f5b, fef;\n\n    var q0 = [\n      [8, 1, 7, 13, 6, 15, 3, 2, 0, 11, 5, 9, 14, 12, 10, 4],\n      [2, 8, 11, 13, 15, 7, 6, 14, 3, 1, 9, 4, 0, 10, 12, 5]\n    ];\n    var q1 = [\n      [14, 12, 11, 8, 1, 2, 3, 5, 15, 4, 10, 6, 7, 0, 9, 13],\n      [1, 14, 2, 11, 4, 12, 3, 7, 6, 13, 10, 5, 15, 9, 0, 8]\n    ];\n    var q2 = [\n      [11, 10, 5, 14, 6, 13, 9, 0, 12, 8, 15, 3, 2, 4, 7, 1],\n      [4, 12, 7, 5, 1, 6, 9, 10, 0, 14, 13, 8, 2, 11, 3, 15]\n    ];\n    var q3 = [\n      [13, 7, 15, 4, 1, 2, 6, 14, 9, 11, 3, 0, 8, 5, 12, 10],\n      [11, 9, 5, 1, 12, 3, 13, 14, 6, 4, 7, 15, 2, 0, 8, 10]\n    ];\n    var ror4 = [0, 8, 1, 9, 2, 10, 3, 11, 4, 12, 5, 13, 6, 14, 7, 15];\n    var ashx = [0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 5, 14, 7];\n    var q = [\n      [],\n      []\n    ];\n    var m = [\n      [],\n      [],\n      [],\n      []\n    ];\n\n    function ffm5b(x) {\n      return x ^ (x >> 2) ^ [0, 90, 180, 238][x & 3];\n    }\n\n    function ffmEf(x) {\n      return x ^ (x >> 1) ^ (x >> 2) ^ [0, 238, 180, 90][x & 3];\n    }\n\n    function mdsRem(p, q) {\n      var i, t, u;\n      for (i = 0; i < 8; i++) {\n        t = q >>> 24;\n        q = ((q << 8) & MAXINT) | p >>> 24;\n        p = (p << 8) & MAXINT;\n        u = t << 1;\n        if (t & 128) {\n          u ^= 333;\n        }\n        q ^= t ^ (u << 16);\n        u ^= t >>> 1;\n        if (t & 1) {\n          u ^= 166;\n        }\n        q ^= u << 24 | u << 8;\n      }\n      return q;\n    }\n\n    function qp(n, x) {\n      var a, b, c, d;\n      a = x >> 4;\n      b = x & 15;\n      c = q0[n][a ^ b];\n      d = q1[n][ror4[b] ^ ashx[a]];\n      return q3[n][ror4[d] ^ ashx[c]] << 4 | q2[n][c ^ d];\n    }\n\n    function hFun(x, key) {\n      var a = getB(x, 0),\n        b = getB(x, 1),\n        c = getB(x, 2),\n        d = getB(x, 3);\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(key[3], 0);\n          b = q[0][b] ^ getB(key[3], 1);\n          c = q[0][c] ^ getB(key[3], 2);\n          d = q[1][d] ^ getB(key[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(key[2], 0);\n          b = q[1][b] ^ getB(key[2], 1);\n          c = q[0][c] ^ getB(key[2], 2);\n          d = q[0][d] ^ getB(key[2], 3);\n        case 2:\n          a = q[0][q[0][a] ^ getB(key[1], 0)] ^ getB(key[0], 0);\n          b = q[0][q[1][b] ^ getB(key[1], 1)] ^ getB(key[0], 1);\n          c = q[1][q[0][c] ^ getB(key[1], 2)] ^ getB(key[0], 2);\n          d = q[1][q[1][d] ^ getB(key[1], 3)] ^ getB(key[0], 3);\n      }\n      return m[0][a] ^ m[1][b] ^ m[2][c] ^ m[3][d];\n    }\n\n    keyBytes = keyBytes.slice(0, 32);\n    i = keyBytes.length;\n    while (i != 16 && i != 24 && i != 32)\n      keyBytes[i++] = 0;\n\n    for (i = 0; i < keyBytes.length; i += 4) {\n      inKey[i >> 2] = getW(keyBytes, i);\n    }\n    for (i = 0; i < 256; i++) {\n      q[0][i] = qp(0, i);\n      q[1][i] = qp(1, i);\n    }\n    for (i = 0; i < 256; i++) {\n      f01 = q[1][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[0][i] = f01 + (f5b << 8) + (fef << 16) + (fef << 24);\n      m[2][i] = f5b + (fef << 8) + (f01 << 16) + (fef << 24);\n      f01 = q[0][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[1][i] = fef + (fef << 8) + (f5b << 16) + (f01 << 24);\n      m[3][i] = f5b + (f01 << 8) + (fef << 16) + (f5b << 24);\n    }\n\n    kLen = inKey.length / 2;\n    for (i = 0; i < kLen; i++) {\n      a = inKey[i + i];\n      meKey[i] = a;\n      b = inKey[i + i + 1];\n      moKey[i] = b;\n      sKey[kLen - i - 1] = mdsRem(a, b);\n    }\n    for (i = 0; i < 40; i += 2) {\n      a = 0x1010101 * i;\n      b = a + 0x1010101;\n      a = hFun(a, meKey);\n      b = rotw(hFun(b, moKey), 8);\n      tfsKey[i] = (a + b) & MAXINT;\n      tfsKey[i + 1] = rotw(a + 2 * b, 9);\n    }\n    for (i = 0; i < 256; i++) {\n      a = b = c = d = i;\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(sKey[3], 0);\n          b = q[0][b] ^ getB(sKey[3], 1);\n          c = q[0][c] ^ getB(sKey[3], 2);\n          d = q[1][d] ^ getB(sKey[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(sKey[2], 0);\n          b = q[1][b] ^ getB(sKey[2], 1);\n          c = q[0][c] ^ getB(sKey[2], 2);\n          d = q[0][d] ^ getB(sKey[2], 3);\n        case 2:\n          tfsM[0][i] = m[0][q[0][q[0][a] ^ getB(sKey[1], 0)] ^ getB(sKey[0], 0)];\n          tfsM[1][i] = m[1][q[0][q[1][b] ^ getB(sKey[1], 1)] ^ getB(sKey[0], 1)];\n          tfsM[2][i] = m[2][q[1][q[0][c] ^ getB(sKey[1], 2)] ^ getB(sKey[0], 2)];\n          tfsM[3][i] = m[3][q[1][q[1][d] ^ getB(sKey[1], 3)] ^ getB(sKey[0], 3)];\n      }\n    }\n  }\n\n  function tfsG0(x) {\n    return tfsM[0][getB(x, 0)] ^ tfsM[1][getB(x, 1)] ^ tfsM[2][getB(x, 2)] ^ tfsM[3][getB(x, 3)];\n  }\n\n  function tfsG1(x) {\n    return tfsM[0][getB(x, 3)] ^ tfsM[1][getB(x, 0)] ^ tfsM[2][getB(x, 1)] ^ tfsM[3][getB(x, 2)];\n  }\n\n  function tfsFrnd(r, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2] ^ (a + b + tfsKey[4 * r + 8]) & MAXINT, 31);\n    blk[3] = rotw(blk[3], 1) ^ (a + 2 * b + tfsKey[4 * r + 9]) & MAXINT;\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0] ^ (a + b + tfsKey[4 * r + 10]) & MAXINT, 31);\n    blk[1] = rotw(blk[1], 1) ^ (a + 2 * b + tfsKey[4 * r + 11]) & MAXINT;\n  }\n\n  function tfsIrnd(i, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2], 1) ^ (a + b + tfsKey[4 * i + 10]) & MAXINT;\n    blk[3] = rotw(blk[3] ^ (a + 2 * b + tfsKey[4 * i + 11]) & MAXINT, 31);\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0], 1) ^ (a + b + tfsKey[4 * i + 8]) & MAXINT;\n    blk[1] = rotw(blk[1] ^ (a + 2 * b + tfsKey[4 * i + 9]) & MAXINT, 31);\n  }\n\n  function tfsClose() {\n    tfsKey = [];\n    tfsM = [\n      [],\n      [],\n      [],\n      []\n    ];\n  }\n\n  function tfsEncrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[0],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[1],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[2],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[3]\n    ];\n    for (var j = 0; j < 8; j++) {\n      tfsFrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[4]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[5]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[6]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[7]);\n    dataOffset += 16;\n    return dataBytes;\n  }\n\n  function tfsDecrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[4],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[5],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[6],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[7]\n    ];\n    for (var j = 7; j >= 0; j--) {\n      tfsIrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[0]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[1]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[2]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[3]);\n    dataOffset += 16;\n  }\n\n  // added by Recurity Labs\n\n  function tfsFinal() {\n    return dataBytes;\n  }\n\n  return {\n    name: \"twofish\",\n    blocksize: 128 / 8,\n    open: tfsInit,\n    close: tfsClose,\n    encrypt: tfsEncrypt,\n    decrypt: tfsDecrypt,\n    // added by Recurity Labs\n    finalize: tfsFinal\n  };\n}\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction TFencrypt(block, key) {\n  var block_copy = [].concat(block);\n  var tf = createTwofish();\n  tf.open(util.str2bin(key), 0);\n  var result = tf.encrypt(block_copy, 0);\n  tf.close();\n  return result;\n}\n\nfunction TF(key) {\n  this.tf = createTwofish();\n  this.tf.open(util.str2bin(key), 0);\n\n  this.encrypt = function(block) {\n    return this.tf.encrypt([].concat(block), 0);\n  }\n}\n\n\nmodule.exports = TF;\nmodule.exports.keySize = TF.prototype.keySize = 32;\nmodule.exports.blockSize = TF.prototype.blockSize = 16;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar random = require('./random.js'),\n  cipher = require('./cipher'),\n  publicKey = require('./public_key'),\n  type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Encrypts data using the specified public key multiprecision integers \n   * and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi[]} if RSA an openpgp_type_mpi; \n   * if elgamal encryption an array of two openpgp_type_mpi is returned; otherwise null\n   */\n  publicKeyEncrypt: function(algo, publicMPIs, data) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          var rsa = new publicKey.rsa();\n          var n = publicMPIs[0].toBigInteger();\n          var e = publicMPIs[1].toBigInteger();\n          var m = data.toBigInteger();\n          return [rsa.encrypt(m, e, n)];\n\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var p = publicMPIs[0].toBigInteger();\n          var g = publicMPIs[1].toBigInteger();\n          var y = publicMPIs[2].toBigInteger();\n          var m = data.toBigInteger();\n          return elgamal.encrypt(m, g, p, y);\n\n        default:\n          return [];\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n  /**\n   * Decrypts data using the specified public key multiprecision integers of the private key,\n   * the specified secretMPIs of the private key and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers \n   * of the public key part of the private key\n   * @param {openpgp_type_mpi[]} secretMPIs Algorithm dependent multiprecision integers \n   * of the private key used\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi} returns a big integer containing the decrypted data; otherwise null\n   */\n\n  publicKeyDecrypt: function(algo, keyIntegers, dataIntegers) {\n    var bn = (function() {\n      switch (algo) {\n        case 'rsa_encrypt_sign':\n        case 'rsa_encrypt':\n          var rsa = new publicKey.rsa();\n          // 0 and 1 are the public key.\n          var d = keyIntegers[2].toBigInteger();\n          var p = keyIntegers[3].toBigInteger();\n          var q = keyIntegers[4].toBigInteger();\n          var u = keyIntegers[5].toBigInteger();\n          var m = dataIntegers[0].toBigInteger();\n          return rsa.decrypt(m, d, p, q, u);\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var x = keyIntegers[3].toBigInteger();\n          var c1 = dataIntegers[0].toBigInteger();\n          var c2 = dataIntegers[1].toBigInteger();\n          var p = keyIntegers[0].toBigInteger();\n          return elgamal.decrypt(c1, c2, p, x);\n        default:\n          return null;\n      }\n    })();\n\n    var result = new type_mpi();\n    result.fromBigInteger(bn);\n    return result;\n  },\n\n  /** Returns the number of integers comprising the private key of an algorithm\n   * @param {openpgp.publickey} algo The public key algorithm\n   * @return {Integer} The number of integers.\n   */\n  getPrivateMpiCount: function(algo) {\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        //   Algorithm-Specific Fields for RSA secret keys:\n        //   - multiprecision integer (MPI) of RSA secret exponent d.\n        //   - MPI of RSA secret prime value p.\n        //   - MPI of RSA secret prime value q (p < q).\n        //   - MPI of u, the multiplicative inverse of p, mod q.\n        return 4;\n      case 'elgamal':\n        // Algorithm-Specific Fields for Elgamal secret keys:\n        //   - MPI of Elgamal secret exponent x.\n        return 1;\n      case 'dsa':\n        // Algorithm-Specific Fields for DSA secret keys:\n        //   - MPI of DSA secret exponent x.\n        return 1;\n      default:\n        throw new Error('Unknown algorithm');\n    }\n  },\n\n  getPublicMpiCount: function(algo) {\n    // - A series of multiprecision integers comprising the key material:\n    //   Algorithm-Specific Fields for RSA public keys:\n    //       - a multiprecision integer (MPI) of RSA public modulus n;\n    //       - an MPI of RSA public encryption exponent e.\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        return 2;\n\n        //   Algorithm-Specific Fields for Elgamal public keys:\n        //     - MPI of Elgamal prime p;\n        //     - MPI of Elgamal group generator g;\n        //     - MPI of Elgamal public key value y (= g**x mod p where x  is secret).\n      case 'elgamal':\n        return 3;\n\n        //   Algorithm-Specific Fields for DSA public keys:\n        //       - MPI of DSA prime p;\n        //       - MPI of DSA group order q (q is a prime divisor of p-1);\n        //       - MPI of DSA group generator g;\n        //       - MPI of DSA public-key value y (= g**x mod p where x  is secret).\n      case 'dsa':\n        return 4;\n\n      default:\n        throw new Error('Unknown algorithm.');\n    }\n  },\n\n  generateMpi: function(algo, bits) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n        case 'rsa_sign':\n          //remember \"publicKey\" refers to the crypto/public_key dir\n          var rsa = new publicKey.rsa();\n          var keyObject = rsa.generate(bits, \"10001\");\n          var output = [];\n          output.push(keyObject.n);\n          output.push(keyObject.ee);\n          output.push(keyObject.d);\n          output.push(keyObject.p);\n          output.push(keyObject.q);\n          output.push(keyObject.u);\n          return output;\n        default:\n          throw new Error('Unsupported algorithm for key generation.');\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n\n  /**\n   * generate random byte prefix as string for the specified algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes with length equal to the block\n   * size of the cipher\n   */\n  getPrefixRandom: function(algo) {\n    return random.getRandomBytes(cipher[algo].blockSize);\n  },\n\n  /**\n   * Generating a session key for the specified symmetric algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes as a string to be used as a key\n   */\n  generateSessionKey: function(algo) {\n    return random.getRandomBytes(cipher[algo].keySize);\n  },\n\n};\n","var sha = require('./sha.js');\n\nmodule.exports = {\n  md5: require('./md5.js'),\n  sha1: sha.sha1,\n  sha256: sha.sha256,\n  sha224: sha.sha224,\n  sha384: sha.sha384,\n  sha512: sha.sha512,\n  ripemd: require('./ripe-md.js'),\n\n  /**\n   * Create a hash on the specified data using the specified algorithm\n   * @param {Integer} algo Hash algorithm type (see RFC4880 9.4)\n   * @param {String} data Data to be hashed\n   * @return {String} hash value\n   */\n  digest: function(algo, data) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return this.md5(data);\n      case 2:\n        // - SHA-1 [FIPS180]\n        return this.sha1(data);\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return this.ripemd(data);\n      case 8:\n        // - SHA256 [FIPS180]\n        return this.sha256(data);\n      case 9:\n        // - SHA384 [FIPS180]\n        return this.sha384(data);\n      case 10:\n        // - SHA512 [FIPS180]\n        return this.sha512(data);\n      case 11:\n        // - SHA224 [FIPS180]\n        return this.sha224(data);\n      default:\n        throw new Error('Invalid hash function.');\n    }\n  },\n\n  /**\n   * Returns the hash size in bytes of the specified hash algorithm type\n   * @param {Integer} algo Hash algorithm type (See RFC4880 9.4)\n   * @return {Integer} Size in bytes of the resulting hash\n   */\n  getHashByteLength: function(algo) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return 16;\n      case 2:\n        // - SHA-1 [FIPS180]\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return 20;\n      case 8:\n        // - SHA256 [FIPS180]\n        return 32;\n      case 9:\n        // - SHA384 [FIPS180]\n        return 48\n      case 10:\n        // - SHA512 [FIPS180]\n        return 64;\n      case 11:\n        // - SHA224 [FIPS180]\n        return 28;\n      default:\n        throw new Error('Invalid hash algorithm.');\n    }\n  }\n\n}\n","/**\n * A fast MD5 JavaScript implementation\n * Copyright (c) 2012 Joseph Myers\n * http://www.myersdaily.org/joseph/javascript/md5-text.html\n *\n * Permission to use, copy, modify, and distribute this software\n * and its documentation for any purposes and without\n * fee is hereby granted provided that this copyright notice\n * appears in all copies.\n *\n * Of course, this soft is provided \"as is\" without express or implied\n * warranty of any kind.\n */\n\nvar util = require('../../util/util.js');\n\nfunction MD5(entree) {\n  var hex = md5(entree);\n  var bin = util.hex2bin(hex);\n  return bin;\n}\n\nfunction md5cycle(x, k) {\n  var a = x[0],\n    b = x[1],\n    c = x[2],\n    d = x[3];\n\n  a = ff(a, b, c, d, k[0], 7, -680876936);\n  d = ff(d, a, b, c, k[1], 12, -389564586);\n  c = ff(c, d, a, b, k[2], 17, 606105819);\n  b = ff(b, c, d, a, k[3], 22, -1044525330);\n  a = ff(a, b, c, d, k[4], 7, -176418897);\n  d = ff(d, a, b, c, k[5], 12, 1200080426);\n  c = ff(c, d, a, b, k[6], 17, -1473231341);\n  b = ff(b, c, d, a, k[7], 22, -45705983);\n  a = ff(a, b, c, d, k[8], 7, 1770035416);\n  d = ff(d, a, b, c, k[9], 12, -1958414417);\n  c = ff(c, d, a, b, k[10], 17, -42063);\n  b = ff(b, c, d, a, k[11], 22, -1990404162);\n  a = ff(a, b, c, d, k[12], 7, 1804603682);\n  d = ff(d, a, b, c, k[13], 12, -40341101);\n  c = ff(c, d, a, b, k[14], 17, -1502002290);\n  b = ff(b, c, d, a, k[15], 22, 1236535329);\n\n  a = gg(a, b, c, d, k[1], 5, -165796510);\n  d = gg(d, a, b, c, k[6], 9, -1069501632);\n  c = gg(c, d, a, b, k[11], 14, 643717713);\n  b = gg(b, c, d, a, k[0], 20, -373897302);\n  a = gg(a, b, c, d, k[5], 5, -701558691);\n  d = gg(d, a, b, c, k[10], 9, 38016083);\n  c = gg(c, d, a, b, k[15], 14, -660478335);\n  b = gg(b, c, d, a, k[4], 20, -405537848);\n  a = gg(a, b, c, d, k[9], 5, 568446438);\n  d = gg(d, a, b, c, k[14], 9, -1019803690);\n  c = gg(c, d, a, b, k[3], 14, -187363961);\n  b = gg(b, c, d, a, k[8], 20, 1163531501);\n  a = gg(a, b, c, d, k[13], 5, -1444681467);\n  d = gg(d, a, b, c, k[2], 9, -51403784);\n  c = gg(c, d, a, b, k[7], 14, 1735328473);\n  b = gg(b, c, d, a, k[12], 20, -1926607734);\n\n  a = hh(a, b, c, d, k[5], 4, -378558);\n  d = hh(d, a, b, c, k[8], 11, -2022574463);\n  c = hh(c, d, a, b, k[11], 16, 1839030562);\n  b = hh(b, c, d, a, k[14], 23, -35309556);\n  a = hh(a, b, c, d, k[1], 4, -1530992060);\n  d = hh(d, a, b, c, k[4], 11, 1272893353);\n  c = hh(c, d, a, b, k[7], 16, -155497632);\n  b = hh(b, c, d, a, k[10], 23, -1094730640);\n  a = hh(a, b, c, d, k[13], 4, 681279174);\n  d = hh(d, a, b, c, k[0], 11, -358537222);\n  c = hh(c, d, a, b, k[3], 16, -722521979);\n  b = hh(b, c, d, a, k[6], 23, 76029189);\n  a = hh(a, b, c, d, k[9], 4, -640364487);\n  d = hh(d, a, b, c, k[12], 11, -421815835);\n  c = hh(c, d, a, b, k[15], 16, 530742520);\n  b = hh(b, c, d, a, k[2], 23, -995338651);\n\n  a = ii(a, b, c, d, k[0], 6, -198630844);\n  d = ii(d, a, b, c, k[7], 10, 1126891415);\n  c = ii(c, d, a, b, k[14], 15, -1416354905);\n  b = ii(b, c, d, a, k[5], 21, -57434055);\n  a = ii(a, b, c, d, k[12], 6, 1700485571);\n  d = ii(d, a, b, c, k[3], 10, -1894986606);\n  c = ii(c, d, a, b, k[10], 15, -1051523);\n  b = ii(b, c, d, a, k[1], 21, -2054922799);\n  a = ii(a, b, c, d, k[8], 6, 1873313359);\n  d = ii(d, a, b, c, k[15], 10, -30611744);\n  c = ii(c, d, a, b, k[6], 15, -1560198380);\n  b = ii(b, c, d, a, k[13], 21, 1309151649);\n  a = ii(a, b, c, d, k[4], 6, -145523070);\n  d = ii(d, a, b, c, k[11], 10, -1120210379);\n  c = ii(c, d, a, b, k[2], 15, 718787259);\n  b = ii(b, c, d, a, k[9], 21, -343485551);\n\n  x[0] = add32(a, x[0]);\n  x[1] = add32(b, x[1]);\n  x[2] = add32(c, x[2]);\n  x[3] = add32(d, x[3]);\n\n}\n\nfunction cmn(q, a, b, x, s, t) {\n  a = add32(add32(a, q), add32(x, t));\n  return add32((a << s) | (a >>> (32 - s)), b);\n}\n\nfunction ff(a, b, c, d, x, s, t) {\n  return cmn((b & c) | ((~b) & d), a, b, x, s, t);\n}\n\nfunction gg(a, b, c, d, x, s, t) {\n  return cmn((b & d) | (c & (~d)), a, b, x, s, t);\n}\n\nfunction hh(a, b, c, d, x, s, t) {\n  return cmn(b ^ c ^ d, a, b, x, s, t);\n}\n\nfunction ii(a, b, c, d, x, s, t) {\n  return cmn(c ^ (b | (~d)), a, b, x, s, t);\n}\n\nfunction md51(s) {\n  txt = '';\n  var n = s.length,\n    state = [1732584193, -271733879, -1732584194, 271733878],\n    i;\n  for (i = 64; i <= s.length; i += 64) {\n    md5cycle(state, md5blk(s.substring(i - 64, i)));\n  }\n  s = s.substring(i - 64);\n  var tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];\n  for (i = 0; i < s.length; i++)\n    tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3);\n  tail[i >> 2] |= 0x80 << ((i % 4) << 3);\n  if (i > 55) {\n    md5cycle(state, tail);\n    for (i = 0; i < 16; i++) tail[i] = 0;\n  }\n  tail[14] = n * 8;\n  md5cycle(state, tail);\n  return state;\n}\n\n/* there needs to be support for Unicode here,\n * unless we pretend that we can redefine the MD-5\n * algorithm for multi-byte characters (perhaps\n * by adding every four 16-bit characters and\n * shortening the sum to 32 bits). Otherwise\n * I suggest performing MD-5 as if every character\n * was two bytes--e.g., 0040 0025 = @%--but then\n * how will an ordinary MD-5 sum be matched?\n * There is no way to standardize text to something\n * like UTF-8 before transformation; speed cost is\n * utterly prohibitive. The JavaScript standard\n * itself needs to look at this: it should start\n * providing access to strings as preformed UTF-8\n * 8-bit unsigned value arrays.\n */\nfunction md5blk(s) { /* I figured global was faster.   */\n  var md5blks = [],\n    i; /* Andy King said do it this way. */\n  for (i = 0; i < 64; i += 4) {\n    md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) <<\n      24);\n  }\n  return md5blks;\n}\n\nvar hex_chr = '0123456789abcdef'.split('');\n\nfunction rhex(n) {\n  var s = '',\n    j = 0;\n  for (; j < 4; j++)\n    s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F];\n  return s;\n}\n\nfunction hex(x) {\n  for (var i = 0; i < x.length; i++)\n    x[i] = rhex(x[i]);\n  return x.join('');\n}\n\nfunction md5(s) {\n  return hex(md51(s));\n}\n\n/* this function is much faster,\nso if possible we use it. Some IEs\nare the only ones I know of that\nneed the idiotic second function,\ngenerated by an if clause.  */\n\nfunction add32(a, b) {\n  return (a + b) & 0xFFFFFFFF;\n}\n\nif (md5('hello') != '5d41402abc4b2a76b9719d911017c592') {\n  function add32(x, y) {\n    var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n      msw = (x >> 16) + (y >> 16) + (lsw >> 16);\n    return (msw << 16) | (lsw & 0xFFFF);\n  }\n}\n\nmodule.exports = MD5\n","/*\n * CryptoMX Tools\n * Copyright (C) 2004 - 2006 Derek Buitenhuis\n *\n * This program is free software; you can redistribute it and/or\n * modify it under the terms of the GNU General Public License\n * as published by the Free Software Foundation; either version 2\n * of the License, or (at your option) any later version.\n *\n * This program is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n * GNU General Public License for more details.\n *\n * You should have received a copy of the GNU General Public License\n * along with this program; if not, write to the Free Software\n * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.\n */\n\n/* Modified by Recurity Labs GmbH\n */\n\nvar RMDsize = 160;\nvar X = new Array();\n\nfunction ROL(x, n) {\n  return new Number((x << n) | (x >>> (32 - n)));\n}\n\nfunction F(x, y, z) {\n  return new Number(x ^ y ^ z);\n}\n\nfunction G(x, y, z) {\n  return new Number((x & y) | (~x & z));\n}\n\nfunction H(x, y, z) {\n  return new Number((x | ~y) ^ z);\n}\n\nfunction I(x, y, z) {\n  return new Number((x & z) | (y & ~z));\n}\n\nfunction J(x, y, z) {\n  return new Number(x ^ (y | ~z));\n}\n\nfunction mixOneRound(a, b, c, d, e, x, s, roundNumber) {\n  switch (roundNumber) {\n    case 0:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n    case 1:\n      a += G(b, c, d) + x + 0x5a827999;\n      break;\n    case 2:\n      a += H(b, c, d) + x + 0x6ed9eba1;\n      break;\n    case 3:\n      a += I(b, c, d) + x + 0x8f1bbcdc;\n      break;\n    case 4:\n      a += J(b, c, d) + x + 0xa953fd4e;\n      break;\n    case 5:\n      a += J(b, c, d) + x + 0x50a28be6;\n      break;\n    case 6:\n      a += I(b, c, d) + x + 0x5c4dd124;\n      break;\n    case 7:\n      a += H(b, c, d) + x + 0x6d703ef3;\n      break;\n    case 8:\n      a += G(b, c, d) + x + 0x7a6d76e9;\n      break;\n    case 9:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n\n    default:\n      document.write(\"Bogus round number\");\n      break;\n  }\n\n  a = ROL(a, s) + e;\n  c = ROL(c, 10);\n\n  a &= 0xffffffff;\n  b &= 0xffffffff;\n  c &= 0xffffffff;\n  d &= 0xffffffff;\n  e &= 0xffffffff;\n\n  var retBlock = new Array();\n  retBlock[0] = a;\n  retBlock[1] = b;\n  retBlock[2] = c;\n  retBlock[3] = d;\n  retBlock[4] = e;\n  retBlock[5] = x;\n  retBlock[6] = s;\n\n  return retBlock;\n}\n\nfunction MDinit(MDbuf) {\n  MDbuf[0] = 0x67452301;\n  MDbuf[1] = 0xefcdab89;\n  MDbuf[2] = 0x98badcfe;\n  MDbuf[3] = 0x10325476;\n  MDbuf[4] = 0xc3d2e1f0;\n}\n\nvar ROLs = [\n  [11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8],\n  [7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12],\n  [11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5],\n  [11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12],\n  [9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6],\n  [8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6],\n  [9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11],\n  [9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5],\n  [15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8],\n  [8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11]\n];\n\nvar indexes = [\n  [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15],\n  [7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8],\n  [3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12],\n  [1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2],\n  [4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13],\n  [5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12],\n  [6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2],\n  [15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13],\n  [8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14],\n  [12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11]\n];\n\nfunction compress(MDbuf, X) {\n  blockA = new Array();\n  blockB = new Array();\n\n  var retBlock;\n\n  for (var i = 0; i < 5; i++) {\n    blockA[i] = new Number(MDbuf[i]);\n    blockB[i] = new Number(MDbuf[i]);\n  }\n\n  var step = 0;\n  for (var j = 0; j < 5; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockA[(step + 0) % 5],\n        blockA[(step + 1) % 5],\n        blockA[(step + 2) % 5],\n        blockA[(step + 3) % 5],\n        blockA[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockA[(step + 0) % 5] = retBlock[0];\n      blockA[(step + 1) % 5] = retBlock[1];\n      blockA[(step + 2) % 5] = retBlock[2];\n      blockA[(step + 3) % 5] = retBlock[3];\n      blockA[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  step = 0;\n  for (var j = 5; j < 10; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockB[(step + 0) % 5],\n        blockB[(step + 1) % 5],\n        blockB[(step + 2) % 5],\n        blockB[(step + 3) % 5],\n        blockB[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockB[(step + 0) % 5] = retBlock[0];\n      blockB[(step + 1) % 5] = retBlock[1];\n      blockB[(step + 2) % 5] = retBlock[2];\n      blockB[(step + 3) % 5] = retBlock[3];\n      blockB[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  blockB[3] += blockA[2] + MDbuf[1];\n  MDbuf[1] = MDbuf[2] + blockA[3] + blockB[4];\n  MDbuf[2] = MDbuf[3] + blockA[4] + blockB[0];\n  MDbuf[3] = MDbuf[4] + blockA[0] + blockB[1];\n  MDbuf[4] = MDbuf[0] + blockA[1] + blockB[2];\n  MDbuf[0] = blockB[3];\n}\n\nfunction zeroX(X) {\n  for (var i = 0; i < 16; i++) {\n    X[i] = 0;\n  }\n}\n\nfunction MDfinish(MDbuf, strptr, lswlen, mswlen) {\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var i = 0; i < (lswlen & 63); i++) {\n    X[i >>> 2] ^= (strptr.charCodeAt(j++) & 255) << (8 * (i & 3));\n  }\n\n  X[(lswlen >>> 2) & 15] ^= 1 << (8 * (lswlen & 3) + 7);\n\n  if ((lswlen & 63) > 55) {\n    compress(MDbuf, X);\n    var X = new Array(16);\n    zeroX(X);\n  }\n\n  X[14] = lswlen << 3;\n  X[15] = (lswlen >>> 29) | (mswlen << 3);\n\n  compress(MDbuf, X);\n}\n\nfunction BYTES_TO_DWORD(fourChars) {\n  var tmp = (fourChars.charCodeAt(3) & 255) << 24;\n  tmp |= (fourChars.charCodeAt(2) & 255) << 16;\n  tmp |= (fourChars.charCodeAt(1) & 255) << 8;\n  tmp |= (fourChars.charCodeAt(0) & 255);\n\n  return tmp;\n}\n\nfunction RMD(message) {\n  var MDbuf = new Array(RMDsize / 32);\n  var hashcode = new Array(RMDsize / 8);\n  var length;\n  var nbytes;\n\n  MDinit(MDbuf);\n  length = message.length;\n\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var nbytes = length; nbytes > 63; nbytes -= 64) {\n    for (var i = 0; i < 16; i++) {\n      X[i] = BYTES_TO_DWORD(message.substr(j, 4));\n      j += 4;\n    }\n    compress(MDbuf, X);\n  }\n\n  MDfinish(MDbuf, message.substr(j), length, 0);\n\n  for (var i = 0; i < RMDsize / 8; i += 4) {\n    hashcode[i] = MDbuf[i >>> 2] & 255;\n    hashcode[i + 1] = (MDbuf[i >>> 2] >>> 8) & 255;\n    hashcode[i + 2] = (MDbuf[i >>> 2] >>> 16) & 255;\n    hashcode[i + 3] = (MDbuf[i >>> 2] >>> 24) & 255;\n  }\n\n  return hashcode;\n}\n\n\nfunction RMDstring(message) {\n  var hashcode = RMD(message);\n  var retString = \"\";\n\n  for (var i = 0; i < RMDsize / 8; i++) {\n    retString += String.fromCharCode(hashcode[i]);\n  }\n\n  return retString;\n}\n\nmodule.exports = RMDstring;\n","/* A JavaScript implementation of the SHA family of hashes, as defined in FIPS \n * PUB 180-2 as well as the corresponding HMAC implementation as defined in\n * FIPS PUB 198a\n *\n * Version 1.3 Copyright Brian Turek 2008-2010\n * Distributed under the BSD License\n * See http://jssha.sourceforge.net/ for more information\n *\n * Several functions taken from Paul Johnson\n */\n\n/* Modified by Recurity Labs GmbH\n * \n * This code has been slightly modified direct string output:\n * - bin2bstr has been added\n * - following wrappers of this library have been added:\n *   - str_sha1\n *   - str_sha256\n *   - str_sha224\n *   - str_sha384\n *   - str_sha512\n */\n\nvar jsSHA = (function() {\n\n  /*\n   * Configurable variables. Defaults typically work\n   */\n  /* Number of Bits Per character (8 for ASCII, 16 for Unicode) */\n  var charSize = 8,\n    /* base-64 pad character. \"=\" for strict RFC compliance */\n    b64pad = \"\",\n    /* hex output format. 0 - lowercase; 1 - uppercase */\n    hexCase = 0,\n\n    /*\n     * Int_64 is a object for 2 32-bit numbers emulating a 64-bit number\n     *\n     * @constructor\n     * @param {Number} msint_32 The most significant 32-bits of a 64-bit number\n     * @param {Number} lsint_32 The least significant 32-bits of a 64-bit number\n     */\n    Int_64 = function(msint_32, lsint_32) {\n      this.highOrder = msint_32;\n      this.lowOrder = lsint_32;\n    },\n\n    /*\n     * Convert a string to an array of big-endian words\n     * If charSize is ASCII, characters >255 have their hi-byte silently\n     * ignored.\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    str2binb = function(str) {\n      var bin = [],\n        mask = (1 << charSize) - 1,\n        length = str.length * charSize,\n        i;\n\n      for (i = 0; i < length; i += charSize) {\n        bin[i >> 5] |= (str.charCodeAt(i / charSize) & mask) <<\n          (32 - charSize - (i % 32));\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert a hex string to an array of big-endian words\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    hex2binb = function(str) {\n      var bin = [],\n        length = str.length,\n        i, num;\n\n      for (i = 0; i < length; i += 2) {\n        num = parseInt(str.substr(i, 2), 16);\n        if (!isNaN(num)) {\n          bin[i >> 3] |= num << (24 - (4 * (i % 8)));\n        } else {\n          return \"INVALID HEX STRING\";\n        }\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert an array of big-endian words to a hex string.\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to hexidecimal\n     *\t representation\n     * @return Hexidecimal representation of the parameter in String form\n     */\n    binb2hex = function(binarray) {\n      var hex_tab = (hexCase) ? \"0123456789ABCDEF\" : \"0123456789abcdef\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, srcByte;\n\n      for (i = 0; i < length; i += 1) {\n        srcByte = binarray[i >> 2] >> ((3 - (i % 4)) * 8);\n        str += hex_tab.charAt((srcByte >> 4) & 0xF) +\n          hex_tab.charAt(srcByte & 0xF);\n      }\n\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a base-64 string\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to base-64\n     *\t representation\n     * @return Base-64 encoded representation of the parameter in String form\n     */\n    binb2b64 = function(binarray) {\n      var tab = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\" +\n        \"0123456789+/\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, j,\n        triplet;\n\n      for (i = 0; i < length; i += 3) {\n        triplet = (((binarray[i >> 2] >> 8 * (3 - i % 4)) & 0xFF) << 16) |\n          (((binarray[i + 1 >> 2] >> 8 * (3 - (i + 1) % 4)) & 0xFF) << 8) |\n          ((binarray[i + 2 >> 2] >> 8 * (3 - (i + 2) % 4)) & 0xFF);\n        for (j = 0; j < 4; j += 1) {\n          if (i * 8 + j * 6 <= binarray.length * 32) {\n            str += tab.charAt((triplet >> 6 * (3 - j)) & 0x3F);\n          } else {\n            str += b64pad;\n          }\n        }\n      }\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a string\n     */\n    binb2str = function(bin) {\n      var str = \"\";\n      var mask = (1 << 8) - 1;\n      for (var i = 0; i < bin.length * 32; i += 8)\n        str += String.fromCharCode((bin[i >> 5] >>> (24 - i % 32)) & mask);\n      return str;\n    },\n    /*\n     * The 32-bit implementation of circular rotate left\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotl_32 = function(x, n) {\n      return (x << n) | (x >>> (32 - n));\n    },\n\n    /*\n     * The 32-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_32 = function(x, n) {\n      return (x >>> n) | (x << (32 - n));\n    },\n\n    /*\n     * The 64-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n        (x.highOrder >>> n) | (x.lowOrder << (32 - n)), (x.lowOrder >>> n) | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n        (x.lowOrder >>> n) | (x.highOrder << (32 - n)), (x.highOrder >>> n) | (x.lowOrder << (32 - n)));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of shift right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_32 = function(x, n) {\n      return x >>> n;\n    },\n\n    /*\n     * The 64-bit implementation of shift right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n          x.highOrder >>> n,\n          x.lowOrder >>> n | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n          0,\n          x.highOrder << (32 - n));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Parity function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    parity_32 = function(x, y, z) {\n      return x ^ y ^ z;\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_32 = function(x, y, z) {\n      return (x & y) ^ (~x & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^ (~x.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^ (~x.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_32 = function(x, y, z) {\n      return (x & y) ^ (x & z) ^ (y & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^\n        (x.highOrder & z.highOrder) ^\n        (y.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^\n        (x.lowOrder & z.lowOrder) ^\n        (y.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_32 = function(x) {\n      return rotr_32(x, 2) ^ rotr_32(x, 13) ^ rotr_32(x, 22);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_64 = function(x) {\n      var rotr28 = rotr_64(x, 28),\n        rotr34 = rotr_64(x, 34),\n        rotr39 = rotr_64(x, 39);\n\n      return new Int_64(\n        rotr28.highOrder ^ rotr34.highOrder ^ rotr39.highOrder,\n        rotr28.lowOrder ^ rotr34.lowOrder ^ rotr39.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_32 = function(x) {\n      return rotr_32(x, 6) ^ rotr_32(x, 11) ^ rotr_32(x, 25);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_64 = function(x) {\n      var rotr14 = rotr_64(x, 14),\n        rotr18 = rotr_64(x, 18),\n        rotr41 = rotr_64(x, 41);\n\n      return new Int_64(\n        rotr14.highOrder ^ rotr18.highOrder ^ rotr41.highOrder,\n        rotr14.lowOrder ^ rotr18.lowOrder ^ rotr41.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_32 = function(x) {\n      return rotr_32(x, 7) ^ rotr_32(x, 18) ^ shr_32(x, 3);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_64 = function(x) {\n      var rotr1 = rotr_64(x, 1),\n        rotr8 = rotr_64(x, 8),\n        shr7 = shr_64(x, 7);\n\n      return new Int_64(\n        rotr1.highOrder ^ rotr8.highOrder ^ shr7.highOrder,\n        rotr1.lowOrder ^ rotr8.lowOrder ^ shr7.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_32 = function(x) {\n      return rotr_32(x, 17) ^ rotr_32(x, 19) ^ shr_32(x, 10);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_64 = function(x) {\n      var rotr19 = rotr_64(x, 19),\n        rotr61 = rotr_64(x, 61),\n        shr6 = shr_64(x, 6);\n\n      return new Int_64(\n        rotr19.highOrder ^ rotr61.highOrder ^ shr6.highOrder,\n        rotr19.lowOrder ^ rotr61.lowOrder ^ shr6.lowOrder);\n    },\n\n    /*\n     * Add two 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument to be added\n     * @param {Number} y The second 32-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_32_2 = function(x, y) {\n      var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n        msw = (x >>> 16) + (y >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add four 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_32_4 = function(a, b, c, d) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add five 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @param {Number} e The fifth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_32_5 = function(a, b, c, d, e) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF) +\n        (e & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (e >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add two 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument to be added\n     * @param {Int_64} y The second 64-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_64_2 = function(x, y) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (x.lowOrder & 0xFFFF) + (y.lowOrder & 0xFFFF);\n      msw = (x.lowOrder >>> 16) + (y.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (x.highOrder & 0xFFFF) + (y.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (x.highOrder >>> 16) + (y.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add four 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_64_4 = function(a, b, c, d) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add five 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @param {Int_64} e The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_64_5 = function(a, b, c, d, e) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF) +\n        (e.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (e.lowOrder >>> 16) +\n        (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) +\n        (e.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) +\n        (e.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Calculates the SHA-1 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} message The binary array representation of the string to\n     *\t hash\n     * @param {Number} messageLen The number of bits in the message\n     * @return The array of integers representing the SHA-1 hash of message\n     */\n    coreSHA1 = function(message, messageLen) {\n      var W = [],\n        a, b, c, d, e, T, ch = ch_32,\n        parity = parity_32,\n        maj = maj_32,\n        rotl = rotl_32,\n        safeAdd_2 = safeAdd_32_2,\n        i, t,\n        safeAdd_5 = safeAdd_32_5,\n        appendedMessageLength,\n        H = [\n            0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0\n        ],\n        K = [\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6\n        ];\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - (messageLen % 32));\n      /* Append length of binary string in the position such that the new\n\t\tlength is a multiple of 512.  Logic does not work for even multiples\n\t\tof 512 but there can never be even multiples of 512 */\n      message[(((messageLen + 65) >> 9) << 4) + 15] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += 16) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n\n        for (t = 0; t < 80; t += 1) {\n          if (t < 16) {\n            W[t] = message[t + i];\n          } else {\n            W[t] = rotl(W[t - 3] ^ W[t - 8] ^ W[t - 14] ^ W[t - 16], 1);\n          }\n\n          if (t < 20) {\n            T = safeAdd_5(rotl(a, 5), ch(b, c, d), e, K[t], W[t]);\n          } else if (t < 40) {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          } else if (t < 60) {\n            T = safeAdd_5(rotl(a, 5), maj(b, c, d), e, K[t], W[t]);\n          } else {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          }\n\n          e = d;\n          d = c;\n          c = rotl(b, 30);\n          b = a;\n          a = T;\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n      }\n\n      return H;\n    },\n\n    /*\n     * Calculates the desired SHA-2 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} The binary array representation of the string to hash\n     * @param {Number} The number of bits in message\n     * @param {String} variant The desired SHA-2 variant\n     * @return The array of integers representing the SHA-2 hash of message\n     */\n    coreSHA2 = function(message, messageLen, variant) {\n      var a, b, c, d, e, f, g, h, T1, T2, H, numRounds, lengthPosition, i, t,\n        binaryStringInc, binaryStringMult, safeAdd_2, safeAdd_4, safeAdd_5,\n        gamma0, gamma1, sigma0, sigma1, ch, maj, Int, K, W = [],\n        appendedMessageLength;\n\n      /* Set up the various function handles and variable for the specific \n       * variant */\n      if (variant === \"SHA-224\" || variant === \"SHA-256\") {\n        /* 32-bit variant */\n        numRounds = 64;\n        lengthPosition = (((messageLen + 65) >> 9) << 4) + 15;\n        binaryStringInc = 16;\n        binaryStringMult = 1;\n        Int = Number;\n        safeAdd_2 = safeAdd_32_2;\n        safeAdd_4 = safeAdd_32_4;\n        safeAdd_5 = safeAdd_32_5;\n        gamma0 = gamma0_32;\n        gamma1 = gamma1_32;\n        sigma0 = sigma0_32;\n        sigma1 = sigma1_32;\n        maj = maj_32;\n        ch = ch_32;\n        K = [\n            0x428A2F98, 0x71374491, 0xB5C0FBCF, 0xE9B5DBA5,\n            0x3956C25B, 0x59F111F1, 0x923F82A4, 0xAB1C5ED5,\n            0xD807AA98, 0x12835B01, 0x243185BE, 0x550C7DC3,\n            0x72BE5D74, 0x80DEB1FE, 0x9BDC06A7, 0xC19BF174,\n            0xE49B69C1, 0xEFBE4786, 0x0FC19DC6, 0x240CA1CC,\n            0x2DE92C6F, 0x4A7484AA, 0x5CB0A9DC, 0x76F988DA,\n            0x983E5152, 0xA831C66D, 0xB00327C8, 0xBF597FC7,\n            0xC6E00BF3, 0xD5A79147, 0x06CA6351, 0x14292967,\n            0x27B70A85, 0x2E1B2138, 0x4D2C6DFC, 0x53380D13,\n            0x650A7354, 0x766A0ABB, 0x81C2C92E, 0x92722C85,\n            0xA2BFE8A1, 0xA81A664B, 0xC24B8B70, 0xC76C51A3,\n            0xD192E819, 0xD6990624, 0xF40E3585, 0x106AA070,\n            0x19A4C116, 0x1E376C08, 0x2748774C, 0x34B0BCB5,\n            0x391C0CB3, 0x4ED8AA4A, 0x5B9CCA4F, 0x682E6FF3,\n            0x748F82EE, 0x78A5636F, 0x84C87814, 0x8CC70208,\n            0x90BEFFFA, 0xA4506CEB, 0xBEF9A3F7, 0xC67178F2\n        ];\n\n        if (variant === \"SHA-224\") {\n          H = [\n              0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939,\n              0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4\n          ];\n        } else {\n          H = [\n              0x6A09E667, 0xBB67AE85, 0x3C6EF372, 0xA54FF53A,\n              0x510E527F, 0x9B05688C, 0x1F83D9AB, 0x5BE0CD19\n          ];\n        }\n      } else if (variant === \"SHA-384\" || variant === \"SHA-512\") {\n        /* 64-bit variant */\n        numRounds = 80;\n        lengthPosition = (((messageLen + 128) >> 10) << 5) + 31;\n        binaryStringInc = 32;\n        binaryStringMult = 2;\n        Int = Int_64;\n        safeAdd_2 = safeAdd_64_2;\n        safeAdd_4 = safeAdd_64_4;\n        safeAdd_5 = safeAdd_64_5;\n        gamma0 = gamma0_64;\n        gamma1 = gamma1_64;\n        sigma0 = sigma0_64;\n        sigma1 = sigma1_64;\n        maj = maj_64;\n        ch = ch_64;\n\n        K = [\n            new Int(0x428a2f98, 0xd728ae22), new Int(0x71374491, 0x23ef65cd),\n            new Int(0xb5c0fbcf, 0xec4d3b2f), new Int(0xe9b5dba5, 0x8189dbbc),\n            new Int(0x3956c25b, 0xf348b538), new Int(0x59f111f1, 0xb605d019),\n            new Int(0x923f82a4, 0xaf194f9b), new Int(0xab1c5ed5, 0xda6d8118),\n            new Int(0xd807aa98, 0xa3030242), new Int(0x12835b01, 0x45706fbe),\n            new Int(0x243185be, 0x4ee4b28c), new Int(0x550c7dc3, 0xd5ffb4e2),\n            new Int(0x72be5d74, 0xf27b896f), new Int(0x80deb1fe, 0x3b1696b1),\n            new Int(0x9bdc06a7, 0x25c71235), new Int(0xc19bf174, 0xcf692694),\n            new Int(0xe49b69c1, 0x9ef14ad2), new Int(0xefbe4786, 0x384f25e3),\n            new Int(0x0fc19dc6, 0x8b8cd5b5), new Int(0x240ca1cc, 0x77ac9c65),\n            new Int(0x2de92c6f, 0x592b0275), new Int(0x4a7484aa, 0x6ea6e483),\n            new Int(0x5cb0a9dc, 0xbd41fbd4), new Int(0x76f988da, 0x831153b5),\n            new Int(0x983e5152, 0xee66dfab), new Int(0xa831c66d, 0x2db43210),\n            new Int(0xb00327c8, 0x98fb213f), new Int(0xbf597fc7, 0xbeef0ee4),\n            new Int(0xc6e00bf3, 0x3da88fc2), new Int(0xd5a79147, 0x930aa725),\n            new Int(0x06ca6351, 0xe003826f), new Int(0x14292967, 0x0a0e6e70),\n            new Int(0x27b70a85, 0x46d22ffc), new Int(0x2e1b2138, 0x5c26c926),\n            new Int(0x4d2c6dfc, 0x5ac42aed), new Int(0x53380d13, 0x9d95b3df),\n            new Int(0x650a7354, 0x8baf63de), new Int(0x766a0abb, 0x3c77b2a8),\n            new Int(0x81c2c92e, 0x47edaee6), new Int(0x92722c85, 0x1482353b),\n            new Int(0xa2bfe8a1, 0x4cf10364), new Int(0xa81a664b, 0xbc423001),\n            new Int(0xc24b8b70, 0xd0f89791), new Int(0xc76c51a3, 0x0654be30),\n            new Int(0xd192e819, 0xd6ef5218), new Int(0xd6990624, 0x5565a910),\n            new Int(0xf40e3585, 0x5771202a), new Int(0x106aa070, 0x32bbd1b8),\n            new Int(0x19a4c116, 0xb8d2d0c8), new Int(0x1e376c08, 0x5141ab53),\n            new Int(0x2748774c, 0xdf8eeb99), new Int(0x34b0bcb5, 0xe19b48a8),\n            new Int(0x391c0cb3, 0xc5c95a63), new Int(0x4ed8aa4a, 0xe3418acb),\n            new Int(0x5b9cca4f, 0x7763e373), new Int(0x682e6ff3, 0xd6b2b8a3),\n            new Int(0x748f82ee, 0x5defb2fc), new Int(0x78a5636f, 0x43172f60),\n            new Int(0x84c87814, 0xa1f0ab72), new Int(0x8cc70208, 0x1a6439ec),\n            new Int(0x90befffa, 0x23631e28), new Int(0xa4506ceb, 0xde82bde9),\n            new Int(0xbef9a3f7, 0xb2c67915), new Int(0xc67178f2, 0xe372532b),\n            new Int(0xca273ece, 0xea26619c), new Int(0xd186b8c7, 0x21c0c207),\n            new Int(0xeada7dd6, 0xcde0eb1e), new Int(0xf57d4f7f, 0xee6ed178),\n            new Int(0x06f067aa, 0x72176fba), new Int(0x0a637dc5, 0xa2c898a6),\n            new Int(0x113f9804, 0xbef90dae), new Int(0x1b710b35, 0x131c471b),\n            new Int(0x28db77f5, 0x23047d84), new Int(0x32caab7b, 0x40c72493),\n            new Int(0x3c9ebe0a, 0x15c9bebc), new Int(0x431d67c4, 0x9c100d4c),\n            new Int(0x4cc5d4be, 0xcb3e42b6), new Int(0x597f299c, 0xfc657e2a),\n            new Int(0x5fcb6fab, 0x3ad6faec), new Int(0x6c44198c, 0x4a475817)\n        ];\n\n        if (variant === \"SHA-384\") {\n          H = [\n              new Int(0xcbbb9d5d, 0xc1059ed8), new Int(0x0629a292a, 0x367cd507),\n              new Int(0x9159015a, 0x3070dd17), new Int(0x0152fecd8, 0xf70e5939),\n              new Int(0x67332667, 0xffc00b31), new Int(0x98eb44a87, 0x68581511),\n              new Int(0xdb0c2e0d, 0x64f98fa7), new Int(0x047b5481d, 0xbefa4fa4)\n          ];\n        } else {\n          H = [\n              new Int(0x6a09e667, 0xf3bcc908), new Int(0xbb67ae85, 0x84caa73b),\n              new Int(0x3c6ef372, 0xfe94f82b), new Int(0xa54ff53a, 0x5f1d36f1),\n              new Int(0x510e527f, 0xade682d1), new Int(0x9b05688c, 0x2b3e6c1f),\n              new Int(0x1f83d9ab, 0xfb41bd6b), new Int(0x5be0cd19, 0x137e2179)\n          ];\n        }\n      }\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - messageLen % 32);\n      /* Append length of binary string in the position such that the new\n       * length is correct */\n      message[lengthPosition] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += binaryStringInc) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n        f = H[5];\n        g = H[6];\n        h = H[7];\n\n        for (t = 0; t < numRounds; t += 1) {\n          if (t < 16) {\n            /* Bit of a hack - for 32-bit, the second term is ignored */\n            W[t] = new Int(message[t * binaryStringMult + i],\n              message[t * binaryStringMult + i + 1]);\n          } else {\n            W[t] = safeAdd_4(\n              gamma1(W[t - 2]), W[t - 7],\n              gamma0(W[t - 15]), W[t - 16]);\n          }\n\n          T1 = safeAdd_5(h, sigma1(e), ch(e, f, g), K[t], W[t]);\n          T2 = safeAdd_2(sigma0(a), maj(a, b, c));\n          h = g;\n          g = f;\n          f = e;\n          e = safeAdd_2(d, T1);\n          d = c;\n          c = b;\n          b = a;\n          a = safeAdd_2(T1, T2);\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n        H[5] = safeAdd_2(f, H[5]);\n        H[6] = safeAdd_2(g, H[6]);\n        H[7] = safeAdd_2(h, H[7]);\n      }\n\n      switch (variant) {\n        case \"SHA-224\":\n          return [\n            H[0], H[1], H[2], H[3],\n            H[4], H[5], H[6]];\n        case \"SHA-256\":\n          return H;\n        case \"SHA-384\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder];\n        case \"SHA-512\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder,\n            H[6].highOrder, H[6].lowOrder,\n            H[7].highOrder, H[7].lowOrder];\n        default:\n          /* This should never be reached */\n          return [];\n      }\n    },\n\n    /*\n     * jsSHA is the workhorse of the library.  Instantiate it with the string to\n     * be hashed as the parameter\n     *\n     * @constructor\n     * @param {String} srcString The string to be hashed\n     * @param {String} inputFormat The format of srcString, ASCII or HEX\n     */\n    jsSHA = function(srcString, inputFormat) {\n\n      this.sha1 = null;\n      this.sha224 = null;\n      this.sha256 = null;\n      this.sha384 = null;\n      this.sha512 = null;\n\n      this.strBinLen = null;\n      this.strToHash = null;\n\n      /* Convert the input string into the correct type */\n      if (\"HEX\" === inputFormat) {\n        if (0 !== (srcString.length % 2)) {\n          return \"TEXT MUST BE IN BYTE INCREMENTS\";\n        }\n        this.strBinLen = srcString.length * 4;\n        this.strToHash = hex2binb(srcString);\n      } else if ((\"ASCII\" === inputFormat) ||\n        ('undefined' === typeof(inputFormat))) {\n        this.strBinLen = srcString.length * charSize;\n        this.strToHash = str2binb(srcString);\n      } else {\n        return \"UNKNOWN TEXT INPUT TYPE\";\n      }\n    };\n\n  jsSHA.prototype = {\n    /*\n     * Returns the desired SHA hash of the string specified at instantiation\n     * using the specified parameters\n     *\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} format The desired output formatting (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHash: function(variant, format) {\n      var formatFunc = null,\n        message = this.strToHash.slice();\n\n      switch (format) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      switch (variant) {\n        case \"SHA-1\":\n          if (null === this.sha1) {\n            this.sha1 = coreSHA1(message, this.strBinLen);\n          }\n          return formatFunc(this.sha1);\n        case \"SHA-224\":\n          if (null === this.sha224) {\n            this.sha224 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha224);\n        case \"SHA-256\":\n          if (null === this.sha256) {\n            this.sha256 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha256);\n        case \"SHA-384\":\n          if (null === this.sha384) {\n            this.sha384 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha384);\n        case \"SHA-512\":\n          if (null === this.sha512) {\n            this.sha512 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha512);\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n    },\n\n    /*\n     * Returns the desired HMAC of the string specified at instantiation\n     * using the key and variant param.\n     *\n     * @param {String} key The key used to calculate the HMAC\n     * @param {String} inputFormat The format of key, ASCII or HEX\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} outputFormat The desired output formatting\n     *\t (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHMAC: function(key, inputFormat, variant, outputFormat) {\n      var formatFunc, keyToUse, blockByteSize, blockBitSize, i,\n        retVal, lastArrayIndex, keyBinLen, hashBitSize,\n        keyWithIPad = [],\n        keyWithOPad = [];\n\n      /* Validate the output format selection */\n      switch (outputFormat) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      /* Validate the hash variant selection and set needed variables */\n      switch (variant) {\n        case \"SHA-1\":\n          blockByteSize = 64;\n          hashBitSize = 160;\n          break;\n        case \"SHA-224\":\n          blockByteSize = 64;\n          hashBitSize = 224;\n          break;\n        case \"SHA-256\":\n          blockByteSize = 64;\n          hashBitSize = 256;\n          break;\n        case \"SHA-384\":\n          blockByteSize = 128;\n          hashBitSize = 384;\n          break;\n        case \"SHA-512\":\n          blockByteSize = 128;\n          hashBitSize = 512;\n          break;\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n\n      /* Validate input format selection */\n      if (\"HEX\" === inputFormat) {\n        /* Nibbles must come in pairs */\n        if (0 !== (key.length % 2)) {\n          return \"KEY MUST BE IN BYTE INCREMENTS\";\n        }\n        keyToUse = hex2binb(key);\n        keyBinLen = key.length * 4;\n      } else if (\"ASCII\" === inputFormat) {\n        keyToUse = str2binb(key);\n        keyBinLen = key.length * charSize;\n      } else {\n        return \"UNKNOWN KEY INPUT TYPE\";\n      }\n\n      /* These are used multiple times, calculate and store them */\n      blockBitSize = blockByteSize * 8;\n      lastArrayIndex = (blockByteSize / 4) - 1;\n\n      /* Figure out what to do with the key based on its size relative to\n       * the hash's block size */\n      if (blockByteSize < (keyBinLen / 8)) {\n        if (\"SHA-1\" === variant) {\n          keyToUse = coreSHA1(keyToUse, keyBinLen);\n        } else {\n          keyToUse = coreSHA2(keyToUse, keyBinLen, variant);\n        }\n        /* For all variants, the block size is bigger than the output\n         * size so there will never be a useful byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      } else if (blockByteSize > (keyBinLen / 8)) {\n        /* If the blockByteSize is greater than the key length, there\n         * will always be at LEAST one \"useless\" byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      }\n\n      /* Create ipad and opad */\n      for (i = 0; i <= lastArrayIndex; i += 1) {\n        keyWithIPad[i] = keyToUse[i] ^ 0x36363636;\n        keyWithOPad[i] = keyToUse[i] ^ 0x5C5C5C5C;\n      }\n\n      /* Calculate the HMAC */\n      if (\"SHA-1\" === variant) {\n        retVal = coreSHA1(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen);\n        retVal = coreSHA1(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize);\n      } else {\n        retVal = coreSHA2(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen, variant);\n        retVal = coreSHA2(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize, variant);\n      }\n\n      return (formatFunc(retVal));\n    }\n  };\n\n  return jsSHA;\n}());\n\nmodule.exports = {\n  sha1: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-1\", \"ASCII\");\n  },\n  sha224: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-224\", \"ASCII\");\n  },\n  sha256: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-256\", \"ASCII\");\n  },\n  sha384: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-384\", \"ASCII\");\n\n  },\n  sha512: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-512\", \"ASCII\");\n  }\n}\n","module.exports = {\n  cipher: require('./cipher'),\n  hash: require('./hash'),\n  cfb: require('./cfb.js'),\n  publicKey: require('./public_key'),\n  signature: require('./signature.js'),\n  random: require('./random.js'),\n  pkcs1: require('./pkcs1.js')\n\n}\n\nvar crypto = require('./crypto.js');\n\nfor (var i in crypto)\n  module.exports[i] = crypto[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * ASN1 object identifiers for hashes (See RFC4880 5.2.2)\n */\nhash_headers = new Array();\nhash_headers[1] = [0x30, 0x20, 0x30, 0x0c, 0x06, 0x08, 0x2a, 0x86, 0x48, 0x86, 0xf7, 0x0d, 0x02, 0x05, 0x05, 0x00, 0x04,\n    0x10\n];\nhash_headers[3] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2B, 0x24, 0x03, 0x02, 0x01, 0x05, 0x00, 0x04, 0x14];\nhash_headers[2] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2b, 0x0e, 0x03, 0x02, 0x1a, 0x05, 0x00, 0x04, 0x14];\nhash_headers[8] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01, 0x05, 0x00,\n    0x04, 0x20\n];\nhash_headers[9] = [0x30, 0x41, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x02, 0x05, 0x00,\n    0x04, 0x30\n];\nhash_headers[10] = [0x30, 0x51, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x03, 0x05,\n    0x00, 0x04, 0x40\n];\nhash_headers[11] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x04, 0x05,\n    0x00, 0x04, 0x1C\n];\n\n\nvar crypto = require('./crypto.js'),\n  random = require('./random.js'),\n  util = require('../util'),\n  BigInteger = require('./public_key/jsbn.js'),\n  hash = require('./hash');\n\nmodule.exports = {\n  eme: {\n    /**\n     * create a EME-PKCS1-v1_5 padding (See RFC4880 13.1.1)\n     * @param {String} message message to be padded\n     * @param {Integer} length Length to the resulting message\n     * @return {String} EME-PKCS1 padded message\n     */\n    encode: function(message, length) {\n      if (message.length > length - 11)\n        return -1;\n      var result = \"\";\n      result += String.fromCharCode(0);\n      result += String.fromCharCode(2);\n      for (var i = 0; i < length - message.length - 3; i++) {\n        result += String.fromCharCode(random.getPseudoRandom(1, 255));\n      }\n      result += String.fromCharCode(0);\n      result += message;\n      return result;\n    },\n\n    /**\n     * decodes a EME-PKCS1-v1_5 padding (See RFC4880 13.1.2)\n     * @param {String} message EME-PKCS1 padded message\n     * @return {String} decoded message \n     */\n    decode: function(message, len) {\n      if (message.length < len)\n        message = String.fromCharCode(0) + message;\n      if (message.length < 12 || message.charCodeAt(0) != 0 || message.charCodeAt(1) != 2)\n        return -1;\n      var i = 2;\n      while (message.charCodeAt(i) != 0 && message.length > i)\n        i++;\n      return message.substring(i + 1, message.length);\n    },\n  },\n\n  emsa: {\n\n    /**\n     * create a EMSA-PKCS1-v1_5 padding (See RFC4880 13.1.3)\n     * @param {Integer} algo Hash algorithm type used\n     * @param {String} data Data to be hashed\n     * @param {Integer} keylength Key size of the public mpi in bytes\n     * @returns {String} Hashcode with pkcs1padding as string\n     */\n    encode: function(algo, data, keylength) {\n      var data2 = \"\";\n      data2 += String.fromCharCode(0x00);\n      data2 += String.fromCharCode(0x01);\n      for (var i = 0; i < (keylength - hash_headers[algo].length - 3 -\n        hash.getHashByteLength(algo)); i++)\n\n        data2 += String.fromCharCode(0xff);\n\n      data2 += String.fromCharCode(0x00);\n\n      for (var i = 0; i < hash_headers[algo].length; i++)\n        data2 += String.fromCharCode(hash_headers[algo][i]);\n\n      data2 += hash.digest(algo, data);\n      return new BigInteger(util.hexstrdump(data2), 16);\n    },\n\n    /**\n     * extract the hash out of an EMSA-PKCS1-v1.5 padding (See RFC4880 13.1.3) \n     * @param {String} data Hash in pkcs1 encoding\n     * @returns {String} The hash as string\n     */\n    decode: function(algo, data) {\n      var i = 0;\n      if (data.charCodeAt(0) == 0) i++;\n      else if (data.charCodeAt(0) != 1) return -1;\n      else i++;\n\n      while (data.charCodeAt(i) == 0xFF) i++;\n      if (data.charCodeAt(i++) != 0) return -1;\n      var j = 0;\n      for (j = 0; j < hash_headers[algo].length && j + i < data.length; j++) {\n        if (data.charCodeAt(j + i) != hash_headers[algo][j]) return -1;\n      }\n      i += j;\n      if (data.substring(i).length < hash.getHashByteLength(algo)) return -1;\n      return data.substring(i);\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// A Digital signature algorithm implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  hashModule = require('../hash'),\n  util = require('../../util');\n\nfunction DSA() {\n  // s1 = ((g**s) mod p) mod q\n  // s1 = ((s**-1)*(sha-1(m)+(s1*x) mod q)\n  function sign(hashalgo, m, g, p, q, x) {\n    // If the output size of the chosen hash is larger than the number of\n    // bits of q, the hash result is truncated to fit by taking the number\n    // of leftmost bits equal to the number of bits of q.  This (possibly\n    // truncated) hash function result is treated as a number and used\n    // directly in the DSA signature algorithm.\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    var k = random.getRandomBigIntegerInRange(BigInteger.ONE.add(BigInteger.ONE), q.subtract(BigInteger.ONE));\n    var s1 = (g.modPow(k, p)).mod(q);\n    var s2 = (k.modInverse(q).multiply(hash.add(x.multiply(s1)))).mod(q);\n    var result = new Array();\n    result[0] = s1.toMPI();\n    result[1] = s2.toMPI();\n    return result;\n  }\n\n  function select_hash_algorithm(q) {\n    var usersetting = openpgp.config.config.prefer_hash_algorithm;\n    /*\n     * 1024-bit key, 160-bit q, SHA-1, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 224-bit q, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     * 3072-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     */\n    switch (Math.round(q.bitLength() / 8)) {\n      case 20:\n        // 1024 bit\n        if (usersetting != 2 &&\n          usersetting > 11 &&\n          usersetting != 10 &&\n          usersetting < 8)\n          return 2; // prefer sha1\n        return usersetting;\n      case 28:\n        // 2048 bit\n        if (usersetting > 11 &&\n          usersetting < 8)\n          return 11;\n        return usersetting;\n      case 32:\n        // 4096 bit // prefer sha224\n        if (usersetting > 10 &&\n          usersetting < 8)\n          return 8; // prefer sha256\n        return usersetting;\n      default:\n        util.print_debug(\"DSA select hash algorithm: returning null for an unknown length of q\");\n        return null;\n\n    }\n  }\n  this.select_hash_algorithm = select_hash_algorithm;\n\n  function verify(hashalgo, s1, s2, m, p, q, g, y) {\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    if (BigInteger.ZERO.compareTo(s1) > 0 ||\n      s1.compareTo(q) > 0 ||\n      BigInteger.ZERO.compareTo(s2) > 0 ||\n      s2.compareTo(q) > 0) {\n      util.print_error(\"invalid DSA Signature\");\n      return null;\n    }\n    var w = s2.modInverse(q);\n    var u1 = hash.multiply(w).mod(q);\n    var u2 = s1.multiply(w).mod(q);\n    return g.modPow(u1, p).multiply(y.modPow(u2, p)).mod(p).mod(q);\n  }\n\n  /*\n\t * unused code. This can be used as a start to write a key generator\n\t * function.\n\t\n\tfunction generateKey(bitcount) {\n\t    var qi = new BigInteger(bitcount, primeCenterie);\n\t    var pi = generateP(q, 512);\n\t    var gi = generateG(p, q, bitcount);\n\t    var xi;\n\t    do {\n\t        xi = new BigInteger(q.bitCount(), rand);\n\t    } while (x.compareTo(BigInteger.ZERO) != 1 && x.compareTo(q) != -1);\n\t    var yi = g.modPow(x, p);\n\t    return {x: xi, q: qi, p: pi, g: gi, y: yi};\n\t}\n\n\tfunction generateP(q, bitlength, randomfn) {\n\t    if (bitlength % 64 != 0) {\n\t    \treturn false;\n\t    }\n\t    var pTemp;\n\t    var pTemp2;\n\t    do {\n\t        pTemp = randomfn(bitcount, true);\n\t        pTemp2 = pTemp.subtract(BigInteger.ONE);\n\t        pTemp = pTemp.subtract(pTemp2.remainder(q));\n\t    } while (!pTemp.isProbablePrime(primeCenterie) || pTemp.bitLength() != l);\n\t    return pTemp;\n\t}\n\t\n\tfunction generateG(p, q, bitlength, randomfn) {\n\t    var aux = p.subtract(BigInteger.ONE);\n\t    var pow = aux.divide(q);\n\t    var gTemp;\n\t    do {\n\t        gTemp = randomfn(bitlength);\n\t    } while (gTemp.compareTo(aux) != -1 && gTemp.compareTo(BigInteger.ONE) != 1);\n\t    return gTemp.modPow(pow, p);\n\t}\n\n\tfunction generateK(q, bitlength, randomfn) {\n\t    var tempK;\n\t    do {\n\t        tempK = randomfn(bitlength, false);\n\t    } while (tempK.compareTo(q) != -1 && tempK.compareTo(BigInteger.ZERO) != 1);\n\t    return tempK;\n\t}\n\n\tfunction generateR(q,p) {\n\t    k = generateK(q);\n\t    var r = g.modPow(k, p).mod(q);\n\t    return r;\n\t}\n\n\tfunction generateS(hashfn,k,r,m,q,x) {\n        var hash = hashfn(m);\n        s = (k.modInverse(q).multiply(hash.add(x.multiply(r)))).mod(q);\n\t    return s;\n\t} */\n  this.sign = sign;\n  this.verify = verify;\n  // this.generate = generateKey;\n}\n\nmodule.exports = DSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// ElGamal implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  util = require('../../util');\n\nfunction Elgamal() {\n\n  function encrypt(m, g, p, y) {\n    //  choose k in {2,...,p-2}\n    var two = BigInteger.ONE.add(BigInteger.ONE);\n    var pMinus2 = p.subtract(two);\n    var k = random.getRandomBigIntegerInRange(two, pMinus2);\n    k = k.mod(pMinus2).add(BigInteger.ONE);\n    var c = [];\n    c[0] = g.modPow(k, p);\n    c[1] = y.modPow(k, p).multiply(m).mod(p);\n    return c;\n  }\n\n  function decrypt(c1, c2, p, x) {\n    util.print_debug(\"Elgamal Decrypt:\\nc1:\" + util.hexstrdump(c1.toMPI()) + \"\\n\" +\n      \"c2:\" + util.hexstrdump(c2.toMPI()) + \"\\n\" +\n      \"p:\" + util.hexstrdump(p.toMPI()) + \"\\n\" +\n      \"x:\" + util.hexstrdump(x.toMPI()));\n    return (c1.modPow(x, p).modInverse(p)).multiply(c2).mod(p);\n    //var c = c1.pow(x).modInverse(p); // c0^-a mod p\n    //return c.multiply(c2).mod(p);\n  }\n\n  // signing and signature verification using Elgamal is not required by OpenPGP.\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n}\n\nmodule.exports = Elgamal;\n","module.exports = {\n  rsa: require('./rsa.js'),\n  elgamal: require('./elgamal.js'),\n  dsa: require('./dsa.js')\n}\n","/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH \n * \n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\nvar util = require('../../util');\n\n// Basic JavaScript BN library - subset useful for RSA encryption.\n\n// Bits per digit\nvar dbits;\n\n// JavaScript engine analysis\nvar canary = 0xdeadbeefcafe;\nvar j_lm = ((canary & 0xffffff) == 0xefcafe);\n\n// (public) Constructor\n\nfunction BigInteger(a, b, c) {\n  if (a != null)\n    if (\"number\" == typeof a) this.fromNumber(a, b, c);\n    else if (b == null && \"string\" != typeof a) this.fromString(a, 256);\n  else this.fromString(a, b);\n}\n\n// return new, unset BigInteger\n\nfunction nbi() {\n  return new BigInteger(null);\n}\n\n// am: Compute w_j += (x*this_i), propagate carries,\n// c is initial carry, returns final carry.\n// c < 3*dvalue, x < 2*dvalue, this_i < dvalue\n// We need to select the fastest one that works in this environment.\n\n// am1: use a single mult and divide to get the high bits,\n// max digit bits should be 26 because\n// max internal value = 2*dvalue^2-2*dvalue (< 2^53)\n\nfunction am1(i, x, w, j, c, n) {\n  while (--n >= 0) {\n    var v = x * this[i++] + w[j] + c;\n    c = Math.floor(v / 0x4000000);\n    w[j++] = v & 0x3ffffff;\n  }\n  return c;\n}\n// am2 avoids a big mult-and-extract completely.\n// Max digit bits should be <= 30 because we do bitwise ops\n// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)\n\nfunction am2(i, x, w, j, c, n) {\n  var xl = x & 0x7fff,\n    xh = x >> 15;\n  while (--n >= 0) {\n    var l = this[i] & 0x7fff;\n    var h = this[i++] >> 15;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x7fff) << 15) + w[j] + (c & 0x3fffffff);\n    c = (l >>> 30) + (m >>> 15) + xh * h + (c >>> 30);\n    w[j++] = l & 0x3fffffff;\n  }\n  return c;\n}\n// Alternately, set max digit bits to 28 since some\n// browsers slow down when dealing with 32-bit numbers.\n\nfunction am3(i, x, w, j, c, n) {\n  var xl = x & 0x3fff,\n    xh = x >> 14;\n  while (--n >= 0) {\n    var l = this[i] & 0x3fff;\n    var h = this[i++] >> 14;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x3fff) << 14) + w[j] + c;\n    c = (l >> 28) + (m >> 14) + xh * h;\n    w[j++] = l & 0xfffffff;\n  }\n  return c;\n}\n/*if(j_lm && (navigator != undefined && \n\tnavigator.appName == \"Microsoft Internet Explorer\")) {\n  BigInteger.prototype.am = am2;\n  dbits = 30;\n}\nelse if(j_lm && (navigator != undefined && navigator.appName != \"Netscape\")) {*/\nBigInteger.prototype.am = am1;\ndbits = 26;\n/*}\nelse { // Mozilla/Netscape seems to prefer am3\n  BigInteger.prototype.am = am3;\n  dbits = 28;\n}*/\n\nBigInteger.prototype.DB = dbits;\nBigInteger.prototype.DM = ((1 << dbits) - 1);\nBigInteger.prototype.DV = (1 << dbits);\n\nvar BI_FP = 52;\nBigInteger.prototype.FV = Math.pow(2, BI_FP);\nBigInteger.prototype.F1 = BI_FP - dbits;\nBigInteger.prototype.F2 = 2 * dbits - BI_FP;\n\n// Digit conversions\nvar BI_RM = \"0123456789abcdefghijklmnopqrstuvwxyz\";\nvar BI_RC = new Array();\nvar rr, vv;\nrr = \"0\".charCodeAt(0);\nfor (vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;\nrr = \"a\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\nrr = \"A\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\n\nfunction int2char(n) {\n  return BI_RM.charAt(n);\n}\n\nfunction intAt(s, i) {\n  var c = BI_RC[s.charCodeAt(i)];\n  return (c == null) ? -1 : c;\n}\n\n// (protected) copy this to r\n\nfunction bnpCopyTo(r) {\n  for (var i = this.t - 1; i >= 0; --i) r[i] = this[i];\n  r.t = this.t;\n  r.s = this.s;\n}\n\n// (protected) set from integer value x, -DV <= x < DV\n\nfunction bnpFromInt(x) {\n  this.t = 1;\n  this.s = (x < 0) ? -1 : 0;\n  if (x > 0) this[0] = x;\n  else if (x < -1) this[0] = x + DV;\n  else this.t = 0;\n}\n\n// return bigint initialized to value\n\nfunction nbv(i) {\n  var r = nbi();\n  r.fromInt(i);\n  return r;\n}\n\n// (protected) set from string and radix\n\nfunction bnpFromString(s, b) {\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 256) k = 8; // byte array\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else {\n    this.fromRadix(s, b);\n    return;\n  }\n  this.t = 0;\n  this.s = 0;\n  var i = s.length,\n    mi = false,\n    sh = 0;\n  while (--i >= 0) {\n    var x = (k == 8) ? s[i] & 0xff : intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\") mi = true;\n      continue;\n    }\n    mi = false;\n    if (sh == 0)\n      this[this.t++] = x;\n    else if (sh + k > this.DB) {\n      this[this.t - 1] |= (x & ((1 << (this.DB - sh)) - 1)) << sh;\n      this[this.t++] = (x >> (this.DB - sh));\n    } else\n      this[this.t - 1] |= x << sh;\n    sh += k;\n    if (sh >= this.DB) sh -= this.DB;\n  }\n  if (k == 8 && (s[0] & 0x80) != 0) {\n    this.s = -1;\n    if (sh > 0) this[this.t - 1] |= ((1 << (this.DB - sh)) - 1) << sh;\n  }\n  this.clamp();\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) clamp off excess high words\n\nfunction bnpClamp() {\n  var c = this.s & this.DM;\n  while (this.t > 0 && this[this.t - 1] == c)--this.t;\n}\n\n// (public) return string representation in given radix\n\nfunction bnToString(b) {\n  if (this.s < 0) return \"-\" + this.negate().toString(b);\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else return this.toRadix(b);\n  var km = (1 << k) - 1,\n    d, m = false,\n    r = \"\",\n    i = this.t;\n  var p = this.DB - (i * this.DB) % k;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) > 0) {\n      m = true;\n      r = int2char(d);\n    }\n    while (i >= 0) {\n      if (p < k) {\n        d = (this[i] & ((1 << p) - 1)) << (k - p);\n        d |= this[--i] >> (p += this.DB - k);\n      } else {\n        d = (this[i] >> (p -= k)) & km;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      if (d > 0) m = true;\n      if (m) r += int2char(d);\n    }\n  }\n  return m ? r : \"0\";\n}\n\n// (public) -this\n\nfunction bnNegate() {\n  var r = nbi();\n  BigInteger.ZERO.subTo(this, r);\n  return r;\n}\n\n// (public) |this|\n\nfunction bnAbs() {\n  return (this.s < 0) ? this.negate() : this;\n}\n\n// (public) return + if this > a, - if this < a, 0 if equal\n\nfunction bnCompareTo(a) {\n  var r = this.s - a.s;\n  if (r != 0) return r;\n  var i = this.t;\n  r = i - a.t;\n  if (r != 0) return r;\n  while (--i >= 0) if ((r = this[i] - a[i]) != 0) return r;\n  return 0;\n}\n\n// returns bit length of the integer x\n\nfunction nbits(x) {\n  var r = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    r += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    r += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    r += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    r += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    r += 1;\n  }\n  return r;\n}\n\n// (public) return the number of bits in \"this\"\n\nfunction bnBitLength() {\n  if (this.t <= 0) return 0;\n  return this.DB * (this.t - 1) + nbits(this[this.t - 1] ^ (this.s & this.DM));\n}\n\n// (protected) r = this << n*DB\n\nfunction bnpDLShiftTo(n, r) {\n  var i;\n  for (i = this.t - 1; i >= 0; --i) r[i + n] = this[i];\n  for (i = n - 1; i >= 0; --i) r[i] = 0;\n  r.t = this.t + n;\n  r.s = this.s;\n}\n\n// (protected) r = this >> n*DB\n\nfunction bnpDRShiftTo(n, r) {\n  for (var i = n; i < this.t; ++i) r[i - n] = this[i];\n  r.t = Math.max(this.t - n, 0);\n  r.s = this.s;\n}\n\n// (protected) r = this << n\n\nfunction bnpLShiftTo(n, r) {\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << cbs) - 1;\n  var ds = Math.floor(n / this.DB),\n    c = (this.s << bs) & this.DM,\n    i;\n  for (i = this.t - 1; i >= 0; --i) {\n    r[i + ds + 1] = (this[i] >> cbs) | c;\n    c = (this[i] & bm) << bs;\n  }\n  for (i = ds - 1; i >= 0; --i) r[i] = 0;\n  r[ds] = c;\n  r.t = this.t + ds + 1;\n  r.s = this.s;\n  r.clamp();\n}\n\n// (protected) r = this >> n\n\nfunction bnpRShiftTo(n, r) {\n  r.s = this.s;\n  var ds = Math.floor(n / this.DB);\n  if (ds >= this.t) {\n    r.t = 0;\n    return;\n  }\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << bs) - 1;\n  r[0] = this[ds] >> bs;\n  for (var i = ds + 1; i < this.t; ++i) {\n    r[i - ds - 1] |= (this[i] & bm) << cbs;\n    r[i - ds] = this[i] >> bs;\n  }\n  if (bs > 0) r[this.t - ds - 1] |= (this.s & bm) << cbs;\n  r.t = this.t - ds;\n  r.clamp();\n}\n\n// (protected) r = this - a\n\nfunction bnpSubTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] - a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c -= a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c -= a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c -= a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c < -1) r[i++] = this.DV + c;\n  else if (c > 0) r[i++] = c;\n  r.t = i;\n  r.clamp();\n}\n\n// (protected) r = this * a, r != this,a (HAC 14.12)\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyTo(a, r) {\n  var x = this.abs(),\n    y = a.abs();\n  var i = x.t;\n  r.t = i + y.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < y.t; ++i) r[i + x.t] = x.am(0, y[i], r, i, 0, x.t);\n  r.s = 0;\n  r.clamp();\n  if (this.s != a.s) BigInteger.ZERO.subTo(r, r);\n}\n\n// (protected) r = this^2, r != this (HAC 14.16)\n\nfunction bnpSquareTo(r) {\n  var x = this.abs();\n  var i = r.t = 2 * x.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < x.t - 1; ++i) {\n    var c = x.am(i, x[i], r, 2 * i, 0, 1);\n    if ((r[i + x.t] += x.am(i + 1, 2 * x[i], r, 2 * i + 1, c, x.t - i - 1)) >= x.DV) {\n      r[i + x.t] -= x.DV;\n      r[i + x.t + 1] = 1;\n    }\n  }\n  if (r.t > 0) r[r.t - 1] += x.am(i, x[i], r, 2 * i, 0, 1);\n  r.s = 0;\n  r.clamp();\n}\n\n// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)\n// r != q, this != m.  q or r may be null.\n\nfunction bnpDivRemTo(m, q, r) {\n  var pm = m.abs();\n  if (pm.t <= 0) return;\n  var pt = this.abs();\n  if (pt.t < pm.t) {\n    if (q != null) q.fromInt(0);\n    if (r != null) this.copyTo(r);\n    return;\n  }\n  if (r == null) r = nbi();\n  var y = nbi(),\n    ts = this.s,\n    ms = m.s;\n  var nsh = this.DB - nbits(pm[pm.t - 1]); // normalize modulus\n  if (nsh > 0) {\n    pm.lShiftTo(nsh, y);\n    pt.lShiftTo(nsh, r);\n  } else {\n    pm.copyTo(y);\n    pt.copyTo(r);\n  }\n  var ys = y.t;\n  var y0 = y[ys - 1];\n  if (y0 == 0) return;\n  var yt = y0 * (1 << this.F1) + ((ys > 1) ? y[ys - 2] >> this.F2 : 0);\n  var d1 = this.FV / yt,\n    d2 = (1 << this.F1) / yt,\n    e = 1 << this.F2;\n  var i = r.t,\n    j = i - ys,\n    t = (q == null) ? nbi() : q;\n  y.dlShiftTo(j, t);\n  if (r.compareTo(t) >= 0) {\n    r[r.t++] = 1;\n    r.subTo(t, r);\n  }\n  BigInteger.ONE.dlShiftTo(ys, t);\n  t.subTo(y, y); // \"negative\" y so we can replace sub with am later\n  while (y.t < ys) y[y.t++] = 0;\n  while (--j >= 0) {\n    // Estimate quotient digit\n    var qd = (r[--i] == y0) ? this.DM : Math.floor(r[i] * d1 + (r[i - 1] + e) * d2);\n    if ((r[i] += y.am(0, qd, r, j, 0, ys)) < qd) { // Try it out\n      y.dlShiftTo(j, t);\n      r.subTo(t, r);\n      while (r[i] < --qd) r.subTo(t, r);\n    }\n  }\n  if (q != null) {\n    r.drShiftTo(ys, q);\n    if (ts != ms) BigInteger.ZERO.subTo(q, q);\n  }\n  r.t = ys;\n  r.clamp();\n  if (nsh > 0) r.rShiftTo(nsh, r); // Denormalize remainder\n  if (ts < 0) BigInteger.ZERO.subTo(r, r);\n}\n\n// (public) this mod a\n\nfunction bnMod(a) {\n  var r = nbi();\n  this.abs().divRemTo(a, null, r);\n  if (this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r, r);\n  return r;\n}\n\n// Modular reduction using \"classic\" algorithm\n\nfunction Classic(m) {\n  this.m = m;\n}\n\nfunction cConvert(x) {\n  if (x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);\n  else return x;\n}\n\nfunction cRevert(x) {\n  return x;\n}\n\nfunction cReduce(x) {\n  x.divRemTo(this.m, null, x);\n}\n\nfunction cMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nfunction cSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\nClassic.prototype.convert = cConvert;\nClassic.prototype.revert = cRevert;\nClassic.prototype.reduce = cReduce;\nClassic.prototype.mulTo = cMulTo;\nClassic.prototype.sqrTo = cSqrTo;\n\n// (protected) return \"-1/this % 2^DB\"; useful for Mont. reduction\n// justification:\n//         xy == 1 (mod m)\n//         xy =  1+km\n//   xy(2-xy) = (1+km)(1-km)\n// x[y(2-xy)] = 1-k^2m^2\n// x[y(2-xy)] == 1 (mod m^2)\n// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2\n// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.\n// JS multiply \"overflows\" differently from C/C++, so care is needed here.\n\nfunction bnpInvDigit() {\n  if (this.t < 1) return 0;\n  var x = this[0];\n  if ((x & 1) == 0) return 0;\n  var y = x & 3; // y == 1/x mod 2^2\n  y = (y * (2 - (x & 0xf) * y)) & 0xf; // y == 1/x mod 2^4\n  y = (y * (2 - (x & 0xff) * y)) & 0xff; // y == 1/x mod 2^8\n  y = (y * (2 - (((x & 0xffff) * y) & 0xffff))) & 0xffff; // y == 1/x mod 2^16\n  // last step - calculate inverse mod DV directly;\n  // assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints\n  y = (y * (2 - x * y % this.DV)) % this.DV; // y == 1/x mod 2^dbits\n  // we really want the negative inverse, and -DV < y < DV\n  return (y > 0) ? this.DV - y : -y;\n}\n\n// Montgomery reduction\n\nfunction Montgomery(m) {\n  this.m = m;\n  this.mp = m.invDigit();\n  this.mpl = this.mp & 0x7fff;\n  this.mph = this.mp >> 15;\n  this.um = (1 << (m.DB - 15)) - 1;\n  this.mt2 = 2 * m.t;\n}\n\n// xR mod m\n\nfunction montConvert(x) {\n  var r = nbi();\n  x.abs().dlShiftTo(this.m.t, r);\n  r.divRemTo(this.m, null, r);\n  if (x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r, r);\n  return r;\n}\n\n// x/R mod m\n\nfunction montRevert(x) {\n  var r = nbi();\n  x.copyTo(r);\n  this.reduce(r);\n  return r;\n}\n\n// x = x/R mod m (HAC 14.32)\n\nfunction montReduce(x) {\n  while (x.t <= this.mt2) // pad x so am has enough room later\n    x[x.t++] = 0;\n  for (var i = 0; i < this.m.t; ++i) {\n    // faster way of calculating u0 = x[i]*mp mod DV\n    var j = x[i] & 0x7fff;\n    var u0 = (j * this.mpl + (((j * this.mph + (x[i] >> 15) * this.mpl) & this.um) << 15)) & x.DM;\n    // use am to combine the multiply-shift-add into one call\n    j = i + this.m.t;\n    x[j] += this.m.am(0, u0, x, i, 0, this.m.t);\n    // propagate carry\n    while (x[j] >= x.DV) {\n      x[j] -= x.DV;\n      x[++j]++;\n    }\n  }\n  x.clamp();\n  x.drShiftTo(this.m.t, x);\n  if (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = \"x^2/R mod m\"; x != r\n\nfunction montSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = \"xy/R mod m\"; x,y != r\n\nfunction montMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nMontgomery.prototype.convert = montConvert;\nMontgomery.prototype.revert = montRevert;\nMontgomery.prototype.reduce = montReduce;\nMontgomery.prototype.mulTo = montMulTo;\nMontgomery.prototype.sqrTo = montSqrTo;\n\n// (protected) true iff this is even\n\nfunction bnpIsEven() {\n  return ((this.t > 0) ? (this[0] & 1) : this.s) == 0;\n}\n\n// (protected) this^e, e < 2^32, doing sqr and mul with \"r\" (HAC 14.79)\n\nfunction bnpExp(e, z) {\n  if (e > 0xffffffff || e < 1) return BigInteger.ONE;\n  var r = nbi(),\n    r2 = nbi(),\n    g = z.convert(this),\n    i = nbits(e) - 1;\n  g.copyTo(r);\n  while (--i >= 0) {\n    z.sqrTo(r, r2);\n    if ((e & (1 << i)) > 0) z.mulTo(r2, g, r);\n    else {\n      var t = r;\n      r = r2;\n      r2 = t;\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) this^e % m, 0 <= e < 2^32\n\nfunction bnModPowInt(e, m) {\n  var z;\n  if (e < 256 || m.isEven()) z = new Classic(m);\n  else z = new Montgomery(m);\n  return this.exp(e, z);\n}\n\n// protected\nBigInteger.prototype.copyTo = bnpCopyTo;\nBigInteger.prototype.fromInt = bnpFromInt;\nBigInteger.prototype.fromString = bnpFromString;\nBigInteger.prototype.clamp = bnpClamp;\nBigInteger.prototype.dlShiftTo = bnpDLShiftTo;\nBigInteger.prototype.drShiftTo = bnpDRShiftTo;\nBigInteger.prototype.lShiftTo = bnpLShiftTo;\nBigInteger.prototype.rShiftTo = bnpRShiftTo;\nBigInteger.prototype.subTo = bnpSubTo;\nBigInteger.prototype.multiplyTo = bnpMultiplyTo;\nBigInteger.prototype.squareTo = bnpSquareTo;\nBigInteger.prototype.divRemTo = bnpDivRemTo;\nBigInteger.prototype.invDigit = bnpInvDigit;\nBigInteger.prototype.isEven = bnpIsEven;\nBigInteger.prototype.exp = bnpExp;\n\n// public\nBigInteger.prototype.toString = bnToString;\nBigInteger.prototype.negate = bnNegate;\nBigInteger.prototype.abs = bnAbs;\nBigInteger.prototype.compareTo = bnCompareTo;\nBigInteger.prototype.bitLength = bnBitLength;\nBigInteger.prototype.mod = bnMod;\nBigInteger.prototype.modPowInt = bnModPowInt;\n\n// \"constants\"\nBigInteger.ZERO = nbv(0);\nBigInteger.ONE = nbv(1);\n\nmodule.exports = BigInteger;\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH\n *\n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\n// Extended JavaScript BN functions, required for RSA private ops.\n\n// Version 1.1: new BigInteger(\"0\", 10) returns \"proper\" zero\n// Version 1.2: square() API, isProbablePrime fix\n\n// (public)\nfunction bnClone() {\n  var r = nbi();\n  this.copyTo(r);\n  return r;\n}\n\n// (public) return value as integer\n\nfunction bnIntValue() {\n  if (this.s < 0) {\n    if (this.t == 1) return this[0] - this.DV;\n    else if (this.t == 0) return -1;\n  } else if (this.t == 1) return this[0];\n  else if (this.t == 0) return 0;\n  // assumes 16 < DB < 32\n  return ((this[1] & ((1 << (32 - this.DB)) - 1)) << this.DB) | this[0];\n}\n\n// (public) return value as byte\n\nfunction bnByteValue() {\n  return (this.t == 0) ? this.s : (this[0] << 24) >> 24;\n}\n\n// (public) return value as short (assumes DB>=16)\n\nfunction bnShortValue() {\n  return (this.t == 0) ? this.s : (this[0] << 16) >> 16;\n}\n\n// (protected) return x s.t. r^x < DV\n\nfunction bnpChunkSize(r) {\n  return Math.floor(Math.LN2 * this.DB / Math.log(r));\n}\n\n// (public) 0 if this == 0, 1 if this > 0\n\nfunction bnSigNum() {\n  if (this.s < 0) return -1;\n  else if (this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;\n  else return 1;\n}\n\n// (protected) convert to radix string\n\nfunction bnpToRadix(b) {\n  if (b == null) b = 10;\n  if (this.signum() == 0 || b < 2 || b > 36) return \"0\";\n  var cs = this.chunkSize(b);\n  var a = Math.pow(b, cs);\n  var d = nbv(a),\n    y = nbi(),\n    z = nbi(),\n    r = \"\";\n  this.divRemTo(d, y, z);\n  while (y.signum() > 0) {\n    r = (a + z.intValue()).toString(b).substr(1) + r;\n    y.divRemTo(d, y, z);\n  }\n  return z.intValue().toString(b) + r;\n}\n\n// (protected) convert from radix string\n\nfunction bnpFromRadix(s, b) {\n  this.fromInt(0);\n  if (b == null) b = 10;\n  var cs = this.chunkSize(b);\n  var d = Math.pow(b, cs),\n    mi = false,\n    j = 0,\n    w = 0;\n  for (var i = 0; i < s.length; ++i) {\n    var x = intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\" && this.signum() == 0) mi = true;\n      continue;\n    }\n    w = b * w + x;\n    if (++j >= cs) {\n      this.dMultiply(d);\n      this.dAddOffset(w, 0);\n      j = 0;\n      w = 0;\n    }\n  }\n  if (j > 0) {\n    this.dMultiply(Math.pow(b, j));\n    this.dAddOffset(w, 0);\n  }\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) alternate constructor\n\nfunction bnpFromNumber(a, b, c) {\n  if (\"number\" == typeof b) {\n    // new BigInteger(int,int,RNG)\n    if (a < 2) this.fromInt(1);\n    else {\n      this.fromNumber(a, c);\n      if (!this.testBit(a - 1)) // force MSB set\n        this.bitwiseTo(BigInteger.ONE.shiftLeft(a - 1), op_or, this);\n      if (this.isEven()) this.dAddOffset(1, 0); // force odd\n      while (!this.isProbablePrime(b)) {\n        this.dAddOffset(2, 0);\n        if (this.bitLength() > a) this.subTo(BigInteger.ONE.shiftLeft(a - 1), this);\n      }\n    }\n  } else {\n    // new BigInteger(int,RNG)\n    var x = new Array(),\n      t = a & 7;\n    x.length = (a >> 3) + 1;\n    b.nextBytes(x);\n    if (t > 0) x[0] &= ((1 << t) - 1);\n    else x[0] = 0;\n    this.fromString(x, 256);\n  }\n}\n\n// (public) convert to bigendian byte array\n\nfunction bnToByteArray() {\n  var i = this.t,\n    r = new Array();\n  r[0] = this.s;\n  var p = this.DB - (i * this.DB) % 8,\n    d, k = 0;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) != (this.s & this.DM) >> p)\n      r[k++] = d | (this.s << (this.DB - p));\n    while (i >= 0) {\n      if (p < 8) {\n        d = (this[i] & ((1 << p) - 1)) << (8 - p);\n        d |= this[--i] >> (p += this.DB - 8);\n      } else {\n        d = (this[i] >> (p -= 8)) & 0xff;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      //if((d&0x80) != 0) d |= -256;\n      //if(k == 0 && (this.s&0x80) != (d&0x80)) ++k;\n      if (k > 0 || d != this.s) r[k++] = d;\n    }\n  }\n  return r;\n}\n\nfunction bnEquals(a) {\n  return (this.compareTo(a) == 0);\n}\n\nfunction bnMin(a) {\n  return (this.compareTo(a) < 0) ? this : a;\n}\n\nfunction bnMax(a) {\n  return (this.compareTo(a) > 0) ? this : a;\n}\n\n// (protected) r = this op a (bitwise)\n\nfunction bnpBitwiseTo(a, op, r) {\n  var i, f, m = Math.min(a.t, this.t);\n  for (i = 0; i < m; ++i) r[i] = op(this[i], a[i]);\n  if (a.t < this.t) {\n    f = a.s & this.DM;\n    for (i = m; i < this.t; ++i) r[i] = op(this[i], f);\n    r.t = this.t;\n  } else {\n    f = this.s & this.DM;\n    for (i = m; i < a.t; ++i) r[i] = op(f, a[i]);\n    r.t = a.t;\n  }\n  r.s = op(this.s, a.s);\n  r.clamp();\n}\n\n// (public) this & a\n\nfunction op_and(x, y) {\n  return x & y;\n}\n\nfunction bnAnd(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_and, r);\n  return r;\n}\n\n// (public) this | a\n\nfunction op_or(x, y) {\n  return x | y;\n}\n\nfunction bnOr(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_or, r);\n  return r;\n}\n\n// (public) this ^ a\n\nfunction op_xor(x, y) {\n  return x ^ y;\n}\n\nfunction bnXor(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_xor, r);\n  return r;\n}\n\n// (public) this & ~a\n\nfunction op_andnot(x, y) {\n  return x & ~y;\n}\n\nfunction bnAndNot(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_andnot, r);\n  return r;\n}\n\n// (public) ~this\n\nfunction bnNot() {\n  var r = nbi();\n  for (var i = 0; i < this.t; ++i) r[i] = this.DM & ~this[i];\n  r.t = this.t;\n  r.s = ~this.s;\n  return r;\n}\n\n// (public) this << n\n\nfunction bnShiftLeft(n) {\n  var r = nbi();\n  if (n < 0) this.rShiftTo(-n, r);\n  else this.lShiftTo(n, r);\n  return r;\n}\n\n// (public) this >> n\n\nfunction bnShiftRight(n) {\n  var r = nbi();\n  if (n < 0) this.lShiftTo(-n, r);\n  else this.rShiftTo(n, r);\n  return r;\n}\n\n// return index of lowest 1-bit in x, x < 2^31\n\nfunction lbit(x) {\n  if (x == 0) return -1;\n  var r = 0;\n  if ((x & 0xffff) == 0) {\n    x >>= 16;\n    r += 16;\n  }\n  if ((x & 0xff) == 0) {\n    x >>= 8;\n    r += 8;\n  }\n  if ((x & 0xf) == 0) {\n    x >>= 4;\n    r += 4;\n  }\n  if ((x & 3) == 0) {\n    x >>= 2;\n    r += 2;\n  }\n  if ((x & 1) == 0)++r;\n  return r;\n}\n\n// (public) returns index of lowest 1-bit (or -1 if none)\n\nfunction bnGetLowestSetBit() {\n  for (var i = 0; i < this.t; ++i)\n    if (this[i] != 0) return i * this.DB + lbit(this[i]);\n  if (this.s < 0) return this.t * this.DB;\n  return -1;\n}\n\n// return number of 1 bits in x\n\nfunction cbit(x) {\n  var r = 0;\n  while (x != 0) {\n    x &= x - 1;\n    ++r;\n  }\n  return r;\n}\n\n// (public) return number of set bits\n\nfunction bnBitCount() {\n  var r = 0,\n    x = this.s & this.DM;\n  for (var i = 0; i < this.t; ++i) r += cbit(this[i] ^ x);\n  return r;\n}\n\n// (public) true iff nth bit is set\n\nfunction bnTestBit(n) {\n  var j = Math.floor(n / this.DB);\n  if (j >= this.t) return (this.s != 0);\n  return ((this[j] & (1 << (n % this.DB))) != 0);\n}\n\n// (protected) this op (1<<n)\n\nfunction bnpChangeBit(n, op) {\n  var r = BigInteger.ONE.shiftLeft(n);\n  this.bitwiseTo(r, op, r);\n  return r;\n}\n\n// (public) this | (1<<n)\n\nfunction bnSetBit(n) {\n  return this.changeBit(n, op_or);\n}\n\n// (public) this & ~(1<<n)\n\nfunction bnClearBit(n) {\n  return this.changeBit(n, op_andnot);\n}\n\n// (public) this ^ (1<<n)\n\nfunction bnFlipBit(n) {\n  return this.changeBit(n, op_xor);\n}\n\n// (protected) r = this + a\n\nfunction bnpAddTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] + a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c += a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c += a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c > 0) r[i++] = c;\n  else if (c < -1) r[i++] = this.DV + c;\n  r.t = i;\n  r.clamp();\n}\n\n// (public) this + a\n\nfunction bnAdd(a) {\n  var r = nbi();\n  this.addTo(a, r);\n  return r;\n}\n\n// (public) this - a\n\nfunction bnSubtract(a) {\n  var r = nbi();\n  this.subTo(a, r);\n  return r;\n}\n\n// (public) this * a\n\nfunction bnMultiply(a) {\n  var r = nbi();\n  this.multiplyTo(a, r);\n  return r;\n}\n\n// (public) this^2\n\nfunction bnSquare() {\n  var r = nbi();\n  this.squareTo(r);\n  return r;\n}\n\n// (public) this / a\n\nfunction bnDivide(a) {\n  var r = nbi();\n  this.divRemTo(a, r, null);\n  return r;\n}\n\n// (public) this % a\n\nfunction bnRemainder(a) {\n  var r = nbi();\n  this.divRemTo(a, null, r);\n  return r;\n}\n\n// (public) [this/a,this%a]\n\nfunction bnDivideAndRemainder(a) {\n  var q = nbi(),\n    r = nbi();\n  this.divRemTo(a, q, r);\n  return new Array(q, r);\n}\n\n// (protected) this *= n, this >= 0, 1 < n < DV\n\nfunction bnpDMultiply(n) {\n  this[this.t] = this.am(0, n - 1, this, 0, 0, this.t);\n  ++this.t;\n  this.clamp();\n}\n\n// (protected) this += n << w words, this >= 0\n\nfunction bnpDAddOffset(n, w) {\n  if (n == 0) return;\n  while (this.t <= w) this[this.t++] = 0;\n  this[w] += n;\n  while (this[w] >= this.DV) {\n    this[w] -= this.DV;\n    if (++w >= this.t) this[this.t++] = 0;\n    ++this[w];\n  }\n}\n\n// A \"null\" reducer\n\nfunction NullExp() {}\n\nfunction nNop(x) {\n  return x;\n}\n\nfunction nMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n}\n\nfunction nSqrTo(x, r) {\n  x.squareTo(r);\n}\n\nNullExp.prototype.convert = nNop;\nNullExp.prototype.revert = nNop;\nNullExp.prototype.mulTo = nMulTo;\nNullExp.prototype.sqrTo = nSqrTo;\n\n// (public) this^e\n\nfunction bnPow(e) {\n  return this.exp(e, new NullExp());\n}\n\n// (protected) r = lower n words of \"this * a\", a.t <= n\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyLowerTo(a, n, r) {\n  var i = Math.min(this.t + a.t, n);\n  r.s = 0; // assumes a,this >= 0\n  r.t = i;\n  while (i > 0) r[--i] = 0;\n  var j;\n  for (j = r.t - this.t; i < j; ++i) r[i + this.t] = this.am(0, a[i], r, i, 0, this.t);\n  for (j = Math.min(a.t, n); i < j; ++i) this.am(0, a[i], r, i, 0, n - i);\n  r.clamp();\n}\n\n// (protected) r = \"this * a\" without lower n words, n > 0\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyUpperTo(a, n, r) {\n  --n;\n  var i = r.t = this.t + a.t - n;\n  r.s = 0; // assumes a,this >= 0\n  while (--i >= 0) r[i] = 0;\n  for (i = Math.max(n - this.t, 0); i < a.t; ++i)\n    r[this.t + i - n] = this.am(n - i, a[i], r, 0, 0, this.t + i - n);\n  r.clamp();\n  r.drShiftTo(1, r);\n}\n\n// Barrett modular reduction\n\nfunction Barrett(m) {\n  // setup Barrett\n  this.r2 = nbi();\n  this.q3 = nbi();\n  BigInteger.ONE.dlShiftTo(2 * m.t, this.r2);\n  this.mu = this.r2.divide(m);\n  this.m = m;\n}\n\nfunction barrettConvert(x) {\n  if (x.s < 0 || x.t > 2 * this.m.t) return x.mod(this.m);\n  else if (x.compareTo(this.m) < 0) return x;\n  else {\n    var r = nbi();\n    x.copyTo(r);\n    this.reduce(r);\n    return r;\n  }\n}\n\nfunction barrettRevert(x) {\n  return x;\n}\n\n// x = x mod m (HAC 14.42)\n\nfunction barrettReduce(x) {\n  x.drShiftTo(this.m.t - 1, this.r2);\n  if (x.t > this.m.t + 1) {\n    x.t = this.m.t + 1;\n    x.clamp();\n  }\n  this.mu.multiplyUpperTo(this.r2, this.m.t + 1, this.q3);\n  this.m.multiplyLowerTo(this.q3, this.m.t + 1, this.r2);\n  while (x.compareTo(this.r2) < 0) x.dAddOffset(1, this.m.t + 1);\n  x.subTo(this.r2, x);\n  while (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = x^2 mod m; x != r\n\nfunction barrettSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = x*y mod m; x,y != r\n\nfunction barrettMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nBarrett.prototype.convert = barrettConvert;\nBarrett.prototype.revert = barrettRevert;\nBarrett.prototype.reduce = barrettReduce;\nBarrett.prototype.mulTo = barrettMulTo;\nBarrett.prototype.sqrTo = barrettSqrTo;\n\n// (public) this^e % m (HAC 14.85)\n\nfunction bnModPow(e, m) {\n  var i = e.bitLength(),\n    k, r = nbv(1),\n    z;\n  if (i <= 0) return r;\n  else if (i < 18) k = 1;\n  else if (i < 48) k = 3;\n  else if (i < 144) k = 4;\n  else if (i < 768) k = 5;\n  else k = 6;\n  if (i < 8)\n    z = new Classic(m);\n  else if (m.isEven())\n    z = new Barrett(m);\n  else\n    z = new Montgomery(m);\n\n  // precomputation\n  var g = new Array(),\n    n = 3,\n    k1 = k - 1,\n    km = (1 << k) - 1;\n  g[1] = z.convert(this);\n  if (k > 1) {\n    var g2 = nbi();\n    z.sqrTo(g[1], g2);\n    while (n <= km) {\n      g[n] = nbi();\n      z.mulTo(g2, g[n - 2], g[n]);\n      n += 2;\n    }\n  }\n\n  var j = e.t - 1,\n    w, is1 = true,\n    r2 = nbi(),\n    t;\n  i = nbits(e[j]) - 1;\n  while (j >= 0) {\n    if (i >= k1) w = (e[j] >> (i - k1)) & km;\n    else {\n      w = (e[j] & ((1 << (i + 1)) - 1)) << (k1 - i);\n      if (j > 0) w |= e[j - 1] >> (this.DB + i - k1);\n    }\n\n    n = k;\n    while ((w & 1) == 0) {\n      w >>= 1;\n      --n;\n    }\n    if ((i -= n) < 0) {\n      i += this.DB;\n      --j;\n    }\n    if (is1) { // ret == 1, don't bother squaring or multiplying it\n      g[w].copyTo(r);\n      is1 = false;\n    } else {\n      while (n > 1) {\n        z.sqrTo(r, r2);\n        z.sqrTo(r2, r);\n        n -= 2;\n      }\n      if (n > 0) z.sqrTo(r, r2);\n      else {\n        t = r;\n        r = r2;\n        r2 = t;\n      }\n      z.mulTo(r2, g[w], r);\n    }\n\n    while (j >= 0 && (e[j] & (1 << i)) == 0) {\n      z.sqrTo(r, r2);\n      t = r;\n      r = r2;\n      r2 = t;\n      if (--i < 0) {\n        i = this.DB - 1;\n        --j;\n      }\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) gcd(this,a) (HAC 14.54)\n\nfunction bnGCD(a) {\n  var x = (this.s < 0) ? this.negate() : this.clone();\n  var y = (a.s < 0) ? a.negate() : a.clone();\n  if (x.compareTo(y) < 0) {\n    var t = x;\n    x = y;\n    y = t;\n  }\n  var i = x.getLowestSetBit(),\n    g = y.getLowestSetBit();\n  if (g < 0) return x;\n  if (i < g) g = i;\n  if (g > 0) {\n    x.rShiftTo(g, x);\n    y.rShiftTo(g, y);\n  }\n  while (x.signum() > 0) {\n    if ((i = x.getLowestSetBit()) > 0) x.rShiftTo(i, x);\n    if ((i = y.getLowestSetBit()) > 0) y.rShiftTo(i, y);\n    if (x.compareTo(y) >= 0) {\n      x.subTo(y, x);\n      x.rShiftTo(1, x);\n    } else {\n      y.subTo(x, y);\n      y.rShiftTo(1, y);\n    }\n  }\n  if (g > 0) y.lShiftTo(g, y);\n  return y;\n}\n\n// (protected) this % n, n < 2^26\n\nfunction bnpModInt(n) {\n  if (n <= 0) return 0;\n  var d = this.DV % n,\n    r = (this.s < 0) ? n - 1 : 0;\n  if (this.t > 0)\n    if (d == 0) r = this[0] % n;\n    else for (var i = this.t - 1; i >= 0; --i) r = (d * r + this[i]) % n;\n  return r;\n}\n\n// (public) 1/this % m (HAC 14.61)\n\nfunction bnModInverse(m) {\n  var ac = m.isEven();\n  if ((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;\n  var u = m.clone(),\n    v = this.clone();\n  var a = nbv(1),\n    b = nbv(0),\n    c = nbv(0),\n    d = nbv(1);\n  while (u.signum() != 0) {\n    while (u.isEven()) {\n      u.rShiftTo(1, u);\n      if (ac) {\n        if (!a.isEven() || !b.isEven()) {\n          a.addTo(this, a);\n          b.subTo(m, b);\n        }\n        a.rShiftTo(1, a);\n      } else if (!b.isEven()) b.subTo(m, b);\n      b.rShiftTo(1, b);\n    }\n    while (v.isEven()) {\n      v.rShiftTo(1, v);\n      if (ac) {\n        if (!c.isEven() || !d.isEven()) {\n          c.addTo(this, c);\n          d.subTo(m, d);\n        }\n        c.rShiftTo(1, c);\n      } else if (!d.isEven()) d.subTo(m, d);\n      d.rShiftTo(1, d);\n    }\n    if (u.compareTo(v) >= 0) {\n      u.subTo(v, u);\n      if (ac) a.subTo(c, a);\n      b.subTo(d, b);\n    } else {\n      v.subTo(u, v);\n      if (ac) c.subTo(a, c);\n      d.subTo(b, d);\n    }\n  }\n  if (v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;\n  if (d.compareTo(m) >= 0) return d.subtract(m);\n  if (d.signum() < 0) d.addTo(m, d);\n  else return d;\n  if (d.signum() < 0) return d.add(m);\n  else return d;\n}\n\nvar lowprimes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97, 101,\n    103, 107, 109, 113, 127, 131, 137, 139, 149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223, 227,\n    229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293, 307, 311, 313, 317, 331, 337, 347, 349, 353, 359,\n    367, 373, 379, 383, 389, 397, 401, 409, 419, 421, 431, 433, 439, 443, 449, 457, 461, 463, 467, 479, 487, 491, 499,\n    503, 509, 521, 523, 541, 547, 557, 563, 569, 571, 577, 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, 643, 647,\n    653, 659, 661, 673, 677, 683, 691, 701, 709, 719, 727, 733, 739, 743, 751, 757, 761, 769, 773, 787, 797, 809, 811,\n    821, 823, 827, 829, 839, 853, 857, 859, 863, 877, 881, 883, 887, 907, 911, 919, 929, 937, 941, 947, 953, 967, 971,\n    977, 983, 991, 997\n];\nvar lplim = (1 << 26) / lowprimes[lowprimes.length - 1];\n\n// (public) test primality with certainty >= 1-.5^t\n\nfunction bnIsProbablePrime(t) {\n  var i, x = this.abs();\n  if (x.t == 1 && x[0] <= lowprimes[lowprimes.length - 1]) {\n    for (i = 0; i < lowprimes.length; ++i)\n      if (x[0] == lowprimes[i]) return true;\n    return false;\n  }\n  if (x.isEven()) return false;\n  i = 1;\n  while (i < lowprimes.length) {\n    var m = lowprimes[i],\n      j = i + 1;\n    while (j < lowprimes.length && m < lplim) m *= lowprimes[j++];\n    m = x.modInt(m);\n    while (i < j) if (m % lowprimes[i++] == 0) return false;\n  }\n  return x.millerRabin(t);\n}\n\n/* added by Recurity Labs */\n\nfunction nbits(x) {\n  var n = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    n += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    n += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    n += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    n += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    n += 1;\n  }\n  return n;\n}\n\nfunction bnToMPI() {\n  var ba = this.toByteArray();\n  var size = (ba.length - 1) * 8 + nbits(ba[0]);\n  var result = \"\";\n  result += String.fromCharCode((size & 0xFF00) >> 8);\n  result += String.fromCharCode(size & 0xFF);\n  result += util.bin2str(ba);\n  return result;\n}\n/* END of addition */\n\n// (protected) true if probably prime (HAC 4.24, Miller-Rabin)\nfunction bnpMillerRabin(t) {\n  var n1 = this.subtract(BigInteger.ONE);\n  var k = n1.getLowestSetBit();\n  if (k <= 0) return false;\n  var r = n1.shiftRight(k);\n  t = (t + 1) >> 1;\n  if (t > lowprimes.length) t = lowprimes.length;\n  var a = nbi();\n  var j, bases = [];\n  for (var i = 0; i < t; ++i) {\n    //Pick bases at random, instead of starting at 2\n    for (;;) {\n      j = lowprimes[Math.floor(Math.random() * lowprimes.length)];\n      if (bases.indexOf(j) == -1) break;\n    }\n    bases.push(j);\n    a.fromInt(j);\n    var y = a.modPow(r, this);\n    if (y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {\n      var j = 1;\n      while (j++ < k && y.compareTo(n1) != 0) {\n        y = y.modPowInt(2, this);\n        if (y.compareTo(BigInteger.ONE) == 0) return false;\n      }\n      if (y.compareTo(n1) != 0) return false;\n    }\n  }\n  return true;\n}\n\nvar BigInteger = require('./jsbn.js');\n\n// protected\nBigInteger.prototype.chunkSize = bnpChunkSize;\nBigInteger.prototype.toRadix = bnpToRadix;\nBigInteger.prototype.fromRadix = bnpFromRadix;\nBigInteger.prototype.fromNumber = bnpFromNumber;\nBigInteger.prototype.bitwiseTo = bnpBitwiseTo;\nBigInteger.prototype.changeBit = bnpChangeBit;\nBigInteger.prototype.addTo = bnpAddTo;\nBigInteger.prototype.dMultiply = bnpDMultiply;\nBigInteger.prototype.dAddOffset = bnpDAddOffset;\nBigInteger.prototype.multiplyLowerTo = bnpMultiplyLowerTo;\nBigInteger.prototype.multiplyUpperTo = bnpMultiplyUpperTo;\nBigInteger.prototype.modInt = bnpModInt;\nBigInteger.prototype.millerRabin = bnpMillerRabin;\n\n// public\nBigInteger.prototype.clone = bnClone;\nBigInteger.prototype.intValue = bnIntValue;\nBigInteger.prototype.byteValue = bnByteValue;\nBigInteger.prototype.shortValue = bnShortValue;\nBigInteger.prototype.signum = bnSigNum;\nBigInteger.prototype.toByteArray = bnToByteArray;\nBigInteger.prototype.equals = bnEquals;\nBigInteger.prototype.min = bnMin;\nBigInteger.prototype.max = bnMax;\nBigInteger.prototype.and = bnAnd;\nBigInteger.prototype.or = bnOr;\nBigInteger.prototype.xor = bnXor;\nBigInteger.prototype.andNot = bnAndNot;\nBigInteger.prototype.not = bnNot;\nBigInteger.prototype.shiftLeft = bnShiftLeft;\nBigInteger.prototype.shiftRight = bnShiftRight;\nBigInteger.prototype.getLowestSetBit = bnGetLowestSetBit;\nBigInteger.prototype.bitCount = bnBitCount;\nBigInteger.prototype.testBit = bnTestBit;\nBigInteger.prototype.setBit = bnSetBit;\nBigInteger.prototype.clearBit = bnClearBit;\nBigInteger.prototype.flipBit = bnFlipBit;\nBigInteger.prototype.add = bnAdd;\nBigInteger.prototype.subtract = bnSubtract;\nBigInteger.prototype.multiply = bnMultiply;\nBigInteger.prototype.divide = bnDivide;\nBigInteger.prototype.remainder = bnRemainder;\nBigInteger.prototype.divideAndRemainder = bnDivideAndRemainder;\nBigInteger.prototype.modPow = bnModPow;\nBigInteger.prototype.modInverse = bnModInverse;\nBigInteger.prototype.pow = bnPow;\nBigInteger.prototype.gcd = bnGCD;\nBigInteger.prototype.isProbablePrime = bnIsProbablePrime;\nBigInteger.prototype.toMPI = bnToMPI;\n\n// JSBN-specific extension\nBigInteger.prototype.square = bnSquare;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// RSA implementation\n\nvar BigInteger = require('./jsbn.js'),\n  util = require('../../util'),\n  random = require('../random.js');\n\nfunction SecureRandom() {\n  function nextBytes(byteArray) {\n    for (var n = 0; n < byteArray.length; n++) {\n      byteArray[n] = random.getSecureRandomOctet();\n    }\n  }\n  this.nextBytes = nextBytes;\n}\n\nfunction RSA() {\n  /**\n   * This function uses jsbn Big Num library to decrypt RSA\n   * @param m\n   *            message\n   * @param d\n   *            RSA d as BigInteger\n   * @param p\n   *            RSA p as BigInteger\n   * @param q\n   *            RSA q as BigInteger\n   * @param u\n   *            RSA u as BigInteger\n   * @return {BigInteger} The decrypted value of the message\n   */\n  function decrypt(m, d, p, q, u) {\n    var xp = m.mod(p).modPow(d.mod(p.subtract(BigInteger.ONE)), p);\n    var xq = m.mod(q).modPow(d.mod(q.subtract(BigInteger.ONE)), q);\n    util.print_debug(\"rsa.js decrypt\\nxpn:\" + util.hexstrdump(xp.toMPI()) + \"\\nxqn:\" + util.hexstrdump(xq.toMPI()));\n\n    var t = xq.subtract(xp);\n    if (t[0] == 0) {\n      t = xp.subtract(xq);\n      t = t.multiply(u).mod(q);\n      t = q.subtract(t);\n    } else {\n      t = t.multiply(u).mod(q);\n    }\n    return t.multiply(p).add(xp);\n  }\n\n  /**\n   * encrypt message\n   * @param m message as BigInteger\n   * @param e public MPI part as BigInteger\n   * @param n public MPI part as BigInteger\n   * @return BigInteger\n   */\n  function encrypt(m, e, n) {\n    return m.modPowInt(e, n);\n  }\n\n  /* Sign and Verify */\n  function sign(m, d, n) {\n    return m.modPow(d, n);\n  }\n\n  function verify(x, e, n) {\n    return x.modPowInt(e, n);\n  }\n\n  // \"empty\" RSA key constructor\n\n  function keyObject() {\n    this.n = null;\n    this.e = 0;\n    this.ee = null;\n    this.d = null;\n    this.p = null;\n    this.q = null;\n    this.dmp1 = null;\n    this.dmq1 = null;\n    this.u = null;\n  }\n\n  // Generate a new random private key B bits long, using public expt E\n\n  function generate(B, E) {\n    var key = new keyObject();\n    var rng = new SecureRandom();\n    var qs = B >> 1;\n    key.e = parseInt(E, 16);\n    key.ee = new BigInteger(E, 16);\n    for (;;) {\n      for (;;) {\n        key.p = new BigInteger(B - qs, 1, rng);\n        if (key.p.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.p.isProbablePrime(10))\n          break;\n      }\n      for (;;) {\n        key.q = new BigInteger(qs, 1, rng);\n        if (key.q.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.q.isProbablePrime(10))\n          break;\n      }\n      if (key.p.compareTo(key.q) <= 0) {\n        var t = key.p;\n        key.p = key.q;\n        key.q = t;\n      }\n      var p1 = key.p.subtract(BigInteger.ONE);\n      var q1 = key.q.subtract(BigInteger.ONE);\n      var phi = p1.multiply(q1);\n      if (phi.gcd(key.ee).compareTo(BigInteger.ONE) == 0) {\n        key.n = key.p.multiply(key.q);\n        key.d = key.ee.modInverse(phi);\n        key.dmp1 = key.d.mod(p1);\n        key.dmq1 = key.d.mod(q1);\n        key.u = key.p.modInverse(key.q);\n        break;\n      }\n    }\n    return key;\n  }\n\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n  this.verify = verify;\n  this.sign = sign;\n  this.generate = generate;\n  this.keyObject = keyObject;\n}\n\nmodule.exports = RSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Retrieve secure random byte string of the specified length\n   * @param {Integer} length Length in bytes to generate\n   * @return {String} Random byte string\n   */\n  getRandomBytes: function(length) {\n    var result = '';\n    for (var i = 0; i < length; i++) {\n      result += String.fromCharCode(this.getSecureRandomOctet());\n    }\n    return result;\n  },\n\n  /**\n   * Return a pseudo-random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A pseudo random number\n   */\n  getPseudoRandom: function(from, to) {\n    return Math.round(Math.random() * (to - from)) + from;\n  },\n\n  /**\n   * Return a secure random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A secure random number\n   */\n  getSecureRandom: function(from, to) {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    var bits = ((to - from)).toString(2).length;\n    while ((buf[0] & (Math.pow(2, bits) - 1)) > (to - from))\n      window.crypto.getRandomValues(buf);\n    return from + (Math.abs(buf[0] & (Math.pow(2, bits) - 1)));\n  },\n\n  getSecureRandomOctet: function() {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    return buf[0] & 0xFF;\n  },\n\n  /**\n   * Create a secure random big integer of bits length\n   * @param {Integer} bits Bit length of the MPI to create\n   * @return {BigInteger} Resulting big integer\n   */\n  getRandomBigInteger: function(bits) {\n    if (bits < 0) {\n      return null;\n    }\n    var numBytes = Math.floor((bits + 7) / 8);\n\n    var randomBits = this.getRandomBytes(numBytes);\n    if (bits % 8 > 0) {\n\n      randomBits = String.fromCharCode(\n      (Math.pow(2, bits % 8) - 1) &\n        randomBits.charCodeAt(0)) +\n        randomBits.substring(1);\n    }\n    var mpi = new type_mpi();\n    mpi.fromBytes(randomBits);\n    return mpi.toBigInteger();\n  },\n\n  getRandomBigIntegerInRange: function(min, max) {\n    if (max.compareTo(min) <= 0) {\n      return;\n    }\n\n    var range = max.subtract(min);\n    var r = this.getRandomBigInteger(range.bitLength());\n    while (r > range) {\n      r = this.getRandomBigInteger(range.bitLength());\n    }\n    return min.add(r);\n  }\n\n};\n","var publicKey = require('./public_key'),\n  pkcs1 = require('./pkcs1.js'),\n  hashModule = require('./hash');\n\nmodule.exports = {\n  /**\n   * \n   * @param {Integer} algo public Key algorithm\n   * @param {Integer} hash_algo Hash algorithm\n   * @param {openpgp_type_mpi[]} msg_MPIs Signature multiprecision integers\n   * @param {openpgp_type_mpi[]} publickey_MPIs Public key multiprecision integers \n   * @param {String} data Data on where the signature was computed on.\n   * @return {Boolean} true if signature (sig_data was equal to data over hash)\n   */\n  verify: function(algo, hash_algo, msg_MPIs, publickey_MPIs, data) {\n    var calc_hash = hashModule.digest(hash_algo, data);\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var n = publickey_MPIs[0].toBigInteger();\n        var e = publickey_MPIs[1].toBigInteger();\n        var x = msg_MPIs[0].toBigInteger();\n        var dopublic = rsa.verify(x, e, n);\n        var hash = pkcs1.emsa.decode(hash_algo, dopublic.toMPI().substring(2));\n        if (hash == -1) {\n          throw new Error('PKCS1 padding in message or key incorrect. Aborting...');\n        }\n        return hash == calc_hash;\n\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error(\"signing with Elgamal is not defined in the OpenPGP standard.\");\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n        var s1 = msg_MPIs[0].toBigInteger();\n        var s2 = msg_MPIs[1].toBigInteger();\n        var p = publickey_MPIs[0].toBigInteger();\n        var q = publickey_MPIs[1].toBigInteger();\n        var g = publickey_MPIs[2].toBigInteger();\n        var y = publickey_MPIs[3].toBigInteger();\n        var m = data;\n        var dopublic = dsa.verify(hash_algo, s1, s2, m, p, q, g, y);\n        return dopublic.compareTo(s1) == 0;\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n\n  },\n\n  /**\n   * Create a signature on data using the specified algorithm\n   * @param {Integer} hash_algo hash Algorithm to use (See RFC4880 9.4)\n   * @param {Integer} algo Asymmetric cipher algorithm to use (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Public key multiprecision integers \n   * of the private key \n   * @param {openpgp_type_mpi[]} secretMPIs Private key multiprecision \n   * integers which is used to sign the data\n   * @param {String} data Data to be signed\n   * @return {openpgp_type_mpi[]}\n   */\n  sign: function(hash_algo, algo, keyIntegers, data) {\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var d = keyIntegers[2].toBigInteger();\n        var n = keyIntegers[0].toBigInteger();\n        var m = pkcs1.emsa.encode(hash_algo,\n          data, keyIntegers[0].byteLength());\n\n        return rsa.sign(m, d, n).toMPI();\n\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n\n        var p = keyIntegers[0].toBigInteger();\n        var q = keyIntegers[1].toBigInteger();\n        var g = keyIntegers[2].toBigInteger();\n        var y = keyIntegers[3].toBigInteger();\n        var x = keyIntegers[4].toBigInteger();\n        var m = data;\n        var result = dsa.sign(hash_algo, m, g, p, q, x);\n\n        return result[0].toString() + result[1].toString();\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error('Signing with Elgamal is not defined in the OpenPGP standard.');\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar base64 = require('./base64.js');\nvar enums = require('../enums.js');\n\n/**\n * Finds out which Ascii Armoring type is used. This is an internal function\n * @param {String} text [String] ascii armored text\n * @returns {Integer} 0 = MESSAGE PART n of m\n *         1 = MESSAGE PART n\n *         2 = SIGNED MESSAGE\n *         3 = PGP MESSAGE\n *         4 = PUBLIC KEY BLOCK\n *         5 = PRIVATE KEY BLOCK\n *         null = unknown\n */\nfunction get_type(text) {\n  var splittedtext = text.split('-----');\n  // BEGIN PGP MESSAGE, PART X/Y\n  // Used for multi-part messages, where the armor is split amongst Y\n  // parts, and this is the Xth part out of Y.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+\\/\\d+/)) {\n    return enums.armor.multipart_section;\n  } else\n  // BEGIN PGP MESSAGE, PART X\n  // Used for multi-part messages, where this is the Xth part of an\n  // unspecified number of parts. Requires the MESSAGE-ID Armor\n  // Header to be used.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+/)) {\n    return enums.armor.multipart_last;\n\n  } else\n  // BEGIN PGP SIGNATURE\n  // Used for detached signatures, OpenPGP/MIME signatures, and\n  // cleartext signatures. Note that PGP 2.x uses BEGIN PGP MESSAGE\n  // for detached signatures.\n  if (splittedtext[1].match(/BEGIN PGP SIGNED MESSAGE/)) {\n    return enums.armor.signed;\n\n  } else\n  // BEGIN PGP MESSAGE\n  // Used for signed, encrypted, or compressed files.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE/)) {\n    return enums.armor.message;\n\n  } else\n  // BEGIN PGP PUBLIC KEY BLOCK\n  // Used for armoring public keys.\n  if (splittedtext[1].match(/BEGIN PGP PUBLIC KEY BLOCK/)) {\n    return enums.armor.public_key;\n\n  } else\n  // BEGIN PGP PRIVATE KEY BLOCK\n  // Used for armoring private keys.\n  if (splittedtext[1].match(/BEGIN PGP PRIVATE KEY BLOCK/)) {\n    return enums.armor.private_key;\n  }\n}\n\n/**\n * Add additional information to the armor version of an OpenPGP binary\n * packet block.\n * @author  Alex\n * @version 2011-12-16\n * @returns {String} The header information\n */\nfunction armor_addheader(options) {\n  var result = \"\";\n  if (options.show_version) {\n    result += \"Version: \" + options.versionstring + '\\r\\n';\n  }\n  if (options.show_comment) {\n    result += \"Comment: \" + options.commentstring + '\\r\\n';\n  }\n  result += '\\r\\n';\n  return result;\n}\n\n\n\n/**\n * Calculates a checksum over the given data and returns it base64 encoded\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {String} Base64 encoded checksum\n */\nfunction getCheckSum(data) {\n  var c = createcrc24(data);\n  var str = \"\" + String.fromCharCode(c >> 16) +\n    String.fromCharCode((c >> 8) & 0xFF) +\n    String.fromCharCode(c & 0xFF);\n  return base64.encode(str);\n}\n\n/**\n * Calculates the checksum over the given data and compares it with the \n * given base64 encoded checksum\n * @param {String} data Data to create a CRC-24 checksum for\n * @param {String} checksum Base64 encoded checksum\n * @return {Boolean} True if the given checksum is correct; otherwise false\n */\nfunction verifyCheckSum(data, checksum) {\n  var c = getCheckSum(data);\n  var d = checksum;\n  return c[0] == d[0] && c[1] == d[1] && c[2] == d[2];\n}\n/**\n * Internal function to calculate a CRC-24 checksum over a given string (data)\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {Integer} The CRC-24 checksum as number\n */\nvar crc_table = [\n    0x00000000, 0x00864cfb, 0x018ad50d, 0x010c99f6, 0x0393e6e1, 0x0315aa1a, 0x021933ec, 0x029f7f17, 0x07a18139,\n    0x0727cdc2, 0x062b5434, 0x06ad18cf, 0x043267d8, 0x04b42b23, 0x05b8b2d5, 0x053efe2e, 0x0fc54e89, 0x0f430272,\n    0x0e4f9b84, 0x0ec9d77f, 0x0c56a868, 0x0cd0e493, 0x0ddc7d65, 0x0d5a319e, 0x0864cfb0, 0x08e2834b, 0x09ee1abd,\n    0x09685646, 0x0bf72951, 0x0b7165aa, 0x0a7dfc5c, 0x0afbb0a7, 0x1f0cd1e9, 0x1f8a9d12, 0x1e8604e4, 0x1e00481f,\n    0x1c9f3708, 0x1c197bf3, 0x1d15e205, 0x1d93aefe, 0x18ad50d0, 0x182b1c2b, 0x192785dd, 0x19a1c926, 0x1b3eb631,\n    0x1bb8faca, 0x1ab4633c, 0x1a322fc7, 0x10c99f60, 0x104fd39b, 0x11434a6d, 0x11c50696, 0x135a7981, 0x13dc357a,\n    0x12d0ac8c, 0x1256e077, 0x17681e59, 0x17ee52a2, 0x16e2cb54, 0x166487af, 0x14fbf8b8, 0x147db443, 0x15712db5,\n    0x15f7614e, 0x3e19a3d2, 0x3e9fef29, 0x3f9376df, 0x3f153a24, 0x3d8a4533, 0x3d0c09c8, 0x3c00903e, 0x3c86dcc5,\n    0x39b822eb, 0x393e6e10, 0x3832f7e6, 0x38b4bb1d, 0x3a2bc40a, 0x3aad88f1, 0x3ba11107, 0x3b275dfc, 0x31dced5b,\n    0x315aa1a0,\n    0x30563856, 0x30d074ad, 0x324f0bba, 0x32c94741, 0x33c5deb7, 0x3343924c, 0x367d6c62, 0x36fb2099, 0x37f7b96f,\n    0x3771f594, 0x35ee8a83, 0x3568c678, 0x34645f8e, 0x34e21375, 0x2115723b, 0x21933ec0, 0x209fa736, 0x2019ebcd,\n    0x228694da, 0x2200d821, 0x230c41d7, 0x238a0d2c, 0x26b4f302, 0x2632bff9, 0x273e260f, 0x27b86af4, 0x252715e3,\n    0x25a15918, 0x24adc0ee, 0x242b8c15, 0x2ed03cb2, 0x2e567049, 0x2f5ae9bf, 0x2fdca544, 0x2d43da53, 0x2dc596a8,\n    0x2cc90f5e, 0x2c4f43a5, 0x2971bd8b, 0x29f7f170, 0x28fb6886, 0x287d247d, 0x2ae25b6a, 0x2a641791, 0x2b688e67,\n    0x2beec29c, 0x7c3347a4, 0x7cb50b5f, 0x7db992a9, 0x7d3fde52, 0x7fa0a145, 0x7f26edbe, 0x7e2a7448, 0x7eac38b3,\n    0x7b92c69d, 0x7b148a66, 0x7a181390, 0x7a9e5f6b, 0x7801207c, 0x78876c87, 0x798bf571, 0x790db98a, 0x73f6092d,\n    0x737045d6, 0x727cdc20, 0x72fa90db, 0x7065efcc, 0x70e3a337, 0x71ef3ac1, 0x7169763a, 0x74578814, 0x74d1c4ef,\n    0x75dd5d19, 0x755b11e2, 0x77c46ef5, 0x7742220e, 0x764ebbf8, 0x76c8f703, 0x633f964d, 0x63b9dab6, 0x62b54340,\n    0x62330fbb,\n    0x60ac70ac, 0x602a3c57, 0x6126a5a1, 0x61a0e95a, 0x649e1774, 0x64185b8f, 0x6514c279, 0x65928e82, 0x670df195,\n    0x678bbd6e, 0x66872498, 0x66016863, 0x6cfad8c4, 0x6c7c943f, 0x6d700dc9, 0x6df64132, 0x6f693e25, 0x6fef72de,\n    0x6ee3eb28, 0x6e65a7d3, 0x6b5b59fd, 0x6bdd1506, 0x6ad18cf0, 0x6a57c00b, 0x68c8bf1c, 0x684ef3e7, 0x69426a11,\n    0x69c426ea, 0x422ae476, 0x42aca88d, 0x43a0317b, 0x43267d80, 0x41b90297, 0x413f4e6c, 0x4033d79a, 0x40b59b61,\n    0x458b654f, 0x450d29b4, 0x4401b042, 0x4487fcb9, 0x461883ae, 0x469ecf55, 0x479256a3, 0x47141a58, 0x4defaaff,\n    0x4d69e604, 0x4c657ff2, 0x4ce33309, 0x4e7c4c1e, 0x4efa00e5, 0x4ff69913, 0x4f70d5e8, 0x4a4e2bc6, 0x4ac8673d,\n    0x4bc4fecb, 0x4b42b230, 0x49ddcd27, 0x495b81dc, 0x4857182a, 0x48d154d1, 0x5d26359f, 0x5da07964, 0x5cace092,\n    0x5c2aac69, 0x5eb5d37e, 0x5e339f85, 0x5f3f0673, 0x5fb94a88, 0x5a87b4a6, 0x5a01f85d, 0x5b0d61ab, 0x5b8b2d50,\n    0x59145247, 0x59921ebc, 0x589e874a, 0x5818cbb1, 0x52e37b16, 0x526537ed, 0x5369ae1b, 0x53efe2e0, 0x51709df7,\n    0x51f6d10c,\n    0x50fa48fa, 0x507c0401, 0x5542fa2f, 0x55c4b6d4, 0x54c82f22, 0x544e63d9, 0x56d11cce, 0x56575035, 0x575bc9c3,\n    0x57dd8538\n];\n\nfunction createcrc24(input) {\n  var crc = 0xB704CE;\n  var index = 0;\n\n  while ((input.length - index) > 16) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 1)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 2)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 3)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 4)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 5)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 6)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 7)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 8)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 9)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 10)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 11)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 12)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 13)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 14)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 15)) & 0xff];\n    index += 16;\n  }\n\n  for (var j = index; j < input.length; j++) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index++)) & 0xff];\n  }\n  return crc & 0xffffff;\n}\n\n/**\n * DeArmor an OpenPGP armored message; verify the checksum and return \n * the encoded bytes\n * @param {String} text OpenPGP armored message\n * @returns {(Boolean|Object)} Either false in case of an error \n * or an object with attribute \"text\" containing the message text\n * and an attribute \"openpgp\" containing the bytes.\n */\nfunction dearmor(text) {\n  text = text.replace(/\\r/g, '');\n\n  var type = get_type(text);\n\n  if (type != 2) {\n    var splittedtext = text.split('-----');\n\n    var data = {\n      openpgp: base64.decode(\n        splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[0]\n        .replace(/\\n- /g, \"\\n\")),\n      type: type\n    };\n\n    if (verifyCheckSum(data.openpgp,\n      splittedtext[2]\n      .split('\\n\\n')[1]\n      .split(\"\\n=\")[1]\n      .split('\\n')[0]))\n\n      return data;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed: '\" + splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[1]\n        .split('\\n')[0] + \"' should be '\" + getCheckSum(data)) + \"'\";\n      return false;\n    }\n  } else {\n    var splittedtext = text.split('-----');\n\n    var result = {\n      text: splittedtext[2]\n        .replace(/\\n- /g, \"\\n\")\n        .split(\"\\n\\n\")[1],\n      openpgp: base64_decode(splittedtext[4]\n        .split(\"\\n\\n\")[1]\n        .split(\"\\n=\")[0]),\n      type: type\n    };\n\n    if (verifyCheckSum(result.openpgp, splittedtext[4]\n      .split(\"\\n\\n\")[1]\n      .split(\"\\n=\")[1]))\n\n      return result;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed\");\n      return false;\n    }\n  }\n}\n\n\n/**\n * Armor an OpenPGP binary packet block\n * @param {Integer} messagetype type of the message\n * @param data\n * @param {Integer} partindex\n * @param {Integer} parttotal\n * @returns {String} Armored text\n */\nfunction armor(messagetype, data, options, partindex, parttotal) {\n  var result = \"\";\n  switch (messagetype) {\n    case enums.armor.multipart_section:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      break;\n    case enums.armor.mutlipart_last:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      break;\n    case enums.armor.signed:\n      result += \"\\r\\n-----BEGIN PGP SIGNED MESSAGE-----\\r\\nHash: \" + data.hash + \"\\r\\n\\r\\n\";\n      result += data.text.replace(/\\n-/g, \"\\n- -\");\n      result += \"\\r\\n-----BEGIN PGP SIGNATURE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data.openpgp);\n      result += \"\\r\\n=\" + getCheckSum(data.openpgp) + \"\\r\\n\";\n      result += \"-----END PGP SIGNATURE-----\\r\\n\";\n      break;\n    case enums.armor.message:\n      result += \"-----BEGIN PGP MESSAGE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE-----\\r\\n\";\n      break;\n    case enums.armor.public_key:\n      result += \"-----BEGIN PGP PUBLIC KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PUBLIC KEY BLOCK-----\\r\\n\\r\\n\";\n      break;\n    case enums.armor.private_key:\n      result += \"-----BEGIN PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      break;\n  }\n\n  return result;\n}\n\nmodule.exports = {\n  encode: armor,\n  decode: dearmor\n};\n","/* OpenPGP radix-64/base64 string encoding/decoding\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.0, check www.haneWIN.de for the latest version\n *\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other materials\n * provided with the application or distribution.\n */\n\nvar b64s = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\n\nfunction s2r(t) {\n  var a, c, n;\n  var r = '',\n    l = 0,\n    s = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = t.charCodeAt(n);\n    if (s == 0) {\n      r += b64s.charAt((c >> 2) & 63);\n      a = (c & 3) << 4;\n    } else if (s == 1) {\n      r += b64s.charAt((a | (c >> 4) & 15));\n      a = (c & 15) << 2;\n    } else if (s == 2) {\n      r += b64s.charAt(a | ((c >> 6) & 3));\n      l += 1;\n      if ((l % 60) == 0)\n        r += \"\\n\";\n      r += b64s.charAt(c & 63);\n    }\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n\n    s += 1;\n    if (s == 3)\n      s = 0;\n  }\n  if (s > 0) {\n    r += b64s.charAt(a);\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n    l += 1;\n  }\n  if (s == 1) {\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n  }\n\n  return r;\n}\n\nfunction r2s(t) {\n  var c, n;\n  var r = '',\n    s = 0,\n    a = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = b64s.indexOf(t.charAt(n));\n    if (c >= 0) {\n      if (s)\n        r += String.fromCharCode(a | (c >> (6 - s)) & 255);\n      s = (s + 2) & 7;\n      a = (c << s) & 255;\n    }\n  }\n  return r;\n}\n\nmodule.exports = {\n  encode: s2r,\n  decode: r2s\n}\n","var enums = {\n\n  /** A string to key specifier type\n   * @enum {Integer}\n   */\n  s2k: {\n    simple: 0,\n    salted: 1,\n    iterated: 3,\n    gnu: 101\n  },\n\n  /** RFC4880, section 9.1 \n   * @enum {String}\n   */\n  publicKey: {\n    rsa_encrypt_sign: 1,\n    rsa_encrypt: 2,\n    rsa_sign: 3,\n    elgamal: 16,\n    dsa: 17\n  },\n\n  /** RFC4880, section 9.2 \n   * @enum {String}\n   */\n  symmetric: {\n    plaintext: 0,\n    /** Not implemented! */\n    idea: 1,\n    tripledes: 2,\n    cast5: 3,\n    blowfish: 4,\n    aes128: 7,\n    aes192: 8,\n    aes256: 9,\n    twofish: 10\n  },\n\n  /** RFC4880, section 9.3\n   * @enum {String}\n   */\n  compression: {\n    uncompressed: 0,\n    /** RFC1951 */\n    zip: 1,\n    /** RFC1950 */\n    zlib: 2,\n    bzip2: 3\n  },\n\n  /** RFC4880, section 9.4\n   * @enum {String}\n   */\n  hash: {\n    md5: 1,\n    sha1: 2,\n    ripemd: 3,\n    sha256: 8,\n    sha384: 9,\n    sha512: 10,\n    sha224: 11\n  },\n\n\n  /**\n   * @enum {String}\n   * A list of packet types and numeric tags associated with them.\n   */\n  packet: {\n    public_key_encrypted_session_key: 1,\n    signature: 2,\n    sym_encrypted_session_key: 3,\n    one_pass_signature: 4,\n    secret_key: 5,\n    public_key: 6,\n    secret_subkey: 7,\n    compressed: 8,\n    symmetrically_encrypted: 9,\n    marker: 10,\n    literal: 11,\n    trust: 12,\n    userid: 13,\n    public_subkey: 14,\n    user_attribute: 17,\n    sym_encrypted_integrity_protected: 18,\n    modification_detection_code: 19\n  },\n\n\n  /**\n   * Data types in the literal packet\n   * @readonly\n   * @enum {String}\n   */\n  literal: {\n    /** Binary data */\n    binary: 'b'.charCodeAt(),\n    /** Text data */\n    text: 't'.charCodeAt(),\n    /** Utf8 data */\n    utf8: 'u'.charCodeAt()\n  },\n\n\n  /** One pass signature packet type\n   * @enum {String} */\n  signature: {\n    /** 0x00: Signature of a binary document. */\n    binary: 0,\n    /** 0x01: Signature of a canonical text document.\n     * Canonicalyzing the document by converting line endings. */\n    text: 1,\n    /** 0x02: Standalone signature.\n     * This signature is a signature of only its own subpacket contents.\n     * It is calculated identically to a signature over a zero-lengh\n     * binary document.  Note that it doesn't make sense to have a V3\n     * standalone signature. */\n    standalone: 2,\n    /** 0x10: Generic certification of a User ID and Public-Key packet.\n     * The issuer of this certification does not make any particular\n     * assertion as to how well the certifier has checked that the owner\n     * of the key is in fact the person described by the User ID. */\n    cert_generic: 16,\n    /** 0x11: Persona certification of a User ID and Public-Key packet.\n     * The issuer of this certification has not done any verification of\n     * the claim that the owner of this key is the User ID specified. */\n    cert_persona: 17,\n    /** 0x12: Casual certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done some casual\n     * verification of the claim of identity. */\n    cert_casual: 18,\n    /** 0x13: Positive certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done substantial\n     * verification of the claim of identity.\n     * \n     * Most OpenPGP implementations make their \"key signatures\" as 0x10\n     * certifications.  Some implementations can issue 0x11-0x13\n     * certifications, but few differentiate between the types. */\n    cert_positive: 19,\n    /** 0x30: Certification revocation signature\n     * This signature revokes an earlier User ID certification signature\n     * (signature class 0x10 through 0x13) or direct-key signature\n     * (0x1F).  It should be issued by the same key that issued the\n     * revoked signature or an authorized revocation key.  The signature\n     * is computed over the same data as the certificate that it\n     * revokes, and should have a later creation date than that\n     * certificate. */\n    cert_revocation: 48,\n    /** 0x18: Subkey Binding Signature\n     * This signature is a statement by the top-level signing key that\n     * indicates that it owns the subkey.  This signature is calculated\n     * directly on the primary key and subkey, and not on any User ID or\n     * other packets.  A signature that binds a signing subkey MUST have\n     * an Embedded Signature subpacket in this binding signature that\n     * contains a 0x19 signature made by the signing subkey on the\n     * primary key and subkey. */\n    subkey_binding: 24,\n    /** 0x19: Primary Key Binding Signature\n\t\t* This signature is a statement by a signing subkey, indicating\n\t\t* that it is owned by the primary key and subkey.  This signature\n\t\t* is calculated the same way as a 0x18 signature: directly on the\n\t\t* primary key and subkey, and not on any User ID or other packets.\n\t\t\n\t\t* When a signature is made over a key, the hash data starts with the\n\t\t* octet 0x99, followed by a two-octet length of the key, and then body\n\t\t* of the key packet.  (Note that this is an old-style packet header for\n\t\t* a key packet with two-octet length.)  A subkey binding signature\n\t\t* (type 0x18) or primary key binding signature (type 0x19) then hashes\n\t\t* the subkey using the same format as the main key (also using 0x99 as\n\t\t* the first octet). */\n    key_binding: 25,\n    /** 0x1F: Signature directly on a key\n     * This signature is calculated directly on a key.  It binds the\n     * information in the Signature subpackets to the key, and is\n     * appropriate to be used for subpackets that provide information\n     * about the key, such as the Revocation Key subpacket.  It is also\n     * appropriate for statements that non-self certifiers want to make\n     * about the key itself, rather than the binding between a key and a\n     * name. */\n    key: 31,\n    /** 0x20: Key revocation signature\n     * The signature is calculated directly on the key being revoked.  A\n     * revoked key is not to be used.  Only revocation signatures by the\n     * key being revoked, or by an authorized revocation key, should be\n     * considered valid revocation signatures.a */\n    key_revocation: 32,\n    /** 0x28: Subkey revocation signature\n     * The signature is calculated directly on the subkey being revoked.\n     * A revoked subkey is not to be used.  Only revocation signatures\n     * by the top-level signature key that is bound to this subkey, or\n     * by an authorized revocation key, should be considered valid\n     * revocation signatures.\n     * Key revocation signatures (types 0x20 and 0x28)\n     * hash only the key being revoked. */\n    subkey_revocation: 40,\n    /** 0x40: Timestamp signature.\n     * This signature is only meaningful for the timestamp contained in\n     * it. */\n    timestamp: 64,\n    /**    0x50: Third-Party Confirmation signature.\n     * This signature is a signature over some other OpenPGP Signature\n     * packet(s).  It is analogous to a notary seal on the signed data.\n     * A third-party signature SHOULD include Signature Target\n     * subpacket(s) to give easy identification.  Note that we really do\n     * mean SHOULD.  There are plausible uses for this (such as a blind\n     * party that only sees the signature, not the key or source\n     * document) that cannot include a target subpacket. */\n    third_party: 80\n  },\n\n  signatureSubpacket: {\n    signature_creation_time: 2,\n    signature_expiration_time: 3,\n    exportable_certification: 4,\n    trust_signature: 5,\n    regular_expression: 6,\n    revocable: 7,\n    reserved: 8,\n    key_expiration_time: 9,\n    placeholder_backwards_compatibility: 10,\n    preferred_symmetric_algorithms: 11,\n    revocation_key: 12,\n    issuer: 16,\n    notification_data: 20,\n    preferred_hash_algorithms: 21,\n    preferred_compression_algorithms: 22,\n    key_server_preferences: 23,\n    preferred_key_server: 24,\n    primary_user_id: 25,\n    policy_uri: 26,\n    key_flags: 27,\n    signers_user_id: 28,\n    reason_for_revocation: 29,\n    features: 30,\n    signature_target: 31,\n    embedded_signature: 32\n  },\n\n  armor: {\n    multipart_section: 0,\n    multipart_last: 1,\n    signed: 2,\n    message: 3,\n    public_key: 4,\n    private_key: 5\n  },\n\n  // Asserts validity and converts from string/integer to integer.\n  write: function(type, e) {\n    if (typeof e == 'number') {\n      e = this.read(type, e);\n    }\n\n    if (type[e] !== undefined) {\n      return type[e];\n    } else throw new Error('Invalid enum value.');\n  },\n  // Converts from an integer to string.\n  read: function(type, e) {\n    for (var i in type)\n      if (type[i] == e) return i;\n\n    throw new Error('Invalid enum value.');\n  }\n}\n\nmodule.exports = enums;\n","module.exports=require('IKGp/G');","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\nvar config = require('./config');\n\n/**\n * @class\n * @classdesc Class that represents an OpenPGP key. Must contain a master key. \n * @param  {packetlist} packetlist [description]\n * Can contain additional subkeys, signatures,\n * user ids, user attributes.\n */\n\n function key(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n\n  /** \n   * Returns the primary key packet (secret or public)\n   * @returns {packet_secret_key|packet_public_key|null} \n   */\n  this.getKeyPacket = function() {\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_key ||\n        this.packets[i].tag == enums.packet.secret_key) {\n        return this.packets[i];\n      }\n    }\n    return null;\n  }\n\n  /** \n   * Returns all the private and public subkey packets\n   * @returns {[public_subkey|secret_subkey]} \n   */\n  this.getSubkeyPackets = function() {\n\n    var subkeys = [];\n\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_subkey ||\n        this.packets[i].tag == enums.packet.secret_subkey) {\n        subkeys.push(this.packets[i]);\n      }\n    }\n\n    return subkeys;\n  }\n\n  /** \n   * Returns all the private and public key and subkey packets\n   * @returns {[public_subkey|secret_subkey|packet_secret_key|packet_public_key]} \n   */\n  this.getAllKeyPackets = function() {\n    return [this.getKeyPacket()].concat(this.getSubkeyPackets());\n  }\n\n  /** \n   * Returns key IDs of all key packets\n   * @returns {[keyid]} \n   */\n  this.getKeyIds = function() {\n    var keyIds = [];\n    var keys = this.getAllKeyPackets();\n    for (var i = 0; i < keys.length; i++) {\n      keyIds.push(keys[i].getKeyId());\n    }\n    return keyIds;\n  }\n\n  /** \n   * Returns key IDs of all key packets in hex\n   * @returns {[String]} \n   */\n  this.getKeyIdsHex = function() {\n    return this.getKeyIds().map(function(keyId) {\n      return keyId.toHex();\n    });\n  }\n\n  /**\n   * Returns first key packet which match to an array of key IDs\n   * @param  {[keyid]} keyIds \n   * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null}       \n   */\n  this.getKeyPacketByIds = function(keyIds) {\n    var keys = this.getAllKeyPackets();\n    for (var i = 0; i < keys.length; i++) {\n      var keyId = keys[i].getKeyId(); \n      for (var j = 0; j < keyIds.length; j++) {\n        if (keyId.equals(keyIds[j])) {\n          //TODO return only verified keys\n          return keys[i];\n        }\n      }\n    }\n    return null;\n  }\n\n  /**\n   * Returns true if this is a public key\n   * @return {Boolean}\n   */\n  this.isPublic = function() {\n    var publicKeyPackets = this.packets.filterByTag(enums.packet.public_key);\n    return publicKeyPackets.length !== 0 ? true : false;\n  }\n\n  /**\n   * Returns true if this is a private key\n   * @return {Boolean}\n   */\n  this.isPrivate = function() {\n    var privateKeyPackets = this.packets.filterByTag(enums.packet.private_key);\n    return privateKeyPackets.length !== 0 ? true : false;\n  }\n\n  /**\n   * Returns first key packet that is available for signing\n   * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null}\n   */\n  this.getSigningKeyPacket = function() {\n\n    var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa];\n\n    signing = signing.map(function(s) {\n      return enums.read(enums.publicKey, s);\n    });\n\n    var keys = this.getAllKeyPackets();\n\n    for (var i = 0; i < keys.length; i++) {\n      if (signing.indexOf(keys[i].algorithm) !== -1) {\n        return keys[i];\n      }\n    }\n\n    return null;\n  }\n\n  /**\n   * Returns preferred signature hash algorithm of this key\n   * @return {String}\n   */\n  function getPreferredSignatureHashAlgorithm() {\n    //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8\n    //separate private key preference from digest preferences\n    return config.prefer_hash_algorithm;\n  }\n\n  /**\n   * Returns the first valid encryption key packet for this key\n   * @returns {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} key packet or null if no encryption key has been found\n   */\n  this.getEncryptionKeyPacket = function() {\n    // V4: by convention subkeys are prefered for encryption service\n    // V3: keys MUST NOT have subkeys\n    var isValidEncryptionKey = function(key) {\n      //TODO evaluate key flags: http://tools.ietf.org/html/rfc4880#section-5.2.3.21\n      return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey,\n        enums.publicKey.rsa_sign);\n      //TODO verify key\n      //&& keys.verifyKey()\n    };\n\n    var subkeys = this.getSubkeyPackets();\n\n    for (var j = 0; j < subkeys.length; j++) {\n      if (isValidEncryptionKey(subkeys[j])) {\n        return subkeys[j];\n      }\n    }\n    // if no valid subkey for encryption, use primary key\n    var primaryKey = this.getKeyPacket();\n    if (isValidEncryptionKey(primaryKey)) {\n      return primaryKey;\n    }\n    return null;\n  }\n\n  /**\n   * Decrypts all secret key and subkey packets\n   * @param  {String} passphrase \n   * @return {undefined}\n   */\n  this.decrypt = function(passphrase) {\n    //TODO boolean return value\n    var keys = this.getAllKeyPackets();\n\n    for (var i in keys)\n      if (keys[i].tag == enums.packet.secret_subkey ||\n        keys[i].tag == enums.packet.secret_key) {\n        keys[i].decrypt(passphrase);\n      }\n  }\n\n\n  // TODO\n  this.verify = function() {\n\n  }\n  // TODO\n  this.revoke = function() {\n\n  }\n\n}\n\n/**\n * reads an OpenPGP armored text and returns a key object\n * @param {String} armoredText text to be parsed\n * @return {key} new key object\n */\nkey.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  //TODO don't accept non-key armored texts\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newKey = new key(packetlist);\n  return newKey;\n}\n\nmodule.exports = key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc The class that deals with storage of the keyring. Currently the only option is to use HTML5 local storage.\n */\nvar keyring = function() {\n  this.armoredPacketlists = [];\n  this.parsedPacketlists = [];\n\n  /**\n   * Initialization routine for the keyring. This method reads the \n   * keyring from HTML5 local storage and initializes this instance.\n   * This method is called by openpgp.init().\n   */\n  function init() {\n    var armoredPacketlists = JSON.parse(window.localStorage.getItem(\"armoredPacketlists\"));\n    if (armoredPacketlists === null || armoredPacketlists.length === 0) {\n      armoredPacketlists = [];\n    }\n    this.armoredPacketlists = armoredPacketlists;\n\n    var packetlist;\n    for (var i = 0; i < armoredPacketlists.length; i++) {\n      packetlist = new packet.list();\n      packetlist.read(armoredPacketlists[i]);\n      this.parsedPacketlists.push(packetlist);\n    }\n  }\n  this.init = init;\n\n  /**\n   * Saves the current state of the keyring to HTML5 local storage.\n   * The privateKeys array and publicKeys array gets Stringified using JSON\n   */\n  function store() {\n    window.localStorage.setItem(\"armoredPacketlists\", JSON.stringify(this.armoredPacketlists));\n  }\n  this.store = store;\n\n  function emailPacketCheck(packet, email) {\n    var emailMatch = false;\n    var packetEmail;\n    email = email.toLowerCase();\n    if (packet.tag == enums.packet.userid) {\n      packetEmail = packet.userid;\n      //we need to get just the email from the userid packet\n      packetEmail = packetEmail.split('<')[1].split('<')[0].trim.toLowerCase();\n      if (packetEmail == email) {\n        emailMatch = true;\n      }\n    }\n    return emailMatch;\n  }\n\n  function idPacketCheck(packet, id) {\n    if (packet.getKeyId && packet.getKeyId().write() == id) {\n      return true;\n    }\n    return false;\n  }\n\n  function helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist) {\n    var packet;\n    for (var l = 0; l < packetlist.length; l++) {\n      packet = packetlist[l];\n      identityMatch = identityFunction(packet, identityInput);\n      if (!packetType) {\n        packetMatch = true;\n      } else if (packet.tag == packetType) {\n        packetMatch = true;\n      }\n      if (packetMatch && identityMatch) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  function checkForIdentityAndPacketMatch(identityFunction, identityInput, packetType) {\n    var results = [];\n    var packetlist;\n    var identityMatch;\n    var packetMatch;\n    for (var p = 0; p < this.parsedPacketlists.length; p++) {\n      identityMatch = false;\n      packetMatch = false;\n      packetlist = this.parsedPacketlists[p];\n      if (helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist)) {\n        results.push(packetlist);\n      }\n    }\n    return results;\n  }\n  this.checkForIdentityAndPacketMatch = checkForIdentityAndPacketMatch;\n\n  /**\n   * searches all public keys in the keyring matching the address or address part of the user ids\n   * @param {String} email_address\n   * @return {openpgp_msg_publickey[]} The public keys associated with provided email address.\n   */\n  function getPublicKeyForAddress(email) {\n    return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.public_key);\n  }\n  this.getPublicKeyForAddress = getPublicKeyForAddress;\n\n  /**\n   * Searches the keyring for a private key containing the specified email address\n   * @param {String} email_address email address to search for\n   * @return {openpgp_msg_privatekey[]} private keys found\n   */\n  function getPrivateKeyForAddress(email_address) {\n    return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.secret_key);\n  }\n  this.getPrivateKeyForAddress = getPrivateKeyForAddress;\n\n  /**\n   * Searches the keyring for public keys having the specified key id\n   * @param {String} keyId provided as string of hex number (lowercase)\n   * @return {openpgp_msg_privatekey[]} public keys found\n   */\n  function getPacketlistsForKeyId(keyId) {\n    return this.checkForIdentityAndPacketMatch(idPacketCheck, keyId);\n  }\n  this.getPacketlistsForKeyId = getPacketlistsForKeyId;\n\n  /**\n   * Imports a packet list (public or private key block) from an ascii armored message \n   * @param {String} armored message to read the packets/key from\n   */\n  function importPacketlist(armored) {\n    this.armoredPacketlists.push(armored);\n\n    var dearmored = armor.decode(armored.replace(/\\r/g, '')).openpgp;\n\n    packetlist = new packet.list();\n    packetlist.read(dearmored);\n    this.parsedPacketlists.push(packetlist);\n\n    return true;\n  }\n  this.importPacketlist = importPacketlist;\n\n  /**\n   * TODO\n   * returns the openpgp_msg_privatekey representation of the public key at public key ring index  \n   * @param {Integer} index the index of the public key within the publicKeys array\n   * @return {openpgp_msg_privatekey} the public key object\n   */\n  function exportPublicKey(index) {\n    return this.publicKey[index];\n  }\n  this.exportPublicKey = exportPublicKey;\n\n  /**\n   * TODO\n   * Removes a public key from the public key keyring at the specified index \n   * @param {Integer} index the index of the public key within the publicKeys array\n   * @return {openpgp_msg_privatekey} The public key object which has been removed\n   */\n  function removePublicKey(index) {\n    var removed = this.publicKeys.splice(index, 1);\n    this.store();\n    return removed;\n  }\n  this.removePublicKey = removePublicKey;\n\n};\n\nmodule.exports = new keyring();\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc A generic message containing one or more literal packets.\n */\n\nfunction message(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n  /**\n   * Returns the key IDs of the public keys to which the session key is encrypted\n   * @return {[keyId]} array of keyid objects\n   */\n  this.getKeyIds = function() {\n    var keyIds = [];\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    pkESKeyPacketlist.forEach(function(packet) {\n      keyIds.push(packet.publicKeyId);\n    });\n    return keyIds;\n  }\n\n  /**\n   * Returns the key IDs in hex of the public keys to which the session key is encrypted\n   * @return {[String]} keyId provided as string of hex numbers (lowercase)\n   */\n  this.getKeyIdsHex = function() {\n    return this.getKeyIds().map(function(keyId) {\n      return keyId.toHex();\n    });\n  }\n\n  /**\n   * Decrypts the message\n   * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with (corresponding to the session key)\n   * @return {[String]} array with plaintext of decrypted messages\n   */\n  this.decrypt = function(privateKeyPacket) {\n    var decryptedMessages = [];\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    for (var i = 0; i < pkESKeyPacketlist.length; i++) {\n      var pkESKeyPacket = pkESKeyPacketlist[i];\n      if (pkESKeyPacket.publicKeyId.equals(privateKeyPacket.getKeyId())) {\n        pkESKeyPacket.decrypt(privateKeyPacket);\n        var symEncryptedPacketlist = this.packets.filter(function(packet) {\n          return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected;\n        });\n        for (var k = 0; k < symEncryptedPacketlist.length; k++) {\n          var symEncryptedPacket = symEncryptedPacketlist[k];\n          symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey);\n          for (var l = 0; l < symEncryptedPacket.packets.length; l++) {\n            var dataPacket = symEncryptedPacket.packets[l];\n            switch (dataPacket.tag) {\n              case enums.packet.literal:\n                decryptedMessages.push(dataPacket.getBytes());\n                break;\n              case enums.packet.compressed:\n                //TODO\n                break;\n              default:\n                //TODO\n            }\n          }\n        }\n        break;\n      }\n    }\n    return decryptedMessages;\n  }\n\n  /**\n   * Decrypts a message and generates user interface message out of the found.\n   * MDC will be verified as well as message signatures\n   * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key)\n   * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {String} plaintext of the message or null on error\n   */\n  function decryptAndVerifySignature(private_key, sessionkey, pubkey) {\n    if (private_key == null || sessionkey == null || sessionkey == \"\")\n      return null;\n    var decrypted = sessionkey.decrypt(this, private_key.keymaterial);\n    if (decrypted == null)\n      return null;\n    var packet;\n    var position = 0;\n    var len = decrypted.length;\n    var validSignatures = new Array();\n    util.print_debug_hexstr_dump(\"openpgp.msg.messge decrypt:\\n\", decrypted);\n\n    var messages = openpgp.read_messages_dearmored({\n      text: decrypted,\n      openpgp: decrypted\n    });\n    for (var m in messages) {\n      if (messages[m].data) {\n        this.text = messages[m].data;\n      }\n      if (messages[m].signature) {\n        validSignatures.push(messages[m].verifySignature(pubkey));\n      }\n    }\n    return {\n      text: this.text,\n      validSignatures: validSignatures\n    };\n  }\n\n  /**\n   * Verifies a message signature. This function can be called after read_message if the message was signed only.\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {boolean} true if the signature was correct; otherwise false\n   */\n  function verifySignature(pubkey) {\n    var result = false;\n    if (this.signature.tagType == 2) {\n      if (!pubkey || pubkey.length == 0) {\n        var pubkey;\n        if (this.signature.version == 4) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.issuerKeyId);\n        } else if (this.signature.version == 3) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.keyId);\n        } else {\n          util.print_error(\"unknown signature type on message!\");\n          return false;\n        }\n      }\n      if (pubkey.length == 0)\n        util.print_warning(\"Unable to verify signature of issuer: \" + util.hexstrdump(this.signature.issuerKeyId) +\n          \". Public key not found in keyring.\");\n      else {\n        for (var i = 0; i < pubkey.length; i++) {\n          var tohash = this.text.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g, \"\\r\\n\");\n          if (this.signature.verify(tohash, pubkey[i])) {\n            util.print_info(\"Found Good Signature from \" + pubkey[i].obj.userIds[0].text + \" (0x\" + util.hexstrdump(\n              pubkey[i].obj.getKeyId()).substring(8) + \")\");\n            result = true;\n          } else {\n            util.print_error(\"Signature verification failed: Bad Signature from \" + pubkey[i].obj.userIds[0].text +\n              \" (0x\" + util.hexstrdump(pubkey[0].obj.getKeyId()).substring(8) + \")\");\n          }\n        }\n      }\n    }\n    return result;\n  }\n}\n\n/**\n * reads an OpenPGP armored message and returns a message object\n * @param {String} armoredText text to be parsed\n * @return {key} new message object\n */\nmessage.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  //TODO don't accept non-message armored texts\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newMessage = new message(packetlist);\n  return newMessage;\n}\n\nmodule.exports = message;","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @fileoverview The openpgp base class should provide all of the functionality \n * to consume the openpgp.js library. All additional classes are documented \n * for extending and developing on top of the base library.\n */\n\nvar armor = require('./encoding/armor.js');\nvar packet = require('./packet');\nvar util = require('./util');\nvar enums = require('./enums.js');\nvar crypto = require('./crypto');\nvar config = require('./config');\n\n/**\n * GPG4Browsers Core interface. A single instance is hold\n * from the beginning. To use this library call \"openpgp.init()\"\n * @alias openpgp\n * @class\n * @classdesc Main Openpgp.js class. Use this to initiate and make all calls to this library.\n */\nfunction _openpgp() {\n  this.tostring = \"\";\n\n  /**\n   * encrypts message with keys\n   * @param  {[key]} keys    array of keys, used to encrypt the message\n   * @param  {String} message text message as native JavaScript string\n   * @return {String}         encrypted ASCII armored message \n   */\n  function encryptMessage(keys, message) {\n\n    var messagePacketlist = new packet.list();\n\n    //TODO get preferred algo from signature\n    var sessionKey = crypto.generateSessionKey(enums.read(enums.symmetric, config.encryption_cipher));\n\n    keys.forEach(function(key) {\n      var encryptionKeyPacket = key.getEncryptionKeyPacket();\n      if (encryptionKeyPacket) {\n        var pkESKeyPacket = new packet.public_key_encrypted_session_key();\n        pkESKeyPacket.publicKeyId = encryptionKeyPacket.getKeyId();\n        pkESKeyPacket.publicKeyAlgorithm = encryptionKeyPacket.algorithm;\n        pkESKeyPacket.sessionKey = sessionKey;\n        //TODO get preferred algo from signature\n        pkESKeyPacket.sessionKeyAlgorithm = enums.read(enums.symmetric, config.encryption_cipher);\n        pkESKeyPacket.encrypt(encryptionKeyPacket);\n        messagePacketlist.push(pkESKeyPacket);\n      }\n    });\n\n    var literalDataPacket = new packet.literal();\n    literalDataPacket.set(message, 'utf8');\n    var literalDataPacketlist = new packet.list();\n    literalDataPacketlist.push(literalDataPacket);\n\n    var symEncryptedPacket;\n    if (config.integrity_protect) {\n      symEncryptedPacket = new packet.sym_encrypted_integrity_protected();\n    } else {\n      symEncryptedPacket = new packet.symmetrically_encrypted();\n    }\n    symEncryptedPacket.packets = literalDataPacketlist;\n    //TODO get preferred algo from signature\n    symEncryptedPacket.encrypt(enums.read(enums.symmetric, config.encryption_cipher), sessionKey);\n    messagePacketlist.push(symEncryptedPacket);\n\n    var armored = armor.encode(enums.armor.message, messagePacketlist.write(), config);\n    return armored;\n  }\n\n  function encryptAndSignMessage(publicKeys, privateKey, message) {\n\n  }\n\n  /**\n   * decrypts message\n   * @param  {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with\n   * @param  {message} message the message object with the encrypted data\n   * @return {String}         decrypted message as as native JavaScript string\n   */\n  function decryptMessage(privateKeyPacket, message) {\n    return message.decrypt(privateKeyPacket);\n  }\n\n  function decryptAndVerifyMessage(privateKey, publicKeys, messagePacketlist) {\n\n  }\n\n  function verifyMessage(publicKeys, messagePacketlist) {\n\n  }\n\n  function signMessage(privateKey, messagePacketlist) {\n\n  }\n\n  /**\n   * TODO: update this doc\n   * generates a new key pair for openpgp. Beta stage. Currently only \n   * supports RSA keys, and no subkeys.\n   * @param {Integer} keyType to indicate what type of key to make. \n   * RSA is 1. Follows algorithms outlined in OpenPGP.\n   * @param {Integer} numBits number of bits for the key creation. (should \n   * be 1024+, generally)\n   * @param {String} userId assumes already in form of \"User Name \n   * <username@email.com>\"\n   * @param {String} passphrase The passphrase used to encrypt the resulting private key\n   * @return {Object} {privateKey: [openpgp_msg_privatekey], \n   * privateKeyArmored: [string], publicKeyArmored: [string]}\n   */\n  function generateKeyPair(keyType, numBits, userId, passphrase) {\n    var packetlist = new packet.list();\n\n    var secretKeyPacket = new packet.secret_key();\n    secretKeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretKeyPacket.generate(numBits);\n    secretKeyPacket.encrypt(passphrase);\n\n    var userIdPacket = new packet.userid();\n    userIdPacket.read(userId);\n\n    var dataToSign = {};\n    dataToSign.userid = userIdPacket;\n    dataToSign.key = secretKeyPacket;\n    var signaturePacket = new packet.signature();\n    signaturePacket.issuerKeyId = secretKeyPacket.getKeyId().write();\n    signaturePacket.signatureType = enums.signature.cert_generic;\n    signaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    signaturePacket.hashAlgorithm = enums.hash.sha256;\n    signaturePacket.sign(secretKeyPacket, dataToSign);\n\n    var secretSubkeyPacket = new packet.secret_subkey();\n    secretSubkeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretSubkeyPacket.generate(numBits);\n    secretSubkeyPacket.encrypt(passphrase);\n\n    dataToSign = {};\n    dataToSign.key = secretKeyPacket;\n    dataToSign.bind = secretSubkeyPacket;\n    var subkeySignaturePacket = new packet.signature();\n    subkeySignaturePacket.issuerKeyId = secretSubkeyPacket.getKeyId().write();\n    subkeySignaturePacket.signatureType = enums.signature.subkey_binding;\n    subkeySignaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    subkeySignaturePacket.hashAlgorithm = enums.hash.sha256;\n    subkeySignaturePacket.sign(secretSubkeyPacket, dataToSign);\n\n    packetlist.push(secretKeyPacket);\n    packetlist.push(userIdPacket);\n    packetlist.push(signaturePacket);\n    packetlist.push(secretSubkeyPacket);\n    packetlist.push(subkeySignaturePacket);\n\n    var armored = armor.encode(enums.armor.private_key, packetlist.write(), this.config);\n    return armored;\n  }\n\n  /**\n   * creates a binary string representation of an encrypted and signed message.\n   * The message will be encrypted with the public keys specified and signed\n   * with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]} Private key \n   * to be used to sign the message\n   * @param {Object[]} publickeys An arraf of {obj: [openpgp_msg_publickey]}\n   * - public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt and sign\n   * @return {String} a binary string representation of the message which \n   * can be OpenPGP armored\n   */\n  function write_signed_and_encrypted_message(privatekey, publickeys, messagetext) {\n    var result = \"\";\n    var i;\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    for (i = 0; i < publickeys.length; i++) {\n      var onepasssignature = new openpgp_packet_onepasssignature();\n      var onepasssigstr = \"\";\n      if (i === 0)\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      else\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      util.print_debug_hexstr_dump(\"onepasssigstr: |\" + onepasssigstr + \"|\\n\", onepasssigstr);\n      var datasignature = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(\n        /\\n/g, \"\\r\\n\"), privatekey);\n      util.print_debug_hexstr_dump(\"datasignature: |\" + datasignature.openpgp + \"|\\n\", datasignature.openpgp);\n      if (i === 0) {\n        result = onepasssigstr + literal + datasignature.openpgp;\n      } else {\n        result = onepasssigstr + result + datasignature.openpgp;\n      }\n    }\n\n    util.print_debug_hexstr_dump(\"signed packet: |\" + result + \"|\\n\", result);\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n  /**\n   * creates a binary string representation of an encrypted message.\n   * The message will be encrypted with the public keys specified \n   * @param {Object[]} publickeys An array of {obj: [openpgp_msg_publickey]}\n   * -public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt\n   * @return {String} a binary string representation of the message\n   * which can be OpenPGP armored\n   */\n  function write_encrypted_message(publickeys, messagetext) {\n    var result = \"\";\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    result = literal;\n\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (var i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n\n  /**\n   * creates a binary string representation a signed message.\n   * The message will be signed with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]}\n   * - the private key to be used to sign the message \n   * @param {String} messagetext message text to sign\n   * @return {Object} {Object: text [String]}, openpgp: {String} a binary\n   *  string representation of the message which can be OpenPGP\n   *   armored(openpgp) and a text representation of the message (text). \n   * This can be directly used to OpenPGP armor the message\n   */\n  function write_signed_message(privatekey, messagetext) {\n    var sig = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/,\n      \"\\r\\n\"), privatekey);\n    var result = {\n      text: messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/, \"\\r\\n\"),\n      openpgp: sig.openpgp,\n      hash: sig.hash\n    };\n    return armor.encode(2, result, null, null);\n  }\n\n  this.generateKeyPair = generateKeyPair;\n  this.write_signed_message = write_signed_message;\n  this.write_signed_and_encrypted_message = write_signed_and_encrypted_message;\n  this.encryptMessage = encryptMessage;\n  this.decryptMessage = decryptMessage;\n\n}\n\nmodule.exports = new _openpgp();\n","var enums = require('../enums.js');\n\n// This is pretty ugly, but browserify needs to have the requires explicitly written.\nmodule.exports = {\n  compressed: require('./compressed.js'),\n  sym_encrypted_integrity_protected: require('./sym_encrypted_integrity_protected.js'),\n  public_key_encrypted_session_key: require('./public_key_encrypted_session_key.js'),\n  sym_encrypted_session_key: require('./sym_encrypted_session_key.js'),\n  literal: require('./literal.js'),\n  public_key: require('./public_key.js'),\n  symmetrically_encrypted: require('./symmetrically_encrypted.js'),\n  marker: require('./marker.js'),\n  public_subkey: require('./public_subkey.js'),\n  user_attribute: require('./user_attribute.js'),\n  one_pass_signature: require('./one_pass_signature.js'),\n  secret_key: require('./secret_key.js'),\n  userid: require('./userid.js'),\n  secret_subkey: require('./secret_subkey.js'),\n  signature: require('./signature.js'),\n  trust: require('./trust.js')\n}\n\nfor (var i in enums.packet) {\n  var packetClass = module.exports[i];\n\n  if (packetClass != undefined)\n    packetClass.prototype.tag = enums.packet[i];\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  JXG = require('../compression/jxg.js'),\n  base64 = require('../encoding/base64.js');\n\n/**\n * @class\n * @classdesc Implementation of the Compressed Data Packet (Tag 8)\n * \n * RFC4880 5.6:\n * The Compressed Data packet contains compressed data.  Typically, this\n * packet is found as the contents of an encrypted packet, or following\n * a Signature or One-Pass Signature packet, and contains a literal data\n * packet.\n */\nmodule.exports = function packet_compressed() {\n  /** @type {packetlist} */\n  this.packets;\n  /** @type {compression} */\n  this.algorithm = 'uncompressed';\n\n  this.compressed = null;\n\n\n  /**\n   * Parsing function for the packet.\n   * @param {String} input Payload of a tag 8 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @parAM {iNTEGER} LEN lENGTH OF the packet or the remaining length of \n   * input at position\n   * @return {openpgp_packet_compressed} Object representation\n   */\n  this.read = function(bytes) {\n    // One octet that gives the algorithm used to compress the packet.\n    this.algorithm = enums.read(enums.compression, bytes.charCodeAt(0));\n\n    // Compressed data, which makes up the remainder of the packet.\n    this.compressed = bytes.substr(1);\n\n    this.decompress();\n  }\n\n\n\n  this.write = function() {\n    if (this.compressed == null)\n      this.compress();\n\n    return String.fromCharCode(enums.write(enums.compression, this.algorithm)) + this.compressed;\n  }\n\n\n  /**\n   * Decompression method for decompressing the compressed data\n   * read by read_packet\n   * @return {String} The decompressed data\n   */\n  this.decompress = function() {\n    var decompressed;\n\n    switch (this.algorithm) {\n      case 'uncompressed':\n        decompressed = this.compressed;\n        break;\n\n      case 'zip':\n        var compData = this.compressed;\n\n        var radix = base64.encode(compData).replace(/\\n/g, \"\");\n        // no header in this case, directly call deflate\n        var jxg_obj = new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(radix));\n\n        decompressed = unescape(jxg_obj.deflate()[0][0]);\n        break;\n\n      case 'zlib':\n        //RFC 1950. Bits 0-3 Compression Method\n        var compressionMethod = this.compressed.charCodeAt(0) % 0x10;\n\n        //Bits 4-7 RFC 1950 are LZ77 Window. Generally this value is 7 == 32k window size.\n        // 2nd Byte in RFC 1950 is for \"FLAGs\" Allows for a Dictionary \n        // (how is this defined). Basic checksum, and compression level.\n\n        if (compressionMethod == 8) { //CM 8 is for DEFLATE, RFC 1951\n          // remove 4 bytes ADLER32 checksum from the end\n          var compData = this.compressed.substring(0, this.compressed.length - 4);\n          var radix = base64.encode(compData).replace(/\\n/g, \"\");\n          //TODO check ADLER32 checksum\n          decompressed = JXG.decompress(radix);\n          break;\n\n        } else {\n          util.print_error(\"Compression algorithm ZLIB only supports \" +\n            \"DEFLATE compression method.\");\n        }\n        break;\n\n      case 'bzip2':\n        // TODO: need to implement this\n        throw new Error('Compression algorithm BZip2 [BZ2] is not implemented.');\n        break;\n\n      default:\n        throw new Error(\"Compression algorithm unknown :\" + this.alogrithm);\n        break;\n    }\n\n    this.packets.read(decompressed);\n  }\n\n  /**\n   * Compress the packet data (member decompressedData)\n   * @param {Integer} type Algorithm to be used // See RFC 4880 9.3\n   * @param {String} data Data to be compressed\n   * @return {String} The compressed data stored in attribute compressedData\n   */\n  this.compress = function() {\n    switch (this.algorithm) {\n\n      case 'uncompressed':\n        // - Uncompressed\n        this.compressed = this.packets.write();\n        break;\n\n      case 'zip':\n        // - ZIP [RFC1951]\n        util.print_error(\"Compression algorithm ZIP [RFC1951] is not implemented.\");\n        break;\n\n      case 'zlib':\n        // - ZLIB [RFC1950]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm ZLIB [RFC1950] is not implemented.\");\n        break;\n\n      case 'bzip2':\n        //  - BZip2 [BZ2]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm BZip2 [BZ2] is not implemented.\");\n        break;\n\n      default:\n        util.print_error(\"Compression algorithm unknown :\" + this.type);\n        break;\n    }\n  }\n};\n","var enums = require('../enums.js');\n\nmodule.exports = {\n  list: require('./packetlist.js'),\n};\n\nvar packets = require('./all_packets.js');\n\nfor (var i in packets)\n  module.exports[i] = packets[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc Implementation of the Literal Data Packet (Tag 11)\n * \n * RFC4880 5.9: A Literal Data packet contains the body of a message; data that\n * is not to be further interpreted.\n */\nmodule.exports = function packet_literal() {\n  this.format = 'utf8';\n  this.data = '';\n  this.date = new Date();\n\n\n  /**\n   * Set the packet data to a javascript native string or a squence of \n   * bytes. Conversion to a proper utf8 encoding takes place when the \n   * packet is written.\n   * @param {String} str Any native javascript string\n   * @param {openpgp_packet_literaldata.format} format \n   */\n  this.set = function(str, format) {\n    this.format = format;\n    this.data = str;\n  }\n\n  /**\n   * Set the packet data to value represented by the provided string\n   * of bytes together with the appropriate conversion format.\n   * @param {String} bytes The string of bytes\n   * @param {openpgp_packet_literaldata.format} format\n   */\n  this.setBytes = function(bytes, format) {\n    this.format = format;\n\n    if (format == 'utf8')\n      bytes = util.decode_utf8(bytes);\n\n    this.data = bytes;\n  }\n\n  /**\n   * Get the byte sequence representing the literal packet data\n   * @returns {String} A sequence of bytes\n   */\n  this.getBytes = function() {\n    if (this.format == 'utf8')\n      return util.encode_utf8(this.data);\n    else\n      return this.data;\n  }\n\n\n\n  /**\n   * Parsing function for a literal data packet (tag 11).\n   * \n   * @param {String} input Payload of a tag 11 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    // - A one-octet field that describes how the data is formatted.\n\n    var format = enums.read(enums.literal, bytes[0].charCodeAt());\n\n    var filename_len = bytes.charCodeAt(1);\n    this.filename = util.decode_utf8(bytes.substr(2, filename_len));\n\n    this.date = util.readDate(bytes.substr(2 + filename_len, 4));\n\n    var data = bytes.substring(6 + filename_len);\n\n    this.setBytes(data, format);\n  }\n\n  /**\n   * Creates a string representation of the packet\n   * \n   * @param {String} data The data to be inserted as body\n   * @return {String} string-representation of the packet\n   */\n  this.write = function() {\n    var filename = util.encode_utf8(\"msg.txt\");\n\n    var data = this.getBytes();\n\n    var result = '';\n    result += String.fromCharCode(enums.write(enums.literal, this.format));\n    result += String.fromCharCode(filename.length);\n    result += filename;\n    result += util.writeDate(this.date);\n    result += data;\n    return result;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the strange \"Marker packet\" (Tag 10)\n * \n * RFC4880 5.8: An experimental version of PGP used this packet as the Literal\n * packet, but no released version of PGP generated Literal packets with this\n * tag. With PGP 5.x, this packet has been reassigned and is reserved for use as\n * the Marker packet.\n * \n * Such a packet MUST be ignored when received.\n */\nfunction packet_marker() {\n  /**\n   * Parsing function for a literal data packet (tag 10).\n   * \n   * @param {String} input Payload of a tag 10 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    if (bytes[0].charCodeAt() == 0x50 && // P\n    bytes[1].charCodeAt() == 0x47 && // G\n    bytes[2].charCodeAt() == 0x50) // P\n      return true;\n    // marker packet does not contain \"PGP\"\n    return false;\n  }\n}\n\nmodule.exports = packet_marker;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the One-Pass Signature Packets (Tag 4)\n * \n * RFC4880 5.4:\n * The One-Pass Signature packet precedes the signed data and contains\n * enough information to allow the receiver to begin calculating any\n * hashes needed to verify the signature.  It allows the Signature\n * packet to be placed at the end of the message, so that the signer\n * can compute the entire signed message in one pass.\n */\n\nvar enums = require('../enums.js'),\n  type_keyid = require('../type/keyid.js');\n\nmodule.exports = function packet_one_pass_signature() {\n  this.version = null; // A one-octet version number.  The current version is 3.\n  this.type = null; // A one-octet signature type.  Signature types are described in RFC4880 Section 5.2.1.\n  this.hashAlgorithm = null; // A one-octet number describing the hash algorithm used. (See RFC4880 9.4)\n  this.publicKeyAlgorithm = null; // A one-octet number describing the public-key algorithm used. (See RFC4880 9.1)\n  this.signingKeyId = null; // An eight-octet number holding the Key ID of the signing key.\n  this.flags = null; //  A one-octet number holding a flag showing whether the signature is nested.  A zero value indicates that the next packet is another One-Pass Signature packet that describes another signature to be applied to the same message data.\n\n  /**\n   * parsing function for a one-pass signature packet (tag 4).\n   * @param {String} bytes payload of a tag 4 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var mypos = 0;\n    // A one-octet version number.  The current version is 3.\n    this.version = bytes.charCodeAt(mypos++);\n\n    // A one-octet signature type.  Signature types are described in\n    //   Section 5.2.1.\n    this.type = enums.read(enums.signature, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the hash algorithm used.\n    this.hashAlgorithm = enums.read(enums.hash, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the public-key algorithm used.\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes.charCodeAt(mypos++));\n\n    // An eight-octet number holding the Key ID of the signing key.\n    this.signingKeyId = new type_keyid();\n    this.signingKeyId.read(bytes.substr(mypos));\n    mypos += 8;\n\n    // A one-octet number holding a flag showing whether the signature\n    //   is nested.  A zero value indicates that the next packet is\n    //   another One-Pass Signature packet that describes another\n    //   signature to be applied to the same message data.\n    this.flags = bytes.charCodeAt(mypos++);\n    return this;\n  }\n\n  /**\n   * creates a string representation of a one-pass signature packet\n   * @param {Integer} type Signature types as described in RFC4880 Section 5.2.1.\n   * @param {Integer} hashalgorithm the hash algorithm used within the signature\n   * @param {openpgp_msg_privatekey} privatekey the private key used to generate the signature\n   * @param {Integer} length length of data to be signed\n   * @param {boolean} nested boolean showing whether the signature is nested. \n   *  \"true\" indicates that the next packet is another One-Pass Signature packet\n   *   that describes another signature to be applied to the same message data. \n   * @return {String} a string representation of a one-pass signature packet\n   */\n  this.write = function(type, hashalgorithm, privatekey, length, nested) {\n    var result = \"\";\n\n    result += String.fromCharCode(3);\n    result += String.fromCharCode(enums.write(enums.signature, type));\n    result += String.fromCharCode(enums.write(enums.hash, this.hashAlgorithm));\n    result += String.fromCharCode(enums.write(enums.publicKey, privatekey.algorithm));\n    result += privatekey.getKeyId().write();\n    if (nested)\n      result += String.fromCharCode(0);\n    else\n      result += String.fromCharCode(1);\n\n    return result;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util');\n\n\nmodule.exports = {\n  readSimpleLength: function(bytes) {\n    var len = 0,\n      offset,\n      type = bytes[0].charCodeAt();\n\n\n    if (type < 192) {\n      len = bytes[0].charCodeAt();\n      offset = 1;\n    } else if (type < 255) {\n      len = ((bytes[0].charCodeAt() - 192) << 8) + (bytes[1].charCodeAt()) + 192;\n      offset = 2;\n    } else if (type == 255) {\n      len = util.readNumber(bytes.substr(1, 4));\n      offset = 5;\n    }\n\n    return {\n      len: len,\n      offset: offset\n    };\n  },\n\n  /**\n   * Encodes a given integer of length to the openpgp length specifier to a\n   * string\n   * \n   * @param {Integer} length The length to encode\n   * @return {String} String with openpgp length representation\n   */\n  writeSimpleLength: function(length) {\n    var result = \"\";\n    if (length < 192) {\n      result += String.fromCharCode(length);\n    } else if (length > 191 && length < 8384) {\n      /*\n       * let a = (total data packet length) - 192 let bc = two octet\n       * representation of a let d = b + 192\n       */\n      result += String.fromCharCode(((length - 192) >> 8) + 192);\n      result += String.fromCharCode((length - 192) & 0xFF);\n    } else {\n      result += String.fromCharCode(255);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Writes a packet header version 4 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeHeader: function(tag_type, length) {\n    /* we're only generating v4 packet headers here */\n    var result = \"\";\n    result += String.fromCharCode(0xC0 | tag_type);\n    result += this.writeSimpleLength(length);\n    return result;\n  },\n\n  /**\n   * Writes a packet header Version 3 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeOldHeader: function(tag_type, length) {\n    var result = \"\";\n    if (length < 256) {\n      result += String.fromCharCode(0x80 | (tag_type << 2));\n      result += String.fromCharCode(length);\n    } else if (length < 65536) {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 1);\n      result += util.writeNumber(length, 2);\n    } else {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 2);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Generic static Packet Parser function\n   * \n   * @param {String} input Input stream as string\n   * @param {integer} position Position to start parsing\n   * @param {integer} len Length of the input from position on\n   * @return {Object} Returns a parsed openpgp_packet\n   */\n  read: function(input, position, len) {\n    // some sanity checks\n    if (input == null || input.length <= position || input.substring(position).length < 2 || (input[position].charCodeAt() &\n      0x80) == 0) {\n      util\n        .print_error(\"Error during parsing. This message / key is probably not containing a valid OpenPGP format.\");\n      return null;\n    }\n    var mypos = position;\n    var tag = -1;\n    var format = -1;\n    var packet_length;\n\n    format = 0; // 0 = old format; 1 = new format\n    if ((input[mypos].charCodeAt() & 0x40) != 0) {\n      format = 1;\n    }\n\n    var packet_length_type;\n    if (format) {\n      // new format header\n      tag = input[mypos].charCodeAt() & 0x3F; // bit 5-0\n    } else {\n      // old format header\n      tag = (input[mypos].charCodeAt() & 0x3F) >> 2; // bit 5-2\n      packet_length_type = input[mypos].charCodeAt() & 0x03; // bit 1-0\n    }\n\n    // header octet parsing done\n    mypos++;\n\n    // parsed length from length field\n    var bodydata = null;\n\n    // used for partial body lengths\n    var real_packet_length = -1;\n    if (!format) {\n      // 4.2.1. Old Format Packet Lengths\n      switch (packet_length_type) {\n        case 0:\n          // The packet has a one-octet length. The header is 2 octets\n          // long.\n          packet_length = input[mypos++].charCodeAt();\n          break;\n        case 1:\n          // The packet has a two-octet length. The header is 3 octets\n          // long.\n          packet_length = (input[mypos++].charCodeAt() << 8) | input[mypos++].charCodeAt();\n          break;\n        case 2:\n          // The packet has a four-octet length. The header is 5\n          // octets long.\n          packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n            8) | input[mypos++].charCodeAt();\n          break;\n        default:\n          // 3 - The packet is of indeterminate length. The header is 1\n          // octet long, and the implementation must determine how long\n          // the packet is. If the packet is in a file, this means that\n          // the packet extends until the end of the file. In general, \n          // an implementation SHOULD NOT use indeterminate-length \n          // packets except where the end of the data will be clear \n          // from the context, and even then it is better to use a \n          // definite length, or a new format header. The new format \n          // headers described below have a mechanism for precisely\n          // encoding data of indeterminate length.\n          packet_length = len;\n          break;\n      }\n\n    } else // 4.2.2. New Format Packet Lengths\n    {\n\n      // 4.2.2.1. One-Octet Lengths\n      if (input[mypos].charCodeAt() < 192) {\n        packet_length = input[mypos++].charCodeAt();\n        util.print_debug(\"1 byte length:\" + packet_length);\n        // 4.2.2.2. Two-Octet Lengths\n      } else if (input[mypos].charCodeAt() >= 192 && input[mypos].charCodeAt() < 224) {\n        packet_length = ((input[mypos++].charCodeAt() - 192) << 8) + (input[mypos++].charCodeAt()) + 192;\n        util.print_debug(\"2 byte length:\" + packet_length);\n        // 4.2.2.4. Partial Body Lengths\n      } else if (input[mypos].charCodeAt() > 223 && input[mypos].charCodeAt() < 255) {\n        packet_length = 1 << (input[mypos++].charCodeAt() & 0x1F);\n        util.print_debug(\"4 byte length:\" + packet_length);\n        // EEEK, we're reading the full data here...\n        var mypos2 = mypos + packet_length;\n        bodydata = input.substring(mypos, mypos + packet_length);\n        while (true) {\n          if (input[mypos2].charCodeAt() < 192) {\n            var tmplen = input[mypos2++].charCodeAt();\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() >= 192 && input[mypos2].charCodeAt() < 224) {\n            var tmplen = ((input[mypos2++].charCodeAt() - 192) << 8) + (input[mypos2++].charCodeAt()) + 192;\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() > 223 && input[mypos2].charCodeAt() < 255) {\n            var tmplen = 1 << (input[mypos2++].charCodeAt() & 0x1F);\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n          } else {\n            mypos2++;\n            var tmplen = (input[mypos2++].charCodeAt() << 24) | (input[mypos2++].charCodeAt() << 16) | (input[mypos2++]\n              .charCodeAt() << 8) | input[mypos2++].charCodeAt();\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            packet_length += tmplen;\n            mypos2 += tmplen;\n            break;\n          }\n        }\n        real_packet_length = mypos2;\n        // 4.2.2.3. Five-Octet Lengths\n      } else {\n        mypos++;\n        packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n          8) | input[mypos++].charCodeAt();\n      }\n    }\n\n    // if there was'nt a partial body length: use the specified\n    // packet_length\n    if (real_packet_length == -1) {\n      real_packet_length = packet_length;\n    }\n\n    if (bodydata == null) {\n      bodydata = input.substring(mypos, mypos + real_packet_length);\n    }\n\n    return {\n      tag: tag,\n      packet: bodydata,\n      offset: mypos + real_packet_length\n    };\n  }\n}\n","var packetParser = require('./packet.js'),\n  packets = require('./all_packets.js'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc This class represents a list of openpgp packets.\n * Take care when iterating over it - the packets themselves\n * are stored as numerical indices.\n */\nmodule.exports = function packetlist() {\n  /** The number of packets contained within the list.\n   * @readonly\n   * @type {Integer} */\n  this.length = 0;\n\n  /**\n   * Reads a stream of binary data and interprents it as a list of packets.\n   * @param {openpgp_bytearray} An array of bytes.\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    while (i < bytes.length) {\n      var parsed = packetParser.read(bytes, i, bytes.length - i);\n      i = parsed.offset;\n\n      var tag = enums.read(enums.packet, parsed.tag);\n      var packet = new packets[tag]();\n\n      this.push(packet);\n\n      packet.read(parsed.packet);\n    }\n  }\n\n  /**\n   * Creates a binary representation of openpgp objects contained within the\n   * class instance.\n   * @returns {openpgp_bytearray} An array of bytes containing valid openpgp packets.\n   */\n  this.write = function() {\n    var bytes = '';\n\n    for (var i = 0; i < this.length; i++) {\n      var packetbytes = this[i].write();\n      bytes += packetParser.writeHeader(this[i].tag, packetbytes.length);\n      bytes += packetbytes;\n    }\n\n    return bytes;\n  }\n\n  /**\n   * Adds a packet to the list. This is the only supported method of doing so;\n   * writing to packetlist[i] directly will result in an error.\n   */\n  this.push = function(packet) {\n    packet.packets = new packetlist();\n\n    this[this.length] = packet;\n    this.length++;\n  }\n\n    /**\n   * Creates a new packetList with all packets that pass the test implemented by the provided function.\n   */\n   this.filter = function(callback) {\n\n    var filtered = new packetlist();\n\n    for (var i = 0; i < this.length; i++) {\n      if (callback(this[i], i, this)) {\n        filtered.push(this[i]);\n      }\n    }\n\n    return filtered; \n   }\n\n   /**\n   * Creates a new packetList with all packets from the given type\n   */\n   this.filterByType = function(packetType) {\n     return this.filter(function(packet) {\n       return packet.tag == packetType;\n     });\n   } \n\n   /**\n    * Executes the provided callback once for each element\n    */\n   this.forEach = function(callback) {\n     for (var i = 0; i < this.length; i++) {\n      callback(this[i]);\n    }\n   }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  type_keyid = require('../type/keyid.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nmodule.exports = function packet_public_key() {\n  /** Key creation date.\n   * @type {Date} */\n  this.created = new Date();\n  /** A list of multiprecision integers\n   * @type {openpgp_type_mpi} */\n  this.mpi = [];\n  /** Public key algorithm\n   * @type {openpgp.publickey} */\n  this.algorithm = 'rsa_sign';\n\n\n  /**\n   * Internal Parser for public keys as specified in RFC 4880 section \n   * 5.5.2 Public-Key Packet Formats\n   * called by read_tag&lt;num&gt;\n   * @param {String} input Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of input\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.readPublicKey = this.read = function(bytes) {\n    // A one-octet version number (3 or 4).\n    var version = bytes[0].charCodeAt();\n\n    if (version == 4) {\n      // - A four-octet number denoting the time that the key was created.\n      this.created = util.readDate(bytes.substr(1, 4));\n\n      // - A one-octet number denoting the public-key algorithm of this key.\n      this.algorithm = enums.read(enums.publicKey, bytes[5].charCodeAt());\n\n      var mpicount = crypto.getPublicMpiCount(this.algorithm);\n      this.mpi = [];\n\n      var bmpi = bytes.substr(6);\n      var p = 0;\n\n      for (var i = 0; i < mpicount && p < bmpi.length; i++) {\n\n        this.mpi[i] = new type_mpi();\n\n        p += this.mpi[i].read(bmpi.substr(p))\n\n        if (p > bmpi.length)\n          util.print_error(\"openpgp.packet.keymaterial.js\\n\" + 'error reading MPI @:' + p);\n      }\n\n      return p + 6;\n    } else {\n      throw new Error('Version ' + version + ' of the key packet is unsupported.');\n    }\n  }\n\n  /*\n   * Same as write_private_key, but has less information because of \n   * public key.\n   * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n   * IE 1 corresponds to RSA.\n   * @param {RSA.keyObject} key\n   * @param timePacket\n   * @return {Object} {body: [string]OpenPGP packet body contents, \n   * header: [string] OpenPGP packet header, string: [string] header+body}\n   */\n  this.writePublicKey = this.write = function() {\n    // Version\n    var result = String.fromCharCode(4);\n    result += util.writeDate(this.created);\n    result += String.fromCharCode(enums.write(enums.publicKey, this.algorithm));\n\n    var mpicount = crypto.getPublicMpiCount(this.algorithm);\n\n    for (var i = 0; i < mpicount; i++) {\n      result += this.mpi[i].write();\n    }\n\n    return result;\n  }\n\n  // Write an old version packet - it's used by some of the internal routines.\n  this.writeOld = function() {\n    var bytes = this.writePublicKey();\n\n    return String.fromCharCode(0x99) +\n      util.writeNumber(bytes.length, 2) +\n      bytes;\n  }\n\n  /**\n   * Calculates the key id of the key \n   * @return {String} A 8 byte key id\n   */\n  this.getKeyId = function() {\n    var keyid = new type_keyid();\n    keyid.read(this.getFingerprint().substr(12, 8));\n    return keyid;\n  }\n\n  /**\n   * Calculates the fingerprint of the key\n   * @return {String} A string containing the fingerprint\n   */\n  this.getFingerprint = function() {\n    var toHash = this.writeOld();\n    return crypto.hash.sha1(toHash, toHash.length);\n  }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_keyid = require('../type/keyid.js'),\n  util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_public_key_encrypted_session_key() {\n  this.version = 3;\n\n  this.publicKeyId = new type_keyid();\n  this.publicKeyAlgorithm = 'rsa_encrypt';\n\n  this.sessionKey = null;\n  this.sessionKeyAlgorithm = 'aes256';\n\n  /** @type {openpgp_type_mpi[]} */\n  this.encrypted = [];\n\n  /**\n   * Parsing function for a publickey encrypted session key packet (tag 1).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n\n    this.version = bytes[0].charCodeAt();\n    this.publicKeyId.read(bytes.substr(1));\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes[9].charCodeAt());\n\n    var i = 10;\n\n    var integerCount = (function(algo) {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          return 1;\n\n        case 'elgamal':\n          return 2;\n\n        default:\n          throw new Error(\"Invalid algorithm.\");\n      }\n    })(this.publicKeyAlgorithm);\n\n    this.encrypted = [];\n\n    for (var j = 0; j < integerCount; j++) {\n      var mpi = new type_mpi();\n      i += mpi.read(bytes.substr(i));\n      this.encrypted.push(mpi);\n    }\n  }\n\n  /**\n   * Create a string representation of a tag 1 packet\n   * \n   * @param {String} publicKeyId\n   *             The public key id corresponding to publicMPIs key as string\n   * @param {openpgp_type_mpi[]} publicMPIs\n   *            Multiprecision integer objects describing the public key\n   * @param {Integer} pubalgo\n   *            The corresponding public key algorithm // See RFC4880 9.1\n   * @param {Integer} symmalgo\n   *            The symmetric cipher algorithm used to encrypt the data \n   *            within an encrypteddatapacket or encryptedintegrity-\n   *            protecteddatapacket \n   *            following this packet //See RFC4880 9.2\n   * @param {String} sessionkey\n   *            A string of randombytes representing the session key\n   * @return {String} The string representation\n   */\n  this.write = function() {\n\n    var result = String.fromCharCode(this.version);\n    result += this.publicKeyId.write();\n    result += String.fromCharCode(\n      enums.write(enums.publicKey, this.publicKeyAlgorithm));\n\n    for (var i = 0; i < this.encrypted.length; i++) {\n      result += this.encrypted[i].write()\n    }\n\n    return result;\n  }\n\n  this.encrypt = function(key) {\n    var data = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm));\n\n    data += this.sessionKey;\n    var checksum = util.calc_checksum(this.sessionKey);\n    data += util.writeNumber(checksum, 2);\n\n    var mpi = new type_mpi();\n    mpi.fromBytes(crypto.pkcs1.eme.encode(\n      data,\n      key.mpi[0].byteLength()));\n\n    this.encrypted = crypto.publicKeyEncrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      mpi);\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(key) {\n    var result = crypto.publicKeyDecrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      this.encrypted).toBytes();\n\n    var checksum = util.readNumber(result.substr(result.length - 2));\n\n    var decoded = crypto.pkcs1.eme.decode(\n      result,\n      key.mpi[0].byteLength());\n\n    var key = decoded.substring(1, decoded.length - 2);\n\n    if (checksum != util.calc_checksum(key)) {\n      throw new Error('Checksum mismatch');\n    } else {\n      this.sessionKey = key;\n      this.sessionKeyAlgorithm =\n        enums.read(enums.symmetric, decoded.charCodeAt(0));\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar public_key = require('./public_key.js');\n\nmodule.exports = function public_subkey() {\n  public_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar publicKey = require('./public_key.js'),\n  enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js'),\n  type_s2k = require('../type/s2k.js');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nfunction packet_secret_key() {\n  publicKey.call(this);\n\n  this.encrypted = null;\n\n\n  function get_hash_len(hash) {\n    if (hash == 'sha1')\n      return 20;\n    else\n      return 2;\n  }\n\n  function get_hash_fn(hash) {\n    if (hash == 'sha1')\n      return crypto.hash.sha1;\n    else\n      return function(c) {\n        return util.writeNumber(util.calc_checksum(c), 2);\n    };\n  }\n\n  // Helper function\n\n  function parse_cleartext_mpi(hash_algorithm, cleartext, algorithm) {\n    var hashlen = get_hash_len(hash_algorithm),\n      hashfn = get_hash_fn(hash_algorithm);\n\n    var hashtext = cleartext.substr(cleartext.length - hashlen);\n    cleartext = cleartext.substr(0, cleartext.length - hashlen);\n\n    var hash = hashfn(cleartext);\n\n    if (hash != hashtext)\n      throw new Error(\"Hash mismatch.\");\n\n    var mpis = crypto.getPrivateMpiCount(algorithm);\n\n    var j = 0;\n    var mpi = [];\n\n    for (var i = 0; i < mpis && j < cleartext.length; i++) {\n      mpi[i] = new type_mpi();\n      j += mpi[i].read(cleartext.substr(j));\n    }\n\n    return mpi;\n  }\n\n  function write_cleartext_mpi(hash_algorithm, algorithm, mpi) {\n    var bytes = '';\n    var discard = crypto.getPublicMpiCount(algorithm);\n\n    for (var i = discard; i < mpi.length; i++) {\n      bytes += mpi[i].write();\n    }\n\n\n    bytes += get_hash_fn(hash_algorithm)(bytes);\n\n    return bytes;\n  }\n\n\n  // 5.5.3.  Secret-Key Packet Formats\n\n  /**\n   * Internal parser for private keys as specified in RFC 4880 section 5.5.3\n   * @param {String} bytes Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of bytes\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.read = function(bytes) {\n    // - A Public-Key or Public-Subkey packet, as described above.\n    var len = this.readPublicKey(bytes);\n\n    bytes = bytes.substr(len);\n\n\n    // - One octet indicating string-to-key usage conventions.  Zero\n    //   indicates that the secret-key data is not encrypted.  255 or 254\n    //   indicates that a string-to-key specifier is being given.  Any\n    //   other value is a symmetric-key encryption algorithm identifier.\n    var isEncrypted = bytes[0].charCodeAt();\n\n    if (isEncrypted) {\n      this.encrypted = bytes;\n    } else {\n\n      // - Plain or encrypted multiprecision integers comprising the secret\n      //   key data.  These algorithm-specific fields are as described\n      //   below.\n\n      this.mpi = this.mpi.concat(parse_cleartext_mpi('mod', bytes.substr(1),\n        this.algorithm));\n    }\n\n  };\n\n  /*\n     * Creates an OpenPGP key packet for the given key. much \n\t * TODO in regards to s2k, subkeys.\n     * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n\t * IE 1 corresponds to RSA.\n     * @param {RSA.keyObject} key\n     * @param passphrase\n     * @param s2kHash\n     * @param symmetricEncryptionAlgorithm\n     * @param timePacket\n     * @return {Object} {body: [string]OpenPGP packet body contents, \n\t\theader: [string] OpenPGP packet header, string: [string] header+body}\n     */\n  this.write = function() {\n    var bytes = this.writePublicKey();\n\n    if (!this.encrypted) {\n      bytes += String.fromCharCode(0);\n\n      bytes += write_cleartext_mpi('mod', this.algorithm, this.mpi);\n    } else {\n      bytes += this.encrypted;\n    }\n\n    return bytes;\n  };\n\n\n\n\n  /** Encrypt the payload. By default, we use aes256 and iterated, salted string\n   * to key specifier\n   * @param {String} passphrase\n   */\n  this.encrypt = function(passphrase) {\n\n    var s2k = new type_s2k(),\n      symmetric = 'aes256',\n      cleartext = write_cleartext_mpi('sha1', this.algorithm, this.mpi),\n      key = produceEncryptionKey(s2k, passphrase, symmetric),\n      blockLen = crypto.cipher[symmetric].blockSize,\n      iv = crypto.random.getRandomBytes(blockLen);\n\n\n    this.encrypted = '';\n    this.encrypted += String.fromCharCode(254);\n    this.encrypted += String.fromCharCode(enums.write(enums.symmetric, symmetric));\n    this.encrypted += s2k.write();\n    this.encrypted += iv;\n\n    this.encrypted += crypto.cfb.normalEncrypt(symmetric, key, cleartext, iv);\n  };\n\n  function produceEncryptionKey(s2k, passphrase, algorithm) {\n    return s2k.produce_key(passphrase,\n      crypto.cipher[algorithm].keySize);\n  }\n\n  /**\n   * Decrypts the private key MPIs which are needed to use the key.\n   * openpgp_packet_keymaterial.hasUnencryptedSecretKeyData should be \n   * false otherwise\n   * a call to this function is not needed\n   * \n   * @param {String} str_passphrase The passphrase for this private key \n   * as string\n   * @return {Boolean} True if the passphrase was correct; false if not\n   */\n  this.decrypt = function(passphrase) {\n    if (!this.encrypted)\n      return;\n\n    var i = 0,\n      symmetric,\n      key;\n\n    var s2k_usage = this.encrypted[i++].charCodeAt();\n\n    // - [Optional] If string-to-key usage octet was 255 or 254, a one-\n    //   octet symmetric encryption algorithm.\n    if (s2k_usage == 255 || s2k_usage == 254) {\n      symmetric = this.encrypted[i++].charCodeAt();\n      symmetric = enums.read(enums.symmetric, symmetric);\n\n      // - [Optional] If string-to-key usage octet was 255 or 254, a\n      //   string-to-key specifier.  The length of the string-to-key\n      //   specifier is implied by its type, as described above.\n      var s2k = new type_s2k();\n      i += s2k.read(this.encrypted.substr(i));\n\n      key = produceEncryptionKey(s2k, passphrase, symmetric);\n    } else {\n      symmetric = s2k_usage;\n      symmetric = enums.read(enums.symmetric, symmetric);\n      key = crypto.hash.md5(passphrase);\n    }\n\n\n    // - [Optional] If secret data is encrypted (string-to-key usage octet\n    //   not zero), an Initial Vector (IV) of the same length as the\n    //   cipher's block size.\n    var iv = this.encrypted.substr(i,\n      crypto.cipher[symmetric].blockSize);\n\n    i += iv.length;\n\n    var cleartext,\n      ciphertext = this.encrypted.substr(i);\n\n    cleartext = crypto.cfb.normalDecrypt(symmetric, key, ciphertext, iv);\n\n    var hash = s2k_usage == 254 ?\n      'sha1' :\n      'mod';\n\n\n    this.mpi = this.mpi.concat(parse_cleartext_mpi(hash, cleartext,\n      this.algorithm));\n  };\n\n  this.generate = function(bits, passphrase) {\n    this.mpi = crypto.generateMpi(this.algorithm, bits);\n  };\n\n}\n\npacket_secret_key.prototype = new publicKey;\n\nmodule.exports = packet_secret_key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar secret_key = require('./secret_key.js');\n\nmodule.exports = function secret_subkey() {\n  secret_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  packet = require('./packet.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js');\n\n/**\n * @class\n * @classdesc Implementation of the Signature Packet (Tag 2)\n * \n * RFC4480 5.2:\n * A Signature packet describes a binding between some public key and\n * some data.  The most common signatures are a signature of a file or a\n * block of text, and a signature that is a certification of a User ID.\n */\nmodule.exports = function packet_signature() {\n\n  this.signatureType = null;\n  this.hashAlgorithm = null;\n  this.publicKeyAlgorithm = null;\n\n  this.signatureData = null;\n  this.signedHashValue = null;\n  this.mpi = null;\n\n  this.created = null;\n  this.signatureExpirationTime = null;\n  this.signatureNeverExpires = null;\n  this.exportable = null;\n  this.trustLevel = null;\n  this.trustAmount = null;\n  this.regularExpression = null;\n  this.revocable = null;\n  this.keyExpirationTime = null;\n  this.keyNeverExpires = null;\n  this.preferredSymmetricAlgorithms = null;\n  this.revocationKeyClass = null;\n  this.revocationKeyAlgorithm = null;\n  this.revocationKeyFingerprint = null;\n  this.issuerKeyId = null;\n  this.notation = {};\n  this.preferredHashAlgorithms = null;\n  this.preferredCompressionAlgorithms = null;\n  this.keyServerPreferences = null;\n  this.preferredKeyServer = null;\n  this.isPrimaryUserID = null;\n  this.policyURI = null;\n  this.keyFlags = null;\n  this.signersUserId = null;\n  this.reasonForRevocationFlag = null;\n  this.reasonForRevocationString = null;\n  this.signatureTargetPublicKeyAlgorithm = null;\n  this.signatureTargetHashAlgorithm = null;\n  this.signatureTargetHash = null;\n  this.embeddedSignature = null;\n\n  this.verified = false;\n\n  /**\n   * parsing function for a signature packet (tag 2).\n   * @param {String} bytes payload of a tag 2 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    var version = bytes[i++].charCodeAt();\n    // switch on version (3 and 4)\n    switch (version) {\n      case 3:\n        // One-octet length of following hashed material. MUST be 5.\n        if (bytes[i++].charCodeAt() != 5)\n          util.print_debug(\"openpgp.packet.signature.js\\n\" +\n            'invalid One-octet length of following hashed material.' +\n            'MUST be 5. @:' + (i - 1));\n\n        var sigpos = i;\n        // One-octet signature type.\n        this.signatureType = bytes[i++].charCodeAt();\n\n        // Four-octet creation time.\n        this.created = util.readDate(bytes.substr(i, 4));\n        i += 4;\n\n        // storing data appended to data which gets verified\n        this.signatureData = bytes.substring(position, i);\n\n        // Eight-octet Key ID of signer.\n        this.issuerKeyId = bytes.substring(i, i + 8);\n        i += 8;\n\n        // One-octet public-key algorithm.\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n\n        // One-octet hash algorithm.\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n        break;\n      case 4:\n        this.signatureType = bytes[i++].charCodeAt();\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n\n\n        function subpackets(bytes, signed) {\n          // Two-octet scalar octet count for following hashed subpacket\n          // data.\n          var subpacket_length = util.readNumber(\n            bytes.substr(0, 2));\n\n          var i = 2;\n\n          // Hashed subpacket data set (zero or more subpackets)\n          var subpacked_read = 0;\n          while (i < 2 + subpacket_length) {\n\n            var len = packet.readSimpleLength(bytes.substr(i));\n            i += len.offset;\n\n            // Since it is trivial to add data to the unhashed portion of \n            // the packet we simply ignore all unauthenticated data.\n            if (signed)\n              this.read_sub_packet(bytes.substr(i, len.len));\n\n            i += len.len;\n          }\n\n          return i;\n        }\n\n        i += subpackets.call(this, bytes.substr(i), true);\n\n        // A V4 signature hashes the packet body\n        // starting from its first field, the version number, through the end\n        // of the hashed subpacket data.  Thus, the fields hashed are the\n        // signature version, the signature type, the public-key algorithm, the\n        // hash algorithm, the hashed subpacket length, and the hashed\n        // subpacket body.\n        this.signatureData = bytes.substr(0, i);\n\n        i += subpackets.call(this, bytes.substr(i), false);\n\n        break;\n      default:\n        throw new Error('Version ' + version + ' of the signature is unsupported.');\n        break;\n    }\n\n    // Two-octet field holding left 16 bits of signed hash value.\n    this.signedHashValue = bytes.substr(i, 2);\n    i += 2;\n\n    this.signature = bytes.substr(i);\n  };\n\n  this.write = function() {\n    return this.signatureData +\n      util.writeNumber(0, 2) + // Number of unsigned subpackets.\n    this.signedHashValue +\n      this.signature;\n  };\n\n  /**\n   * Signs provided data. This needs to be done prior to serialization.\n   * @param {Object} data Contains packets to be signed.\n   * @param {openpgp_msg_privatekey} privatekey private key used to sign the message. \n   */\n  this.sign = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var result = String.fromCharCode(4);\n    result += String.fromCharCode(signatureType);\n    result += String.fromCharCode(publicKeyAlgorithm);\n    result += String.fromCharCode(hashAlgorithm);\n\n    //Calculate subpackets\n    var creationTimeSubpacket = write_sub_packet(enums.signatureSubpacket.signature_creation_time,\n      util.writeDate(new Date()));\n\n    var issuerSubpacket = write_sub_packet(enums.signatureSubpacket.issuer, key.getKeyId().write());\n\n    // Add subpackets here\n    result += util.writeNumber(creationTimeSubpacket.length + issuerSubpacket.length, 2);\n    result += creationTimeSubpacket + issuerSubpacket;\n\n    this.signatureData = result;\n\n    var trailer = this.calculateTrailer();\n\n    var toHash = this.toSign(signatureType, data) +\n      this.signatureData + trailer;\n\n    var hash = crypto.hash.digest(hashAlgorithm, toHash);\n\n    this.signedHashValue = hash.substr(0, 2);\n\n    this.signature = crypto.signature.sign(hashAlgorithm,\n      publicKeyAlgorithm, key.mpi, toHash);\n  };\n\n  /**\n   * creates a string representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   * @param {Integer} type subpacket signature type. Signature types as described \n   * in RFC4880 Section 5.2.3.2\n   * @param {String} data data to be included\n   * @return {String} a string-representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   */\n  function write_sub_packet(type, data) {\n    var result = \"\";\n    result += packet.writeSimpleLength(data.length + 1);\n    result += String.fromCharCode(type);\n    result += data;\n    return result;\n  }\n\n  // V4 signature sub packets\n\n  this.read_sub_packet = function(bytes) {\n    var mypos = 0;\n\n    function read_array(prop, bytes) {\n      this[prop] = [];\n\n      for (var i = 0; i < bytes.length; i++) {\n        this[prop].push(bytes[i].charCodeAt());\n      }\n    }\n\n    // The leftwost bit denotes a \"critical\" packet, but we ignore it.\n    var type = bytes[mypos++].charCodeAt() & 0x7F;\n\n    // subpacket type\n    switch (type) {\n      case 2:\n        // Signature Creation Time\n        this.created = util.readDate(bytes.substr(mypos));\n        break;\n      case 3:\n        // Signature Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.signatureNeverExpires = time.getTime() == 0;\n        this.signatureExpirationTime = time;\n\n        break;\n      case 4:\n        // Exportable Certification\n        this.exportable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 5:\n        // Trust Signature\n        this.trustLevel = bytes[mypos++].charCodeAt();\n        this.trustAmount = bytes[mypos++].charCodeAt();\n        break;\n      case 6:\n        // Regular Expression\n        this.regularExpression = bytes.substr(mypos);\n        break;\n      case 7:\n        // Revocable\n        this.revocable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 9:\n        // Key Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.keyExpirationTime = time;\n        this.keyNeverExpires = time.getTime() == 0;\n\n        break;\n      case 11:\n        // Preferred Symmetric Algorithms\n        this.preferredSymmetricAlgorithms = [];\n\n        while (mypos != bytes.length) {\n          this.preferredSymmetricAlgorithms.push(bytes[mypos++].charCodeAt());\n        }\n\n        break;\n      case 12:\n        // Revocation Key\n        // (1 octet of class, 1 octet of public-key algorithm ID, 20\n        // octets of\n        // fingerprint)\n        this.revocationKeyClass = bytes[mypos++].charCodeAt();\n        this.revocationKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.revocationKeyFingerprint = bytes.substr(mypos, 20);\n        break;\n\n      case 16:\n        // Issuer\n        this.issuerKeyId = bytes.substr(mypos, 8);\n        break;\n\n      case 20:\n        // Notation Data\n        // We don't know how to handle anything but a text flagged data.\n        if (bytes[mypos].charCodeAt() == 0x80) {\n\n          // We extract key/value tuple from the byte stream.\n          mypos += 4;\n          var m = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n          var n = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n\n          var name = bytes.substr(mypos, m),\n            value = bytes.substr(mypos + m, n);\n\n          this.notation[name] = value;\n        } else throw new Error(\"Unsupported notation flag.\");\n        break;\n      case 21:\n        // Preferred Hash Algorithms\n        read_array.call(this, 'preferredHashAlgorithms', bytes.substr(mypos));\n        break;\n      case 22:\n        // Preferred Compression Algorithms\n        read_array.call(this, 'preferredCompressionAlgorithms ', bytes.substr(mypos));\n        break;\n      case 23:\n        // Key Server Preferences\n        read_array.call(this, 'keyServerPreferencess', bytes.substr(mypos));\n        break;\n      case 24:\n        // Preferred Key Server\n        this.preferredKeyServer = bytes.substr(mypos);\n        break;\n      case 25:\n        // Primary User ID\n        this.isPrimaryUserID = bytes[mypos++] != 0;\n        break;\n      case 26:\n        // Policy URI\n        this.policyURI = bytes.substr(mypos);\n        break;\n      case 27:\n        // Key Flags\n        read_array.call(this, 'keyFlags', bytes.substr(mypos));\n        break;\n      case 28:\n        // Signer's User ID\n        this.signersUserId += bytes.substr(mypos);\n        break;\n      case 29:\n        // Reason for Revocation\n        this.reasonForRevocationFlag = bytes[mypos++].charCodeAt();\n        this.reasonForRevocationString = bytes.substr(mypos);\n        break;\n      case 30:\n        // Features\n        read_array.call(this, 'features', bytes.substr(mypos));\n        break;\n      case 31:\n        // Signature Target\n        // (1 octet public-key algorithm, 1 octet hash algorithm, N octets hash)\n        this.signatureTargetPublicKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.signatureTargetHashAlgorithm = bytes[mypos++].charCodeAt();\n\n        var len = crypto.getHashByteLength(this.signatureTargetHashAlgorithm);\n\n        this.signatureTargetHash = bytes.substr(mypos, len);\n        break;\n      case 32:\n        // Embedded Signature\n        this.embeddedSignature = new packet_signature();\n        this.embeddedSignature.read(bytes.substr(mypos));\n        break;\n      default:\n        util.print_error(\"openpgp.packet.signature.js\\n\" +\n          'unknown signature subpacket type ' + type + \" @:\" + mypos +\n          \" subplen:\" + subplen + \" len:\" + len);\n        break;\n    }\n  };\n\n  // Produces data to produce signature on\n  this.toSign = function(type, data) {\n    var t = enums.signature;\n\n    switch (type) {\n      case t.binary:\n        return data.getBytes();\n\n      case t.text:\n        return this.toSign(t.binary, data)\n          .replace(/\\r\\n/g, '\\n')\n          .replace(/\\n/g, '\\r\\n');\n\n      case t.standalone:\n        return '';\n\n      case t.cert_generic:\n      case t.cert_persona:\n      case t.cert_casual:\n      case t.cert_positive:\n      case t.cert_revocation:\n        {\n          var packet, tag;\n\n          if (data.userid !== undefined) {\n            tag = 0xB4;\n            packet = data.userid;\n          } else if (data.userattribute !== undefined) {\n            tag = 0xD1;\n            packet = data.userattribute;\n          } else throw new Error('Either a userid or userattribute packet needs to be ' +\n              'supplied for certification.');\n\n          var bytes = packet.write();\n\n          return this.toSign(t.key, data) +\n            String.fromCharCode(tag) +\n            util.writeNumber(bytes.length, 4) +\n            bytes;\n        }\n      case t.subkey_binding:\n      case t.key_binding:\n        {\n          return this.toSign(t.key, data) + this.toSign(t.key, {\n            key: data.bind\n          });\n        }\n      case t.key:\n        {\n          if (data.key == undefined)\n            throw new Error('Key packet is required for this sigtature.');\n\n          return data.key.writeOld();\n        }\n      case t.key_revocation:\n      case t.subkey_revocation:\n        return this.toSign(t.key, data);\n      case t.timestamp:\n        return '';\n      case t.thrid_party:\n        throw new Error('Not implemented');\n        break;\n      default:\n        throw new Error('Unknown signature type.')\n    }\n  }\n\n\n  this.calculateTrailer = function() {\n    // calculating the trailer\n    var trailer = '';\n    trailer += String.fromCharCode(4); // Version\n    trailer += String.fromCharCode(0xFF);\n    trailer += util.writeNumber(this.signatureData.length, 4);\n    return trailer\n  }\n\n\n  /**\n   * verifys the signature packet. Note: not signature types are implemented\n   * @param {String} data data which on the signature applies\n   * @param {openpgp_msg_privatekey} key the public key to verify the signature\n   * @return {boolean} True if message is verified, else false.\n   */\n  this.verify = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var bytes = this.toSign(signatureType, data),\n      trailer = this.calculateTrailer();\n\n\n    var mpicount = 0;\n    // Algorithm-Specific Fields for RSA signatures:\n    // \t    - multiprecision number (MPI) of RSA signature value m**d mod n.\n    if (publicKeyAlgorithm > 0 && publicKeyAlgorithm < 4)\n      mpicount = 1;\n    //    Algorithm-Specific Fields for DSA signatures:\n    //      - MPI of DSA value r.\n    //      - MPI of DSA value s.\n    else if (publicKeyAlgorithm == 17)\n      mpicount = 2;\n\n    var mpi = [],\n      i = 0;\n    for (var j = 0; j < mpicount; j++) {\n      mpi[j] = new type_mpi();\n      i += mpi[j].read(this.signature.substr(i));\n    }\n\n    this.verified = crypto.signature.verify(publicKeyAlgorithm,\n      hashAlgorithm, mpi, key.mpi,\n      bytes + this.signatureData + trailer);\n\n    return this.verified;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Sym. Encrypted Integrity Protected Data \n * Packet (Tag 18)\n * \n * RFC4880 5.13: The Symmetrically Encrypted Integrity Protected Data packet is\n * a variant of the Symmetrically Encrypted Data packet. It is a new feature\n * created for OpenPGP that addresses the problem of detecting a modification to\n * encrypted data. It is used in combination with a Modification Detection Code\n * packet.\n */\n\nmodule.exports = function packet_sym_encrypted_integrity_protected() {\n  /** The encrypted payload. */\n  this.encrypted = null; // string\n  /** @type {Boolean}\n   * If after decrypting the packet this is set to true,\n   * a modification has been detected and thus the contents\n   * should be discarded.\n   */\n  this.modification = false;\n  this.packets;\n\n\n  this.read = function(bytes) {\n    // - A one-octet version number. The only currently defined value is\n    // 1.\n    var version = bytes[0].charCodeAt();\n\n    if (version != 1) {\n      throw new Error('Invalid packet version.');\n    }\n\n    // - Encrypted data, the output of the selected symmetric-key cipher\n    //   operating in Cipher Feedback mode with shift amount equal to the\n    //   block size of the cipher (CFB-n where n is the block size).\n    this.encrypted = bytes.substr(1);\n  }\n\n  this.write = function() {\n\n    return String.fromCharCode(1) // Version\n    + this.encrypted;\n  }\n\n  this.encrypt = function(sessionKeyAlgorithm, key) {\n    var bytes = this.packets.write()\n\n    var prefixrandom = crypto.getPrefixRandom(sessionKeyAlgorithm);\n    var prefix = prefixrandom + prefixrandom.charAt(prefixrandom.length - 2) + prefixrandom.charAt(prefixrandom.length -\n      1)\n\n    var tohash = bytes;\n\n\n    // Modification detection code packet.\n    tohash += String.fromCharCode(0xD3);\n    tohash += String.fromCharCode(0x14);\n\n\n    tohash += crypto.hash.sha1(prefix + tohash);\n\n\n    this.encrypted = crypto.cfb.encrypt(prefixrandom,\n      sessionKeyAlgorithm, tohash, key, false).substring(0,\n      prefix.length + tohash.length);\n  }\n\n  /**\n   * Decrypts the encrypted data contained in this object read_packet must\n   * have been called before\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *            The selected symmetric encryption algorithm to be used\n   * @param {String} key The key of cipher blocksize length to be used\n   * @return {String} The decrypted data of this packet\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, false);\n\n\n    // there must be a modification detection code packet as the\n    // last packet and everything gets hashed except the hash itself\n    this.hash = crypto.hash.sha1(\n      crypto.cfb.mdc(sessionKeyAlgorithm, key, this.encrypted) + decrypted.substring(0, decrypted.length - 20));\n\n\n    var mdc = decrypted.substr(decrypted.length - 20, 20);\n\n    if (this.hash != mdc) {\n      throw new Error('Modification detected.');\n    } else\n      this.packets.read(decrypted.substr(0, decrypted.length - 22));\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_s2k = require('../type/s2k.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_sym_encrypted_session_key() {\n  this.tag = 3;\n  this.sessionKeyEncryptionAlgorithm = null;\n  this.sessionKeyAlgorithm = 'aes256';\n  this.encrypted = null;\n  this.s2k = new type_s2k();\n\n  /**\n   * Parsing function for a symmetric encrypted session key packet (tag 3).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    // A one-octet version number. The only currently defined version is 4.\n    this.version = bytes[0].charCodeAt();\n\n    // A one-octet number describing the symmetric algorithm used.\n    var algo = enums.read(enums.symmetric, bytes[1].charCodeAt());\n\n    // A string-to-key (S2K) specifier, length as defined above.\n    var s2klength = this.s2k.read(bytes.substr(2));\n\n    // Optionally, the encrypted session key itself, which is decrypted\n    // with the string-to-key object.\n    var done = s2klength + 2;\n\n    if (done < bytes.length) {\n      this.encrypted = bytes.substr(done);\n      this.sessionKeyEncryptionAlgorithm = algo\n    } else\n      this.sessionKeyAlgorithm = algo;\n  }\n\n  this.write = function() {\n    var algo = this.encrypted == null ?\n      this.sessionKeyAlgorithm :\n      this.sessionKeyEncryptionAlgorithm;\n\n    var bytes = String.fromCharCode(this.version) +\n      String.fromCharCode(enums.write(enums.symmetric, algo)) +\n      this.s2k.write();\n\n    if (this.encrypted != null)\n      bytes += this.encrypted;\n    return bytes;\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(passphrase) {\n    var algo = this.sessionKeyEncryptionAlgorithm != null ?\n      this.sessionKeyEncryptionAlgorithm :\n      this.sessionKeyAlgorithm;\n\n\n    var length = crypto.cipher[algo].keySize;\n    var key = this.s2k.produce_key(passphrase, length);\n\n    if (this.encrypted == null) {\n      this.sessionKey = key;\n\n    } else {\n      var decrypted = crypto.cfb.decrypt(\n        this.sessionKeyEncryptionAlgorithm, key, this.encrypted, true);\n\n      this.sessionKeyAlgorithm = enums.read(enums.symmetric,\n        decrypted[0].keyCodeAt());\n\n      this.sessionKey = decrypted.substr(1);\n    }\n  }\n\n  this.encrypt = function(passphrase) {\n    var length = crypto.getKeyLength(this.sessionKeyEncryptionAlgorithm);\n    var key = this.s2k.produce_key(passphrase, length);\n\n    var private_key = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm)) +\n\n    crypto.getRandomBytes(\n      crypto.getKeyLength(this.sessionKeyAlgorithm));\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(this.sessionKeyEncryptionAlgorithm),\n      this.sessionKeyEncryptionAlgorithm, key, private_key, true);\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Symmetrically Encrypted Data Packet (Tag 9)\n * \n * RFC4880 5.7: The Symmetrically Encrypted Data packet contains data encrypted\n * with a symmetric-key algorithm. When it has been decrypted, it contains other\n * packets (usually a literal data packet or compressed data packet, but in\n * theory other Symmetrically Encrypted Data packets or sequences of packets\n * that form whole OpenPGP messages).\n */\n\nmodule.exports = function packet_symmetrically_encrypted() {\n  this.encrypted = null;\n  /** Decrypted packets contained within. \n   * @type {openpgp_packetlist} */\n  this.packets;\n\n  this.read = function(bytes) {\n    this.encrypted = bytes;\n  }\n\n  this.write = function() {\n    return this.encrypted;\n  }\n\n  /**\n   * Symmetrically decrypt the packet data\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *             Symmetric key algorithm to use // See RFC4880 9.2\n   * @param {String} key\n   *             Key as string with the corresponding length to the\n   *            algorithm\n   * @return The decrypted data;\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, true);\n\n    this.packets.read(decrypted);\n  }\n\n  this.encrypt = function(algo, key) {\n    var data = this.packets.write();\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(algo), algo, data, key, true);\n  }\n};\n","module.exports = function packet_trust() {\n\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/** \n * @class\n * @classdesc Implementation of the User Attribute Packet (Tag 17)\n *  The User Attribute packet is a variation of the User ID packet.  It\n *  is capable of storing more types of data than the User ID packet,\n *  which is limited to text.  Like the User ID packet, a User Attribute\n *  packet may be certified by the key owner (\"self-signed\") or any other\n *  key owner who cares to certify it.  Except as noted, a User Attribute\n *  packet may be used anywhere that a User ID packet may be used.\n *\n *  While User Attribute packets are not a required part of the OpenPGP\n *  standard, implementations SHOULD provide at least enough\n *  compatibility to properly handle a certification signature on the\n *  User Attribute packet.  A simple way to do this is by treating the\n *  User Attribute packet as a User ID packet with opaque contents, but\n *  an implementation may use any method desired.\n */\nmodule.exports = function packet_user_attribute() {\n  this.tag = 17;\n  this.attributes = [];\n\n  /**\n   * parsing function for a user attribute packet (tag 17).\n   * @param {String} input payload of a tag 17 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    while (i < bytes.length) {\n      var len = openpgp_packet.read_simple_length(bytes);\n\n      i += len.offset;\n      this.attributes.push(bytes.substr(i, len.len));\n      i += len.len;\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of the User ID Packet (Tag 13)\n * A User ID packet consists of UTF-8 text that is intended to represent\n * the name and email address of the key holder.  By convention, it\n * includes an RFC 2822 [RFC2822] mail name-addr, but there are no\n * restrictions on its content.  The packet length in the header\n * specifies the length of the User ID. \n */\nmodule.exports = function packet_userid() {\n  /** @type {String} A string containing the user id. Usually in the form\n   * John Doe <john@example.com> \n   */\n  this.userid = '';\n\n\n  /**\n   * Parsing function for a user id packet (tag 13).\n   * @param {String} input payload of a tag 13 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input \n   * at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    this.userid = util.decode_utf8(bytes);\n  }\n\n  /**\n   * Creates a string representation of the user id packet\n   * @param {String} user_id the user id as string (\"John Doe <john.doe@mail.us\")\n   * @return {String} string representation\n   */\n  this.write = function() {\n    return util.encode_utf8(this.userid);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of type key id (RFC4880 3.3)\n *  A Key ID is an eight-octet scalar that identifies a key.\n   Implementations SHOULD NOT assume that Key IDs are unique.  The\n   section \"Enhanced Key Formats\" below describes how Key IDs are\n   formed.\n */\nmodule.exports = function keyid() {\n\n  this.bytes = '';\n\n  for (var i = 0; i < 8; i++) {\n    this.bytes += String.fromCharCode(0);\n  }\n  /**\n   * Parsing method for a key id\n   * @param {String} input Input to read the key id from \n   * @param {integer} position Position where to start reading the key \n   * id from input\n   * @return {openpgp_type_keyid} This object\n   */\n  this.read = function(bytes) {\n    this.bytes = bytes.substr(0, 8);\n  }\n\n  this.write = function() {\n    return this.bytes;\n  }\n\n  this.toHex = function() {\n    return util.hexstrdump(this.bytes);\n  }\n\n  this.equals = function(keyid) {\n    return this.bytes == keyid.bytes;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n// Hint: We hold our MPIs as an array of octets in big endian format preceeding a two\n// octet scalar: MPI: [a,b,c,d,e,f]\n// - MPI size: (a << 8) | b \n// - MPI = c | d << 8 | e << ((MPI.length -2)*8) | f ((MPI.length -2)*8)\n\nvar BigInteger = require('../crypto/public_key/jsbn.js'),\n  util = require('../util');\n\n/**\n * @class\n * @classdescImplementation of type MPI (RFC4880 3.2)\n * Multiprecision integers (also called MPIs) are unsigned integers used\n * to hold large integers such as the ones used in cryptographic\n * calculations.\n * An MPI consists of two pieces: a two-octet scalar that is the length\n * of the MPI in bits followed by a string of octets that contain the\n * actual integer.\n */\nmodule.exports = function mpi() {\n  /** An implementation dependent integer */\n  this.data = null;\n\n  /**\n   * Parsing function for a mpi (RFC 4880 3.2).\n   * @param {String} input Payload of mpi data\n   * @param {Integer} position Position to start reading from the input \n   * string\n   * @param {Integer} len Length of the packet or the remaining length of \n   * input at position\n   * @return {openpgp_type_mpi} Object representation\n   */\n  this.read = function(bytes) {\n    var bits = (bytes[0].charCodeAt() << 8) | bytes[1].charCodeAt();\n\n    // Additional rules:\n    //\n    //    The size of an MPI is ((MPI.length + 7) / 8) + 2 octets.\n    //\n    //    The length field of an MPI describes the length starting from its\n    //\t  most significant non-zero bit.  Thus, the MPI [00 02 01] is not\n    //    formed correctly.  It should be [00 01 01].\n\n    // TODO: Verification of this size method! This size calculation as\n    // \t\t specified above is not applicable in JavaScript\n    var bytelen = Math.ceil(bits / 8);\n\n    var raw = bytes.substr(2, bytelen);\n    this.fromBytes(raw);\n\n    return 2 + bytelen;\n  }\n\n  this.fromBytes = function(bytes) {\n    this.data = new BigInteger(util.hexstrdump(bytes), 16);\n  }\n\n  this.toBytes = function() {\n    return this.write().substr(2);\n  }\n\n  this.byteLength = function() {\n    return this.toBytes().length;\n  }\n\n  /**\n   * Converts the mpi object to a string as specified in RFC4880 3.2\n   * @return {String} mpi Byte representation\n   */\n  this.write = function() {\n    return this.data.toMPI();\n  }\n\n  this.toBigInteger = function() {\n    return this.data.clone();\n  }\n\n  this.fromBigInteger = function(bn) {\n    this.data = bn.clone();\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the String-to-key specifier (RFC4880 3.7)\n * String-to-key (S2K) specifiers are used to convert passphrase strings\n   into symmetric-key encryption/decryption keys.  They are used in two\n   places, currently: to encrypt the secret part of private keys in the\n   private keyring, and to convert passphrases to encryption keys for\n   symmetrically encrypted messages.\n */\nmodule.exports = function s2k() {\n  /** @type {openpgp.hash} */\n  this.algorithm = 'sha256';\n  /** @type {openpgp_type_s2k.type} */\n  this.type = 'iterated';\n  this.c = 96;\n  /** @type {openpgp_bytearray} \n   * Eight bytes of salt. */\n  this.salt = crypto.random.getRandomBytes(8);\n\n\n  // Exponen bias, defined in RFC4880\n  var expbias = 6;\n\n  this.get_count = function() {\n    return (16 + (this.c & 15)) << ((this.c >> 4) + expbias);\n  }\n\n  /**\n   * Parsing function for a string-to-key specifier (RFC 4880 3.7).\n   * @param {String} input Payload of string-to-key specifier\n   * @return {Integer} Actual length of the object\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    this.type = enums.read(enums.s2k, bytes[i++].charCodeAt());\n    this.algorithm = enums.read(enums.hash, bytes[i++].charCodeAt());\n\n    switch (this.type) {\n      case 'simple':\n        break;\n\n      case 'salted':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n        break;\n\n      case 'iterated':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n\n        // Octet 10: count, a one-octet, coded value\n        this.c = bytes[i++].charCodeAt();\n        break;\n\n      case 'gnu':\n        if (bytes.substr(i, 3) == \"GNU\") {\n          i += 3; // GNU\n          var gnuExtType = 1000 + bytes[i++].charCodeAt();\n          if (gnuExtType == 1001) {\n            this.type = gnuExtType;\n            // GnuPG extension mode 1001 -- don't write secret key at all\n          } else {\n            throw new Error(\"Unknown s2k gnu protection mode.\");\n          }\n        } else {\n          throw new Error(\"Unknown s2k type.\");\n        }\n        break;\n\n      default:\n        throw new Error(\"Unknown s2k type.\");\n        break;\n    }\n\n    return i;\n  }\n\n\n  /**\n   * writes an s2k hash based on the inputs.\n   * @return {String} Produced key of hashAlgorithm hash length\n   */\n  this.write = function() {\n    var bytes = String.fromCharCode(enums.write(enums.s2k, this.type));\n    bytes += String.fromCharCode(enums.write(enums.hash, this.algorithm));\n\n    switch (this.type) {\n      case 'simple':\n        break;\n      case 'salted':\n        bytes += this.salt;\n        break;\n      case 'iterated':\n        bytes += this.salt;\n        bytes += String.fromCharCode(this.c);\n        break;\n    };\n\n    return bytes;\n  }\n\n  /**\n   * Produces a key using the specified passphrase and the defined \n   * hashAlgorithm \n   * @param {String} passphrase Passphrase containing user input\n   * @return {String} Produced key with a length corresponding to \n   * hashAlgorithm hash length\n   */\n  this.produce_key = function(passphrase, numBytes) {\n    passphrase = util.encode_utf8(passphrase);\n\n    function round(prefix, s2k) {\n      var algorithm = enums.write(enums.hash, s2k.algorithm);\n\n      switch (s2k.type) {\n        case 'simple':\n          return crypto.hash.digest(algorithm, prefix + passphrase);\n\n        case 'salted':\n          return crypto.hash.digest(algorithm,\n            prefix + s2k.salt + passphrase);\n\n        case 'iterated':\n          var isp = [],\n            count = s2k.get_count();\n          data = s2k.salt + passphrase;\n\n          while (isp.length * data.length < count)\n            isp.push(data);\n\n          isp = isp.join('');\n\n          if (isp.length > count)\n            isp = isp.substr(0, count);\n\n          return crypto.hash.digest(algorithm, prefix + isp);\n      };\n    }\n\n    var result = '',\n      prefix = '';\n\n    while (result.length <= numBytes) {\n      result += round(prefix, this);\n      prefix += String.fromCharCode(0);\n    }\n\n    return result.substr(0, numBytes);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar Util = function() {\n\n  this.readNumber = function(bytes) {\n    var n = 0;\n\n    for (var i = 0; i < bytes.length; i++) {\n      n <<= 8;\n      n += bytes[i].charCodeAt();\n    }\n\n    return n;\n  };\n\n  this.writeNumber = function(n, bytes) {\n    var b = '';\n    for (var i = 0; i < bytes; i++) {\n      b += String.fromCharCode((n >> (8 * (bytes - i - 1))) & 0xFF);\n    }\n\n    return b;\n  };\n\n\n\n  this.readDate = function(bytes) {\n    var n = this.readNumber(bytes);\n    var d = new Date();\n    d.setTime(n * 1000);\n    return d;\n  };\n\n  this.writeDate = function(time) {\n    var numeric = Math.round(time.getTime() / 1000);\n\n    return this.writeNumber(numeric, 4);\n  };\n\n  this.emailRegEx =\n    /[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?/;\n\n  this.debug = false;\n\n  this.hexdump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    var i = 0;\n    while (c < e) {\n      h = str.charCodeAt(c++).toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\" \" + h);\n      i++;\n      if (i % 32 == 0)\n        r.push(\"\\n           \");\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create hexstring from a binary\n   * @param {String} str String to convert\n   * @return {String} String containing the hexadecimal values\n   */\n  this.hexstrdump = function(str) {\n    if (str == null)\n      return \"\";\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].charCodeAt().toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create binary string from a hex encoded string\n   * @param {String} str Hex string to convert\n   * @return {String} String containing the binary values\n   */\n  this.hex2bin = function(hex) {\n    var str = '';\n    for (var i = 0; i < hex.length; i += 2)\n      str += String.fromCharCode(parseInt(hex.substr(i, 2), 16));\n    return str;\n  };\n\n  /**\n   * Creating a hex string from an binary array of integers (0..255)\n   * @param {String} str Array of bytes to convert\n   * @return {String} Hexadecimal representation of the array\n   */\n  this.hexidump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n\n  /**\n   * Convert a native javascript string to a string of utf8 bytes\n   * @param {String} str The string to convert\n   * @return {String} A valid squence of utf8 bytes\n   */\n  this.encode_utf8 = function(str) {\n    return unescape(encodeURIComponent(str));\n  };\n\n  /**\n   * Convert a string of utf8 bytes to a native javascript string\n   * @param {String} utf8 A valid squence of utf8 bytes\n   * @return {String} A native javascript string\n   */\n  this.decode_utf8 = function(utf8) {\n    try {\n      return decodeURIComponent(escape(utf8));\n    } catch (e) {\n      return utf8;\n    }\n  };\n\n  var str2bin = function(str, result) {\n    for (var i = 0; i < str.length; i++) {\n      result[i] = str.charCodeAt(i);\n    }\n\n    return result;\n  };\n\n  var bin2str = function(bin) {\n    var result = [];\n\n    for (var i = 0; i < bin.length; i++) {\n      result.push(String.fromCharCode(bin[i]));\n    }\n\n    return result.join('');\n  };\n\n  /**\n   * Convert a string to an array of integers(0.255)\n   * @param {String} str String to convert\n   * @return {Integer[]} An array of (binary) integers\n   */\n  this.str2bin = function(str) {\n    return str2bin(str, new Array(str.length));\n  };\n\n\n  /**\n   * Convert an array of integers(0.255) to a string \n   * @param {Integer[]} bin An array of (binary) integers to convert\n   * @return {String} The string representation of the array\n   */\n  this.bin2str = bin2str;\n\n  /**\n   * Convert a string to a Uint8Array\n   * @param {String} str String to convert\n   * @return {Uint8Array} The array of (binary) integers\n   */\n  this.str2Uint8Array = function(str) {\n    return str2bin(str, new Uint8Array(new ArrayBuffer(str.length)));\n  };\n\n  /**\n   * Convert a Uint8Array to a string. This currently functions \n   * the same as bin2str. \n   * @param {Uint8Array} bin An array of (binary) integers to convert\n   * @return {String} String representation of the array\n   */\n  this.Uint8Array2str = bin2str;\n\n  /**\n   * Calculates a 16bit sum of a string by adding each character \n   * codes modulus 65535\n   * @param {String} text String to create a sum of\n   * @return {Integer} An integer containing the sum of all character \n   * codes % 65535\n   */\n  this.calc_checksum = function(text) {\n    var checksum = {\n      s: 0,\n      add: function(sadd) {\n        this.s = (this.s + sadd) % 65536;\n      }\n    };\n    for (var i = 0; i < text.length; i++) {\n      checksum.add(text.charCodeAt(i));\n    }\n    return checksum.s;\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug = function(str) {\n    if (this.debug) {\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * Different than print_debug because will call hexstrdump iff necessary.\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug_hexstr_dump = function(str, strToHex) {\n    if (this.debug) {\n      str = str + this.hexstrdump(strToHex);\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print an error message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the error message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded error message\n   */\n  this.print_error = function(str) {\n    if (this.debug)\n      throw str;\n    console.log(str);\n  };\n\n  /**\n   * Helper function to print an info message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'.\n   * @param {String} str String of the info message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded info message\n   */\n  this.print_info = function(str) {\n    if (this.debug)\n      console.log(str);\n  };\n\n  this.print_warning = function(str) {\n    console.log(str);\n  };\n\n  this.getLeftNBits = function(string, bitcount) {\n    var rest = bitcount % 8;\n    if (rest == 0)\n      return string.substring(0, bitcount / 8);\n    var bytes = (bitcount - rest) / 8 + 1;\n    var result = string.substring(0, bytes);\n    return this.shiftRight(result, 8 - rest); // +String.fromCharCode(string.charCodeAt(bytes -1) << (8-rest) & 0xFF);\n  };\n\n  /**\n   * Shifting a string to n bits right\n   * @param {String} value The string to shift\n   * @param {Integer} bitcount Amount of bits to shift (MUST be smaller \n   * than 9)\n   * @return {String} Resulting string. \n   */\n  this.shiftRight = function(value, bitcount) {\n    var temp = util.str2bin(value);\n    if (bitcount % 8 != 0) {\n      for (var i = temp.length - 1; i >= 0; i--) {\n        temp[i] >>= bitcount % 8;\n        if (i > 0)\n          temp[i] |= (temp[i - 1] << (8 - (bitcount % 8))) & 0xFF;\n      }\n    } else {\n      return value;\n    }\n    return util.bin2str(temp);\n  };\n\n  /**\n   * Return the algorithm type as string\n   * @return {String} String representing the message type\n   */\n  this.get_hashAlgorithmString = function(algo) {\n    switch (algo) {\n      case 1:\n        return \"MD5\";\n      case 2:\n        return \"SHA1\";\n      case 3:\n        return \"RIPEMD160\";\n      case 8:\n        return \"SHA256\";\n      case 9:\n        return \"SHA384\";\n      case 10:\n        return \"SHA512\";\n      case 11:\n        return \"SHA224\";\n    }\n    return \"unknown\";\n  };\n\n};\n\n/**\n * an instance that should be used. \n */\nmodule.exports = new Util();\n"]} ; \ No newline at end of file diff --git a/src/index.js b/src/index.js index bf51fb1d..2febb3ca 100644 --- a/src/index.js +++ b/src/index.js @@ -1,6 +1,7 @@ module.exports = require('./openpgp.js'); module.exports.key = require('./key.js'); +module.exports.keyring = require('./keyring.js'); module.exports.message = require('./message.js'); module.exports.util = require('./util'); module.exports.packet = require('./packet'); diff --git a/src/key.js b/src/key.js index 5d9d70d8..bdbb073f 100644 --- a/src/key.js +++ b/src/key.js @@ -18,6 +18,7 @@ var packet = require('./packet'); var enums = require('./enums.js'); var armor = require('./encoding/armor.js'); +var config = require('./config'); /** * @class @@ -32,9 +33,11 @@ var armor = require('./encoding/armor.js'); this.packets = packetlist || new packet.list(); - /** Returns the primary key (secret or public) - * @returns {packet_secret_key|packet_public_key|null} */ - this.getKey = function() { + /** + * Returns the primary key packet (secret or public) + * @returns {packet_secret_key|packet_public_key|null} + */ + this.getKeyPacket = function() { for (var i = 0; i < this.packets.length; i++) { if (this.packets[i].tag == enums.packet.public_key || this.packets[i].tag == enums.packet.secret_key) { @@ -42,11 +45,13 @@ var armor = require('./encoding/armor.js'); } } return null; - }; + } - /** Returns all the private and public subkeys - * @returns {[public_subkey|secret_subkey]} */ - this.getSubkeys = function() { + /** + * Returns all the private and public subkey packets + * @returns {[public_subkey|secret_subkey]} + */ + this.getSubkeyPackets = function() { var subkeys = []; @@ -58,31 +63,81 @@ var armor = require('./encoding/armor.js'); } return subkeys; - }; - - this.getAllKeys = function() { - return [this.getKey()].concat(this.getSubkeys()); - }; - - this.getKeyids = function() { - var keyids = []; - var keys = this.getAllKeys(); - for (var i = 0; i < keys.length; i++) { - keyids.push(keys[i].getKeyId()); - } - return keyids; - }; - - this.getKeyById = function(keyid) { - var keys = this.getAllKeys(); - for (var i = 0; i < keys.length; i++) { - if (keys[i].getKeyId().equals(keyid)) { - return keys[i]; - } - } } - this.getSigningKey = function() { + /** + * Returns all the private and public key and subkey packets + * @returns {[public_subkey|secret_subkey|packet_secret_key|packet_public_key]} + */ + this.getAllKeyPackets = function() { + return [this.getKeyPacket()].concat(this.getSubkeyPackets()); + } + + /** + * Returns key IDs of all key packets + * @returns {[keyid]} + */ + this.getKeyIds = function() { + var keyIds = []; + var keys = this.getAllKeyPackets(); + for (var i = 0; i < keys.length; i++) { + keyIds.push(keys[i].getKeyId()); + } + return keyIds; + } + + /** + * Returns key IDs of all key packets in hex + * @returns {[String]} + */ + this.getKeyIdsHex = function() { + return this.getKeyIds().map(function(keyId) { + return keyId.toHex(); + }); + } + + /** + * Returns first key packet which match to an array of key IDs + * @param {[keyid]} keyIds + * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} + */ + this.getKeyPacketByIds = function(keyIds) { + var keys = this.getAllKeyPackets(); + for (var i = 0; i < keys.length; i++) { + var keyId = keys[i].getKeyId(); + for (var j = 0; j < keyIds.length; j++) { + if (keyId.equals(keyIds[j])) { + //TODO return only verified keys + return keys[i]; + } + } + } + return null; + } + + /** + * Returns true if this is a public key + * @return {Boolean} + */ + this.isPublic = function() { + var publicKeyPackets = this.packets.filterByTag(enums.packet.public_key); + return publicKeyPackets.length !== 0 ? true : false; + } + + /** + * Returns true if this is a private key + * @return {Boolean} + */ + this.isPrivate = function() { + var privateKeyPackets = this.packets.filterByTag(enums.packet.private_key); + return privateKeyPackets.length !== 0 ? true : false; + } + + /** + * Returns first key packet that is available for signing + * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} + */ + this.getSigningKeyPacket = function() { var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa]; @@ -90,7 +145,7 @@ var armor = require('./encoding/armor.js'); return enums.read(enums.publicKey, s); }); - var keys = this.getAllKeys(); + var keys = this.getAllKeyPackets(); for (var i = 0; i < keys.length; i++) { if (signing.indexOf(keys[i].algorithm) !== -1) { @@ -99,38 +154,34 @@ var armor = require('./encoding/armor.js'); } return null; - }; - - function getPreferredSignatureHashAlgorithm() { - var pkey = this.getSigningKey(); - if (pkey === null) { - util.print_error("private key is for encryption only! Cannot create a signature."); - return null; - } - if (pkey.publicKey.publicKeyAlgorithm == 17) { - var dsa = new DSA(); - return dsa.select_hash_algorithm(pkey.publicKey.MPIs[1].toBigInteger()); // q - } - //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8 - //separate private key preference from digest preferences - return openpgp.config.config.prefer_hash_algorithm; } /** - * Finds an encryption key for this key - * @returns null if no encryption key has been found + * Returns preferred signature hash algorithm of this key + * @return {String} */ - this.getEncryptionKey = function() { + function getPreferredSignatureHashAlgorithm() { + //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8 + //separate private key preference from digest preferences + return config.prefer_hash_algorithm; + } + + /** + * Returns the first valid encryption key packet for this key + * @returns {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} key packet or null if no encryption key has been found + */ + this.getEncryptionKeyPacket = function() { // V4: by convention subkeys are prefered for encryption service // V3: keys MUST NOT have subkeys var isValidEncryptionKey = function(key) { + //TODO evaluate key flags: http://tools.ietf.org/html/rfc4880#section-5.2.3.21 return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey, enums.publicKey.rsa_sign); //TODO verify key //&& keys.verifyKey() }; - var subkeys = this.getSubkeys(); + var subkeys = this.getSubkeyPackets(); for (var j = 0; j < subkeys.length; j++) { if (isValidEncryptionKey(subkeys[j])) { @@ -138,31 +189,40 @@ var armor = require('./encoding/armor.js'); } } // if no valid subkey for encryption, use primary key - var primaryKey = this.getKey(); + var primaryKey = this.getKeyPacket(); if (isValidEncryptionKey(primaryKey)) { return primaryKey; } return null; - }; + } + /** + * Decrypts all secret key and subkey packets + * @param {String} passphrase + * @return {undefined} + */ this.decrypt = function(passphrase) { - var keys = this.getAllKeys(); + //TODO boolean return value + var keys = this.getAllKeyPackets(); for (var i in keys) if (keys[i].tag == enums.packet.secret_subkey || keys[i].tag == enums.packet.secret_key) { keys[i].decrypt(passphrase); } - }; + } - // TODO need to implement this + // TODO + this.verify = function() { - function revoke() { + } + // TODO + this.revoke = function() { } -}; +} /** * reads an OpenPGP armored text and returns a key object @@ -171,6 +231,7 @@ var armor = require('./encoding/armor.js'); */ key.readArmored = function(armoredText) { //TODO how do we want to handle bad text? Exception throwing + //TODO don't accept non-key armored texts var input = armor.decode(armoredText).openpgp; var packetlist = new packet.list(); packetlist.read(input); diff --git a/src/openpgp.keyring.js b/src/keyring.js similarity index 100% rename from src/openpgp.keyring.js rename to src/keyring.js diff --git a/src/message.js b/src/message.js index 873660b8..09174c8e 100644 --- a/src/message.js +++ b/src/message.js @@ -30,67 +30,60 @@ function message(packetlist) { /** * Returns the key IDs of the public keys to which the session key is encrypted - * @return {[keyId]} keyId provided as string of hex number (lowercase) + * @return {[keyId]} array of keyid objects */ this.getKeyIds = function() { var keyIds = []; var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key); pkESKeyPacketlist.forEach(function(packet) { - keyIds.push(packet.publicKeyId.toHex()); + keyIds.push(packet.publicKeyId); }); return keyIds; } - function generic_decrypt(packets, passphrase) { - var sessionkey; - - for (var i = 0; i < packets.length; i++) { - if (packets[i].tag == openpgp_packet.tags.public_key_encrypted_session_key) { - var key = openpgp.keyring.getKeyById(packets[i].public_key_id); - - } - } - + /** + * Returns the key IDs in hex of the public keys to which the session key is encrypted + * @return {[String]} keyId provided as string of hex numbers (lowercase) + */ + this.getKeyIdsHex = function() { + return this.getKeyIds().map(function(keyId) { + return keyId.toHex(); + }); } /** - * Decrypts a message and generates user interface message out of the found. - * MDC will be verified as well as message signatures - * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key) - * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message - * @return {} plaintext of the message or null on error + * Decrypts the message + * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with (corresponding to the session key) + * @return {[String]} array with plaintext of decrypted messages */ - this.decrypt = function(key) { + this.decrypt = function(privateKeyPacket) { var decryptedMessages = []; - var keyids = key.getKeyids(); var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key); - outer: for (var i = 0; i < pkESKeyPacketlist.length; i++) { + for (var i = 0; i < pkESKeyPacketlist.length; i++) { var pkESKeyPacket = pkESKeyPacketlist[i]; - for (var j = 0; j < keyids.length; j++) { - if (pkESKeyPacket.publicKeyId.equals(keyids[j])) { - pkESKeyPacket.decrypt(key.getKeyById(pkESKeyPacket.publicKeyId)); - var symEncryptedPacketlist = this.packets.filter(function(packet) { - return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected; - }); - for (var k = 0; k < symEncryptedPacketlist.length; k++) { - var symEncryptedPacket = symEncryptedPacketlist[k]; - symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey); - for (var l = 0; l < symEncryptedPacket.packets.length; l++) { - var dataPacket = symEncryptedPacket.packets[l]; - switch (dataPacket.tag) { - case enums.packet.literal: - decryptedMessages.push(dataPacket.getBytes()); - break; - case enums.packet.compressed: - //TODO - break; - default: - //TODO - } + if (pkESKeyPacket.publicKeyId.equals(privateKeyPacket.getKeyId())) { + pkESKeyPacket.decrypt(privateKeyPacket); + var symEncryptedPacketlist = this.packets.filter(function(packet) { + return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected; + }); + for (var k = 0; k < symEncryptedPacketlist.length; k++) { + var symEncryptedPacket = symEncryptedPacketlist[k]; + symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey); + for (var l = 0; l < symEncryptedPacket.packets.length; l++) { + var dataPacket = symEncryptedPacket.packets[l]; + switch (dataPacket.tag) { + case enums.packet.literal: + decryptedMessages.push(dataPacket.getBytes()); + break; + case enums.packet.compressed: + //TODO + break; + default: + //TODO } } - break outer; } + break; } } return decryptedMessages; @@ -181,6 +174,7 @@ function message(packetlist) { */ message.readArmored = function(armoredText) { //TODO how do we want to handle bad text? Exception throwing + //TODO don't accept non-message armored texts var input = armor.decode(armoredText).openpgp; var packetlist = new packet.list(); packetlist.read(input); diff --git a/src/openpgp.js b/src/openpgp.js index c142b5f2..d2ac1069 100644 --- a/src/openpgp.js +++ b/src/openpgp.js @@ -52,15 +52,15 @@ function _openpgp() { var sessionKey = crypto.generateSessionKey(enums.read(enums.symmetric, config.encryption_cipher)); keys.forEach(function(key) { - var encryptionKey = key.getEncryptionKey(); - if (encryptionKey) { + var encryptionKeyPacket = key.getEncryptionKeyPacket(); + if (encryptionKeyPacket) { var pkESKeyPacket = new packet.public_key_encrypted_session_key(); - pkESKeyPacket.publicKeyId = encryptionKey.getKeyId(); - pkESKeyPacket.publicKeyAlgorithm = encryptionKey.algorithm; + pkESKeyPacket.publicKeyId = encryptionKeyPacket.getKeyId(); + pkESKeyPacket.publicKeyAlgorithm = encryptionKeyPacket.algorithm; pkESKeyPacket.sessionKey = sessionKey; //TODO get preferred algo from signature pkESKeyPacket.sessionKeyAlgorithm = enums.read(enums.symmetric, config.encryption_cipher); - pkESKeyPacket.encrypt(encryptionKey); + pkESKeyPacket.encrypt(encryptionKeyPacket); messagePacketlist.push(pkESKeyPacket); } }); @@ -91,12 +91,12 @@ function _openpgp() { /** * decrypts message - * @param {[key]} decrypted privateKey + * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with * @param {message} message the message object with the encrypted data * @return {String} decrypted message as as native JavaScript string */ - function decryptMessage(privateKey, message) { - return message.decrypt(privateKey); + function decryptMessage(privateKeyPacket, message) { + return message.decrypt(privateKeyPacket); } function decryptAndVerifyMessage(privateKey, publicKeys, messagePacketlist) { diff --git a/src/util/util.js b/src/util/util.js index e05916f4..10044cf4 100644 --- a/src/util/util.js +++ b/src/util/util.js @@ -140,7 +140,11 @@ var Util = function() { * @return {String} A native javascript string */ this.decode_utf8 = function(utf8) { - return decodeURIComponent(escape(utf8)); + try { + return decodeURIComponent(escape(utf8)); + } catch (e) { + return utf8; + } }; var str2bin = function(str, result) { diff --git a/test/crypto/cipher/aes.js b/test/crypto/cipher/aes.js index 9f527eb2..d2fff50e 100644 --- a/test/crypto/cipher/aes.js +++ b/test/crypto/cipher/aes.js @@ -7,7 +7,7 @@ unit.register("AES Rijndael cipher test with test vectors from ecb_tbl.txt", fun var result = new Array(); function test_aes(input, key, output) { - var aes = new openpgp.cipher.aes128(util.bin2str(key)); + var aes = new openpgp.crypto.cipher.aes128(util.bin2str(key)); var result = util.bin2str(aes.encrypt(input)); diff --git a/test/crypto/cipher/blowfish.js b/test/crypto/cipher/blowfish.js index eb44f020..9a64e80d 100644 --- a/test/crypto/cipher/blowfish.js +++ b/test/crypto/cipher/blowfish.js @@ -3,11 +3,11 @@ var unit = require('../../unit.js'); unit.register("Blowfish cipher test with test vectors from http://www.schneier.com/code/vectors.txt", function() { var openpgp = require('../../../'), util = openpgp.util, - BFencrypt = openpgp.cipher.blowfish; + BFencrypt = openpgp.crypto.cipher.blowfish; var result = []; function test_bf(input, key, output) { - var blowfish = new openpgp.cipher.blowfish(util.bin2str(key)); + var blowfish = new openpgp.crypto.cipher.blowfish(util.bin2str(key)); var result = util.bin2str(blowfish.encrypt(input)); return (util.hexstrdump(result) == util.hexstrdump(util.bin2str(output))); diff --git a/test/crypto/cipher/cast5.js b/test/crypto/cipher/cast5.js index ce98519a..2e060675 100644 --- a/test/crypto/cipher/cast5.js +++ b/test/crypto/cipher/cast5.js @@ -6,7 +6,7 @@ unit.register("CAST-128 cipher test with test vectors from RFC2144", function() var result = []; function test_cast(input, key, output) { - var cast5 = new openpgp.cipher.cast5(util.bin2str(key)); + var cast5 = new openpgp.crypto.cipher.cast5(util.bin2str(key)); var result = util.bin2str(cast5.encrypt(input)); return util.hexstrdump(result) == util.hexstrdump(util.bin2str(output)); diff --git a/test/crypto/cipher/des.js b/test/crypto/cipher/des.js index 973deb42..68521678 100644 --- a/test/crypto/cipher/des.js +++ b/test/crypto/cipher/des.js @@ -1,33 +1,33 @@ -var unit = require('../../unit.js'); - -unit.register("TripleDES (EDE) cipher test with test vectors from http://csrc.nist.gov/publications/nistpubs/800-20/800-20.pdf", function() { - var openpgp = require('../../../'), - util = openpgp.util; - - var result = []; - var key = util.bin2str([1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1]); - var testvectors = [[[0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x95,0xF8,0xA5,0xE5,0xDD,0x31,0xD9,0x00]], - [[0x40,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xDD,0x7F,0x12,0x1C,0xA5,0x01,0x56,0x19]], - [[0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x2E,0x86,0x53,0x10,0x4F,0x38,0x34,0xEA]], - [[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4B,0xD3,0x88,0xFF,0x6C,0xD8,0x1D,0x4F]], - [[0x08,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x20,0xB9,0xE7,0x67,0xB2,0xFB,0x14,0x56]], - [[0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x55,0x57,0x93,0x80,0xD7,0x71,0x38,0xEF]], - [[0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x6C,0xC5,0xDE,0xFA,0xAF,0x04,0x51,0x2F]], - [[0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x0D,0x9F,0x27,0x9B,0xA5,0xD8,0x72,0x60]], - [[0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x00],[0xD9,0x03,0x1B,0x02,0x71,0xBD,0x5A,0x0A]], - [[0x00,0x40,0x00,0x00,0x00,0x00,0x00,0x00],[0x42,0x42,0x50,0xB3,0x7C,0x3D,0xD9,0x51]], - [[0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00],[0xB8,0x06,0x1B,0x7E,0xCD,0x9A,0x21,0xE5]], - [[0x00,0x10,0x00,0x00,0x00,0x00,0x00,0x00],[0xF1,0x5D,0x0F,0x28,0x6B,0x65,0xBD,0x28]], - [[0x00,0x08,0x00,0x00,0x00,0x00,0x00,0x00],[0xAD,0xD0,0xCC,0x8D,0x6E,0x5D,0xEB,0xA1]], - [[0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00],[0xE6,0xD5,0xF8,0x27,0x52,0xAD,0x63,0xD1]], - [[0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00],[0xEC,0xBF,0xE3,0xBD,0x3F,0x59,0x1A,0x5E]], - [[0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x00],[0xF3,0x56,0x83,0x43,0x79,0xD1,0x65,0xCD]], - [[0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00],[0x2B,0x9F,0x98,0x2F,0x20,0x03,0x7F,0xA9]], - [[0x00,0x00,0x40,0x00,0x00,0x00,0x00,0x00],[0x88,0x9D,0xE0,0x68,0xA1,0x6F,0x0B,0xE6]], - [[0x00,0x00,0x20,0x00,0x00,0x00,0x00,0x00],[0xE1,0x9E,0x27,0x5D,0x84,0x6A,0x12,0x98]], - [[0x00,0x00,0x10,0x00,0x00,0x00,0x00,0x00],[0x32,0x9A,0x8E,0xD5,0x23,0xD7,0x1A,0xEC]], - [[0x00,0x00,0x08,0x00,0x00,0x00,0x00,0x00],[0xE7,0xFC,0xE2,0x25,0x57,0xD2,0x3C,0x97]], - [[0x00,0x00,0x04,0x00,0x00,0x00,0x00,0x00],[0x12,0xA9,0xF5,0x81,0x7F,0xF2,0xD6,0x5D]], +var unit = require('../../unit.js'); + +unit.register("TripleDES (EDE) cipher test with test vectors from http://csrc.nist.gov/publications/nistpubs/800-20/800-20.pdf", function() { + var openpgp = require('../../../'), + util = openpgp.util; + + var result = []; + var key = util.bin2str([1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1]); + var testvectors = [[[0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x95,0xF8,0xA5,0xE5,0xDD,0x31,0xD9,0x00]], + [[0x40,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xDD,0x7F,0x12,0x1C,0xA5,0x01,0x56,0x19]], + [[0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x2E,0x86,0x53,0x10,0x4F,0x38,0x34,0xEA]], + [[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4B,0xD3,0x88,0xFF,0x6C,0xD8,0x1D,0x4F]], + [[0x08,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x20,0xB9,0xE7,0x67,0xB2,0xFB,0x14,0x56]], + [[0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x55,0x57,0x93,0x80,0xD7,0x71,0x38,0xEF]], + [[0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x6C,0xC5,0xDE,0xFA,0xAF,0x04,0x51,0x2F]], + [[0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x0D,0x9F,0x27,0x9B,0xA5,0xD8,0x72,0x60]], + [[0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x00],[0xD9,0x03,0x1B,0x02,0x71,0xBD,0x5A,0x0A]], + [[0x00,0x40,0x00,0x00,0x00,0x00,0x00,0x00],[0x42,0x42,0x50,0xB3,0x7C,0x3D,0xD9,0x51]], + [[0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00],[0xB8,0x06,0x1B,0x7E,0xCD,0x9A,0x21,0xE5]], + [[0x00,0x10,0x00,0x00,0x00,0x00,0x00,0x00],[0xF1,0x5D,0x0F,0x28,0x6B,0x65,0xBD,0x28]], + [[0x00,0x08,0x00,0x00,0x00,0x00,0x00,0x00],[0xAD,0xD0,0xCC,0x8D,0x6E,0x5D,0xEB,0xA1]], + [[0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00],[0xE6,0xD5,0xF8,0x27,0x52,0xAD,0x63,0xD1]], + [[0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00],[0xEC,0xBF,0xE3,0xBD,0x3F,0x59,0x1A,0x5E]], + [[0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x00],[0xF3,0x56,0x83,0x43,0x79,0xD1,0x65,0xCD]], + [[0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00],[0x2B,0x9F,0x98,0x2F,0x20,0x03,0x7F,0xA9]], + [[0x00,0x00,0x40,0x00,0x00,0x00,0x00,0x00],[0x88,0x9D,0xE0,0x68,0xA1,0x6F,0x0B,0xE6]], + [[0x00,0x00,0x20,0x00,0x00,0x00,0x00,0x00],[0xE1,0x9E,0x27,0x5D,0x84,0x6A,0x12,0x98]], + [[0x00,0x00,0x10,0x00,0x00,0x00,0x00,0x00],[0x32,0x9A,0x8E,0xD5,0x23,0xD7,0x1A,0xEC]], + [[0x00,0x00,0x08,0x00,0x00,0x00,0x00,0x00],[0xE7,0xFC,0xE2,0x25,0x57,0xD2,0x3C,0x97]], + [[0x00,0x00,0x04,0x00,0x00,0x00,0x00,0x00],[0x12,0xA9,0xF5,0x81,0x7F,0xF2,0xD6,0x5D]], [[0x00,0x00,0x02,0x00,0x00,0x00,0x00,0x00],[0xA4,0x84,0xC3,0xAD,0x38,0xDC,0x9C,0x19]], [[0x00,0x00,0x01,0x00,0x00,0x00,0x00,0x00],[0xFB,0xE0,0x0A,0x8A,0x1E,0xF8,0xAD,0x72]], [[0x00,0x00,0x00,0x80,0x00,0x00,0x00,0x00],[0x75,0x0D,0x07,0x94,0x07,0x52,0x13,0x63]], @@ -70,38 +70,38 @@ unit.register("TripleDES (EDE) cipher test with test vectors from http://csrc.ni [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x04],[0xD2,0xFD,0x88,0x67,0xD5,0x0D,0x2D,0xFE]], [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x02],[0x06,0xE7,0xEA,0x22,0xCE,0x92,0x70,0x8F]], [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01],[0x16,0x6B,0x40,0xB4,0x4A,0xBA,0x4B,0xD6]]]; - - var res = true; - var j = 0; - for (var i = 0; i < testvectors.length; i++) { - var des = new openpgp.cipher.des(key); - - var encr = util.bin2str(des.encrypt(testvectors[i][0], key)); - var res2 = encr == util.bin2str(testvectors[i][1]); - - res &= res2; - - if (!res2) { - result[j] = new unit.result("Testing vector with block " + - util.hexidump(testvectors[i][0]) + - " and key " + util.hexstrdump(key) + - " should be " + util.hexidump(testvectors[i][1]) + " != " + - util.hexidump(encr), - false); - j++; - } - } - if (res) { - result[j] = new unit.result("All 3DES EDE test vectors completed", true); - } - return result; -}); - - + + var res = true; + var j = 0; + for (var i = 0; i < testvectors.length; i++) { + var des = new openpgp.crypto.cipher.des(key); + + var encr = util.bin2str(des.encrypt(testvectors[i][0], key)); + var res2 = encr == util.bin2str(testvectors[i][1]); + + res &= res2; + + if (!res2) { + result[j] = new unit.result("Testing vector with block " + + util.hexidump(testvectors[i][0]) + + " and key " + util.hexstrdump(key) + + " should be " + util.hexidump(testvectors[i][1]) + " != " + + util.hexidump(encr), + false); + j++; + } + } + if (res) { + result[j] = new unit.result("All 3DES EDE test vectors completed", true); + } + return result; +}); + + unit.register("DES encrypt/decrypt padding tests", function () { - var openpgp = require('../../../'), - util = openpgp.util; - + var openpgp = require('../../../'), + util = openpgp.util; + var result = []; var key = util.bin2str([0x01, 0x23, 0x45, 0x67, 0x89, 0xAB, 0xCD, 0xEF]); var testvectors = new Array(); @@ -130,7 +130,7 @@ unit.register("DES encrypt/decrypt padding tests", function () { [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0xCA, 0x59, 0x61, 0x3A, 0x83, 0x23, 0x26, 0xDD]], [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9]]]; - var des = new openpgp.cipher.originalDes(key); + var des = new openpgp.crypto.cipher.originalDes(key); var res = true; var j = 0; diff --git a/test/crypto/cipher/twofish.js b/test/crypto/cipher/twofish.js index 7f412aa5..033464e5 100644 --- a/test/crypto/cipher/twofish.js +++ b/test/crypto/cipher/twofish.js @@ -5,7 +5,7 @@ unit.register("Twofish test with test vectors from http://www.schneier.com/code/ util = openpgp.util; function TFencrypt(block, key) { - var tf = new openpgp.cipher.twofish(key); + var tf = new openpgp.crypto.cipher.twofish(key); return tf.encrypt(block); } diff --git a/test/crypto/hash/md5.js b/test/crypto/hash/md5.js index a03d8e31..e662dc6c 100644 --- a/test/crypto/hash/md5.js +++ b/test/crypto/hash/md5.js @@ -3,7 +3,7 @@ var unit = require('../../unit.js'); unit.register("MD5 test with test vectors from RFC 1321", function() { var openpgp = require('../../../'), util = openpgp.util, - MD5 = openpgp.hash.md5; + MD5 = openpgp.crypto.hash.md5; var result = new Array(); result[0] = new unit.result("MD5 (\"\") = d41d8cd98f00b204e9800998ecf8427e", diff --git a/test/crypto/hash/ripemd.js b/test/crypto/hash/ripemd.js index 555328e1..db87b1f9 100644 --- a/test/crypto/hash/ripemd.js +++ b/test/crypto/hash/ripemd.js @@ -4,7 +4,7 @@ unit.register("RIPE-MD 160 bits test with test vectors from http://homes.esat.ku var openpgp = require('../../../'), util = openpgp.util, - RMDstring = openpgp.hash.ripemd; + RMDstring = openpgp.crypto.hash.ripemd; var result = new Array(); result[0] = new unit.result("RMDstring (\"\") = 9c1185a5c5e9fc54612808977ee8f548b2258d31", diff --git a/test/crypto/hash/sha.js b/test/crypto/hash/sha.js index d6ce149d..3e0a7f57 100644 --- a/test/crypto/hash/sha.js +++ b/test/crypto/hash/sha.js @@ -2,9 +2,9 @@ var unit = require('../../unit.js'); unit.register("SHA* test with test vectors from NIST FIPS 180-2", function() { - var openpgp = require('../../../src'), + var openpgp = require('../../../'), util = openpgp.util, - hash = openpgp.hash; + hash = openpgp.crypto.hash; var result = new Array(); diff --git a/test/crypto/openpgp.crypto.js b/test/crypto/openpgp.crypto.js index a12e11f6..bd3fb3cc 100644 --- a/test/crypto/openpgp.crypto.js +++ b/test/crypto/openpgp.crypto.js @@ -225,14 +225,14 @@ unit.register("Functional testing of openpgp.crypto.* methods", function() { //Originally we passed public and secret MPI separately, now they are joined. Is this what we want to do long term? // RSA - var RSAsignedData = openpgp.signature.sign(2, 1, RSApubMPIs.concat(RSAsecMPIs), "foobar"); + var RSAsignedData = openpgp.crypto.signature.sign(2, 1, RSApubMPIs.concat(RSAsecMPIs), "foobar"); var RSAsignedDataMPI = new openpgp.mpi(); RSAsignedDataMPI.read(RSAsignedData); result[0] = new unit.result("Testing RSA Sign and Verify", - openpgp.signature.verify(1, 2, [RSAsignedDataMPI], RSApubMPIs, "foobar")); + openpgp.crypto.signature.verify(1, 2, [RSAsignedDataMPI], RSApubMPIs, "foobar")); // DSA - var DSAsignedData = openpgp.signature.sign(2, 17, DSApubMPIs.concat(DSAsecMPIs), "foobar"); + var DSAsignedData = openpgp.crypto.signature.sign(2, 17, DSApubMPIs.concat(DSAsecMPIs), "foobar"); var DSAmsgMPIs = []; DSAmsgMPIs[0] = new openpgp.mpi(); @@ -240,31 +240,31 @@ unit.register("Functional testing of openpgp.crypto.* methods", function() { DSAmsgMPIs[0].read(DSAsignedData.substring(0,34)); DSAmsgMPIs[1].read(DSAsignedData.substring(34,68)); result[1] = new unit.result("Testing DSA Sign and Verify", - openpgp.signature.verify(17, 2, DSAmsgMPIs, DSApubMPIs, "foobar")); + openpgp.crypto.signature.verify(17, 2, DSAmsgMPIs, DSApubMPIs, "foobar")); var symmAlgo = "aes256"; // AES256 - var symmKey = openpgp.generateSessionKey(symmAlgo); - var symmencDataOCFB = openpgp.cfb.encrypt(openpgp.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, true); - var symmencDataCFB = openpgp.cfb.encrypt(openpgp.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, false); + var symmKey = openpgp.crypto.generateSessionKey(symmAlgo); + var symmencDataOCFB = openpgp.crypto.cfb.encrypt(openpgp.crypto.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, true); + var symmencDataCFB = openpgp.crypto.cfb.encrypt(openpgp.crypto.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, false); result[2] = new unit.result("Testing symmetric encrypt and decrypt with OpenPGP CFB resync", - openpgp.cfb.decrypt(symmAlgo,symmKey,symmencDataOCFB,true) == "foobarfoobar1234567890"); + openpgp.crypto.cfb.decrypt(symmAlgo,symmKey,symmencDataOCFB,true) == "foobarfoobar1234567890"); result[3] = new unit.result("Testing symmetric encrypt and decrypt without OpenPGP CFB resync (used in modification detection code \"MDC\" packets)", - openpgp.cfb.decrypt(symmAlgo,symmKey,symmencDataCFB,false) == "foobarfoobar1234567890"); + openpgp.crypto.cfb.decrypt(symmAlgo,symmKey,symmencDataCFB,false) == "foobarfoobar1234567890"); var RSAUnencryptedData = new openpgp.mpi(); - RSAUnencryptedData.fromBytes(openpgp.pkcs1.eme.encode(symmKey, RSApubMPIs[0].byteLength())); - var RSAEncryptedData = openpgp.publicKeyEncrypt("rsa_encrypt_sign", RSApubMPIs, RSAUnencryptedData); + RSAUnencryptedData.fromBytes(openpgp.crypto.pkcs1.eme.encode(symmKey, RSApubMPIs[0].byteLength())); + var RSAEncryptedData = openpgp.crypto.publicKeyEncrypt("rsa_encrypt_sign", RSApubMPIs, RSAUnencryptedData); result[4] = new unit.result("Testing asymmetric encrypt and decrypt using RSA with eme_pkcs1 padding", - openpgp.pkcs1.eme.decode(openpgp.publicKeyDecrypt("rsa_encrypt_sign", RSApubMPIs.concat(RSAsecMPIs), RSAEncryptedData).write().substring(2), RSApubMPIs[0].byteLength()) == symmKey); + openpgp.crypto.pkcs1.eme.decode(openpgp.crypto.publicKeyDecrypt("rsa_encrypt_sign", RSApubMPIs.concat(RSAsecMPIs), RSAEncryptedData).write().substring(2), RSApubMPIs[0].byteLength()) == symmKey); var ElgamalUnencryptedData = new openpgp.mpi(); - ElgamalUnencryptedData.fromBytes(openpgp.pkcs1.eme.encode(symmKey, ElgamalpubMPIs[0].byteLength())); - var ElgamalEncryptedData = openpgp.publicKeyEncrypt("elgamal", ElgamalpubMPIs, ElgamalUnencryptedData); + ElgamalUnencryptedData.fromBytes(openpgp.crypto.pkcs1.eme.encode(symmKey, ElgamalpubMPIs[0].byteLength())); + var ElgamalEncryptedData = openpgp.crypto.publicKeyEncrypt("elgamal", ElgamalpubMPIs, ElgamalUnencryptedData); result[5] = new unit.result("Testing asymmetric encrypt and decrypt using Elgamal with eme_pkcs1 padding", - openpgp.pkcs1.eme.decode(openpgp.publicKeyDecrypt("elgamal", ElgamalpubMPIs.concat(ElgamalsecMPIs), ElgamalEncryptedData).write().substring(2), ElgamalpubMPIs[0].byteLength()) == symmKey); + openpgp.crypto.pkcs1.eme.decode(openpgp.crypto.publicKeyDecrypt("elgamal", ElgamalpubMPIs.concat(ElgamalsecMPIs), ElgamalEncryptedData).write().substring(2), ElgamalpubMPIs[0].byteLength()) == symmKey); return result; }); diff --git a/test/crypto/openpgp.sigcheck.js b/test/crypto/openpgp.sigcheck.js index c175a365..3dc57ba1 100644 --- a/test/crypto/openpgp.sigcheck.js +++ b/test/crypto/openpgp.sigcheck.js @@ -2,9 +2,9 @@ var unit = require('../unit.js'); unit.register("Testing of binary signature checking", function() { var openpgp = require('../../'); - var keyring = require('../../src/openpgp.keyring.js'); + var keyring = openpgp.keyring; var result = []; - var priv_key = openpgp.readArmoredPackets([ + var priv_key = openpgp.key.readArmored([ '-----BEGIN PGP PRIVATE KEY BLOCK-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -32,8 +32,8 @@ unit.register("Testing of binary signature checking", function() { 'AKC8omYPPomN1E/UJFfXdLDIMi5LoA==', '=LSrW', '-----END PGP PRIVATE KEY BLOCK-----' - ].join("\n")); - var pub_key = openpgp.readArmoredPackets( + ].join("\n")).packets; + var pub_key = openpgp.key.readArmored( [ '-----BEGIN PGP PUBLIC KEY BLOCK-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -51,8 +51,8 @@ unit.register("Testing of binary signature checking", function() { 'EplqEakMckCtikEnpxYe', '=b2Ln', '-----END PGP PUBLIC KEY BLOCK-----' - ].join("\n")); - var msg = openpgp.readArmoredPackets([ + ].join("\n")).packets; + var msg = openpgp.message.readArmored([ '-----BEGIN PGP MESSAGE-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -68,7 +68,7 @@ unit.register("Testing of binary signature checking", function() { 'aOU=', '=4iGt', '-----END PGP MESSAGE-----' - ].join("\n")); + ].join("\n")).packets; //TODO need both? priv_key[0].decrypt("abcd"); priv_key[3].decrypt("abcd"); @@ -80,7 +80,7 @@ unit.register("Testing of binary signature checking", function() { // exercises the GnuPG s2k type 1001 extension: // the secrets on the primary key have been stripped. - var priv_key_gnupg_ext = openpgp.readArmoredPackets([ + var priv_key_gnupg_ext = openpgp.key.readArmored([ '-----BEGIN PGP PRIVATE KEY BLOCK-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -107,7 +107,7 @@ unit.register("Testing of binary signature checking", function() { 'iY3UT9QkV9d0sMgyLkug', '=GQsY', '-----END PGP PRIVATE KEY BLOCK-----', - ].join("\n")); + ].join("\n")).packets; priv_key_gnupg_ext[3].decrypt("abcd"); msg[0].decrypt(priv_key_gnupg_ext[3]); msg[1].decrypt(msg[0].sessionKeyAlgorithm, msg[0].sessionKey); @@ -149,7 +149,7 @@ unit.register("Testing of binary signature checking", function() { '-----END PGP PUBLIC KEY BLOCK-----' ].join("\n")); - var msg2 = openpgp.readArmoredPackets([ + var msg2 = openpgp.message.readArmored([ '-----BEGIN PGP MESSAGE-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -158,7 +158,7 @@ unit.register("Testing of binary signature checking", function() { 'AJ9zh4zsK4GIPuEu81YPNmHsju7DYg==', '=WaSx', '-----END PGP MESSAGE-----' - ].join("\n")); + ].join("\n")).packets; var packetlists = keyring.getPacketlistsForKeyId(msg2[0].signingKeyId.write()); var pubKey = packetlists[0]; msg2[2].verify(pubKey[3], msg2[1]); diff --git a/test/general/openpgp.basic.js b/test/general/openpgp.basic.js index d50c8d7a..28ee4396 100644 --- a/test/general/openpgp.basic.js +++ b/test/general/openpgp.basic.js @@ -2,7 +2,7 @@ var unit = require('../unit.js'); unit.register("Key generation/encryption/decryption", function() { var openpgp = require('../../'); - var keyring = require('../../src/openpgp.keyring.js'); + var keyring = openpgp.keyring; var result = []; var testHelper = function(passphrase, userid, message) { var key = openpgp.generateKeyPair(openpgp.enums.publicKey.rsa_encrypt_sign, 512, @@ -12,46 +12,31 @@ unit.register("Key generation/encryption/decryption", function() { + 'userid: ' + userid + '\n' + 'message: ' + message; - var keyPacketlist = openpgp.readArmoredPackets(key); debugger; - var privKey = openpgp.getKeyFromPacketlist(keyPacketlist, passphrase); + var privKey = openpgp.key.readArmored(key); - var encrypted = openpgp.encryptMessage(keyPacketlist, message); + var encrypted = openpgp.encryptMessage([privKey], message); - openpgp.keyring.importPublicKey(key.publicKeyArmored); - - var msg = openpgp.read_message(encrypted); - var keymat = null; - var sesskey = null; + var msg = openpgp.message.readArmored(encrypted); // Find the private (sub)key for the session key of the message - for (var i = 0; i< msg[0].sessionKeys.length; i++) { - if (priv_key.privateKeyPacket.publicKey.getKeyId().write() == msg[0].sessionKeys[i].keyId.bytes) { - keymat = { key: priv_key, keymaterial: priv_key.privateKeyPacket}; - sesskey = msg[0].sessionKeys[i]; - break; - } - for (var j = 0; j < priv_key.subKeys.length; j++) { - if (priv_key.subKeys[j].publicKey.getKeyId().write() == msg[0].sessionKeys[i].keyId.bytes) { - keymat = { key: priv_key, keymaterial: priv_key.subKeys[j]}; - sesskey = msg[0].sessionKeys[i]; - break; - } - } + var privKeyPacket = privKey.getKeyPacketByIds(msg.getKeyIds()); + + if (!privKeyPacket) { + return new unit.result("No private key found!" + info, false); } - var decrypted = ''; - if (keymat !== null) { - if (!keymat.keymaterial.decryptSecretMPIs(passphrase)) { - return new test_result("Password for secrect key was incorrect!", - + info, false); + try { + if (!privKeyPacket.decrypt(passphrase)) { + return new unit.result("Password for secrect key was incorrect!" + info, false); } - - decrypted = msg[0].decrypt(keymat, sesskey); - } else { - return new test_result("No private key found!" + info, false); + } catch (e) { + return new unit.result("Exception on decrypt of private key packet!" + info, false); } - return new test_result(message + ' == ' + decrypted + info, message == decrypted); + + var decrypted = msg.decrypt(privKeyPacket)[0]; + + return new unit.result(message + ' == ' + decrypted + info, message == decrypted); }; result.push(testHelper('password', 'Test McTestington ', 'hello world')); @@ -61,7 +46,9 @@ unit.register("Key generation/encryption/decryption", function() { }); unit.register("Encryption/decryption", function() { - var openpgp = require('../../src'); + var openpgp = require('../../'); + + var result = []; var pub_key = ['-----BEGIN PGP PUBLIC KEY BLOCK-----', @@ -155,10 +142,14 @@ unit.register("Encryption/decryption", function() { var privKey = openpgp.key.readArmored(priv_key); - privKey.decrypt('hello world'); + var privKeyPacket = privKey.getKeyPacketByIds(message.getKeyIds()); - var decrypted = openpgp.decryptMessage(privKey, message); + privKeyPacket.decrypt('hello world'); - return new test_result('Encrypt plain text and afterwards decrypt leads to same result', plaintext == decrypted); + var decrypted = openpgp.decryptMessage(privKeyPacket, message); + + result[0] = new unit.result('Encrypt plain text and afterwards decrypt leads to same result', plaintext == decrypted); + + return result; }); diff --git a/test/test-bundle.js b/test/test-bundle.js index 9133b9dd..347eb174 100644 --- a/test/test-bundle.js +++ b/test/test-bundle.js @@ -9080,6 +9080,7 @@ module.exports = enums; module.exports = require('./openpgp.js'); module.exports.key = require('./key.js'); +module.exports.keyring = require('./keyring.js'); module.exports.message = require('./message.js'); module.exports.util = require('./util'); module.exports.packet = require('./packet'); @@ -9091,7 +9092,251 @@ module.exports.enums = require('./enums.js'); module.exports.config = require('./config'); module.exports.crypto = require('./crypto'); -},{"./config":3,"./crypto":16,"./encoding/armor.js":25,"./enums.js":27,"./key.js":29,"./message.js":30,"./openpgp.js":31,"./packet":35,"./type/keyid.js":53,"./type/mpi.js":54,"./type/s2k.js":55,"./util":56}],29:[function(require,module,exports){ +},{"./config":3,"./crypto":16,"./encoding/armor.js":25,"./enums.js":27,"./key.js":29,"./keyring.js":30,"./message.js":31,"./openpgp.js":32,"./packet":35,"./type/keyid.js":53,"./type/mpi.js":54,"./type/s2k.js":55,"./util":56}],29:[function(require,module,exports){ +// GPG4Browsers - An OpenPGP implementation in javascript +// Copyright (C) 2011 Recurity Labs GmbH +// +// This library is free software; you can redistribute it and/or +// modify it under the terms of the GNU Lesser General Public +// License as published by the Free Software Foundation; either +// version 2.1 of the License, or (at your option) any later version. +// +// This library is distributed in the hope that it will be useful, +// but WITHOUT ANY WARRANTY; without even the implied warranty of +// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU +// Lesser General Public License for more details. +// +// You should have received a copy of the GNU Lesser General Public +// License along with this library; if not, write to the Free Software +// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA + +var packet = require('./packet'); +var enums = require('./enums.js'); +var armor = require('./encoding/armor.js'); +var config = require('./config'); + +/** + * @class + * @classdesc Class that represents an OpenPGP key. Must contain a master key. + * @param {packetlist} packetlist [description] + * Can contain additional subkeys, signatures, + * user ids, user attributes. + */ + + function key(packetlist) { + + this.packets = packetlist || new packet.list(); + + + /** + * Returns the primary key packet (secret or public) + * @returns {packet_secret_key|packet_public_key|null} + */ + this.getKeyPacket = function() { + for (var i = 0; i < this.packets.length; i++) { + if (this.packets[i].tag == enums.packet.public_key || + this.packets[i].tag == enums.packet.secret_key) { + return this.packets[i]; + } + } + return null; + } + + /** + * Returns all the private and public subkey packets + * @returns {[public_subkey|secret_subkey]} + */ + this.getSubkeyPackets = function() { + + var subkeys = []; + + for (var i = 0; i < this.packets.length; i++) { + if (this.packets[i].tag == enums.packet.public_subkey || + this.packets[i].tag == enums.packet.secret_subkey) { + subkeys.push(this.packets[i]); + } + } + + return subkeys; + } + + /** + * Returns all the private and public key and subkey packets + * @returns {[public_subkey|secret_subkey|packet_secret_key|packet_public_key]} + */ + this.getAllKeyPackets = function() { + return [this.getKeyPacket()].concat(this.getSubkeyPackets()); + } + + /** + * Returns key IDs of all key packets + * @returns {[keyid]} + */ + this.getKeyIds = function() { + var keyIds = []; + var keys = this.getAllKeyPackets(); + for (var i = 0; i < keys.length; i++) { + keyIds.push(keys[i].getKeyId()); + } + return keyIds; + } + + /** + * Returns key IDs of all key packets in hex + * @returns {[String]} + */ + this.getKeyIdsHex = function() { + return this.getKeyIds().map(function(keyId) { + return keyId.toHex(); + }); + } + + /** + * Returns first key packet which match to an array of key IDs + * @param {[keyid]} keyIds + * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} + */ + this.getKeyPacketByIds = function(keyIds) { + var keys = this.getAllKeyPackets(); + for (var i = 0; i < keys.length; i++) { + var keyId = keys[i].getKeyId(); + for (var j = 0; j < keyIds.length; j++) { + if (keyId.equals(keyIds[j])) { + //TODO return only verified keys + return keys[i]; + } + } + } + return null; + } + + /** + * Returns true if this is a public key + * @return {Boolean} + */ + this.isPublic = function() { + var publicKeyPackets = this.packets.filterByTag(enums.packet.public_key); + return publicKeyPackets.length !== 0 ? true : false; + } + + /** + * Returns true if this is a private key + * @return {Boolean} + */ + this.isPrivate = function() { + var privateKeyPackets = this.packets.filterByTag(enums.packet.private_key); + return privateKeyPackets.length !== 0 ? true : false; + } + + /** + * Returns first key packet that is available for signing + * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} + */ + this.getSigningKeyPacket = function() { + + var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa]; + + signing = signing.map(function(s) { + return enums.read(enums.publicKey, s); + }); + + var keys = this.getAllKeyPackets(); + + for (var i = 0; i < keys.length; i++) { + if (signing.indexOf(keys[i].algorithm) !== -1) { + return keys[i]; + } + } + + return null; + } + + /** + * Returns preferred signature hash algorithm of this key + * @return {String} + */ + function getPreferredSignatureHashAlgorithm() { + //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8 + //separate private key preference from digest preferences + return config.prefer_hash_algorithm; + } + + /** + * Returns the first valid encryption key packet for this key + * @returns {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} key packet or null if no encryption key has been found + */ + this.getEncryptionKeyPacket = function() { + // V4: by convention subkeys are prefered for encryption service + // V3: keys MUST NOT have subkeys + var isValidEncryptionKey = function(key) { + //TODO evaluate key flags: http://tools.ietf.org/html/rfc4880#section-5.2.3.21 + return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey, + enums.publicKey.rsa_sign); + //TODO verify key + //&& keys.verifyKey() + }; + + var subkeys = this.getSubkeyPackets(); + + for (var j = 0; j < subkeys.length; j++) { + if (isValidEncryptionKey(subkeys[j])) { + return subkeys[j]; + } + } + // if no valid subkey for encryption, use primary key + var primaryKey = this.getKeyPacket(); + if (isValidEncryptionKey(primaryKey)) { + return primaryKey; + } + return null; + } + + /** + * Decrypts all secret key and subkey packets + * @param {String} passphrase + * @return {undefined} + */ + this.decrypt = function(passphrase) { + //TODO boolean return value + var keys = this.getAllKeyPackets(); + + for (var i in keys) + if (keys[i].tag == enums.packet.secret_subkey || + keys[i].tag == enums.packet.secret_key) { + keys[i].decrypt(passphrase); + } + } + + + // TODO + this.verify = function() { + + } + // TODO + this.revoke = function() { + + } + +} + +/** + * reads an OpenPGP armored text and returns a key object + * @param {String} armoredText text to be parsed + * @return {key} new key object + */ +key.readArmored = function(armoredText) { + //TODO how do we want to handle bad text? Exception throwing + //TODO don't accept non-key armored texts + var input = armor.decode(armoredText).openpgp; + var packetlist = new packet.list(); + packetlist.read(input); + var newKey = new key(packetlist); + return newKey; +} + +module.exports = key; + +},{"./config":3,"./encoding/armor.js":25,"./enums.js":27,"./packet":35}],30:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -9115,166 +9360,174 @@ var armor = require('./encoding/armor.js'); /** * @class - * @classdesc Class that represents an OpenPGP key. Must contain a master key. - * @param {packetlist} packetlist [description] - * Can contain additional subkeys, signatures, - * user ids, user attributes. + * @classdesc The class that deals with storage of the keyring. Currently the only option is to use HTML5 local storage. */ - - function key(packetlist) { - - this.packets = packetlist || new packet.list(); - - - /** Returns the primary key (secret or public) - * @returns {packet_secret_key|packet_public_key|null} */ - this.getKey = function() { - for (var i = 0; i < this.packets.length; i++) { - if (this.packets[i].tag == enums.packet.public_key || - this.packets[i].tag == enums.packet.secret_key) { - return this.packets[i]; - } - } - return null; - }; - - /** Returns all the private and public subkeys - * @returns {[public_subkey|secret_subkey]} */ - this.getSubkeys = function() { - - var subkeys = []; - - for (var i = 0; i < this.packets.length; i++) { - if (this.packets[i].tag == enums.packet.public_subkey || - this.packets[i].tag == enums.packet.secret_subkey) { - subkeys.push(this.packets[i]); - } - } - - return subkeys; - }; - - this.getAllKeys = function() { - return [this.getKey()].concat(this.getSubkeys()); - }; - - this.getKeyids = function() { - var keyids = []; - var keys = this.getAllKeys(); - for (var i = 0; i < keys.length; i++) { - keyids.push(keys[i].getKeyId()); - } - return keyids; - }; - - this.getKeyById = function(keyid) { - var keys = this.getAllKeys(); - for (var i = 0; i < keys.length; i++) { - if (keys[i].getKeyId().equals(keyid)) { - return keys[i]; - } - } - } - - this.getSigningKey = function() { - - var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa]; - - signing = signing.map(function(s) { - return enums.read(enums.publicKey, s); - }); - - var keys = this.getAllKeys(); - - for (var i = 0; i < keys.length; i++) { - if (signing.indexOf(keys[i].algorithm) !== -1) { - return keys[i]; - } - } - - return null; - }; - - function getPreferredSignatureHashAlgorithm() { - var pkey = this.getSigningKey(); - if (pkey === null) { - util.print_error("private key is for encryption only! Cannot create a signature."); - return null; - } - if (pkey.publicKey.publicKeyAlgorithm == 17) { - var dsa = new DSA(); - return dsa.select_hash_algorithm(pkey.publicKey.MPIs[1].toBigInteger()); // q - } - //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8 - //separate private key preference from digest preferences - return openpgp.config.config.prefer_hash_algorithm; - } +var keyring = function() { + this.armoredPacketlists = []; + this.parsedPacketlists = []; /** - * Finds an encryption key for this key - * @returns null if no encryption key has been found + * Initialization routine for the keyring. This method reads the + * keyring from HTML5 local storage and initializes this instance. + * This method is called by openpgp.init(). */ - this.getEncryptionKey = function() { - // V4: by convention subkeys are prefered for encryption service - // V3: keys MUST NOT have subkeys - var isValidEncryptionKey = function(key) { - return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey, - enums.publicKey.rsa_sign); - //TODO verify key - //&& keys.verifyKey() - }; - - var subkeys = this.getSubkeys(); - - for (var j = 0; j < subkeys.length; j++) { - if (isValidEncryptionKey(subkeys[j])) { - return subkeys[j]; - } + function init() { + var armoredPacketlists = JSON.parse(window.localStorage.getItem("armoredPacketlists")); + if (armoredPacketlists === null || armoredPacketlists.length === 0) { + armoredPacketlists = []; } - // if no valid subkey for encryption, use primary key - var primaryKey = this.getKey(); - if (isValidEncryptionKey(primaryKey)) { - return primaryKey; + this.armoredPacketlists = armoredPacketlists; + + var packetlist; + for (var i = 0; i < armoredPacketlists.length; i++) { + packetlist = new packet.list(); + packetlist.read(armoredPacketlists[i]); + this.parsedPacketlists.push(packetlist); } - return null; - }; - - this.decrypt = function(passphrase) { - var keys = this.getAllKeys(); - - for (var i in keys) - if (keys[i].tag == enums.packet.secret_subkey || - keys[i].tag == enums.packet.secret_key) { - keys[i].decrypt(passphrase); - } - }; - - - // TODO need to implement this - - function revoke() { - } + this.init = init; + + /** + * Saves the current state of the keyring to HTML5 local storage. + * The privateKeys array and publicKeys array gets Stringified using JSON + */ + function store() { + window.localStorage.setItem("armoredPacketlists", JSON.stringify(this.armoredPacketlists)); + } + this.store = store; + + function emailPacketCheck(packet, email) { + var emailMatch = false; + var packetEmail; + email = email.toLowerCase(); + if (packet.tag == enums.packet.userid) { + packetEmail = packet.userid; + //we need to get just the email from the userid packet + packetEmail = packetEmail.split('<')[1].split('<')[0].trim.toLowerCase(); + if (packetEmail == email) { + emailMatch = true; + } + } + return emailMatch; + } + + function idPacketCheck(packet, id) { + if (packet.getKeyId && packet.getKeyId().write() == id) { + return true; + } + return false; + } + + function helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist) { + var packet; + for (var l = 0; l < packetlist.length; l++) { + packet = packetlist[l]; + identityMatch = identityFunction(packet, identityInput); + if (!packetType) { + packetMatch = true; + } else if (packet.tag == packetType) { + packetMatch = true; + } + if (packetMatch && identityMatch) { + return true; + } + } + return false; + } + + function checkForIdentityAndPacketMatch(identityFunction, identityInput, packetType) { + var results = []; + var packetlist; + var identityMatch; + var packetMatch; + for (var p = 0; p < this.parsedPacketlists.length; p++) { + identityMatch = false; + packetMatch = false; + packetlist = this.parsedPacketlists[p]; + if (helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist)) { + results.push(packetlist); + } + } + return results; + } + this.checkForIdentityAndPacketMatch = checkForIdentityAndPacketMatch; + + /** + * searches all public keys in the keyring matching the address or address part of the user ids + * @param {String} email_address + * @return {openpgp_msg_publickey[]} The public keys associated with provided email address. + */ + function getPublicKeyForAddress(email) { + return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.public_key); + } + this.getPublicKeyForAddress = getPublicKeyForAddress; + + /** + * Searches the keyring for a private key containing the specified email address + * @param {String} email_address email address to search for + * @return {openpgp_msg_privatekey[]} private keys found + */ + function getPrivateKeyForAddress(email_address) { + return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.secret_key); + } + this.getPrivateKeyForAddress = getPrivateKeyForAddress; + + /** + * Searches the keyring for public keys having the specified key id + * @param {String} keyId provided as string of hex number (lowercase) + * @return {openpgp_msg_privatekey[]} public keys found + */ + function getPacketlistsForKeyId(keyId) { + return this.checkForIdentityAndPacketMatch(idPacketCheck, keyId); + } + this.getPacketlistsForKeyId = getPacketlistsForKeyId; + + /** + * Imports a packet list (public or private key block) from an ascii armored message + * @param {String} armored message to read the packets/key from + */ + function importPacketlist(armored) { + this.armoredPacketlists.push(armored); + + var dearmored = armor.decode(armored.replace(/\r/g, '')).openpgp; + + packetlist = new packet.list(); + packetlist.read(dearmored); + this.parsedPacketlists.push(packetlist); + + return true; + } + this.importPacketlist = importPacketlist; + + /** + * TODO + * returns the openpgp_msg_privatekey representation of the public key at public key ring index + * @param {Integer} index the index of the public key within the publicKeys array + * @return {openpgp_msg_privatekey} the public key object + */ + function exportPublicKey(index) { + return this.publicKey[index]; + } + this.exportPublicKey = exportPublicKey; + + /** + * TODO + * Removes a public key from the public key keyring at the specified index + * @param {Integer} index the index of the public key within the publicKeys array + * @return {openpgp_msg_privatekey} The public key object which has been removed + */ + function removePublicKey(index) { + var removed = this.publicKeys.splice(index, 1); + this.store(); + return removed; + } + this.removePublicKey = removePublicKey; }; -/** - * reads an OpenPGP armored text and returns a key object - * @param {String} armoredText text to be parsed - * @return {key} new key object - */ -key.readArmored = function(armoredText) { - //TODO how do we want to handle bad text? Exception throwing - var input = armor.decode(armoredText).openpgp; - var packetlist = new packet.list(); - packetlist.read(input); - var newKey = new key(packetlist); - return newKey; -} +module.exports = new keyring(); -module.exports = key; - -},{"./encoding/armor.js":25,"./enums.js":27,"./packet":35}],30:[function(require,module,exports){ +},{"./encoding/armor.js":25,"./enums.js":27,"./packet":35}],31:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -9307,67 +9560,60 @@ function message(packetlist) { /** * Returns the key IDs of the public keys to which the session key is encrypted - * @return {[keyId]} keyId provided as string of hex number (lowercase) + * @return {[keyId]} array of keyid objects */ this.getKeyIds = function() { var keyIds = []; var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key); pkESKeyPacketlist.forEach(function(packet) { - keyIds.push(packet.publicKeyId.toHex()); + keyIds.push(packet.publicKeyId); }); return keyIds; } - function generic_decrypt(packets, passphrase) { - var sessionkey; - - for (var i = 0; i < packets.length; i++) { - if (packets[i].tag == openpgp_packet.tags.public_key_encrypted_session_key) { - var key = openpgp.keyring.getKeyById(packets[i].public_key_id); - - } - } - + /** + * Returns the key IDs in hex of the public keys to which the session key is encrypted + * @return {[String]} keyId provided as string of hex numbers (lowercase) + */ + this.getKeyIdsHex = function() { + return this.getKeyIds().map(function(keyId) { + return keyId.toHex(); + }); } /** - * Decrypts a message and generates user interface message out of the found. - * MDC will be verified as well as message signatures - * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key) - * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message - * @return {} plaintext of the message or null on error + * Decrypts the message + * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with (corresponding to the session key) + * @return {[String]} array with plaintext of decrypted messages */ - this.decrypt = function(key) { + this.decrypt = function(privateKeyPacket) { var decryptedMessages = []; - var keyids = key.getKeyids(); var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key); - outer: for (var i = 0; i < pkESKeyPacketlist.length; i++) { + for (var i = 0; i < pkESKeyPacketlist.length; i++) { var pkESKeyPacket = pkESKeyPacketlist[i]; - for (var j = 0; j < keyids.length; j++) { - if (pkESKeyPacket.publicKeyId.equals(keyids[j])) { - pkESKeyPacket.decrypt(key.getKeyById(pkESKeyPacket.publicKeyId)); - var symEncryptedPacketlist = this.packets.filter(function(packet) { - return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected; - }); - for (var k = 0; k < symEncryptedPacketlist.length; k++) { - var symEncryptedPacket = symEncryptedPacketlist[k]; - symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey); - for (var l = 0; l < symEncryptedPacket.packets.length; l++) { - var dataPacket = symEncryptedPacket.packets[l]; - switch (dataPacket.tag) { - case enums.packet.literal: - decryptedMessages.push(dataPacket.getBytes()); - break; - case enums.packet.compressed: - //TODO - break; - default: - //TODO - } + if (pkESKeyPacket.publicKeyId.equals(privateKeyPacket.getKeyId())) { + pkESKeyPacket.decrypt(privateKeyPacket); + var symEncryptedPacketlist = this.packets.filter(function(packet) { + return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected; + }); + for (var k = 0; k < symEncryptedPacketlist.length; k++) { + var symEncryptedPacket = symEncryptedPacketlist[k]; + symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey); + for (var l = 0; l < symEncryptedPacket.packets.length; l++) { + var dataPacket = symEncryptedPacket.packets[l]; + switch (dataPacket.tag) { + case enums.packet.literal: + decryptedMessages.push(dataPacket.getBytes()); + break; + case enums.packet.compressed: + //TODO + break; + default: + //TODO } } - break outer; } + break; } } return decryptedMessages; @@ -9458,6 +9704,7 @@ function message(packetlist) { */ message.readArmored = function(armoredText) { //TODO how do we want to handle bad text? Exception throwing + //TODO don't accept non-message armored texts var input = armor.decode(armoredText).openpgp; var packetlist = new packet.list(); packetlist.read(input); @@ -9466,7 +9713,7 @@ message.readArmored = function(armoredText) { } module.exports = message; -},{"./encoding/armor.js":25,"./enums.js":27,"./packet":35}],31:[function(require,module,exports){ +},{"./encoding/armor.js":25,"./enums.js":27,"./packet":35}],32:[function(require,module,exports){ // GPG4Browsers - An OpenPGP implementation in javascript // Copyright (C) 2011 Recurity Labs GmbH // @@ -9521,15 +9768,15 @@ function _openpgp() { var sessionKey = crypto.generateSessionKey(enums.read(enums.symmetric, config.encryption_cipher)); keys.forEach(function(key) { - var encryptionKey = key.getEncryptionKey(); - if (encryptionKey) { + var encryptionKeyPacket = key.getEncryptionKeyPacket(); + if (encryptionKeyPacket) { var pkESKeyPacket = new packet.public_key_encrypted_session_key(); - pkESKeyPacket.publicKeyId = encryptionKey.getKeyId(); - pkESKeyPacket.publicKeyAlgorithm = encryptionKey.algorithm; + pkESKeyPacket.publicKeyId = encryptionKeyPacket.getKeyId(); + pkESKeyPacket.publicKeyAlgorithm = encryptionKeyPacket.algorithm; pkESKeyPacket.sessionKey = sessionKey; //TODO get preferred algo from signature pkESKeyPacket.sessionKeyAlgorithm = enums.read(enums.symmetric, config.encryption_cipher); - pkESKeyPacket.encrypt(encryptionKey); + pkESKeyPacket.encrypt(encryptionKeyPacket); messagePacketlist.push(pkESKeyPacket); } }); @@ -9560,12 +9807,12 @@ function _openpgp() { /** * decrypts message - * @param {[key]} decrypted privateKey + * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with * @param {message} message the message object with the encrypted data * @return {String} decrypted message as as native JavaScript string */ - function decryptMessage(privateKey, message) { - return message.decrypt(privateKey); + function decryptMessage(privateKeyPacket, message) { + return message.decrypt(privateKeyPacket); } function decryptAndVerifyMessage(privateKey, publicKeys, messagePacketlist) { @@ -9784,198 +10031,7 @@ function _openpgp() { module.exports = new _openpgp(); -},{"./config":3,"./crypto":16,"./encoding/armor.js":25,"./enums.js":27,"./packet":35,"./util":56}],32:[function(require,module,exports){ -// GPG4Browsers - An OpenPGP implementation in javascript -// Copyright (C) 2011 Recurity Labs GmbH -// -// This library is free software; you can redistribute it and/or -// modify it under the terms of the GNU Lesser General Public -// License as published by the Free Software Foundation; either -// version 2.1 of the License, or (at your option) any later version. -// -// This library is distributed in the hope that it will be useful, -// but WITHOUT ANY WARRANTY; without even the implied warranty of -// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU -// Lesser General Public License for more details. -// -// You should have received a copy of the GNU Lesser General Public -// License along with this library; if not, write to the Free Software -// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA - -var packet = require('./packet'); -var enums = require('./enums.js'); -var armor = require('./encoding/armor.js'); - -/** - * @class - * @classdesc The class that deals with storage of the keyring. Currently the only option is to use HTML5 local storage. - */ -var keyring = function() { - this.armoredPacketlists = []; - this.parsedPacketlists = []; - - /** - * Initialization routine for the keyring. This method reads the - * keyring from HTML5 local storage and initializes this instance. - * This method is called by openpgp.init(). - */ - function init() { - var armoredPacketlists = JSON.parse(window.localStorage.getItem("armoredPacketlists")); - if (armoredPacketlists === null || armoredPacketlists.length === 0) { - armoredPacketlists = []; - } - this.armoredPacketlists = armoredPacketlists; - - var packetlist; - for (var i = 0; i < armoredPacketlists.length; i++) { - packetlist = new packet.list(); - packetlist.read(armoredPacketlists[i]); - this.parsedPacketlists.push(packetlist); - } - } - this.init = init; - - /** - * Saves the current state of the keyring to HTML5 local storage. - * The privateKeys array and publicKeys array gets Stringified using JSON - */ - function store() { - window.localStorage.setItem("armoredPacketlists", JSON.stringify(this.armoredPacketlists)); - } - this.store = store; - - function emailPacketCheck(packet, email) { - var emailMatch = false; - var packetEmail; - email = email.toLowerCase(); - if (packet.tag == enums.packet.userid) { - packetEmail = packet.userid; - //we need to get just the email from the userid packet - packetEmail = packetEmail.split('<')[1].split('<')[0].trim.toLowerCase(); - if (packetEmail == email) { - emailMatch = true; - } - } - return emailMatch; - } - - function idPacketCheck(packet, id) { - if (packet.getKeyId && packet.getKeyId().write() == id) { - return true; - } - return false; - } - - function helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist) { - var packet; - for (var l = 0; l < packetlist.length; l++) { - packet = packetlist[l]; - identityMatch = identityFunction(packet, identityInput); - if (!packetType) { - packetMatch = true; - } else if (packet.tag == packetType) { - packetMatch = true; - } - if (packetMatch && identityMatch) { - return true; - } - } - return false; - } - - function checkForIdentityAndPacketMatch(identityFunction, identityInput, packetType) { - var results = []; - var packetlist; - var identityMatch; - var packetMatch; - for (var p = 0; p < this.parsedPacketlists.length; p++) { - identityMatch = false; - packetMatch = false; - packetlist = this.parsedPacketlists[p]; - if (helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist)) { - results.push(packetlist); - } - } - return results; - } - this.checkForIdentityAndPacketMatch = checkForIdentityAndPacketMatch; - - /** - * searches all public keys in the keyring matching the address or address part of the user ids - * @param {String} email_address - * @return {openpgp_msg_publickey[]} The public keys associated with provided email address. - */ - function getPublicKeyForAddress(email) { - return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.public_key); - } - this.getPublicKeyForAddress = getPublicKeyForAddress; - - /** - * Searches the keyring for a private key containing the specified email address - * @param {String} email_address email address to search for - * @return {openpgp_msg_privatekey[]} private keys found - */ - function getPrivateKeyForAddress(email_address) { - return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.secret_key); - } - this.getPrivateKeyForAddress = getPrivateKeyForAddress; - - /** - * Searches the keyring for public keys having the specified key id - * @param {String} keyId provided as string of hex number (lowercase) - * @return {openpgp_msg_privatekey[]} public keys found - */ - function getPacketlistsForKeyId(keyId) { - return this.checkForIdentityAndPacketMatch(idPacketCheck, keyId); - } - this.getPacketlistsForKeyId = getPacketlistsForKeyId; - - /** - * Imports a packet list (public or private key block) from an ascii armored message - * @param {String} armored message to read the packets/key from - */ - function importPacketlist(armored) { - this.armoredPacketlists.push(armored); - - var dearmored = armor.decode(armored.replace(/\r/g, '')).openpgp; - - packetlist = new packet.list(); - packetlist.read(dearmored); - this.parsedPacketlists.push(packetlist); - - return true; - } - this.importPacketlist = importPacketlist; - - /** - * TODO - * returns the openpgp_msg_privatekey representation of the public key at public key ring index - * @param {Integer} index the index of the public key within the publicKeys array - * @return {openpgp_msg_privatekey} the public key object - */ - function exportPublicKey(index) { - return this.publicKey[index]; - } - this.exportPublicKey = exportPublicKey; - - /** - * TODO - * Removes a public key from the public key keyring at the specified index - * @param {Integer} index the index of the public key within the publicKeys array - * @return {openpgp_msg_privatekey} The public key object which has been removed - */ - function removePublicKey(index) { - var removed = this.publicKeys.splice(index, 1); - this.store(); - return removed; - } - this.removePublicKey = removePublicKey; - -}; - -module.exports = new keyring(); - -},{"./encoding/armor.js":25,"./enums.js":27,"./packet":35}],33:[function(require,module,exports){ +},{"./config":3,"./crypto":16,"./encoding/armor.js":25,"./enums.js":27,"./packet":35,"./util":56}],33:[function(require,module,exports){ var enums = require('../enums.js'); // This is pretty ugly, but browserify needs to have the requires explicitly written. @@ -12892,7 +12948,11 @@ var Util = function() { * @return {String} A native javascript string */ this.decode_utf8 = function(utf8) { - return decodeURIComponent(escape(utf8)); + try { + return decodeURIComponent(escape(utf8)); + } catch (e) { + return utf8; + } }; var str2bin = function(str, result) { @@ -13107,7 +13167,7 @@ unit.register("AES Rijndael cipher test with test vectors from ecb_tbl.txt", fun var result = new Array(); function test_aes(input, key, output) { - var aes = new openpgp.cipher.aes128(util.bin2str(key)); + var aes = new openpgp.crypto.cipher.aes128(util.bin2str(key)); var result = util.bin2str(aes.encrypt(input)); @@ -13226,11 +13286,11 @@ var unit = require('../../unit.js'); unit.register("Blowfish cipher test with test vectors from http://www.schneier.com/code/vectors.txt", function() { var openpgp = require('../../../'), util = openpgp.util, - BFencrypt = openpgp.cipher.blowfish; + BFencrypt = openpgp.crypto.cipher.blowfish; var result = []; function test_bf(input, key, output) { - var blowfish = new openpgp.cipher.blowfish(util.bin2str(key)); + var blowfish = new openpgp.crypto.cipher.blowfish(util.bin2str(key)); var result = util.bin2str(blowfish.encrypt(input)); return (util.hexstrdump(result) == util.hexstrdump(util.bin2str(output))); @@ -13298,7 +13358,7 @@ unit.register("CAST-128 cipher test with test vectors from RFC2144", function() var result = []; function test_cast(input, key, output) { - var cast5 = new openpgp.cipher.cast5(util.bin2str(key)); + var cast5 = new openpgp.crypto.cipher.cast5(util.bin2str(key)); var result = util.bin2str(cast5.encrypt(input)); return util.hexstrdump(result) == util.hexstrdump(util.bin2str(output)); @@ -13317,36 +13377,36 @@ unit.register("CAST-128 cipher test with test vectors from RFC2144", function() }); },{"../../../":28,"../../unit.js":70}],60:[function(require,module,exports){ -var unit = require('../../unit.js'); - -unit.register("TripleDES (EDE) cipher test with test vectors from http://csrc.nist.gov/publications/nistpubs/800-20/800-20.pdf", function() { - var openpgp = require('../../../'), - util = openpgp.util; - - var result = []; - var key = util.bin2str([1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1]); - var testvectors = [[[0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x95,0xF8,0xA5,0xE5,0xDD,0x31,0xD9,0x00]], - [[0x40,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xDD,0x7F,0x12,0x1C,0xA5,0x01,0x56,0x19]], - [[0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x2E,0x86,0x53,0x10,0x4F,0x38,0x34,0xEA]], - [[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4B,0xD3,0x88,0xFF,0x6C,0xD8,0x1D,0x4F]], - [[0x08,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x20,0xB9,0xE7,0x67,0xB2,0xFB,0x14,0x56]], - [[0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x55,0x57,0x93,0x80,0xD7,0x71,0x38,0xEF]], - [[0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x6C,0xC5,0xDE,0xFA,0xAF,0x04,0x51,0x2F]], - [[0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x0D,0x9F,0x27,0x9B,0xA5,0xD8,0x72,0x60]], - [[0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x00],[0xD9,0x03,0x1B,0x02,0x71,0xBD,0x5A,0x0A]], - [[0x00,0x40,0x00,0x00,0x00,0x00,0x00,0x00],[0x42,0x42,0x50,0xB3,0x7C,0x3D,0xD9,0x51]], - [[0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00],[0xB8,0x06,0x1B,0x7E,0xCD,0x9A,0x21,0xE5]], - [[0x00,0x10,0x00,0x00,0x00,0x00,0x00,0x00],[0xF1,0x5D,0x0F,0x28,0x6B,0x65,0xBD,0x28]], - [[0x00,0x08,0x00,0x00,0x00,0x00,0x00,0x00],[0xAD,0xD0,0xCC,0x8D,0x6E,0x5D,0xEB,0xA1]], - [[0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00],[0xE6,0xD5,0xF8,0x27,0x52,0xAD,0x63,0xD1]], - [[0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00],[0xEC,0xBF,0xE3,0xBD,0x3F,0x59,0x1A,0x5E]], - [[0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x00],[0xF3,0x56,0x83,0x43,0x79,0xD1,0x65,0xCD]], - [[0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00],[0x2B,0x9F,0x98,0x2F,0x20,0x03,0x7F,0xA9]], - [[0x00,0x00,0x40,0x00,0x00,0x00,0x00,0x00],[0x88,0x9D,0xE0,0x68,0xA1,0x6F,0x0B,0xE6]], - [[0x00,0x00,0x20,0x00,0x00,0x00,0x00,0x00],[0xE1,0x9E,0x27,0x5D,0x84,0x6A,0x12,0x98]], - [[0x00,0x00,0x10,0x00,0x00,0x00,0x00,0x00],[0x32,0x9A,0x8E,0xD5,0x23,0xD7,0x1A,0xEC]], - [[0x00,0x00,0x08,0x00,0x00,0x00,0x00,0x00],[0xE7,0xFC,0xE2,0x25,0x57,0xD2,0x3C,0x97]], - [[0x00,0x00,0x04,0x00,0x00,0x00,0x00,0x00],[0x12,0xA9,0xF5,0x81,0x7F,0xF2,0xD6,0x5D]], +var unit = require('../../unit.js'); + +unit.register("TripleDES (EDE) cipher test with test vectors from http://csrc.nist.gov/publications/nistpubs/800-20/800-20.pdf", function() { + var openpgp = require('../../../'), + util = openpgp.util; + + var result = []; + var key = util.bin2str([1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1]); + var testvectors = [[[0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x95,0xF8,0xA5,0xE5,0xDD,0x31,0xD9,0x00]], + [[0x40,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xDD,0x7F,0x12,0x1C,0xA5,0x01,0x56,0x19]], + [[0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x2E,0x86,0x53,0x10,0x4F,0x38,0x34,0xEA]], + [[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4B,0xD3,0x88,0xFF,0x6C,0xD8,0x1D,0x4F]], + [[0x08,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x20,0xB9,0xE7,0x67,0xB2,0xFB,0x14,0x56]], + [[0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x55,0x57,0x93,0x80,0xD7,0x71,0x38,0xEF]], + [[0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x6C,0xC5,0xDE,0xFA,0xAF,0x04,0x51,0x2F]], + [[0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x0D,0x9F,0x27,0x9B,0xA5,0xD8,0x72,0x60]], + [[0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x00],[0xD9,0x03,0x1B,0x02,0x71,0xBD,0x5A,0x0A]], + [[0x00,0x40,0x00,0x00,0x00,0x00,0x00,0x00],[0x42,0x42,0x50,0xB3,0x7C,0x3D,0xD9,0x51]], + [[0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00],[0xB8,0x06,0x1B,0x7E,0xCD,0x9A,0x21,0xE5]], + [[0x00,0x10,0x00,0x00,0x00,0x00,0x00,0x00],[0xF1,0x5D,0x0F,0x28,0x6B,0x65,0xBD,0x28]], + [[0x00,0x08,0x00,0x00,0x00,0x00,0x00,0x00],[0xAD,0xD0,0xCC,0x8D,0x6E,0x5D,0xEB,0xA1]], + [[0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00],[0xE6,0xD5,0xF8,0x27,0x52,0xAD,0x63,0xD1]], + [[0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00],[0xEC,0xBF,0xE3,0xBD,0x3F,0x59,0x1A,0x5E]], + [[0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x00],[0xF3,0x56,0x83,0x43,0x79,0xD1,0x65,0xCD]], + [[0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00],[0x2B,0x9F,0x98,0x2F,0x20,0x03,0x7F,0xA9]], + [[0x00,0x00,0x40,0x00,0x00,0x00,0x00,0x00],[0x88,0x9D,0xE0,0x68,0xA1,0x6F,0x0B,0xE6]], + [[0x00,0x00,0x20,0x00,0x00,0x00,0x00,0x00],[0xE1,0x9E,0x27,0x5D,0x84,0x6A,0x12,0x98]], + [[0x00,0x00,0x10,0x00,0x00,0x00,0x00,0x00],[0x32,0x9A,0x8E,0xD5,0x23,0xD7,0x1A,0xEC]], + [[0x00,0x00,0x08,0x00,0x00,0x00,0x00,0x00],[0xE7,0xFC,0xE2,0x25,0x57,0xD2,0x3C,0x97]], + [[0x00,0x00,0x04,0x00,0x00,0x00,0x00,0x00],[0x12,0xA9,0xF5,0x81,0x7F,0xF2,0xD6,0x5D]], [[0x00,0x00,0x02,0x00,0x00,0x00,0x00,0x00],[0xA4,0x84,0xC3,0xAD,0x38,0xDC,0x9C,0x19]], [[0x00,0x00,0x01,0x00,0x00,0x00,0x00,0x00],[0xFB,0xE0,0x0A,0x8A,0x1E,0xF8,0xAD,0x72]], [[0x00,0x00,0x00,0x80,0x00,0x00,0x00,0x00],[0x75,0x0D,0x07,0x94,0x07,0x52,0x13,0x63]], @@ -13389,38 +13449,38 @@ unit.register("TripleDES (EDE) cipher test with test vectors from http://csrc.ni [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x04],[0xD2,0xFD,0x88,0x67,0xD5,0x0D,0x2D,0xFE]], [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x02],[0x06,0xE7,0xEA,0x22,0xCE,0x92,0x70,0x8F]], [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01],[0x16,0x6B,0x40,0xB4,0x4A,0xBA,0x4B,0xD6]]]; - - var res = true; - var j = 0; - for (var i = 0; i < testvectors.length; i++) { - var des = new openpgp.cipher.des(key); - - var encr = util.bin2str(des.encrypt(testvectors[i][0], key)); - var res2 = encr == util.bin2str(testvectors[i][1]); - - res &= res2; - - if (!res2) { - result[j] = new unit.result("Testing vector with block " + - util.hexidump(testvectors[i][0]) + - " and key " + util.hexstrdump(key) + - " should be " + util.hexidump(testvectors[i][1]) + " != " + - util.hexidump(encr), - false); - j++; - } - } - if (res) { - result[j] = new unit.result("All 3DES EDE test vectors completed", true); - } - return result; -}); - - + + var res = true; + var j = 0; + for (var i = 0; i < testvectors.length; i++) { + var des = new openpgp.crypto.cipher.des(key); + + var encr = util.bin2str(des.encrypt(testvectors[i][0], key)); + var res2 = encr == util.bin2str(testvectors[i][1]); + + res &= res2; + + if (!res2) { + result[j] = new unit.result("Testing vector with block " + + util.hexidump(testvectors[i][0]) + + " and key " + util.hexstrdump(key) + + " should be " + util.hexidump(testvectors[i][1]) + " != " + + util.hexidump(encr), + false); + j++; + } + } + if (res) { + result[j] = new unit.result("All 3DES EDE test vectors completed", true); + } + return result; +}); + + unit.register("DES encrypt/decrypt padding tests", function () { - var openpgp = require('../../../'), - util = openpgp.util; - + var openpgp = require('../../../'), + util = openpgp.util; + var result = []; var key = util.bin2str([0x01, 0x23, 0x45, 0x67, 0x89, 0xAB, 0xCD, 0xEF]); var testvectors = new Array(); @@ -13449,7 +13509,7 @@ unit.register("DES encrypt/decrypt padding tests", function () { [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0xCA, 0x59, 0x61, 0x3A, 0x83, 0x23, 0x26, 0xDD]], [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9]]]; - var des = new openpgp.cipher.originalDes(key); + var des = new openpgp.crypto.cipher.originalDes(key); var res = true; var j = 0; @@ -13491,7 +13551,7 @@ unit.register("Twofish test with test vectors from http://www.schneier.com/code/ util = openpgp.util; function TFencrypt(block, key) { - var tf = new openpgp.cipher.twofish(key); + var tf = new openpgp.crypto.cipher.twofish(key); return tf.encrypt(block); } @@ -13559,7 +13619,7 @@ var unit = require('../../unit.js'); unit.register("MD5 test with test vectors from RFC 1321", function() { var openpgp = require('../../../'), util = openpgp.util, - MD5 = openpgp.hash.md5; + MD5 = openpgp.crypto.hash.md5; var result = new Array(); result[0] = new unit.result("MD5 (\"\") = d41d8cd98f00b204e9800998ecf8427e", @@ -13584,7 +13644,7 @@ unit.register("RIPE-MD 160 bits test with test vectors from http://homes.esat.ku var openpgp = require('../../../'), util = openpgp.util, - RMDstring = openpgp.hash.ripemd; + RMDstring = openpgp.crypto.hash.ripemd; var result = new Array(); result[0] = new unit.result("RMDstring (\"\") = 9c1185a5c5e9fc54612808977ee8f548b2258d31", @@ -13603,9 +13663,9 @@ var unit = require('../../unit.js'); unit.register("SHA* test with test vectors from NIST FIPS 180-2", function() { - var openpgp = require('../../../src'), + var openpgp = require('../../../'), util = openpgp.util, - hash = openpgp.hash; + hash = openpgp.crypto.hash; var result = new Array(); @@ -13632,7 +13692,7 @@ unit.register("SHA* test with test vectors from NIST FIPS 180-2", function() { return result; }); -},{"../../../src":28,"../../unit.js":70}],65:[function(require,module,exports){ +},{"../../../":28,"../../unit.js":70}],65:[function(require,module,exports){ var unit = require('../unit.js'); unit.register("Functional testing of openpgp.crypto.* methods", function() { @@ -13860,14 +13920,14 @@ unit.register("Functional testing of openpgp.crypto.* methods", function() { //Originally we passed public and secret MPI separately, now they are joined. Is this what we want to do long term? // RSA - var RSAsignedData = openpgp.signature.sign(2, 1, RSApubMPIs.concat(RSAsecMPIs), "foobar"); + var RSAsignedData = openpgp.crypto.signature.sign(2, 1, RSApubMPIs.concat(RSAsecMPIs), "foobar"); var RSAsignedDataMPI = new openpgp.mpi(); RSAsignedDataMPI.read(RSAsignedData); result[0] = new unit.result("Testing RSA Sign and Verify", - openpgp.signature.verify(1, 2, [RSAsignedDataMPI], RSApubMPIs, "foobar")); + openpgp.crypto.signature.verify(1, 2, [RSAsignedDataMPI], RSApubMPIs, "foobar")); // DSA - var DSAsignedData = openpgp.signature.sign(2, 17, DSApubMPIs.concat(DSAsecMPIs), "foobar"); + var DSAsignedData = openpgp.crypto.signature.sign(2, 17, DSApubMPIs.concat(DSAsecMPIs), "foobar"); var DSAmsgMPIs = []; DSAmsgMPIs[0] = new openpgp.mpi(); @@ -13875,31 +13935,31 @@ unit.register("Functional testing of openpgp.crypto.* methods", function() { DSAmsgMPIs[0].read(DSAsignedData.substring(0,34)); DSAmsgMPIs[1].read(DSAsignedData.substring(34,68)); result[1] = new unit.result("Testing DSA Sign and Verify", - openpgp.signature.verify(17, 2, DSAmsgMPIs, DSApubMPIs, "foobar")); + openpgp.crypto.signature.verify(17, 2, DSAmsgMPIs, DSApubMPIs, "foobar")); var symmAlgo = "aes256"; // AES256 - var symmKey = openpgp.generateSessionKey(symmAlgo); - var symmencDataOCFB = openpgp.cfb.encrypt(openpgp.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, true); - var symmencDataCFB = openpgp.cfb.encrypt(openpgp.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, false); + var symmKey = openpgp.crypto.generateSessionKey(symmAlgo); + var symmencDataOCFB = openpgp.crypto.cfb.encrypt(openpgp.crypto.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, true); + var symmencDataCFB = openpgp.crypto.cfb.encrypt(openpgp.crypto.getPrefixRandom(symmAlgo), symmAlgo, "foobarfoobar1234567890", symmKey, false); result[2] = new unit.result("Testing symmetric encrypt and decrypt with OpenPGP CFB resync", - openpgp.cfb.decrypt(symmAlgo,symmKey,symmencDataOCFB,true) == "foobarfoobar1234567890"); + openpgp.crypto.cfb.decrypt(symmAlgo,symmKey,symmencDataOCFB,true) == "foobarfoobar1234567890"); result[3] = new unit.result("Testing symmetric encrypt and decrypt without OpenPGP CFB resync (used in modification detection code \"MDC\" packets)", - openpgp.cfb.decrypt(symmAlgo,symmKey,symmencDataCFB,false) == "foobarfoobar1234567890"); + openpgp.crypto.cfb.decrypt(symmAlgo,symmKey,symmencDataCFB,false) == "foobarfoobar1234567890"); var RSAUnencryptedData = new openpgp.mpi(); - RSAUnencryptedData.fromBytes(openpgp.pkcs1.eme.encode(symmKey, RSApubMPIs[0].byteLength())); - var RSAEncryptedData = openpgp.publicKeyEncrypt("rsa_encrypt_sign", RSApubMPIs, RSAUnencryptedData); + RSAUnencryptedData.fromBytes(openpgp.crypto.pkcs1.eme.encode(symmKey, RSApubMPIs[0].byteLength())); + var RSAEncryptedData = openpgp.crypto.publicKeyEncrypt("rsa_encrypt_sign", RSApubMPIs, RSAUnencryptedData); result[4] = new unit.result("Testing asymmetric encrypt and decrypt using RSA with eme_pkcs1 padding", - openpgp.pkcs1.eme.decode(openpgp.publicKeyDecrypt("rsa_encrypt_sign", RSApubMPIs.concat(RSAsecMPIs), RSAEncryptedData).write().substring(2), RSApubMPIs[0].byteLength()) == symmKey); + openpgp.crypto.pkcs1.eme.decode(openpgp.crypto.publicKeyDecrypt("rsa_encrypt_sign", RSApubMPIs.concat(RSAsecMPIs), RSAEncryptedData).write().substring(2), RSApubMPIs[0].byteLength()) == symmKey); var ElgamalUnencryptedData = new openpgp.mpi(); - ElgamalUnencryptedData.fromBytes(openpgp.pkcs1.eme.encode(symmKey, ElgamalpubMPIs[0].byteLength())); - var ElgamalEncryptedData = openpgp.publicKeyEncrypt("elgamal", ElgamalpubMPIs, ElgamalUnencryptedData); + ElgamalUnencryptedData.fromBytes(openpgp.crypto.pkcs1.eme.encode(symmKey, ElgamalpubMPIs[0].byteLength())); + var ElgamalEncryptedData = openpgp.crypto.publicKeyEncrypt("elgamal", ElgamalpubMPIs, ElgamalUnencryptedData); result[5] = new unit.result("Testing asymmetric encrypt and decrypt using Elgamal with eme_pkcs1 padding", - openpgp.pkcs1.eme.decode(openpgp.publicKeyDecrypt("elgamal", ElgamalpubMPIs.concat(ElgamalsecMPIs), ElgamalEncryptedData).write().substring(2), ElgamalpubMPIs[0].byteLength()) == symmKey); + openpgp.crypto.pkcs1.eme.decode(openpgp.crypto.publicKeyDecrypt("elgamal", ElgamalpubMPIs.concat(ElgamalsecMPIs), ElgamalEncryptedData).write().substring(2), ElgamalpubMPIs[0].byteLength()) == symmKey); return result; }); @@ -13909,9 +13969,9 @@ var unit = require('../unit.js'); unit.register("Testing of binary signature checking", function() { var openpgp = require('../../'); - var keyring = require('../../src/openpgp.keyring.js'); + var keyring = openpgp.keyring; var result = []; - var priv_key = openpgp.readArmoredPackets([ + var priv_key = openpgp.key.readArmored([ '-----BEGIN PGP PRIVATE KEY BLOCK-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -13939,8 +13999,8 @@ unit.register("Testing of binary signature checking", function() { 'AKC8omYPPomN1E/UJFfXdLDIMi5LoA==', '=LSrW', '-----END PGP PRIVATE KEY BLOCK-----' - ].join("\n")); - var pub_key = openpgp.readArmoredPackets( + ].join("\n")).packets; + var pub_key = openpgp.key.readArmored( [ '-----BEGIN PGP PUBLIC KEY BLOCK-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -13958,8 +14018,8 @@ unit.register("Testing of binary signature checking", function() { 'EplqEakMckCtikEnpxYe', '=b2Ln', '-----END PGP PUBLIC KEY BLOCK-----' - ].join("\n")); - var msg = openpgp.readArmoredPackets([ + ].join("\n")).packets; + var msg = openpgp.message.readArmored([ '-----BEGIN PGP MESSAGE-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -13975,7 +14035,7 @@ unit.register("Testing of binary signature checking", function() { 'aOU=', '=4iGt', '-----END PGP MESSAGE-----' - ].join("\n")); + ].join("\n")).packets; //TODO need both? priv_key[0].decrypt("abcd"); priv_key[3].decrypt("abcd"); @@ -13987,7 +14047,7 @@ unit.register("Testing of binary signature checking", function() { // exercises the GnuPG s2k type 1001 extension: // the secrets on the primary key have been stripped. - var priv_key_gnupg_ext = openpgp.readArmoredPackets([ + var priv_key_gnupg_ext = openpgp.key.readArmored([ '-----BEGIN PGP PRIVATE KEY BLOCK-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -14014,7 +14074,7 @@ unit.register("Testing of binary signature checking", function() { 'iY3UT9QkV9d0sMgyLkug', '=GQsY', '-----END PGP PRIVATE KEY BLOCK-----', - ].join("\n")); + ].join("\n")).packets; priv_key_gnupg_ext[3].decrypt("abcd"); msg[0].decrypt(priv_key_gnupg_ext[3]); msg[1].decrypt(msg[0].sessionKeyAlgorithm, msg[0].sessionKey); @@ -14056,7 +14116,7 @@ unit.register("Testing of binary signature checking", function() { '-----END PGP PUBLIC KEY BLOCK-----' ].join("\n")); - var msg2 = openpgp.readArmoredPackets([ + var msg2 = openpgp.message.readArmored([ '-----BEGIN PGP MESSAGE-----', 'Version: GnuPG v1.4.11 (GNU/Linux)', '', @@ -14065,7 +14125,7 @@ unit.register("Testing of binary signature checking", function() { 'AJ9zh4zsK4GIPuEu81YPNmHsju7DYg==', '=WaSx', '-----END PGP MESSAGE-----' - ].join("\n")); + ].join("\n")).packets; var packetlists = keyring.getPacketlistsForKeyId(msg2[0].signingKeyId.write()); var pubKey = packetlists[0]; msg2[2].verify(pubKey[3], msg2[1]); @@ -14077,12 +14137,12 @@ unit.register("Testing of binary signature checking", function() { }); -},{"../../":28,"../../src/openpgp.keyring.js":32,"../unit.js":70}],67:[function(require,module,exports){ +},{"../../":28,"../unit.js":70}],67:[function(require,module,exports){ var unit = require('../unit.js'); unit.register("Key generation/encryption/decryption", function() { var openpgp = require('../../'); - var keyring = require('../../src/openpgp.keyring.js'); + var keyring = openpgp.keyring; var result = []; var testHelper = function(passphrase, userid, message) { var key = openpgp.generateKeyPair(openpgp.enums.publicKey.rsa_encrypt_sign, 512, @@ -14092,46 +14152,31 @@ unit.register("Key generation/encryption/decryption", function() { + 'userid: ' + userid + '\n' + 'message: ' + message; - var keyPacketlist = openpgp.readArmoredPackets(key); debugger; - var privKey = openpgp.getKeyFromPacketlist(keyPacketlist, passphrase); + var privKey = openpgp.key.readArmored(key); - var encrypted = openpgp.encryptMessage(keyPacketlist, message); + var encrypted = openpgp.encryptMessage([privKey], message); - openpgp.keyring.importPublicKey(key.publicKeyArmored); - - var msg = openpgp.read_message(encrypted); - var keymat = null; - var sesskey = null; + var msg = openpgp.message.readArmored(encrypted); // Find the private (sub)key for the session key of the message - for (var i = 0; i< msg[0].sessionKeys.length; i++) { - if (priv_key.privateKeyPacket.publicKey.getKeyId().write() == msg[0].sessionKeys[i].keyId.bytes) { - keymat = { key: priv_key, keymaterial: priv_key.privateKeyPacket}; - sesskey = msg[0].sessionKeys[i]; - break; - } - for (var j = 0; j < priv_key.subKeys.length; j++) { - if (priv_key.subKeys[j].publicKey.getKeyId().write() == msg[0].sessionKeys[i].keyId.bytes) { - keymat = { key: priv_key, keymaterial: priv_key.subKeys[j]}; - sesskey = msg[0].sessionKeys[i]; - break; - } - } + var privKeyPacket = privKey.getKeyPacketByIds(msg.getKeyIds()); + + if (!privKeyPacket) { + return new unit.result("No private key found!" + info, false); } - var decrypted = ''; - if (keymat !== null) { - if (!keymat.keymaterial.decryptSecretMPIs(passphrase)) { - return new test_result("Password for secrect key was incorrect!", - + info, false); + try { + if (!privKeyPacket.decrypt(passphrase)) { + return new unit.result("Password for secrect key was incorrect!" + info, false); } - - decrypted = msg[0].decrypt(keymat, sesskey); - } else { - return new test_result("No private key found!" + info, false); + } catch (e) { + return new unit.result("Exception on decrypt of private key packet!" + info, false); } - return new test_result(message + ' == ' + decrypted + info, message == decrypted); + + var decrypted = msg.decrypt(privKeyPacket)[0]; + + return new unit.result(message + ' == ' + decrypted + info, message == decrypted); }; result.push(testHelper('password', 'Test McTestington ', 'hello world')); @@ -14141,7 +14186,9 @@ unit.register("Key generation/encryption/decryption", function() { }); unit.register("Encryption/decryption", function() { - var openpgp = require('../../src'); + var openpgp = require('../../'); + + var result = []; var pub_key = ['-----BEGIN PGP PUBLIC KEY BLOCK-----', @@ -14235,15 +14282,19 @@ unit.register("Encryption/decryption", function() { var privKey = openpgp.key.readArmored(priv_key); - privKey.decrypt('hello world'); + var privKeyPacket = privKey.getKeyPacketByIds(message.getKeyIds()); - var decrypted = openpgp.decryptMessage(privKey, message); + privKeyPacket.decrypt('hello world'); - return new test_result('Encrypt plain text and afterwards decrypt leads to same result', plaintext == decrypted); + var decrypted = openpgp.decryptMessage(privKeyPacket, message); + + result[0] = new unit.result('Encrypt plain text and afterwards decrypt leads to same result', plaintext == decrypted); + + return result; }); -},{"../../":28,"../../src":28,"../../src/openpgp.keyring.js":32,"../unit.js":70}],"test-bundle.js":[function(require,module,exports){ +},{"../../":28,"../unit.js":70}],"test-bundle.js":[function(require,module,exports){ module.exports=require('QjPZ1q'); },{}],"QjPZ1q":[function(require,module,exports){ @@ -14315,5 +14366,5 @@ module.exports = { },{"__browserify_process":1}]},{},[]) -//@ sourceMappingURL=data:application/json;base64,{"version":3,"file":"generated.js","sources":["/home/toberndo/dev/node/lib/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js","/home/toberndo/dev/openpgpjs-devel/src/compression/jxg.js","/home/toberndo/dev/openpgpjs-devel/src/config/config.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cfb.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/aes.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/blowfish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/cast5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/des.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/twofish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/crypto.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/md5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/ripe-md.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/sha.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/pkcs1.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/dsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/elgamal.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/jsbn.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/rsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/random.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/signature.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/armor.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/base64.js","/home/toberndo/dev/openpgpjs-devel/src/enums.js","/home/toberndo/dev/openpgpjs-devel/src/index.js","/home/toberndo/dev/openpgpjs-devel/src/key.js","/home/toberndo/dev/openpgpjs-devel/src/message.js","/home/toberndo/dev/openpgpjs-devel/src/openpgp.js","/home/toberndo/dev/openpgpjs-devel/src/openpgp.keyring.js","/home/toberndo/dev/openpgpjs-devel/src/packet/all_packets.js","/home/toberndo/dev/openpgpjs-devel/src/packet/compressed.js","/home/toberndo/dev/openpgpjs-devel/src/packet/index.js","/home/toberndo/dev/openpgpjs-devel/src/packet/literal.js","/home/toberndo/dev/openpgpjs-devel/src/packet/marker.js","/home/toberndo/dev/openpgpjs-devel/src/packet/one_pass_signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packet.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packetlist.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_integrity_protected.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/symmetrically_encrypted.js","/home/toberndo/dev/openpgpjs-devel/src/packet/trust.js","/home/toberndo/dev/openpgpjs-devel/src/packet/user_attribute.js","/home/toberndo/dev/openpgpjs-devel/src/packet/userid.js","/home/toberndo/dev/openpgpjs-devel/src/type/keyid.js","/home/toberndo/dev/openpgpjs-devel/src/type/mpi.js","/home/toberndo/dev/openpgpjs-devel/src/type/s2k.js","/home/toberndo/dev/openpgpjs-devel/src/util/util.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/aes.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/blowfish.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/cast5.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/des.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/twofish.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/hash/md5.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/hash/ripemd.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/hash/sha.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/openpgp.crypto.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/openpgp.sigcheck.js","/home/toberndo/dev/openpgpjs-devel/test/general/openpgp.basic.js","/home/toberndo/dev/openpgpjs-devel/test/test-all.js","/home/toberndo/dev/openpgpjs-devel/test/unit.js"],"names":[],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3fA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzqDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpEA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1VA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;;;;;;;;;;;;;;;;;;;;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourcesContent":["// shim for using process in browser\n\nvar process = module.exports = {};\n\nprocess.nextTick = (function () {\n    var canSetImmediate = typeof window !== 'undefined'\n    && window.setImmediate;\n    var canPost = typeof window !== 'undefined'\n    && window.postMessage && window.addEventListener\n    ;\n\n    if (canSetImmediate) {\n        return function (f) { return window.setImmediate(f) };\n    }\n\n    if (canPost) {\n        var queue = [];\n        window.addEventListener('message', function (ev) {\n            if (ev.source === window && ev.data === 'process-tick') {\n                ev.stopPropagation();\n                if (queue.length > 0) {\n                    var fn = queue.shift();\n                    fn();\n                }\n            }\n        }, true);\n\n        return function nextTick(fn) {\n            queue.push(fn);\n            window.postMessage('process-tick', '*');\n        };\n    }\n\n    return function nextTick(fn) {\n        setTimeout(fn, 0);\n    };\n})();\n\nprocess.title = 'browser';\nprocess.browser = true;\nprocess.env = {};\nprocess.argv = [];\n\nprocess.binding = function (name) {\n    throw new Error('process.binding is not supported');\n}\n\n// TODO(shtylman)\nprocess.cwd = function () { return '/' };\nprocess.chdir = function (dir) {\n    throw new Error('process.chdir is not supported');\n};\n","JXG = {\n  exists: (function(undefined) {\n    return function(v) {\n      return !(v === undefined || v === null);\n    }\n  })()\n};\nJXG.decompress = function(str) {\n  return unescape((new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(str))).unzip()[0][0]);\n};\n/*\n    Copyright 2008-2012\n        Matthias Ehmann,\n        Michael Gerhaeuser,\n        Carsten Miller,\n        Bianca Valentin,\n        Alfred Wassermann,\n        Peter Wilfahrt\n\n    This file is part of JSXGraph.\n    \n    Dual licensed under the Apache License Version 2.0, or LGPL Version 3 licenses.\n\n    You should have received a copy of the GNU Lesser General Public License\n    along with JSXCompressor.  If not, see <http://www.gnu.org/licenses/>.\n    \n    You should have received a copy of the Apache License along with JSXCompressor.  \n    If not, see <http://www.apache.org/licenses/>.\n\n*/\n\n/**\n * @class Util class\n * @classdesc Utilities for uncompressing and base64 decoding\n * Class for gunzipping, unzipping and base64 decoding of files.\n * It is used for reading GEONExT, Geogebra and Intergeo files.\n *\n * Only Huffman codes are decoded in gunzip.\n * The code is based on the source code for gunzip.c by Pasi Ojala \n * {@link http://www.cs.tut.fi/~albert/Dev/gunzip/gunzip.c}\n * {@link http://www.cs.tut.fi/~albert}\n */\nJXG.Util = {};\n\n/**\n * Unzip zip files\n */\nJXG.Util.Unzip = function(barray) {\n  var outputArr = [],\n    output = \"\",\n    debug = false,\n    gpflags,\n    files = 0,\n    unzipped = [],\n    crc,\n    buf32k = new Array(32768),\n    bIdx = 0,\n    modeZIP = false,\n\n    CRC, SIZE,\n\n    bitReverse = [\n        0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0,\n        0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0,\n        0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,\n        0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8,\n        0x04, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4,\n        0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,\n        0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec,\n        0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc,\n        0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,\n        0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2,\n        0x0a, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea,\n        0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,\n        0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6,\n        0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6,\n        0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,\n        0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe,\n        0x01, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1,\n        0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,\n        0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9,\n        0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9,\n        0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,\n        0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5,\n        0x0d, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed,\n        0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,\n        0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3,\n        0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3,\n        0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,\n        0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb,\n        0x07, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7,\n        0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,\n        0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef,\n        0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff\n    ],\n\n    cplens = [\n        3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31,\n        35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0\n    ],\n\n    cplext = [\n        0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2,\n        3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 99, 99\n    ],\n    /* 99==invalid */\n\n    cpdist = [\n        0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0007, 0x0009, 0x000d,\n        0x0011, 0x0019, 0x0021, 0x0031, 0x0041, 0x0061, 0x0081, 0x00c1,\n        0x0101, 0x0181, 0x0201, 0x0301, 0x0401, 0x0601, 0x0801, 0x0c01,\n        0x1001, 0x1801, 0x2001, 0x3001, 0x4001, 0x6001\n    ],\n\n    cpdext = [\n        0, 0, 0, 0, 1, 1, 2, 2,\n        3, 3, 4, 4, 5, 5, 6, 6,\n        7, 7, 8, 8, 9, 9, 10, 10,\n        11, 11, 12, 12, 13, 13\n    ],\n\n    border = [16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15],\n\n    bA = barray,\n\n    bytepos = 0,\n    bitpos = 0,\n    bb = 1,\n    bits = 0,\n\n    NAMEMAX = 256,\n\n    nameBuf = [],\n\n    fileout;\n\n  function readByte() {\n    bits += 8;\n    if (bytepos < bA.length) {\n      //if (debug)\n      //    document.write(bytepos+\": \"+bA[bytepos]+\"<br>\");\n      return bA[bytepos++];\n    } else\n      return -1;\n  };\n\n  function byteAlign() {\n    bb = 1;\n  };\n\n  function readBit() {\n    var carry;\n    bits++;\n    carry = (bb & 1);\n    bb >>= 1;\n    if (bb == 0) {\n      bb = readByte();\n      carry = (bb & 1);\n      bb = (bb >> 1) | 0x80;\n    }\n    return carry;\n  };\n\n  function readBits(a) {\n    var res = 0,\n      i = a;\n\n    while (i--) {\n      res = (res << 1) | readBit();\n    }\n    if (a) {\n      res = bitReverse[res] >> (8 - a);\n    }\n    return res;\n  };\n\n  function flushBuffer() {\n    //document.write('FLUSHBUFFER:'+buf32k);\n    bIdx = 0;\n  };\n\n  function addBuffer(a) {\n    SIZE++;\n    //CRC=updcrc(a,crc);\n    buf32k[bIdx++] = a;\n    outputArr.push(String.fromCharCode(a));\n    //output+=String.fromCharCode(a);\n    if (bIdx == 0x8000) {\n      //document.write('ADDBUFFER:'+buf32k);\n      bIdx = 0;\n    }\n  };\n\n  function HufNode() {\n    this.b0 = 0;\n    this.b1 = 0;\n    this.jump = null;\n    this.jumppos = -1;\n  };\n\n  var LITERALS = 288;\n\n  var literalTree = new Array(LITERALS);\n  var distanceTree = new Array(32);\n  var treepos = 0;\n  var Places = null;\n  var Places2 = null;\n\n  var impDistanceTree = new Array(64);\n  var impLengthTree = new Array(64);\n\n  var len = 0;\n  var fpos = new Array(17);\n  fpos[0] = 0;\n  var flens;\n  var fmax;\n\n  function IsPat() {\n    while (1) {\n      if (fpos[len] >= fmax)\n        return -1;\n      if (flens[fpos[len]] == len)\n        return fpos[len]++;\n      fpos[len]++;\n    }\n  };\n\n  function Rec() {\n    var curplace = Places[treepos];\n    var tmp;\n    if (debug)\n      document.write(\"<br>len:\" + len + \" treepos:\" + treepos);\n    if (len == 17) { //war 17\n      return -1;\n    }\n    treepos++;\n    len++;\n\n    tmp = IsPat();\n    if (debug)\n      document.write(\"<br>IsPat \" + tmp);\n    if (tmp >= 0) {\n      curplace.b0 = tmp; /* leaf cell for 0-bit */\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n    } else {\n      /* Not a Leaf cell */\n      curplace.b0 = 0x8000;\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n      if (Rec())\n        return -1;\n    }\n    tmp = IsPat();\n    if (tmp >= 0) {\n      curplace.b1 = tmp; /* leaf cell for 1-bit */\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = null; /* Just for the display routine */\n    } else {\n      /* Not a Leaf cell */\n      curplace.b1 = 0x8000;\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = Places[treepos];\n      curplace.jumppos = treepos;\n      if (Rec())\n        return -1;\n    }\n    len--;\n    return 0;\n  };\n\n  function CreateTree(currentTree, numval, lengths, show) {\n    var i;\n    /* Create the Huffman decode tree/table */\n    //document.write(\"<br>createtree<br>\");\n    if (debug)\n      document.write(\"currentTree \" + currentTree + \" numval \" + numval + \" lengths \" + lengths + \" show \" + show);\n    Places = currentTree;\n    treepos = 0;\n    flens = lengths;\n    fmax = numval;\n    for (i = 0; i < 17; i++)\n      fpos[i] = 0;\n    len = 0;\n    if (Rec()) {\n      //fprintf(stderr, \"invalid huffman tree\\n\");\n      if (debug)\n        alert(\"invalid huffman tree\\n\");\n      return -1;\n    }\n    if (debug) {\n      document.write('<br>Tree: ' + Places.length);\n      for (var a = 0; a < 32; a++) {\n        document.write(\"Places[\" + a + \"].b0=\" + Places[a].b0 + \"<br>\");\n        document.write(\"Places[\" + a + \"].b1=\" + Places[a].b1 + \"<br>\");\n      }\n    }\n\n    /*if(show) {\n            var tmp;\n            for(tmp=currentTree;tmp<Places;tmp++) {\n                fprintf(stdout, \"0x%03x  0x%03x (0x%04x)\",tmp-currentTree, tmp->jump?tmp->jump-currentTree:0,(tmp->jump?tmp->jump-currentTree:0)*6+0xcf0);\n                if(!(tmp.b0 & 0x8000)) {\n                    //fprintf(stdout, \"  0x%03x (%c)\", tmp->b0,(tmp->b0<256 && isprint(tmp->b0))?tmp->b0:'�');\n                }\n                if(!(tmp.b1 & 0x8000)) {\n                    if((tmp.b0 & 0x8000))\n                        fprintf(stdout, \"           \");\n                    fprintf(stdout, \"  0x%03x (%c)\", tmp->b1,(tmp->b1<256 && isprint(tmp->b1))?tmp->b1:'�');\n                }\n                fprintf(stdout, \"\\n\");\n            }\n        }*/\n    return 0;\n  };\n\n  function DecodeValue(currentTree) {\n    var len, i,\n      xtreepos = 0,\n      X = currentTree[xtreepos],\n      b;\n\n    /* decode one symbol of the data */\n    while (1) {\n      b = readBit();\n      if (debug)\n        document.write(\"b=\" + b);\n      if (b) {\n        if (!(X.b1 & 0x8000)) {\n          if (debug)\n            document.write(\"ret1\");\n          return X.b1; /* If leaf node, return data */\n        }\n        X = X.jump;\n        len = currentTree.length;\n        for (i = 0; i < len; i++) {\n          if (currentTree[i] === X) {\n            xtreepos = i;\n            break;\n          }\n        }\n        //xtreepos++;\n      } else {\n        if (!(X.b0 & 0x8000)) {\n          if (debug)\n            document.write(\"ret2\");\n          return X.b0; /* If leaf node, return data */\n        }\n        //X++; //??????????????????\n        xtreepos++;\n        X = currentTree[xtreepos];\n      }\n    }\n    if (debug)\n      document.write(\"ret3\");\n    return -1;\n  };\n\n  function DeflateLoop() {\n    var last, c, type, i, len;\n\n    do {\n      /*if((last = readBit())){\n            fprintf(errfp, \"Last Block: \");\n        } else {\n            fprintf(errfp, \"Not Last Block: \");\n        }*/\n      last = readBit();\n      type = readBits(2);\n      switch (type) {\n        case 0:\n          if (debug)\n            alert(\"Stored\\n\");\n          break;\n        case 1:\n          if (debug)\n            alert(\"Fixed Huffman codes\\n\");\n          break;\n        case 2:\n          if (debug)\n            alert(\"Dynamic Huffman codes\\n\");\n          break;\n        case 3:\n          if (debug)\n            alert(\"Reserved block type!!\\n\");\n          break;\n        default:\n          if (debug)\n            alert(\"Unexpected value %d!\\n\", type);\n          break;\n      }\n\n      if (type == 0) {\n        var blockLen, cSum;\n\n        // Stored \n        byteAlign();\n        blockLen = readByte();\n        blockLen |= (readByte() << 8);\n\n        cSum = readByte();\n        cSum |= (readByte() << 8);\n\n        if (((blockLen ^ ~cSum) & 0xffff)) {\n          document.write(\"BlockLen checksum mismatch\\n\");\n        }\n        while (blockLen--) {\n          c = readByte();\n          addBuffer(c);\n        }\n      } else if (type == 1) {\n        var j;\n\n        /* Fixed Huffman tables -- fixed decode routine */\n        while (1) {\n          /*\n                256    0000000        0\n                :   :     :\n                279    0010111        23\n                0   00110000    48\n                :    :      :\n                143    10111111    191\n                280 11000000    192\n                :    :      :\n                287 11000111    199\n                144    110010000    400\n                :    :       :\n                255    111111111    511\n    \n                Note the bit order!\n                */\n\n          j = (bitReverse[readBits(7)] >> 1);\n          if (j > 23) {\n            j = (j << 1) | readBit(); /* 48..255 */\n\n            if (j > 199) { /* 200..255 */\n              j -= 128; /*  72..127 */\n              j = (j << 1) | readBit(); /* 144..255 << */\n            } else { /*  48..199 */\n              j -= 48; /*   0..151 */\n              if (j > 143) {\n                j = j + 136; /* 280..287 << */\n                /*   0..143 << */\n              }\n            }\n          } else { /*   0..23 */\n            j += 256; /* 256..279 << */\n          }\n          if (j < 256) {\n            addBuffer(j);\n            //document.write(\"out:\"+String.fromCharCode(j));\n            /*fprintf(errfp, \"@%d %02x\\n\", SIZE, j);*/\n          } else if (j == 256) {\n            /* EOF */\n            break;\n          } else {\n            var len, dist;\n\n            j -= 256 + 1; /* bytes + EOF */\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = bitReverse[readBits(5)] >> 3;\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n\n            /*fprintf(errfp, \"@%d (l%02x,d%04x)\\n\", SIZE, len, dist);*/\n            for (j = 0; j < len; j++) {\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          }\n        } // while\n      } else if (type == 2) {\n        var j, n, literalCodes, distCodes, lenCodes;\n        var ll = new Array(288 + 32); // \"static\" just to preserve stack\n\n        // Dynamic Huffman tables \n\n        literalCodes = 257 + readBits(5);\n        distCodes = 1 + readBits(5);\n        lenCodes = 4 + readBits(4);\n        //document.write(\"<br>param: \"+literalCodes+\" \"+distCodes+\" \"+lenCodes+\"<br>\");\n        for (j = 0; j < 19; j++) {\n          ll[j] = 0;\n        }\n\n        // Get the decode tree code lengths\n\n        //document.write(\"<br>\");\n        for (j = 0; j < lenCodes; j++) {\n          ll[border[j]] = readBits(3);\n          //document.write(ll[border[j]]+\" \");\n        }\n        //fprintf(errfp, \"\\n\");\n        //document.write('<br>ll:'+ll);\n        len = distanceTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        if (CreateTree(distanceTree, 19, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug) {\n          document.write(\"<br>distanceTree\");\n          for (var a = 0; a < distanceTree.length; a++) {\n            document.write(\"<br>\" + distanceTree[a].b0 + \" \" + distanceTree[a].b1 + \" \" + distanceTree[a].jump + \" \" +\n              distanceTree[a].jumppos);\n            /*if (distanceTree[a].jumppos!=-1)\n                    \tdocument.write(\" \"+distanceTree[a].jump.b0+\" \"+distanceTree[a].jump.b1);\n                \t*/\n          }\n        }\n        //document.write('<BR>tree created');\n\n        //read in literal and distance code lengths\n        n = literalCodes + distCodes;\n        i = 0;\n        var z = -1;\n        if (debug)\n          document.write(\"<br>n=\" + n + \" bits: \" + bits + \"<br>\");\n        while (i < n) {\n          z++;\n          j = DecodeValue(distanceTree);\n          if (debug)\n            document.write(\"<br>\" + z + \" i:\" + i + \" decode: \" + j + \"    bits \" + bits + \"<br>\");\n          if (j < 16) { // length of code in bits (0..15)\n            ll[i++] = j;\n          } else if (j == 16) { // repeat last length 3 to 6 times \n            var l;\n            j = 3 + readBits(2);\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            l = i ? ll[i - 1] : 0;\n            while (j--) {\n              ll[i++] = l;\n            }\n          } else {\n            if (j == 17) { // 3 to 10 zero length codes\n              j = 3 + readBits(3);\n            } else { // j == 18: 11 to 138 zero length codes \n              j = 11 + readBits(7);\n            }\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            while (j--) {\n              ll[i++] = 0;\n            }\n          }\n        }\n        /*for(j=0; j<literalCodes+distCodes; j++) {\n                //fprintf(errfp, \"%d \", ll[j]);\n                if ((j&7)==7)\n                    fprintf(errfp, \"\\n\");\n            }\n            fprintf(errfp, \"\\n\");*/\n        // Can overwrite tree decode tree as it is not used anymore\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          literalTree[i] = new HufNode();\n        if (CreateTree(literalTree, literalCodes, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        var ll2 = new Array();\n        for (i = literalCodes; i < ll.length; i++) {\n          ll2[i - literalCodes] = ll[i];\n        }\n        if (CreateTree(distanceTree, distCodes, ll2, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug)\n          document.write(\"<br>literalTree\");\n        outer: while (1) {\n          j = DecodeValue(literalTree);\n          if (j >= 256) { // In C64: if carry set\n            var len, dist;\n            j -= 256;\n            if (j == 0) {\n              // EOF\n              break;\n            }\n            j--;\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = DecodeValue(distanceTree);\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n            while (len--) {\n              if (bIdx - dist < 0) {\n                break outer;\n              }\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          } else {\n            addBuffer(j);\n          }\n        }\n      }\n    } while (!last);\n    flushBuffer();\n\n    byteAlign();\n    return 0;\n  };\n\n  JXG.Util.Unzip.prototype.unzipFile = function(name) {\n    var i;\n    this.unzip();\n    //alert(unzipped[0][1]);\n    for (i = 0; i < unzipped.length; i++) {\n      if (unzipped[i][1] == name) {\n        return unzipped[i][0];\n      }\n    }\n\n  };\n\n  JXG.Util.Unzip.prototype.deflate = function() {\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    DeflateLoop();\n    if (debug)\n      alert(outputArr.join(''));\n    unzipped[files] = new Array(2);\n    unzipped[files][0] = outputArr.join('');\n    unzipped[files][1] = \"DEFLATE\";\n    files++;\n    return unzipped;\n  }\n\n  JXG.Util.Unzip.prototype.unzip = function() {\n    //convertToByteArray(input);\n    if (debug)\n      alert(bA);\n    /*for (i=0;i<bA.length*8;i++){\n\t\tdocument.write(readBit());\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}*/\n    /*for (i=0;i<bA.length;i++){\n\t\tdocument.write(readByte()+\" \");\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}\n\tfor (i=0;i<bA.length;i++){\n\t\tdocument.write(bA[i]+\" \");\n\t\tif ((i+1)%16==0)\n\t\t\tdocument.write(\"<br>\");\n\t}\t\n\t*/\n    //alert(bA);\n    nextFile();\n    return unzipped;\n  };\n\n  function nextFile() {\n    if (debug)\n      alert(\"NEXTFILE\");\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    tmp[0] = readByte();\n    tmp[1] = readByte();\n    if (debug)\n      alert(\"type: \" + tmp[0] + \" \" + tmp[1]);\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"da\", 16)) { //GZIP\n      if (debug)\n        alert(\"GEONExT-GZIP\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"geonext.gxt\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"9c\", 16)) { //ZLIB\n      if (debug)\n        alert(\"ZLIB\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"ZLIB\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"1f\", 16) && tmp[1] == parseInt(\"8b\", 16)) { //GZIP\n      if (debug)\n        alert(\"GZIP\");\n      //DeflateLoop();\n      skipdir();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"file\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"50\", 16) && tmp[1] == parseInt(\"4b\", 16)) { //ZIP\n      modeZIP = true;\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n      if (tmp[2] == parseInt(\"3\", 16) && tmp[3] == parseInt(\"4\", 16)) {\n        //MODE_ZIP\n        tmp[0] = readByte();\n        tmp[1] = readByte();\n        if (debug)\n          alert(\"ZIP-Version: \" + tmp[1] + \" \" + tmp[0] / 10 + \".\" + tmp[0] % 10);\n\n        gpflags = readByte();\n        gpflags |= (readByte() << 8);\n        if (debug)\n          alert(\"gpflags: \" + gpflags);\n\n        var method = readByte();\n        method |= (readByte() << 8);\n        if (debug)\n          alert(\"method: \" + method);\n\n        readByte();\n        readByte();\n        readByte();\n        readByte();\n\n        var crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n\n        var compSize = readByte();\n        compSize |= (readByte() << 8);\n        compSize |= (readByte() << 16);\n        compSize |= (readByte() << 24);\n\n        var size = readByte();\n        size |= (readByte() << 8);\n        size |= (readByte() << 16);\n        size |= (readByte() << 24);\n\n        if (debug)\n          alert(\"local CRC: \" + crc + \"\\nlocal Size: \" + size + \"\\nlocal CompSize: \" + compSize);\n\n        var filelen = readByte();\n        filelen |= (readByte() << 8);\n\n        var extralen = readByte();\n        extralen |= (readByte() << 8);\n\n        if (debug)\n          alert(\"filelen \" + filelen);\n        i = 0;\n        nameBuf = [];\n        while (filelen--) {\n          var c = readByte();\n          if (c == \"/\" | c == \":\") {\n            i = 0;\n          } else if (i < NAMEMAX - 1)\n            nameBuf[i++] = String.fromCharCode(c);\n        }\n        if (debug)\n          alert(\"nameBuf: \" + nameBuf);\n\n        //nameBuf[i] = \"\\0\";\n        if (!fileout)\n          fileout = nameBuf;\n\n        var i = 0;\n        while (i < extralen) {\n          c = readByte();\n          i++;\n        }\n\n        CRC = 0xffffffff;\n        SIZE = 0;\n\n        if (size = 0 && fileOut.charAt(fileout.length - 1) == \"/\") {\n          //skipdir\n          if (debug)\n            alert(\"skipdir\");\n        }\n        if (method == 8) {\n          DeflateLoop();\n          if (debug)\n            alert(outputArr.join(''));\n          unzipped[files] = new Array(2);\n          unzipped[files][0] = outputArr.join('');\n          unzipped[files][1] = nameBuf.join('');\n          files++;\n          //return outputArr.join('');\n        }\n        skipdir();\n      }\n    }\n  };\n\n  function skipdir() {\n    var crc,\n      tmp = [],\n      compSize, size, os, i, c;\n\n    if ((gpflags & 8)) {\n      tmp[0] = readByte();\n      tmp[1] = readByte();\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n\n      if (tmp[0] == parseInt(\"50\", 16) &&\n        tmp[1] == parseInt(\"4b\", 16) &&\n        tmp[2] == parseInt(\"07\", 16) &&\n        tmp[3] == parseInt(\"08\", 16)) {\n        crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n      } else {\n        crc = tmp[0] | (tmp[1] << 8) | (tmp[2] << 16) | (tmp[3] << 24);\n      }\n\n      compSize = readByte();\n      compSize |= (readByte() << 8);\n      compSize |= (readByte() << 16);\n      compSize |= (readByte() << 24);\n\n      size = readByte();\n      size |= (readByte() << 8);\n      size |= (readByte() << 16);\n      size |= (readByte() << 24);\n\n      if (debug)\n        alert(\"CRC:\");\n    }\n\n    if (modeZIP)\n      nextFile();\n\n    tmp[0] = readByte();\n    if (tmp[0] != 8) {\n      if (debug)\n        alert(\"Unknown compression method!\");\n      return 0;\n    }\n\n    gpflags = readByte();\n    if (debug) {\n      if ((gpflags & ~(parseInt(\"1f\", 16))))\n        alert(\"Unknown flags set!\");\n    }\n\n    readByte();\n    readByte();\n    readByte();\n    readByte();\n\n    readByte();\n    os = readByte();\n\n    if ((gpflags & 4)) {\n      tmp[0] = readByte();\n      tmp[2] = readByte();\n      len = tmp[0] + 256 * tmp[1];\n      if (debug)\n        alert(\"Extra field size: \" + len);\n      for (i = 0; i < len; i++)\n        readByte();\n    }\n\n    if ((gpflags & 8)) {\n      i = 0;\n      nameBuf = [];\n      while (c = readByte()) {\n        if (c == \"7\" || c == \":\")\n          i = 0;\n        if (i < NAMEMAX - 1)\n          nameBuf[i++] = c;\n      }\n      //nameBuf[i] = \"\\0\";\n      if (debug)\n        alert(\"original file name: \" + nameBuf);\n    }\n\n    if ((gpflags & 16)) {\n      while (c = readByte()) {\n        //FILE COMMENT\n      }\n    }\n\n    if ((gpflags & 2)) {\n      readByte();\n      readByte();\n    }\n\n    DeflateLoop();\n\n    crc = readByte();\n    crc |= (readByte() << 8);\n    crc |= (readByte() << 16);\n    crc |= (readByte() << 24);\n\n    size = readByte();\n    size |= (readByte() << 8);\n    size |= (readByte() << 16);\n    size |= (readByte() << 24);\n\n    if (modeZIP)\n      nextFile();\n\n  };\n\n};\n\n/**\n *  Base64 encoding / decoding\n *  {@link http://www.webtoolkit.info/}\n */\nJXG.Util.Base64 = {\n\n  // private property\n  _keyStr: \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=\",\n\n  // public method for encoding\n  encode: function(input) {\n    var output = [],\n      chr1, chr2, chr3, enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = JXG.Util.Base64._utf8_encode(input);\n\n    while (i < input.length) {\n\n      chr1 = input.charCodeAt(i++);\n      chr2 = input.charCodeAt(i++);\n      chr3 = input.charCodeAt(i++);\n\n      enc1 = chr1 >> 2;\n      enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);\n      enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);\n      enc4 = chr3 & 63;\n\n      if (isNaN(chr2)) {\n        enc3 = enc4 = 64;\n      } else if (isNaN(chr3)) {\n        enc4 = 64;\n      }\n\n      output.push([this._keyStr.charAt(enc1),\n          this._keyStr.charAt(enc2),\n          this._keyStr.charAt(enc3),\n          this._keyStr.charAt(enc4)\n      ].join(''));\n    }\n\n    return output.join('');\n  },\n\n  // public method for decoding\n  decode: function(input, utf8) {\n    var output = [],\n      chr1, chr2, chr3,\n      enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = input.replace(/[^A-Za-z0-9\\+\\/\\=]/g, \"\");\n\n    while (i < input.length) {\n\n      enc1 = this._keyStr.indexOf(input.charAt(i++));\n      enc2 = this._keyStr.indexOf(input.charAt(i++));\n      enc3 = this._keyStr.indexOf(input.charAt(i++));\n      enc4 = this._keyStr.indexOf(input.charAt(i++));\n\n      chr1 = (enc1 << 2) | (enc2 >> 4);\n      chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);\n      chr3 = ((enc3 & 3) << 6) | enc4;\n\n      output.push(String.fromCharCode(chr1));\n\n      if (enc3 != 64) {\n        output.push(String.fromCharCode(chr2));\n      }\n      if (enc4 != 64) {\n        output.push(String.fromCharCode(chr3));\n      }\n    }\n\n    output = output.join('');\n\n    if (utf8) {\n      output = JXG.Util.Base64._utf8_decode(output);\n    }\n    return output;\n\n  },\n\n  // private method for UTF-8 encoding\n  _utf8_encode: function(string) {\n    string = string.replace(/\\r\\n/g, \"\\n\");\n    var utftext = \"\";\n\n    for (var n = 0; n < string.length; n++) {\n\n      var c = string.charCodeAt(n);\n\n      if (c < 128) {\n        utftext += String.fromCharCode(c);\n      } else if ((c > 127) && (c < 2048)) {\n        utftext += String.fromCharCode((c >> 6) | 192);\n        utftext += String.fromCharCode((c & 63) | 128);\n      } else {\n        utftext += String.fromCharCode((c >> 12) | 224);\n        utftext += String.fromCharCode(((c >> 6) & 63) | 128);\n        utftext += String.fromCharCode((c & 63) | 128);\n      }\n\n    }\n\n    return utftext;\n  },\n\n  // private method for UTF-8 decoding\n  _utf8_decode: function(utftext) {\n    var string = [],\n      i = 0,\n      c = 0,\n      c2 = 0,\n      c3 = 0;\n\n    while (i < utftext.length) {\n      c = utftext.charCodeAt(i);\n      if (c < 128) {\n        string.push(String.fromCharCode(c));\n        i++;\n      } else if ((c > 191) && (c < 224)) {\n        c2 = utftext.charCodeAt(i + 1);\n        string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n        i += 2;\n      } else {\n        c2 = utftext.charCodeAt(i + 1);\n        c3 = utftext.charCodeAt(i + 2);\n        string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n        i += 3;\n      }\n    }\n    return string.join('');\n  },\n\n  _destrip: function(stripped, wrap) {\n    var lines = [],\n      lineno, i,\n      destripped = [];\n\n    if (wrap == null)\n      wrap = 76;\n\n    stripped.replace(/ /g, \"\");\n    lineno = stripped.length / wrap;\n    for (i = 0; i < lineno; i++)\n      lines[i] = stripped.substr(i * wrap, wrap);\n    if (lineno != stripped.length / wrap)\n      lines[lines.length] = stripped.substr(lineno * wrap, stripped.length - (lineno * wrap));\n\n    for (i = 0; i < lines.length; i++)\n      destripped.push(lines[i]);\n    return destripped.join('\\n');\n  },\n\n  decodeAsArray: function(input) {\n    var dec = this.decode(input),\n      ar = [],\n      i;\n    for (i = 0; i < dec.length; i++) {\n      ar[i] = dec.charCodeAt(i);\n    }\n    return ar;\n  },\n\n  decodeGEONExT: function(input) {\n    return decodeAsArray(destrip(input), false);\n  }\n};\n\n/**\n * @private\n */\nJXG.Util.asciiCharCodeAt = function(str, i) {\n  var c = str.charCodeAt(i);\n  if (c > 255) {\n    switch (c) {\n      case 8364:\n        c = 128;\n        break;\n      case 8218:\n        c = 130;\n        break;\n      case 402:\n        c = 131;\n        break;\n      case 8222:\n        c = 132;\n        break;\n      case 8230:\n        c = 133;\n        break;\n      case 8224:\n        c = 134;\n        break;\n      case 8225:\n        c = 135;\n        break;\n      case 710:\n        c = 136;\n        break;\n      case 8240:\n        c = 137;\n        break;\n      case 352:\n        c = 138;\n        break;\n      case 8249:\n        c = 139;\n        break;\n      case 338:\n        c = 140;\n        break;\n      case 381:\n        c = 142;\n        break;\n      case 8216:\n        c = 145;\n        break;\n      case 8217:\n        c = 146;\n        break;\n      case 8220:\n        c = 147;\n        break;\n      case 8221:\n        c = 148;\n        break;\n      case 8226:\n        c = 149;\n        break;\n      case 8211:\n        c = 150;\n        break;\n      case 8212:\n        c = 151;\n        break;\n      case 732:\n        c = 152;\n        break;\n      case 8482:\n        c = 153;\n        break;\n      case 353:\n        c = 154;\n        break;\n      case 8250:\n        c = 155;\n        break;\n      case 339:\n        c = 156;\n        break;\n      case 382:\n        c = 158;\n        break;\n      case 376:\n        c = 159;\n        break;\n      default:\n        break;\n    }\n  }\n  return c;\n};\n\n/**\n * Decoding string into utf-8\n * @param {String} string to decode\n * @return {String} utf8 decoded string\n */\nJXG.Util.utf8Decode = function(utftext) {\n  var string = [];\n  var i = 0;\n  var c = 0,\n    c1 = 0,\n    c2 = 0,\n    c3;\n  if (!JXG.exists(utftext)) return '';\n\n  while (i < utftext.length) {\n    c = utftext.charCodeAt(i);\n\n    if (c < 128) {\n      string.push(String.fromCharCode(c));\n      i++;\n    } else if ((c > 191) && (c < 224)) {\n      c2 = utftext.charCodeAt(i + 1);\n      string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n      i += 2;\n    } else {\n      c2 = utftext.charCodeAt(i + 1);\n      c3 = utftext.charCodeAt(i + 2);\n      string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n      i += 3;\n    }\n  };\n  return string.join('');\n};\n\n/**\n * Generate a random uuid.\n * http://www.broofa.com\n * mailto:robert@broofa.com\n *\n * Copyright (c) 2010 Robert Kieffer\n * Dual licensed under the MIT and GPL licenses.\n *\n * EXAMPLES:\n *   >>> Math.uuid()\n *   \"92329D39-6F5C-4520-ABFC-AAB64544E172\"\n */\nJXG.Util.genUUID = function() {\n  // Private array of chars to use\n  var chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''),\n    uuid = new Array(36),\n    rnd = 0,\n    r;\n\n  for (var i = 0; i < 36; i++) {\n    if (i == 8 || i == 13 || i == 18 || i == 23) {\n      uuid[i] = '-';\n    } else if (i == 14) {\n      uuid[i] = '4';\n    } else {\n      if (rnd <= 0x02) rnd = 0x2000000 + (Math.random() * 0x1000000) | 0;\n      r = rnd & 0xf;\n      rnd = rnd >> 4;\n      uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r];\n    }\n  }\n\n  return uuid.join('');\n};\n\n\nmodule.exports = JXG;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js');\n\n/**\n *\n * This object contains configuration values and implements\n * storing and retrieving configuration them from HTML5 local storage.\n *\n * This object can be accessed after calling openpgp.init()\n * using openpgp.config\n * Stored config parameters can be accessed using\n * openpgp.config.config\n * @class\n * @classdesc Implementation of the GPG4Browsers config object\n */\nvar config = function() {\n  /**\n   * @property {Integer} prefer_hash_algorithm\n   * @property {Integer} encryption_cipher\n   * @property {Integer} compression\n   * @property {Boolean} show_version\n   * @property {Boolean} show_comment\n   * @property {Boolean} integrity_protect\n   * @property {String} keyserver\n   */\n  this.prefer_hash_algorithm = enums.hash.sha256;\n  this.encryption_cipher = enums.symmetric.aes256;\n  this.compression = enums.compression.zip;\n  this.show_version = true;\n  this.show_comment = true;\n  this.integrity_protect = true;\n  this.keyserver = \"keyserver.linux.it\"; // \"pgp.mit.edu:11371\"\n\n  this.versionstring = \"OpenPGP.js VERSION\";\n  this.commentstring = \"http://openpgpjs.org\";\n\n  /**\n   * If enabled, debug messages will be printed\n   */\n  this.debug = false;\n\n};\n\nmodule.exports = new config();\n","// Modified by Recurity Labs GmbH \n\n// modified version of http://www.hanewin.net/encrypt/PGdecode.js:\n\n/* OpenPGP encryption using RSA/AES\n * Copyright 2005-2006 Herbert Hanewinkel, www.haneWIN.de\n * version 2.0, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../util'),\n  cipher = require('./cipher');\n\nmodule.exports = {\n\n  /**\n   * An array of bytes, that is integers with values from 0 to 255\n   * @typedef {(Array|Uint8Array)} openpgp_byte_array\n   */\n\n  /**\n   * Block cipher function\n   * @callback openpgp_cipher_block_fn\n   * @param {openpgp_byte_array} block A block to perform operations on\n   * @param {openpgp_byte_array} key to use in encryption/decryption\n   * @return {openpgp_byte_array} Encrypted/decrypted block\n   */\n\n\n  // --------------------------------------\n  /**\n   * This function encrypts a given with the specified prefixrandom \n   * using the specified blockcipher to encrypt a message\n   * @param {String} prefixrandom random bytes of block_size length provided \n   *  as a string to be used in prefixing the data\n   * @param {openpgp_cipher_block_fn} blockcipherfn the algorithm encrypt function to encrypt\n   *  data in one block_size encryption. \n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext data to be encrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to encrypt the data. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Encryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the encrypted data\n   */\n  encrypt: function(prefixrandom, cipherfn, plaintext, key, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var FR = new Array(block_size);\n    var FRE = new Array(block_size);\n\n    prefixrandom = prefixrandom + prefixrandom.charAt(block_size - 2) + prefixrandom.charAt(block_size - 1);\n    util.print_debug(\"prefixrandom:\" + util.hexstrdump(prefixrandom));\n    var ciphertext = \"\";\n    // 1.  The feedback register (FR) is set to the IV, which is all zeros.\n    for (var i = 0; i < block_size; i++) FR[i] = 0;\n\n    // 2.  FR is encrypted to produce FRE (FR Encrypted).  This is the\n    //     encryption of an all-zero value.\n    FRE = cipherfn.encrypt(FR);\n    // 3.  FRE is xored with the first BS octets of random data prefixed to\n    //     the plaintext to produce C[1] through C[BS], the first BS octets\n    //     of ciphertext.\n    for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ prefixrandom.charCodeAt(i));\n\n    // 4.  FR is loaded with C[1] through C[BS].\n    for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n\n    // 5.  FR is encrypted to produce FRE, the encryption of the first BS\n    // \t   octets of ciphertext.\n    FRE = cipherfn.encrypt(FR);\n\n    // 6.  The left two octets of FRE get xored with the next two octets of\n    //     data that were prefixed to the plaintext.  This produces C[BS+1]\n    //     and C[BS+2], the next two octets of ciphertext.\n    ciphertext += String.fromCharCode(FRE[0] ^ prefixrandom.charCodeAt(block_size));\n    ciphertext += String.fromCharCode(FRE[1] ^ prefixrandom.charCodeAt(block_size + 1));\n\n    if (resync) {\n      // 7.  (The resync step) FR is loaded with C3-C10.\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i + 2);\n    } else {\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n    }\n    // 8.  FR is encrypted to produce FRE.\n    FRE = cipherfn.encrypt(FR, key);\n\n    if (resync) {\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 0; i < block_size; i++)\n        ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      for (n = block_size + 2; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(n + i);\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        // next 8 octets of ciphertext.  These are loaded into FR and the\n        // process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt((n - 2) +\n            i));\n      }\n    } else {\n      plaintext = \"  \" + plaintext;\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 2; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      var tempCiphertext = ciphertext.substring(0, 2 * block_size).split('');\n      var tempCiphertextString = ciphertext.substring(block_size);\n      for (n = block_size; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = tempCiphertextString.charCodeAt(i);\n        tempCiphertextString = '';\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        //     next 8 octets of ciphertext.  These are loaded into FR and the\n        //     process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) {\n          tempCiphertext.push(String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i)));\n          tempCiphertextString += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i));\n        }\n      }\n      ciphertext = tempCiphertext.join('');\n\n    }\n\n    ciphertext = ciphertext.substring(0, plaintext.length + 2 + block_size);\n\n    return ciphertext;\n  },\n\n  /**\n   * Decrypts the prefixed data for the Modification Detection Code (MDC) computation\n   * @param {openpgp_block_cipher_fn} cipherfn.encrypt Cipher function to use\n   * @param {Integer} block_size Blocksize of the algorithm\n   * @param {openpgp_byte_array} key The key for encryption\n   * @param {String} ciphertext The encrypted data\n   * @return {String} plaintext Data of D(ciphertext) with blocksize length +2\n   */\n  mdc: function(cipherfn, key, ciphertext) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i;\n\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock);\n\n    return util.bin2str(iblock) +\n      String.fromCharCode(ablock[0] ^ ciphertext.charCodeAt(block_size)) +\n      String.fromCharCode(ablock[1] ^ ciphertext.charCodeAt(block_size + 1));\n  },\n  /**\n   * This function decrypts a given plaintext using the specified\n   * blockcipher to decrypt a message\n   * @param {openpgp_cipher_block_fn} blockcipherfn The algorithm _encrypt_ function to encrypt\n   *  data in one block_size encryption.\n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext ciphertext to be decrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to decrypt the ciphertext. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Decryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the plaintext data\n   */\n\n  decrypt: function(cipherfn, key, ciphertext, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i, n = '';\n    var text = [];\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock, key);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock, key);\n\n    // test check octets\n    if (iblock[block_size - 2] != (ablock[0] ^ ciphertext.charCodeAt(block_size)) || iblock[block_size - 1] != (ablock[\n      1] ^ ciphertext.charCodeAt(block_size + 1))) {\n      throw new Error('Invalid data.');\n    }\n\n    /*  RFC4880: Tag 18 and Resync:\n\t\t *  [...] Unlike the Symmetrically Encrypted Data Packet, no\n\t\t *  special CFB resynchronization is done after encrypting this prefix\n\t\t *  data.  See \"OpenPGP CFB Mode\" below for more details.\n\n\t\t */\n\n    if (resync) {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i + 2);\n      for (n = block_size + 2; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    } else {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i);\n      for (n = block_size; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    }\n\n    var n = resync ? 0 : 2;\n\n    text = text.join('');\n\n    text = text.substring(n, ciphertext.length - block_size - 2 + n);\n\n\n    return text;\n  },\n\n\n  normalEncrypt: function(cipherfn, key, plaintext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blocki = \"\";\n    var blockc = \"\";\n    var pos = 0;\n    var cyphertext = [];\n    var tempBlock = [];\n    blockc = iv.substring(0, block_size);\n    while (plaintext.length > block_size * pos) {\n      var encblock = cipherfn.encrypt(util.str2bin(blockc));\n      blocki = plaintext.substring((pos * block_size), (pos * block_size) + block_size);\n      for (var i = 0; i < blocki.length; i++)\n        tempBlock.push(String.fromCharCode(blocki.charCodeAt(i) ^ encblock[i]));\n      blockc = tempBlock.join('');\n      tempBlock = [];\n      cyphertext.push(blockc);\n      pos++;\n    }\n    return cyphertext.join('');\n  },\n\n  normalDecrypt: function(cipherfn, key, ciphertext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blockp = \"\";\n    var pos = 0;\n    var plaintext = [];\n    var offset = 0;\n    if (iv == null)\n      for (var i = 0; i < block_size; i++) blockp += String.fromCharCode(0);\n    else\n      blockp = iv.substring(0, block_size);\n    while (ciphertext.length > (block_size * pos)) {\n      var decblock = cipherfn.encrypt(util.str2bin(blockp));\n      blockp = ciphertext.substring((pos * (block_size)) + offset, (pos * (block_size)) + (block_size) + offset);\n      for (var i = 0; i < blockp.length; i++) {\n        plaintext.push(String.fromCharCode(blockp.charCodeAt(i) ^ decblock[i]));\n      }\n      pos++;\n    }\n\n    return plaintext.join('');\n  }\n}\n","/* Rijndael (AES) Encryption\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.1, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../../util');\n\n// The round constants used in subkey expansion\nvar Rcon = [\n    0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36, 0x6c, 0xd8,\n    0xab, 0x4d, 0x9a, 0x2f, 0x5e, 0xbc, 0x63, 0xc6, 0x97, 0x35, 0x6a, 0xd4,\n    0xb3, 0x7d, 0xfa, 0xef, 0xc5, 0x91\n];\n\n// Precomputed lookup table for the SBox\nvar S = [\n    99, 124, 119, 123, 242, 107, 111, 197, 48, 1, 103, 43, 254, 215, 171,\n    118, 202, 130, 201, 125, 250, 89, 71, 240, 173, 212, 162, 175, 156, 164,\n    114, 192, 183, 253, 147, 38, 54, 63, 247, 204, 52, 165, 229, 241, 113,\n    216, 49, 21, 4, 199, 35, 195, 24, 150, 5, 154, 7, 18, 128, 226,\n    235, 39, 178, 117, 9, 131, 44, 26, 27, 110, 90, 160, 82, 59, 214,\n    179, 41, 227, 47, 132, 83, 209, 0, 237, 32, 252, 177, 91, 106, 203,\n    190, 57, 74, 76, 88, 207, 208, 239, 170, 251, 67, 77, 51, 133, 69,\n    249, 2, 127, 80, 60, 159, 168, 81, 163, 64, 143, 146, 157, 56, 245,\n    188, 182, 218, 33, 16, 255, 243, 210, 205, 12, 19, 236, 95, 151, 68,\n    23, 196, 167, 126, 61, 100, 93, 25, 115, 96, 129, 79, 220, 34, 42,\n    144, 136, 70, 238, 184, 20, 222, 94, 11, 219, 224, 50, 58, 10, 73,\n    6, 36, 92, 194, 211, 172, 98, 145, 149, 228, 121, 231, 200, 55, 109,\n    141, 213, 78, 169, 108, 86, 244, 234, 101, 122, 174, 8, 186, 120, 37,\n    46, 28, 166, 180, 198, 232, 221, 116, 31, 75, 189, 139, 138, 112, 62,\n    181, 102, 72, 3, 246, 14, 97, 53, 87, 185, 134, 193, 29, 158, 225,\n    248, 152, 17, 105, 217, 142, 148, 155, 30, 135, 233, 206, 85, 40, 223,\n    140, 161, 137, 13, 191, 230, 66, 104, 65, 153, 45, 15, 176, 84, 187,\n    22\n];\n\nvar T1 = [\n    0xa56363c6, 0x847c7cf8, 0x997777ee, 0x8d7b7bf6,\n    0x0df2f2ff, 0xbd6b6bd6, 0xb16f6fde, 0x54c5c591,\n    0x50303060, 0x03010102, 0xa96767ce, 0x7d2b2b56,\n    0x19fefee7, 0x62d7d7b5, 0xe6abab4d, 0x9a7676ec,\n    0x45caca8f, 0x9d82821f, 0x40c9c989, 0x877d7dfa,\n    0x15fafaef, 0xeb5959b2, 0xc947478e, 0x0bf0f0fb,\n    0xecadad41, 0x67d4d4b3, 0xfda2a25f, 0xeaafaf45,\n    0xbf9c9c23, 0xf7a4a453, 0x967272e4, 0x5bc0c09b,\n    0xc2b7b775, 0x1cfdfde1, 0xae93933d, 0x6a26264c,\n    0x5a36366c, 0x413f3f7e, 0x02f7f7f5, 0x4fcccc83,\n    0x5c343468, 0xf4a5a551, 0x34e5e5d1, 0x08f1f1f9,\n    0x937171e2, 0x73d8d8ab, 0x53313162, 0x3f15152a,\n    0x0c040408, 0x52c7c795, 0x65232346, 0x5ec3c39d,\n    0x28181830, 0xa1969637, 0x0f05050a, 0xb59a9a2f,\n    0x0907070e, 0x36121224, 0x9b80801b, 0x3de2e2df,\n    0x26ebebcd, 0x6927274e, 0xcdb2b27f, 0x9f7575ea,\n    0x1b090912, 0x9e83831d, 0x742c2c58, 0x2e1a1a34,\n    0x2d1b1b36, 0xb26e6edc, 0xee5a5ab4, 0xfba0a05b,\n    0xf65252a4, 0x4d3b3b76, 0x61d6d6b7, 0xceb3b37d,\n    0x7b292952, 0x3ee3e3dd, 0x712f2f5e, 0x97848413,\n    0xf55353a6, 0x68d1d1b9, 0x00000000, 0x2cededc1,\n    0x60202040, 0x1ffcfce3, 0xc8b1b179, 0xed5b5bb6,\n    0xbe6a6ad4, 0x46cbcb8d, 0xd9bebe67, 0x4b393972,\n    0xde4a4a94, 0xd44c4c98, 0xe85858b0, 0x4acfcf85,\n    0x6bd0d0bb, 0x2aefefc5, 0xe5aaaa4f, 0x16fbfbed,\n    0xc5434386, 0xd74d4d9a, 0x55333366, 0x94858511,\n    0xcf45458a, 0x10f9f9e9, 0x06020204, 0x817f7ffe,\n    0xf05050a0, 0x443c3c78, 0xba9f9f25, 0xe3a8a84b,\n    0xf35151a2, 0xfea3a35d, 0xc0404080, 0x8a8f8f05,\n    0xad92923f, 0xbc9d9d21, 0x48383870, 0x04f5f5f1,\n    0xdfbcbc63, 0xc1b6b677, 0x75dadaaf, 0x63212142,\n    0x30101020, 0x1affffe5, 0x0ef3f3fd, 0x6dd2d2bf,\n    0x4ccdcd81, 0x140c0c18, 0x35131326, 0x2fececc3,\n    0xe15f5fbe, 0xa2979735, 0xcc444488, 0x3917172e,\n    0x57c4c493, 0xf2a7a755, 0x827e7efc, 0x473d3d7a,\n    0xac6464c8, 0xe75d5dba, 0x2b191932, 0x957373e6,\n    0xa06060c0, 0x98818119, 0xd14f4f9e, 0x7fdcdca3,\n    0x66222244, 0x7e2a2a54, 0xab90903b, 0x8388880b,\n    0xca46468c, 0x29eeeec7, 0xd3b8b86b, 0x3c141428,\n    0x79dedea7, 0xe25e5ebc, 0x1d0b0b16, 0x76dbdbad,\n    0x3be0e0db, 0x56323264, 0x4e3a3a74, 0x1e0a0a14,\n    0xdb494992, 0x0a06060c, 0x6c242448, 0xe45c5cb8,\n    0x5dc2c29f, 0x6ed3d3bd, 0xefacac43, 0xa66262c4,\n    0xa8919139, 0xa4959531, 0x37e4e4d3, 0x8b7979f2,\n    0x32e7e7d5, 0x43c8c88b, 0x5937376e, 0xb76d6dda,\n    0x8c8d8d01, 0x64d5d5b1, 0xd24e4e9c, 0xe0a9a949,\n    0xb46c6cd8, 0xfa5656ac, 0x07f4f4f3, 0x25eaeacf,\n    0xaf6565ca, 0x8e7a7af4, 0xe9aeae47, 0x18080810,\n    0xd5baba6f, 0x887878f0, 0x6f25254a, 0x722e2e5c,\n    0x241c1c38, 0xf1a6a657, 0xc7b4b473, 0x51c6c697,\n    0x23e8e8cb, 0x7cdddda1, 0x9c7474e8, 0x211f1f3e,\n    0xdd4b4b96, 0xdcbdbd61, 0x868b8b0d, 0x858a8a0f,\n    0x907070e0, 0x423e3e7c, 0xc4b5b571, 0xaa6666cc,\n    0xd8484890, 0x05030306, 0x01f6f6f7, 0x120e0e1c,\n    0xa36161c2, 0x5f35356a, 0xf95757ae, 0xd0b9b969,\n    0x91868617, 0x58c1c199, 0x271d1d3a, 0xb99e9e27,\n    0x38e1e1d9, 0x13f8f8eb, 0xb398982b, 0x33111122,\n    0xbb6969d2, 0x70d9d9a9, 0x898e8e07, 0xa7949433,\n    0xb69b9b2d, 0x221e1e3c, 0x92878715, 0x20e9e9c9,\n    0x49cece87, 0xff5555aa, 0x78282850, 0x7adfdfa5,\n    0x8f8c8c03, 0xf8a1a159, 0x80898909, 0x170d0d1a,\n    0xdabfbf65, 0x31e6e6d7, 0xc6424284, 0xb86868d0,\n    0xc3414182, 0xb0999929, 0x772d2d5a, 0x110f0f1e,\n    0xcbb0b07b, 0xfc5454a8, 0xd6bbbb6d, 0x3a16162c\n];\n\nvar T2 = [\n    0x6363c6a5, 0x7c7cf884, 0x7777ee99, 0x7b7bf68d,\n    0xf2f2ff0d, 0x6b6bd6bd, 0x6f6fdeb1, 0xc5c59154,\n    0x30306050, 0x01010203, 0x6767cea9, 0x2b2b567d,\n    0xfefee719, 0xd7d7b562, 0xabab4de6, 0x7676ec9a,\n    0xcaca8f45, 0x82821f9d, 0xc9c98940, 0x7d7dfa87,\n    0xfafaef15, 0x5959b2eb, 0x47478ec9, 0xf0f0fb0b,\n    0xadad41ec, 0xd4d4b367, 0xa2a25ffd, 0xafaf45ea,\n    0x9c9c23bf, 0xa4a453f7, 0x7272e496, 0xc0c09b5b,\n    0xb7b775c2, 0xfdfde11c, 0x93933dae, 0x26264c6a,\n    0x36366c5a, 0x3f3f7e41, 0xf7f7f502, 0xcccc834f,\n    0x3434685c, 0xa5a551f4, 0xe5e5d134, 0xf1f1f908,\n    0x7171e293, 0xd8d8ab73, 0x31316253, 0x15152a3f,\n    0x0404080c, 0xc7c79552, 0x23234665, 0xc3c39d5e,\n    0x18183028, 0x969637a1, 0x05050a0f, 0x9a9a2fb5,\n    0x07070e09, 0x12122436, 0x80801b9b, 0xe2e2df3d,\n    0xebebcd26, 0x27274e69, 0xb2b27fcd, 0x7575ea9f,\n    0x0909121b, 0x83831d9e, 0x2c2c5874, 0x1a1a342e,\n    0x1b1b362d, 0x6e6edcb2, 0x5a5ab4ee, 0xa0a05bfb,\n    0x5252a4f6, 0x3b3b764d, 0xd6d6b761, 0xb3b37dce,\n    0x2929527b, 0xe3e3dd3e, 0x2f2f5e71, 0x84841397,\n    0x5353a6f5, 0xd1d1b968, 0x00000000, 0xededc12c,\n    0x20204060, 0xfcfce31f, 0xb1b179c8, 0x5b5bb6ed,\n    0x6a6ad4be, 0xcbcb8d46, 0xbebe67d9, 0x3939724b,\n    0x4a4a94de, 0x4c4c98d4, 0x5858b0e8, 0xcfcf854a,\n    0xd0d0bb6b, 0xefefc52a, 0xaaaa4fe5, 0xfbfbed16,\n    0x434386c5, 0x4d4d9ad7, 0x33336655, 0x85851194,\n    0x45458acf, 0xf9f9e910, 0x02020406, 0x7f7ffe81,\n    0x5050a0f0, 0x3c3c7844, 0x9f9f25ba, 0xa8a84be3,\n    0x5151a2f3, 0xa3a35dfe, 0x404080c0, 0x8f8f058a,\n    0x92923fad, 0x9d9d21bc, 0x38387048, 0xf5f5f104,\n    0xbcbc63df, 0xb6b677c1, 0xdadaaf75, 0x21214263,\n    0x10102030, 0xffffe51a, 0xf3f3fd0e, 0xd2d2bf6d,\n    0xcdcd814c, 0x0c0c1814, 0x13132635, 0xececc32f,\n    0x5f5fbee1, 0x979735a2, 0x444488cc, 0x17172e39,\n    0xc4c49357, 0xa7a755f2, 0x7e7efc82, 0x3d3d7a47,\n    0x6464c8ac, 0x5d5dbae7, 0x1919322b, 0x7373e695,\n    0x6060c0a0, 0x81811998, 0x4f4f9ed1, 0xdcdca37f,\n    0x22224466, 0x2a2a547e, 0x90903bab, 0x88880b83,\n    0x46468cca, 0xeeeec729, 0xb8b86bd3, 0x1414283c,\n    0xdedea779, 0x5e5ebce2, 0x0b0b161d, 0xdbdbad76,\n    0xe0e0db3b, 0x32326456, 0x3a3a744e, 0x0a0a141e,\n    0x494992db, 0x06060c0a, 0x2424486c, 0x5c5cb8e4,\n    0xc2c29f5d, 0xd3d3bd6e, 0xacac43ef, 0x6262c4a6,\n    0x919139a8, 0x959531a4, 0xe4e4d337, 0x7979f28b,\n    0xe7e7d532, 0xc8c88b43, 0x37376e59, 0x6d6ddab7,\n    0x8d8d018c, 0xd5d5b164, 0x4e4e9cd2, 0xa9a949e0,\n    0x6c6cd8b4, 0x5656acfa, 0xf4f4f307, 0xeaeacf25,\n    0x6565caaf, 0x7a7af48e, 0xaeae47e9, 0x08081018,\n    0xbaba6fd5, 0x7878f088, 0x25254a6f, 0x2e2e5c72,\n    0x1c1c3824, 0xa6a657f1, 0xb4b473c7, 0xc6c69751,\n    0xe8e8cb23, 0xdddda17c, 0x7474e89c, 0x1f1f3e21,\n    0x4b4b96dd, 0xbdbd61dc, 0x8b8b0d86, 0x8a8a0f85,\n    0x7070e090, 0x3e3e7c42, 0xb5b571c4, 0x6666ccaa,\n    0x484890d8, 0x03030605, 0xf6f6f701, 0x0e0e1c12,\n    0x6161c2a3, 0x35356a5f, 0x5757aef9, 0xb9b969d0,\n    0x86861791, 0xc1c19958, 0x1d1d3a27, 0x9e9e27b9,\n    0xe1e1d938, 0xf8f8eb13, 0x98982bb3, 0x11112233,\n    0x6969d2bb, 0xd9d9a970, 0x8e8e0789, 0x949433a7,\n    0x9b9b2db6, 0x1e1e3c22, 0x87871592, 0xe9e9c920,\n    0xcece8749, 0x5555aaff, 0x28285078, 0xdfdfa57a,\n    0x8c8c038f, 0xa1a159f8, 0x89890980, 0x0d0d1a17,\n    0xbfbf65da, 0xe6e6d731, 0x424284c6, 0x6868d0b8,\n    0x414182c3, 0x999929b0, 0x2d2d5a77, 0x0f0f1e11,\n    0xb0b07bcb, 0x5454a8fc, 0xbbbb6dd6, 0x16162c3a\n];\n\nvar T3 = [\n    0x63c6a563, 0x7cf8847c, 0x77ee9977, 0x7bf68d7b,\n    0xf2ff0df2, 0x6bd6bd6b, 0x6fdeb16f, 0xc59154c5,\n    0x30605030, 0x01020301, 0x67cea967, 0x2b567d2b,\n    0xfee719fe, 0xd7b562d7, 0xab4de6ab, 0x76ec9a76,\n    0xca8f45ca, 0x821f9d82, 0xc98940c9, 0x7dfa877d,\n    0xfaef15fa, 0x59b2eb59, 0x478ec947, 0xf0fb0bf0,\n    0xad41ecad, 0xd4b367d4, 0xa25ffda2, 0xaf45eaaf,\n    0x9c23bf9c, 0xa453f7a4, 0x72e49672, 0xc09b5bc0,\n    0xb775c2b7, 0xfde11cfd, 0x933dae93, 0x264c6a26,\n    0x366c5a36, 0x3f7e413f, 0xf7f502f7, 0xcc834fcc,\n    0x34685c34, 0xa551f4a5, 0xe5d134e5, 0xf1f908f1,\n    0x71e29371, 0xd8ab73d8, 0x31625331, 0x152a3f15,\n    0x04080c04, 0xc79552c7, 0x23466523, 0xc39d5ec3,\n    0x18302818, 0x9637a196, 0x050a0f05, 0x9a2fb59a,\n    0x070e0907, 0x12243612, 0x801b9b80, 0xe2df3de2,\n    0xebcd26eb, 0x274e6927, 0xb27fcdb2, 0x75ea9f75,\n    0x09121b09, 0x831d9e83, 0x2c58742c, 0x1a342e1a,\n    0x1b362d1b, 0x6edcb26e, 0x5ab4ee5a, 0xa05bfba0,\n    0x52a4f652, 0x3b764d3b, 0xd6b761d6, 0xb37dceb3,\n    0x29527b29, 0xe3dd3ee3, 0x2f5e712f, 0x84139784,\n    0x53a6f553, 0xd1b968d1, 0x00000000, 0xedc12ced,\n    0x20406020, 0xfce31ffc, 0xb179c8b1, 0x5bb6ed5b,\n    0x6ad4be6a, 0xcb8d46cb, 0xbe67d9be, 0x39724b39,\n    0x4a94de4a, 0x4c98d44c, 0x58b0e858, 0xcf854acf,\n    0xd0bb6bd0, 0xefc52aef, 0xaa4fe5aa, 0xfbed16fb,\n    0x4386c543, 0x4d9ad74d, 0x33665533, 0x85119485,\n    0x458acf45, 0xf9e910f9, 0x02040602, 0x7ffe817f,\n    0x50a0f050, 0x3c78443c, 0x9f25ba9f, 0xa84be3a8,\n    0x51a2f351, 0xa35dfea3, 0x4080c040, 0x8f058a8f,\n    0x923fad92, 0x9d21bc9d, 0x38704838, 0xf5f104f5,\n    0xbc63dfbc, 0xb677c1b6, 0xdaaf75da, 0x21426321,\n    0x10203010, 0xffe51aff, 0xf3fd0ef3, 0xd2bf6dd2,\n    0xcd814ccd, 0x0c18140c, 0x13263513, 0xecc32fec,\n    0x5fbee15f, 0x9735a297, 0x4488cc44, 0x172e3917,\n    0xc49357c4, 0xa755f2a7, 0x7efc827e, 0x3d7a473d,\n    0x64c8ac64, 0x5dbae75d, 0x19322b19, 0x73e69573,\n    0x60c0a060, 0x81199881, 0x4f9ed14f, 0xdca37fdc,\n    0x22446622, 0x2a547e2a, 0x903bab90, 0x880b8388,\n    0x468cca46, 0xeec729ee, 0xb86bd3b8, 0x14283c14,\n    0xdea779de, 0x5ebce25e, 0x0b161d0b, 0xdbad76db,\n    0xe0db3be0, 0x32645632, 0x3a744e3a, 0x0a141e0a,\n    0x4992db49, 0x060c0a06, 0x24486c24, 0x5cb8e45c,\n    0xc29f5dc2, 0xd3bd6ed3, 0xac43efac, 0x62c4a662,\n    0x9139a891, 0x9531a495, 0xe4d337e4, 0x79f28b79,\n    0xe7d532e7, 0xc88b43c8, 0x376e5937, 0x6ddab76d,\n    0x8d018c8d, 0xd5b164d5, 0x4e9cd24e, 0xa949e0a9,\n    0x6cd8b46c, 0x56acfa56, 0xf4f307f4, 0xeacf25ea,\n    0x65caaf65, 0x7af48e7a, 0xae47e9ae, 0x08101808,\n    0xba6fd5ba, 0x78f08878, 0x254a6f25, 0x2e5c722e,\n    0x1c38241c, 0xa657f1a6, 0xb473c7b4, 0xc69751c6,\n    0xe8cb23e8, 0xdda17cdd, 0x74e89c74, 0x1f3e211f,\n    0x4b96dd4b, 0xbd61dcbd, 0x8b0d868b, 0x8a0f858a,\n    0x70e09070, 0x3e7c423e, 0xb571c4b5, 0x66ccaa66,\n    0x4890d848, 0x03060503, 0xf6f701f6, 0x0e1c120e,\n    0x61c2a361, 0x356a5f35, 0x57aef957, 0xb969d0b9,\n    0x86179186, 0xc19958c1, 0x1d3a271d, 0x9e27b99e,\n    0xe1d938e1, 0xf8eb13f8, 0x982bb398, 0x11223311,\n    0x69d2bb69, 0xd9a970d9, 0x8e07898e, 0x9433a794,\n    0x9b2db69b, 0x1e3c221e, 0x87159287, 0xe9c920e9,\n    0xce8749ce, 0x55aaff55, 0x28507828, 0xdfa57adf,\n    0x8c038f8c, 0xa159f8a1, 0x89098089, 0x0d1a170d,\n    0xbf65dabf, 0xe6d731e6, 0x4284c642, 0x68d0b868,\n    0x4182c341, 0x9929b099, 0x2d5a772d, 0x0f1e110f,\n    0xb07bcbb0, 0x54a8fc54, 0xbb6dd6bb, 0x162c3a16\n];\n\nvar T4 = [\n    0xc6a56363, 0xf8847c7c, 0xee997777, 0xf68d7b7b,\n    0xff0df2f2, 0xd6bd6b6b, 0xdeb16f6f, 0x9154c5c5,\n    0x60503030, 0x02030101, 0xcea96767, 0x567d2b2b,\n    0xe719fefe, 0xb562d7d7, 0x4de6abab, 0xec9a7676,\n    0x8f45caca, 0x1f9d8282, 0x8940c9c9, 0xfa877d7d,\n    0xef15fafa, 0xb2eb5959, 0x8ec94747, 0xfb0bf0f0,\n    0x41ecadad, 0xb367d4d4, 0x5ffda2a2, 0x45eaafaf,\n    0x23bf9c9c, 0x53f7a4a4, 0xe4967272, 0x9b5bc0c0,\n    0x75c2b7b7, 0xe11cfdfd, 0x3dae9393, 0x4c6a2626,\n    0x6c5a3636, 0x7e413f3f, 0xf502f7f7, 0x834fcccc,\n    0x685c3434, 0x51f4a5a5, 0xd134e5e5, 0xf908f1f1,\n    0xe2937171, 0xab73d8d8, 0x62533131, 0x2a3f1515,\n    0x080c0404, 0x9552c7c7, 0x46652323, 0x9d5ec3c3,\n    0x30281818, 0x37a19696, 0x0a0f0505, 0x2fb59a9a,\n    0x0e090707, 0x24361212, 0x1b9b8080, 0xdf3de2e2,\n    0xcd26ebeb, 0x4e692727, 0x7fcdb2b2, 0xea9f7575,\n    0x121b0909, 0x1d9e8383, 0x58742c2c, 0x342e1a1a,\n    0x362d1b1b, 0xdcb26e6e, 0xb4ee5a5a, 0x5bfba0a0,\n    0xa4f65252, 0x764d3b3b, 0xb761d6d6, 0x7dceb3b3,\n    0x527b2929, 0xdd3ee3e3, 0x5e712f2f, 0x13978484,\n    0xa6f55353, 0xb968d1d1, 0x00000000, 0xc12ceded,\n    0x40602020, 0xe31ffcfc, 0x79c8b1b1, 0xb6ed5b5b,\n    0xd4be6a6a, 0x8d46cbcb, 0x67d9bebe, 0x724b3939,\n    0x94de4a4a, 0x98d44c4c, 0xb0e85858, 0x854acfcf,\n    0xbb6bd0d0, 0xc52aefef, 0x4fe5aaaa, 0xed16fbfb,\n    0x86c54343, 0x9ad74d4d, 0x66553333, 0x11948585,\n    0x8acf4545, 0xe910f9f9, 0x04060202, 0xfe817f7f,\n    0xa0f05050, 0x78443c3c, 0x25ba9f9f, 0x4be3a8a8,\n    0xa2f35151, 0x5dfea3a3, 0x80c04040, 0x058a8f8f,\n    0x3fad9292, 0x21bc9d9d, 0x70483838, 0xf104f5f5,\n    0x63dfbcbc, 0x77c1b6b6, 0xaf75dada, 0x42632121,\n    0x20301010, 0xe51affff, 0xfd0ef3f3, 0xbf6dd2d2,\n    0x814ccdcd, 0x18140c0c, 0x26351313, 0xc32fecec,\n    0xbee15f5f, 0x35a29797, 0x88cc4444, 0x2e391717,\n    0x9357c4c4, 0x55f2a7a7, 0xfc827e7e, 0x7a473d3d,\n    0xc8ac6464, 0xbae75d5d, 0x322b1919, 0xe6957373,\n    0xc0a06060, 0x19988181, 0x9ed14f4f, 0xa37fdcdc,\n    0x44662222, 0x547e2a2a, 0x3bab9090, 0x0b838888,\n    0x8cca4646, 0xc729eeee, 0x6bd3b8b8, 0x283c1414,\n    0xa779dede, 0xbce25e5e, 0x161d0b0b, 0xad76dbdb,\n    0xdb3be0e0, 0x64563232, 0x744e3a3a, 0x141e0a0a,\n    0x92db4949, 0x0c0a0606, 0x486c2424, 0xb8e45c5c,\n    0x9f5dc2c2, 0xbd6ed3d3, 0x43efacac, 0xc4a66262,\n    0x39a89191, 0x31a49595, 0xd337e4e4, 0xf28b7979,\n    0xd532e7e7, 0x8b43c8c8, 0x6e593737, 0xdab76d6d,\n    0x018c8d8d, 0xb164d5d5, 0x9cd24e4e, 0x49e0a9a9,\n    0xd8b46c6c, 0xacfa5656, 0xf307f4f4, 0xcf25eaea,\n    0xcaaf6565, 0xf48e7a7a, 0x47e9aeae, 0x10180808,\n    0x6fd5baba, 0xf0887878, 0x4a6f2525, 0x5c722e2e,\n    0x38241c1c, 0x57f1a6a6, 0x73c7b4b4, 0x9751c6c6,\n    0xcb23e8e8, 0xa17cdddd, 0xe89c7474, 0x3e211f1f,\n    0x96dd4b4b, 0x61dcbdbd, 0x0d868b8b, 0x0f858a8a,\n    0xe0907070, 0x7c423e3e, 0x71c4b5b5, 0xccaa6666,\n    0x90d84848, 0x06050303, 0xf701f6f6, 0x1c120e0e,\n    0xc2a36161, 0x6a5f3535, 0xaef95757, 0x69d0b9b9,\n    0x17918686, 0x9958c1c1, 0x3a271d1d, 0x27b99e9e,\n    0xd938e1e1, 0xeb13f8f8, 0x2bb39898, 0x22331111,\n    0xd2bb6969, 0xa970d9d9, 0x07898e8e, 0x33a79494,\n    0x2db69b9b, 0x3c221e1e, 0x15928787, 0xc920e9e9,\n    0x8749cece, 0xaaff5555, 0x50782828, 0xa57adfdf,\n    0x038f8c8c, 0x59f8a1a1, 0x09808989, 0x1a170d0d,\n    0x65dabfbf, 0xd731e6e6, 0x84c64242, 0xd0b86868,\n    0x82c34141, 0x29b09999, 0x5a772d2d, 0x1e110f0f,\n    0x7bcbb0b0, 0xa8fc5454, 0x6dd6bbbb, 0x2c3a1616\n];\n\nfunction B0(x) {\n  return (x & 255);\n}\n\nfunction B1(x) {\n  return ((x >> 8) & 255);\n}\n\nfunction B2(x) {\n  return ((x >> 16) & 255);\n}\n\nfunction B3(x) {\n  return ((x >> 24) & 255);\n}\n\nfunction F1(x0, x1, x2, x3) {\n  return B1(T1[x0 & 255]) | (B1(T1[(x1 >> 8) & 255]) << 8) | (B1(T1[(x2 >> 16) & 255]) << 16) | (B1(T1[x3 >>> 24]) <<\n    24);\n}\n\nfunction packBytes(octets) {\n  var i, j;\n  var len = octets.length;\n  var b = new Array(len / 4);\n\n  if (!octets || len % 4) return;\n\n  for (i = 0, j = 0; j < len; j += 4)\n    b[i++] = octets[j] | (octets[j + 1] << 8) | (octets[j + 2] << 16) | (octets[j + 3] << 24);\n\n  return b;\n}\n\nfunction unpackBytes(packed) {\n  var j;\n  var i = 0,\n    l = packed.length;\n  var r = new Array(l * 4);\n\n  for (j = 0; j < l; j++) {\n    r[i++] = B0(packed[j]);\n    r[i++] = B1(packed[j]);\n    r[i++] = B2(packed[j]);\n    r[i++] = B3(packed[j]);\n  }\n  return r;\n}\n\n// ------------------------------------------------\n\nvar maxkc = 8;\nvar maxrk = 14;\n\nfunction keyExpansion(key) {\n  var kc, i, j, r, t;\n  var rounds;\n  var keySched = new Array(maxrk + 1);\n  var keylen = key.length;\n  var k = new Array(maxkc);\n  var tk = new Array(maxkc);\n  var rconpointer = 0;\n\n  if (keylen == 16) {\n    rounds = 10;\n    kc = 4;\n  } else if (keylen == 24) {\n    rounds = 12;\n    kc = 6;\n  } else if (keylen == 32) {\n    rounds = 14;\n    kc = 8;\n  } else {\n    util.print_error('aes.js: Invalid key-length for AES key:' + keylen);\n    return;\n  }\n\n  for (i = 0; i < maxrk + 1; i++) keySched[i] = new Array(4);\n\n  for (i = 0, j = 0; j < keylen; j++, i += 4)\n    k[j] = key.charCodeAt(i) | (key.charCodeAt(i + 1) << 8) | (key.charCodeAt(i + 2) << 16) | (key.charCodeAt(i + 3) <<\n      24);\n\n  for (j = kc - 1; j >= 0; j--) tk[j] = k[j];\n\n  r = 0;\n  t = 0;\n  for (j = 0;\n  (j < kc) && (r < rounds + 1);) {\n    for (;\n    (j < kc) && (t < 4); j++, t++) {\n      keySched[r][t] = tk[j];\n    }\n    if (t == 4) {\n      r++;\n      t = 0;\n    }\n  }\n\n  while (r < rounds + 1) {\n    var temp = tk[kc - 1];\n\n    tk[0] ^= S[B1(temp)] | (S[B2(temp)] << 8) | (S[B3(temp)] << 16) | (S[B0(temp)] << 24);\n    tk[0] ^= Rcon[rconpointer++];\n\n    if (kc != 8) {\n      for (j = 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    } else {\n      for (j = 1; j < kc / 2; j++) tk[j] ^= tk[j - 1];\n\n      temp = tk[kc / 2 - 1];\n      tk[kc / 2] ^= S[B0(temp)] | (S[B1(temp)] << 8) | (S[B2(temp)] << 16) | (S[B3(temp)] << 24);\n\n      for (j = kc / 2 + 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    }\n\n    for (j = 0;\n    (j < kc) && (r < rounds + 1);) {\n      for (;\n      (j < kc) && (t < 4); j++, t++) {\n        keySched[r][t] = tk[j];\n      }\n      if (t == 4) {\n        r++;\n        t = 0;\n      }\n    }\n  }\n  this.rounds = rounds;\n  this.rk = keySched;\n  return this;\n}\n\nfunction AESencrypt(block, ctx) {\n  var r;\n  var t0, t1, t2, t3;\n\n  var b = packBytes(block);\n  var rounds = ctx.rounds;\n  var b0 = b[0];\n  var b1 = b[1];\n  var b2 = b[2];\n  var b3 = b[3];\n\n  for (r = 0; r < rounds - 1; r++) {\n    t0 = b0 ^ ctx.rk[r][0];\n    t1 = b1 ^ ctx.rk[r][1];\n    t2 = b2 ^ ctx.rk[r][2];\n    t3 = b3 ^ ctx.rk[r][3];\n\n    b0 = T1[t0 & 255] ^ T2[(t1 >> 8) & 255] ^ T3[(t2 >> 16) & 255] ^ T4[t3 >>> 24];\n    b1 = T1[t1 & 255] ^ T2[(t2 >> 8) & 255] ^ T3[(t3 >> 16) & 255] ^ T4[t0 >>> 24];\n    b2 = T1[t2 & 255] ^ T2[(t3 >> 8) & 255] ^ T3[(t0 >> 16) & 255] ^ T4[t1 >>> 24];\n    b3 = T1[t3 & 255] ^ T2[(t0 >> 8) & 255] ^ T3[(t1 >> 16) & 255] ^ T4[t2 >>> 24];\n  }\n\n  // last round is special\n  r = rounds - 1;\n\n  t0 = b0 ^ ctx.rk[r][0];\n  t1 = b1 ^ ctx.rk[r][1];\n  t2 = b2 ^ ctx.rk[r][2];\n  t3 = b3 ^ ctx.rk[r][3];\n\n  b[0] = F1(t0, t1, t2, t3) ^ ctx.rk[rounds][0];\n  b[1] = F1(t1, t2, t3, t0) ^ ctx.rk[rounds][1];\n  b[2] = F1(t2, t3, t0, t1) ^ ctx.rk[rounds][2];\n  b[3] = F1(t3, t0, t1, t2) ^ ctx.rk[rounds][3];\n\n  return unpackBytes(b);\n}\n\nfunction makeClass(length) {\n\n  var c = function(key) {\n    this.key = keyExpansion(key);\n\n    this.encrypt = function(block) {\n      return AESencrypt(block, this.key);\n    }\n  }\n\n  c.blockSize = c.prototype.blockSize = 16;\n  c.keySize = c.prototype.keySize = length / 8;\n\n  return c;\n}\n\nmodule.exports = {}\n\nvar types = [128, 192, 256];\n\nfor (var i in types) {\n  module.exports[types[i]] = makeClass(types[i]);\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Originally written by nklein software (nklein.com)\n */\n\n/* \n * Javascript implementation based on Bruce Schneier's reference implementation.\n *\n *\n * The constructor doesn't do much of anything.  It's just here\n * so we can start defining properties and methods and such.\n */\nfunction Blowfish() {};\n\n/*\n * Declare the block size so that protocols know what size\n * Initialization Vector (IV) they will need.\n */\nBlowfish.prototype.BLOCKSIZE = 8;\n\n/*\n * These are the default SBOXES.\n */\nBlowfish.prototype.SBOXES = [\n  [\n      0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96,\n      0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16,\n      0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658,\n      0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013,\n      0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e,\n      0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60,\n      0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6,\n      0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a,\n      0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c,\n      0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193,\n      0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1,\n      0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239,\n      0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a,\n      0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3,\n      0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176,\n      0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe,\n      0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706,\n      0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b,\n      0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b,\n      0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463,\n      0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c,\n      0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3,\n      0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a,\n      0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8,\n      0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760,\n      0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db,\n      0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8,\n      0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b,\n      0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33,\n      0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4,\n      0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0,\n      0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c,\n      0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777,\n      0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299,\n      0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705,\n      0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf,\n      0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e,\n      0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa,\n      0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9,\n      0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915,\n      0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f,\n      0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664,\n      0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a\n  ],\n  [\n      0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d,\n      0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1,\n      0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65,\n      0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1,\n      0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9,\n      0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737,\n      0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d,\n      0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd,\n      0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc,\n      0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41,\n      0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908,\n      0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af,\n      0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124,\n      0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c,\n      0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908,\n      0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd,\n      0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b,\n      0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e,\n      0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa,\n      0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a,\n      0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d,\n      0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66,\n      0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5,\n      0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84,\n      0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96,\n      0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14,\n      0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca,\n      0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7,\n      0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77,\n      0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99,\n      0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054,\n      0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73,\n      0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea,\n      0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105,\n      0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646,\n      0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285,\n      0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea,\n      0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb,\n      0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e,\n      0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc,\n      0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd,\n      0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20,\n      0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7\n  ],\n  [\n      0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7,\n      0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af,\n      0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af,\n      0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504,\n      0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4,\n      0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee,\n      0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec,\n      0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b,\n      0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332,\n      0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527,\n      0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58,\n      0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c,\n      0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22,\n      0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17,\n      0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60,\n      0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115,\n      0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99,\n      0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0,\n      0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74,\n      0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d,\n      0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3,\n      0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3,\n      0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979,\n      0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c,\n      0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa,\n      0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a,\n      0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086,\n      0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc,\n      0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24,\n      0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2,\n      0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84,\n      0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c,\n      0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09,\n      0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10,\n      0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe,\n      0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027,\n      0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0,\n      0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634,\n      0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188,\n      0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc,\n      0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8,\n      0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837,\n      0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0\n  ],\n  [\n      0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742,\n      0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b,\n      0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79,\n      0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6,\n      0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a,\n      0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4,\n      0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1,\n      0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59,\n      0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797,\n      0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28,\n      0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6,\n      0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28,\n      0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba,\n      0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a,\n      0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5,\n      0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f,\n      0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce,\n      0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680,\n      0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd,\n      0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb,\n      0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb,\n      0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370,\n      0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc,\n      0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048,\n      0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc,\n      0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9,\n      0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a,\n      0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f,\n      0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a,\n      0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1,\n      0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b,\n      0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e,\n      0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e,\n      0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f,\n      0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623,\n      0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc,\n      0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a,\n      0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6,\n      0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3,\n      0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060,\n      0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c,\n      0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f,\n      0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6\n  ]\n];\n\n//*\n//* This is the default PARRAY\n//*\nBlowfish.prototype.PARRAY = [\n    0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0,\n    0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c,\n    0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b\n];\n\n//*\n//* This is the number of rounds the cipher will go\n//*\nBlowfish.prototype.NN = 16;\n\n//*\n//* This function is needed to get rid of problems\n//* with the high-bit getting set.  If we don't do\n//* this, then sometimes ( aa & 0x00FFFFFFFF ) is not\n//* equal to ( bb & 0x00FFFFFFFF ) even when they\n//* agree bit-for-bit for the first 32 bits.\n//*\nBlowfish.prototype._clean = function(xx) {\n  if (xx < 0) {\n    var yy = xx & 0x7FFFFFFF;\n    xx = yy + 0x80000000;\n  }\n  return xx;\n};\n\n//*\n//* This is the mixing function that uses the sboxes\n//*\nBlowfish.prototype._F = function(xx) {\n  var aa;\n  var bb;\n  var cc;\n  var dd;\n  var yy;\n\n  dd = xx & 0x00FF;\n  xx >>>= 8;\n  cc = xx & 0x00FF;\n  xx >>>= 8;\n  bb = xx & 0x00FF;\n  xx >>>= 8;\n  aa = xx & 0x00FF;\n\n  yy = this.sboxes[0][aa] + this.sboxes[1][bb];\n  yy = yy ^ this.sboxes[2][cc];\n  yy = yy + this.sboxes[3][dd];\n\n  return yy;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and does NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._encrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = 0; ii < this.NN; ++ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[this.NN + 0];\n  dataR = dataR ^ this.parray[this.NN + 1];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a vector of numbers and turns them\n//* into long words so that they can be processed by the\n//* real algorithm.\n//*\n//* Maybe I should make the real algorithm above take a vector\n//* instead.  That will involve more looping, but it won't require\n//* the F() method to deconstruct the vector.\n//*\nBlowfish.prototype.encrypt_block = function(vector) {\n  var ii;\n  var vals = [0, 0];\n  var off = this.BLOCKSIZE / 2;\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    vals[0] = (vals[0] << 8) | (vector[ii + 0] & 0x00FF);\n    vals[1] = (vals[1] << 8) | (vector[ii + off] & 0x00FF);\n  }\n\n  this._encrypt_block(vals);\n\n  var ret = [];\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    ret[ii + 0] = (vals[0] >>> (24 - 8 * (ii)) & 0x00FF);\n    ret[ii + off] = (vals[1] >>> (24 - 8 * (ii)) & 0x00FF);\n    // vals[ 0 ] = ( vals[ 0 ] >>> 8 );\n    // vals[ 1 ] = ( vals[ 1 ] >>> 8 );\n  }\n\n  return ret;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and undoes NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._decrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = this.NN + 1; ii > 1; --ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[1];\n  dataR = dataR ^ this.parray[0];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a key array and initializes the\n//* sboxes and parray for this encryption.\n//*\nBlowfish.prototype.init = function(key) {\n  var ii;\n  var jj = 0;\n\n  this.parray = [];\n  for (ii = 0; ii < this.NN + 2; ++ii) {\n    var data = 0x00000000;\n    var kk;\n    for (kk = 0; kk < 4; ++kk) {\n      data = (data << 8) | (key[jj] & 0x00FF);\n      if (++jj >= key.length) {\n        jj = 0;\n      }\n    }\n    this.parray[ii] = this.PARRAY[ii] ^ data;\n  }\n\n  this.sboxes = [];\n  for (ii = 0; ii < 4; ++ii) {\n    this.sboxes[ii] = [];\n    for (jj = 0; jj < 256; ++jj) {\n      this.sboxes[ii][jj] = this.SBOXES[ii][jj];\n    }\n  }\n\n  var vals = [0x00000000, 0x00000000];\n\n  for (ii = 0; ii < this.NN + 2; ii += 2) {\n    this._encrypt_block(vals);\n    this.parray[ii + 0] = vals[0];\n    this.parray[ii + 1] = vals[1];\n  }\n\n  for (ii = 0; ii < 4; ++ii) {\n    for (jj = 0; jj < 256; jj += 2) {\n      this._encrypt_block(vals);\n      this.sboxes[ii][jj + 0] = vals[0];\n      this.sboxes[ii][jj + 1] = vals[1];\n    }\n  }\n};\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction BFencrypt(block, key) {\n  var bf = new Blowfish();\n  bf.init(util.str2bin(key));\n  return bf.encrypt_block(block);\n}\n\nfunction BF(key) {\n  this.bf = new Blowfish();\n  this.bf.init(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.bf.encrypt_block(block);\n  }\n}\n\n\nmodule.exports = BF;\nmodule.exports.keySize = BF.prototype.keySize = 16;\nmodule.exports.blockSize = BF.prototype.blockSize = 16;\n","// Use of this source code is governed by a BSD-style\n// license that can be found in the LICENSE file.\n\n// Copyright 2010 pjacobs@xeekr.com . All rights reserved.\n\n// Modified by Recurity Labs GmbH\n\n// fixed/modified by Herbert Hanewinkel, www.haneWIN.de\n// check www.haneWIN.de for the latest version\n\n// cast5.js is a Javascript implementation of CAST-128, as defined in RFC 2144.\n// CAST-128 is a common OpenPGP cipher.\n\n\n// CAST5 constructor\n\n\n\nfunction openpgp_symenc_cast5() {\n  this.BlockSize = 8;\n  this.KeySize = 16;\n\n  this.setKey = function(key) {\n    this.masking = new Array(16);\n    this.rotate = new Array(16);\n\n    this.reset();\n\n    if (key.length == this.KeySize) {\n      this.keySchedule(key);\n    } else {\n      util.print_error('cast5.js: CAST-128: keys must be 16 bytes');\n      return false;\n    }\n    return true;\n  };\n\n  this.reset = function() {\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = 0;\n      this.rotate[i] = 0;\n    }\n  };\n\n  this.getBlockSize = function() {\n    return BlockSize;\n  };\n\n  this.encrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >>> 16) & 255;\n      dst[i + 6] = (l >>> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n\n  this.decrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >> 16) & 255;\n      dst[i + 6] = (l >> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n  var scheduleA = new Array(4);\n\n  scheduleA[0] = new Array(4);\n  scheduleA[0][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 0x8);\n  scheduleA[0][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[0][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[0][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n  scheduleA[1] = new Array(4);\n  scheduleA[1][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[1][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[1][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[1][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  scheduleA[2] = new Array(4);\n  scheduleA[2][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 8);\n  scheduleA[2][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[2][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[2][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n\n  scheduleA[3] = new Array(4);\n  scheduleA[3][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[3][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[3][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[3][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  var scheduleB = new Array(4);\n\n  scheduleB[0] = new Array(4);\n  scheduleB[0][0] = new Array(16 + 8, 16 + 9, 16 + 7, 16 + 6, 16 + 2);\n  scheduleB[0][1] = new Array(16 + 0xa, 16 + 0xb, 16 + 5, 16 + 4, 16 + 6);\n  scheduleB[0][2] = new Array(16 + 0xc, 16 + 0xd, 16 + 3, 16 + 2, 16 + 9);\n  scheduleB[0][3] = new Array(16 + 0xe, 16 + 0xf, 16 + 1, 16 + 0, 16 + 0xc);\n\n  scheduleB[1] = new Array(4);\n  scheduleB[1][0] = new Array(3, 2, 0xc, 0xd, 8);\n  scheduleB[1][1] = new Array(1, 0, 0xe, 0xf, 0xd);\n  scheduleB[1][2] = new Array(7, 6, 8, 9, 3);\n  scheduleB[1][3] = new Array(5, 4, 0xa, 0xb, 7);\n\n\n  scheduleB[2] = new Array(4);\n  scheduleB[2][0] = new Array(16 + 3, 16 + 2, 16 + 0xc, 16 + 0xd, 16 + 9);\n  scheduleB[2][1] = new Array(16 + 1, 16 + 0, 16 + 0xe, 16 + 0xf, 16 + 0xc);\n  scheduleB[2][2] = new Array(16 + 7, 16 + 6, 16 + 8, 16 + 9, 16 + 2);\n  scheduleB[2][3] = new Array(16 + 5, 16 + 4, 16 + 0xa, 16 + 0xb, 16 + 6);\n\n\n  scheduleB[3] = new Array(4);\n  scheduleB[3][0] = new Array(8, 9, 7, 6, 3);\n  scheduleB[3][1] = new Array(0xa, 0xb, 5, 4, 7);\n  scheduleB[3][2] = new Array(0xc, 0xd, 3, 2, 8);\n  scheduleB[3][3] = new Array(0xe, 0xf, 1, 0, 0xd);\n\n  // changed 'in' to 'inn' (in javascript 'in' is a reserved word)\n  this.keySchedule = function(inn) {\n    var t = new Array(8);\n    var k = new Array(32);\n\n    for (var i = 0; i < 4; i++) {\n      var j = i * 4;\n      t[i] = inn[j] << 24 | inn[j + 1] << 16 | inn[j + 2] << 8 | inn[j + 3];\n    }\n\n    var x = [6, 7, 4, 5];\n    var ki = 0;\n\n    for (var half = 0; half < 2; half++) {\n      for (var round = 0; round < 4; round++) {\n        for (var j = 0; j < 4; j++) {\n          var a = scheduleA[round][j];\n          var w = t[a[1]];\n\n          w ^= sBox[4][(t[a[2] >>> 2] >>> (24 - 8 * (a[2] & 3))) & 0xff];\n          w ^= sBox[5][(t[a[3] >>> 2] >>> (24 - 8 * (a[3] & 3))) & 0xff];\n          w ^= sBox[6][(t[a[4] >>> 2] >>> (24 - 8 * (a[4] & 3))) & 0xff];\n          w ^= sBox[7][(t[a[5] >>> 2] >>> (24 - 8 * (a[5] & 3))) & 0xff];\n          w ^= sBox[x[j]][(t[a[6] >>> 2] >>> (24 - 8 * (a[6] & 3))) & 0xff];\n          t[a[0]] = w;\n        }\n\n        for (var j = 0; j < 4; j++) {\n          var b = scheduleB[round][j];\n          var w = sBox[4][(t[b[0] >>> 2] >>> (24 - 8 * (b[0] & 3))) & 0xff];\n\n          w ^= sBox[5][(t[b[1] >>> 2] >>> (24 - 8 * (b[1] & 3))) & 0xff];\n          w ^= sBox[6][(t[b[2] >>> 2] >>> (24 - 8 * (b[2] & 3))) & 0xff];\n          w ^= sBox[7][(t[b[3] >>> 2] >>> (24 - 8 * (b[3] & 3))) & 0xff];\n          w ^= sBox[4 + j][(t[b[4] >>> 2] >>> (24 - 8 * (b[4] & 3))) & 0xff];\n          k[ki] = w;\n          ki++;\n        }\n      }\n    }\n\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = k[i];\n      this.rotate[i] = k[16 + i] & 0x1f;\n    }\n  };\n\n  // These are the three 'f' functions. See RFC 2144, section 2.2.\n\n  function f1(d, m, r) {\n    var t = m + d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] ^ sBox[1][(I >>> 16) & 255]) - sBox[2][(I >>> 8) & 255]) + sBox[3][I & 255];\n  }\n\n  function f2(d, m, r) {\n    var t = m ^ d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] - sBox[1][(I >>> 16) & 255]) + sBox[2][(I >>> 8) & 255]) ^ sBox[3][I & 255];\n  }\n\n  function f3(d, m, r) {\n    var t = m - d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] + sBox[1][(I >>> 16) & 255]) ^ sBox[2][(I >>> 8) & 255]) - sBox[3][I & 255];\n  }\n\n  var sBox = new Array(8);\n  sBox[0] = new Array(\n    0x30fb40d4, 0x9fa0ff0b, 0x6beccd2f, 0x3f258c7a, 0x1e213f2f, 0x9c004dd3, 0x6003e540, 0xcf9fc949,\n    0xbfd4af27, 0x88bbbdb5, 0xe2034090, 0x98d09675, 0x6e63a0e0, 0x15c361d2, 0xc2e7661d, 0x22d4ff8e,\n    0x28683b6f, 0xc07fd059, 0xff2379c8, 0x775f50e2, 0x43c340d3, 0xdf2f8656, 0x887ca41a, 0xa2d2bd2d,\n    0xa1c9e0d6, 0x346c4819, 0x61b76d87, 0x22540f2f, 0x2abe32e1, 0xaa54166b, 0x22568e3a, 0xa2d341d0,\n    0x66db40c8, 0xa784392f, 0x004dff2f, 0x2db9d2de, 0x97943fac, 0x4a97c1d8, 0x527644b7, 0xb5f437a7,\n    0xb82cbaef, 0xd751d159, 0x6ff7f0ed, 0x5a097a1f, 0x827b68d0, 0x90ecf52e, 0x22b0c054, 0xbc8e5935,\n    0x4b6d2f7f, 0x50bb64a2, 0xd2664910, 0xbee5812d, 0xb7332290, 0xe93b159f, 0xb48ee411, 0x4bff345d,\n    0xfd45c240, 0xad31973f, 0xc4f6d02e, 0x55fc8165, 0xd5b1caad, 0xa1ac2dae, 0xa2d4b76d, 0xc19b0c50,\n    0x882240f2, 0x0c6e4f38, 0xa4e4bfd7, 0x4f5ba272, 0x564c1d2f, 0xc59c5319, 0xb949e354, 0xb04669fe,\n    0xb1b6ab8a, 0xc71358dd, 0x6385c545, 0x110f935d, 0x57538ad5, 0x6a390493, 0xe63d37e0, 0x2a54f6b3,\n    0x3a787d5f, 0x6276a0b5, 0x19a6fcdf, 0x7a42206a, 0x29f9d4d5, 0xf61b1891, 0xbb72275e, 0xaa508167,\n    0x38901091, 0xc6b505eb, 0x84c7cb8c, 0x2ad75a0f, 0x874a1427, 0xa2d1936b, 0x2ad286af, 0xaa56d291,\n    0xd7894360, 0x425c750d, 0x93b39e26, 0x187184c9, 0x6c00b32d, 0x73e2bb14, 0xa0bebc3c, 0x54623779,\n    0x64459eab, 0x3f328b82, 0x7718cf82, 0x59a2cea6, 0x04ee002e, 0x89fe78e6, 0x3fab0950, 0x325ff6c2,\n    0x81383f05, 0x6963c5c8, 0x76cb5ad6, 0xd49974c9, 0xca180dcf, 0x380782d5, 0xc7fa5cf6, 0x8ac31511,\n    0x35e79e13, 0x47da91d0, 0xf40f9086, 0xa7e2419e, 0x31366241, 0x051ef495, 0xaa573b04, 0x4a805d8d,\n    0x548300d0, 0x00322a3c, 0xbf64cddf, 0xba57a68e, 0x75c6372b, 0x50afd341, 0xa7c13275, 0x915a0bf5,\n    0x6b54bfab, 0x2b0b1426, 0xab4cc9d7, 0x449ccd82, 0xf7fbf265, 0xab85c5f3, 0x1b55db94, 0xaad4e324,\n    0xcfa4bd3f, 0x2deaa3e2, 0x9e204d02, 0xc8bd25ac, 0xeadf55b3, 0xd5bd9e98, 0xe31231b2, 0x2ad5ad6c,\n    0x954329de, 0xadbe4528, 0xd8710f69, 0xaa51c90f, 0xaa786bf6, 0x22513f1e, 0xaa51a79b, 0x2ad344cc,\n    0x7b5a41f0, 0xd37cfbad, 0x1b069505, 0x41ece491, 0xb4c332e6, 0x032268d4, 0xc9600acc, 0xce387e6d,\n    0xbf6bb16c, 0x6a70fb78, 0x0d03d9c9, 0xd4df39de, 0xe01063da, 0x4736f464, 0x5ad328d8, 0xb347cc96,\n    0x75bb0fc3, 0x98511bfb, 0x4ffbcc35, 0xb58bcf6a, 0xe11f0abc, 0xbfc5fe4a, 0xa70aec10, 0xac39570a,\n    0x3f04442f, 0x6188b153, 0xe0397a2e, 0x5727cb79, 0x9ceb418f, 0x1cacd68d, 0x2ad37c96, 0x0175cb9d,\n    0xc69dff09, 0xc75b65f0, 0xd9db40d8, 0xec0e7779, 0x4744ead4, 0xb11c3274, 0xdd24cb9e, 0x7e1c54bd,\n    0xf01144f9, 0xd2240eb1, 0x9675b3fd, 0xa3ac3755, 0xd47c27af, 0x51c85f4d, 0x56907596, 0xa5bb15e6,\n    0x580304f0, 0xca042cf1, 0x011a37ea, 0x8dbfaadb, 0x35ba3e4a, 0x3526ffa0, 0xc37b4d09, 0xbc306ed9,\n    0x98a52666, 0x5648f725, 0xff5e569d, 0x0ced63d0, 0x7c63b2cf, 0x700b45e1, 0xd5ea50f1, 0x85a92872,\n    0xaf1fbda7, 0xd4234870, 0xa7870bf3, 0x2d3b4d79, 0x42e04198, 0x0cd0ede7, 0x26470db8, 0xf881814c,\n    0x474d6ad7, 0x7c0c5e5c, 0xd1231959, 0x381b7298, 0xf5d2f4db, 0xab838653, 0x6e2f1e23, 0x83719c9e,\n    0xbd91e046, 0x9a56456e, 0xdc39200c, 0x20c8c571, 0x962bda1c, 0xe1e696ff, 0xb141ab08, 0x7cca89b9,\n    0x1a69e783, 0x02cc4843, 0xa2f7c579, 0x429ef47d, 0x427b169c, 0x5ac9f049, 0xdd8f0f00, 0x5c8165bf);\n\n  sBox[1] = new Array(\n    0x1f201094, 0xef0ba75b, 0x69e3cf7e, 0x393f4380, 0xfe61cf7a, 0xeec5207a, 0x55889c94, 0x72fc0651,\n    0xada7ef79, 0x4e1d7235, 0xd55a63ce, 0xde0436ba, 0x99c430ef, 0x5f0c0794, 0x18dcdb7d, 0xa1d6eff3,\n    0xa0b52f7b, 0x59e83605, 0xee15b094, 0xe9ffd909, 0xdc440086, 0xef944459, 0xba83ccb3, 0xe0c3cdfb,\n    0xd1da4181, 0x3b092ab1, 0xf997f1c1, 0xa5e6cf7b, 0x01420ddb, 0xe4e7ef5b, 0x25a1ff41, 0xe180f806,\n    0x1fc41080, 0x179bee7a, 0xd37ac6a9, 0xfe5830a4, 0x98de8b7f, 0x77e83f4e, 0x79929269, 0x24fa9f7b,\n    0xe113c85b, 0xacc40083, 0xd7503525, 0xf7ea615f, 0x62143154, 0x0d554b63, 0x5d681121, 0xc866c359,\n    0x3d63cf73, 0xcee234c0, 0xd4d87e87, 0x5c672b21, 0x071f6181, 0x39f7627f, 0x361e3084, 0xe4eb573b,\n    0x602f64a4, 0xd63acd9c, 0x1bbc4635, 0x9e81032d, 0x2701f50c, 0x99847ab4, 0xa0e3df79, 0xba6cf38c,\n    0x10843094, 0x2537a95e, 0xf46f6ffe, 0xa1ff3b1f, 0x208cfb6a, 0x8f458c74, 0xd9e0a227, 0x4ec73a34,\n    0xfc884f69, 0x3e4de8df, 0xef0e0088, 0x3559648d, 0x8a45388c, 0x1d804366, 0x721d9bfd, 0xa58684bb,\n    0xe8256333, 0x844e8212, 0x128d8098, 0xfed33fb4, 0xce280ae1, 0x27e19ba5, 0xd5a6c252, 0xe49754bd,\n    0xc5d655dd, 0xeb667064, 0x77840b4d, 0xa1b6a801, 0x84db26a9, 0xe0b56714, 0x21f043b7, 0xe5d05860,\n    0x54f03084, 0x066ff472, 0xa31aa153, 0xdadc4755, 0xb5625dbf, 0x68561be6, 0x83ca6b94, 0x2d6ed23b,\n    0xeccf01db, 0xa6d3d0ba, 0xb6803d5c, 0xaf77a709, 0x33b4a34c, 0x397bc8d6, 0x5ee22b95, 0x5f0e5304,\n    0x81ed6f61, 0x20e74364, 0xb45e1378, 0xde18639b, 0x881ca122, 0xb96726d1, 0x8049a7e8, 0x22b7da7b,\n    0x5e552d25, 0x5272d237, 0x79d2951c, 0xc60d894c, 0x488cb402, 0x1ba4fe5b, 0xa4b09f6b, 0x1ca815cf,\n    0xa20c3005, 0x8871df63, 0xb9de2fcb, 0x0cc6c9e9, 0x0beeff53, 0xe3214517, 0xb4542835, 0x9f63293c,\n    0xee41e729, 0x6e1d2d7c, 0x50045286, 0x1e6685f3, 0xf33401c6, 0x30a22c95, 0x31a70850, 0x60930f13,\n    0x73f98417, 0xa1269859, 0xec645c44, 0x52c877a9, 0xcdff33a6, 0xa02b1741, 0x7cbad9a2, 0x2180036f,\n    0x50d99c08, 0xcb3f4861, 0xc26bd765, 0x64a3f6ab, 0x80342676, 0x25a75e7b, 0xe4e6d1fc, 0x20c710e6,\n    0xcdf0b680, 0x17844d3b, 0x31eef84d, 0x7e0824e4, 0x2ccb49eb, 0x846a3bae, 0x8ff77888, 0xee5d60f6,\n    0x7af75673, 0x2fdd5cdb, 0xa11631c1, 0x30f66f43, 0xb3faec54, 0x157fd7fa, 0xef8579cc, 0xd152de58,\n    0xdb2ffd5e, 0x8f32ce19, 0x306af97a, 0x02f03ef8, 0x99319ad5, 0xc242fa0f, 0xa7e3ebb0, 0xc68e4906,\n    0xb8da230c, 0x80823028, 0xdcdef3c8, 0xd35fb171, 0x088a1bc8, 0xbec0c560, 0x61a3c9e8, 0xbca8f54d,\n    0xc72feffa, 0x22822e99, 0x82c570b4, 0xd8d94e89, 0x8b1c34bc, 0x301e16e6, 0x273be979, 0xb0ffeaa6,\n    0x61d9b8c6, 0x00b24869, 0xb7ffce3f, 0x08dc283b, 0x43daf65a, 0xf7e19798, 0x7619b72f, 0x8f1c9ba4,\n    0xdc8637a0, 0x16a7d3b1, 0x9fc393b7, 0xa7136eeb, 0xc6bcc63e, 0x1a513742, 0xef6828bc, 0x520365d6,\n    0x2d6a77ab, 0x3527ed4b, 0x821fd216, 0x095c6e2e, 0xdb92f2fb, 0x5eea29cb, 0x145892f5, 0x91584f7f,\n    0x5483697b, 0x2667a8cc, 0x85196048, 0x8c4bacea, 0x833860d4, 0x0d23e0f9, 0x6c387e8a, 0x0ae6d249,\n    0xb284600c, 0xd835731d, 0xdcb1c647, 0xac4c56ea, 0x3ebd81b3, 0x230eabb0, 0x6438bc87, 0xf0b5b1fa,\n    0x8f5ea2b3, 0xfc184642, 0x0a036b7a, 0x4fb089bd, 0x649da589, 0xa345415e, 0x5c038323, 0x3e5d3bb9,\n    0x43d79572, 0x7e6dd07c, 0x06dfdf1e, 0x6c6cc4ef, 0x7160a539, 0x73bfbe70, 0x83877605, 0x4523ecf1);\n\n  sBox[2] = new Array(\n    0x8defc240, 0x25fa5d9f, 0xeb903dbf, 0xe810c907, 0x47607fff, 0x369fe44b, 0x8c1fc644, 0xaececa90,\n    0xbeb1f9bf, 0xeefbcaea, 0xe8cf1950, 0x51df07ae, 0x920e8806, 0xf0ad0548, 0xe13c8d83, 0x927010d5,\n    0x11107d9f, 0x07647db9, 0xb2e3e4d4, 0x3d4f285e, 0xb9afa820, 0xfade82e0, 0xa067268b, 0x8272792e,\n    0x553fb2c0, 0x489ae22b, 0xd4ef9794, 0x125e3fbc, 0x21fffcee, 0x825b1bfd, 0x9255c5ed, 0x1257a240,\n    0x4e1a8302, 0xbae07fff, 0x528246e7, 0x8e57140e, 0x3373f7bf, 0x8c9f8188, 0xa6fc4ee8, 0xc982b5a5,\n    0xa8c01db7, 0x579fc264, 0x67094f31, 0xf2bd3f5f, 0x40fff7c1, 0x1fb78dfc, 0x8e6bd2c1, 0x437be59b,\n    0x99b03dbf, 0xb5dbc64b, 0x638dc0e6, 0x55819d99, 0xa197c81c, 0x4a012d6e, 0xc5884a28, 0xccc36f71,\n    0xb843c213, 0x6c0743f1, 0x8309893c, 0x0feddd5f, 0x2f7fe850, 0xd7c07f7e, 0x02507fbf, 0x5afb9a04,\n    0xa747d2d0, 0x1651192e, 0xaf70bf3e, 0x58c31380, 0x5f98302e, 0x727cc3c4, 0x0a0fb402, 0x0f7fef82,\n    0x8c96fdad, 0x5d2c2aae, 0x8ee99a49, 0x50da88b8, 0x8427f4a0, 0x1eac5790, 0x796fb449, 0x8252dc15,\n    0xefbd7d9b, 0xa672597d, 0xada840d8, 0x45f54504, 0xfa5d7403, 0xe83ec305, 0x4f91751a, 0x925669c2,\n    0x23efe941, 0xa903f12e, 0x60270df2, 0x0276e4b6, 0x94fd6574, 0x927985b2, 0x8276dbcb, 0x02778176,\n    0xf8af918d, 0x4e48f79e, 0x8f616ddf, 0xe29d840e, 0x842f7d83, 0x340ce5c8, 0x96bbb682, 0x93b4b148,\n    0xef303cab, 0x984faf28, 0x779faf9b, 0x92dc560d, 0x224d1e20, 0x8437aa88, 0x7d29dc96, 0x2756d3dc,\n    0x8b907cee, 0xb51fd240, 0xe7c07ce3, 0xe566b4a1, 0xc3e9615e, 0x3cf8209d, 0x6094d1e3, 0xcd9ca341,\n    0x5c76460e, 0x00ea983b, 0xd4d67881, 0xfd47572c, 0xf76cedd9, 0xbda8229c, 0x127dadaa, 0x438a074e,\n    0x1f97c090, 0x081bdb8a, 0x93a07ebe, 0xb938ca15, 0x97b03cff, 0x3dc2c0f8, 0x8d1ab2ec, 0x64380e51,\n    0x68cc7bfb, 0xd90f2788, 0x12490181, 0x5de5ffd4, 0xdd7ef86a, 0x76a2e214, 0xb9a40368, 0x925d958f,\n    0x4b39fffa, 0xba39aee9, 0xa4ffd30b, 0xfaf7933b, 0x6d498623, 0x193cbcfa, 0x27627545, 0x825cf47a,\n    0x61bd8ba0, 0xd11e42d1, 0xcead04f4, 0x127ea392, 0x10428db7, 0x8272a972, 0x9270c4a8, 0x127de50b,\n    0x285ba1c8, 0x3c62f44f, 0x35c0eaa5, 0xe805d231, 0x428929fb, 0xb4fcdf82, 0x4fb66a53, 0x0e7dc15b,\n    0x1f081fab, 0x108618ae, 0xfcfd086d, 0xf9ff2889, 0x694bcc11, 0x236a5cae, 0x12deca4d, 0x2c3f8cc5,\n    0xd2d02dfe, 0xf8ef5896, 0xe4cf52da, 0x95155b67, 0x494a488c, 0xb9b6a80c, 0x5c8f82bc, 0x89d36b45,\n    0x3a609437, 0xec00c9a9, 0x44715253, 0x0a874b49, 0xd773bc40, 0x7c34671c, 0x02717ef6, 0x4feb5536,\n    0xa2d02fff, 0xd2bf60c4, 0xd43f03c0, 0x50b4ef6d, 0x07478cd1, 0x006e1888, 0xa2e53f55, 0xb9e6d4bc,\n    0xa2048016, 0x97573833, 0xd7207d67, 0xde0f8f3d, 0x72f87b33, 0xabcc4f33, 0x7688c55d, 0x7b00a6b0,\n    0x947b0001, 0x570075d2, 0xf9bb88f8, 0x8942019e, 0x4264a5ff, 0x856302e0, 0x72dbd92b, 0xee971b69,\n    0x6ea22fde, 0x5f08ae2b, 0xaf7a616d, 0xe5c98767, 0xcf1febd2, 0x61efc8c2, 0xf1ac2571, 0xcc8239c2,\n    0x67214cb8, 0xb1e583d1, 0xb7dc3e62, 0x7f10bdce, 0xf90a5c38, 0x0ff0443d, 0x606e6dc6, 0x60543a49,\n    0x5727c148, 0x2be98a1d, 0x8ab41738, 0x20e1be24, 0xaf96da0f, 0x68458425, 0x99833be5, 0x600d457d,\n    0x282f9350, 0x8334b362, 0xd91d1120, 0x2b6d8da0, 0x642b1e31, 0x9c305a00, 0x52bce688, 0x1b03588a,\n    0xf7baefd5, 0x4142ed9c, 0xa4315c11, 0x83323ec5, 0xdfef4636, 0xa133c501, 0xe9d3531c, 0xee353783);\n\n  sBox[3] = new Array(\n    0x9db30420, 0x1fb6e9de, 0xa7be7bef, 0xd273a298, 0x4a4f7bdb, 0x64ad8c57, 0x85510443, 0xfa020ed1,\n    0x7e287aff, 0xe60fb663, 0x095f35a1, 0x79ebf120, 0xfd059d43, 0x6497b7b1, 0xf3641f63, 0x241e4adf,\n    0x28147f5f, 0x4fa2b8cd, 0xc9430040, 0x0cc32220, 0xfdd30b30, 0xc0a5374f, 0x1d2d00d9, 0x24147b15,\n    0xee4d111a, 0x0fca5167, 0x71ff904c, 0x2d195ffe, 0x1a05645f, 0x0c13fefe, 0x081b08ca, 0x05170121,\n    0x80530100, 0xe83e5efe, 0xac9af4f8, 0x7fe72701, 0xd2b8ee5f, 0x06df4261, 0xbb9e9b8a, 0x7293ea25,\n    0xce84ffdf, 0xf5718801, 0x3dd64b04, 0xa26f263b, 0x7ed48400, 0x547eebe6, 0x446d4ca0, 0x6cf3d6f5,\n    0x2649abdf, 0xaea0c7f5, 0x36338cc1, 0x503f7e93, 0xd3772061, 0x11b638e1, 0x72500e03, 0xf80eb2bb,\n    0xabe0502e, 0xec8d77de, 0x57971e81, 0xe14f6746, 0xc9335400, 0x6920318f, 0x081dbb99, 0xffc304a5,\n    0x4d351805, 0x7f3d5ce3, 0xa6c866c6, 0x5d5bcca9, 0xdaec6fea, 0x9f926f91, 0x9f46222f, 0x3991467d,\n    0xa5bf6d8e, 0x1143c44f, 0x43958302, 0xd0214eeb, 0x022083b8, 0x3fb6180c, 0x18f8931e, 0x281658e6,\n    0x26486e3e, 0x8bd78a70, 0x7477e4c1, 0xb506e07c, 0xf32d0a25, 0x79098b02, 0xe4eabb81, 0x28123b23,\n    0x69dead38, 0x1574ca16, 0xdf871b62, 0x211c40b7, 0xa51a9ef9, 0x0014377b, 0x041e8ac8, 0x09114003,\n    0xbd59e4d2, 0xe3d156d5, 0x4fe876d5, 0x2f91a340, 0x557be8de, 0x00eae4a7, 0x0ce5c2ec, 0x4db4bba6,\n    0xe756bdff, 0xdd3369ac, 0xec17b035, 0x06572327, 0x99afc8b0, 0x56c8c391, 0x6b65811c, 0x5e146119,\n    0x6e85cb75, 0xbe07c002, 0xc2325577, 0x893ff4ec, 0x5bbfc92d, 0xd0ec3b25, 0xb7801ab7, 0x8d6d3b24,\n    0x20c763ef, 0xc366a5fc, 0x9c382880, 0x0ace3205, 0xaac9548a, 0xeca1d7c7, 0x041afa32, 0x1d16625a,\n    0x6701902c, 0x9b757a54, 0x31d477f7, 0x9126b031, 0x36cc6fdb, 0xc70b8b46, 0xd9e66a48, 0x56e55a79,\n    0x026a4ceb, 0x52437eff, 0x2f8f76b4, 0x0df980a5, 0x8674cde3, 0xedda04eb, 0x17a9be04, 0x2c18f4df,\n    0xb7747f9d, 0xab2af7b4, 0xefc34d20, 0x2e096b7c, 0x1741a254, 0xe5b6a035, 0x213d42f6, 0x2c1c7c26,\n    0x61c2f50f, 0x6552daf9, 0xd2c231f8, 0x25130f69, 0xd8167fa2, 0x0418f2c8, 0x001a96a6, 0x0d1526ab,\n    0x63315c21, 0x5e0a72ec, 0x49bafefd, 0x187908d9, 0x8d0dbd86, 0x311170a7, 0x3e9b640c, 0xcc3e10d7,\n    0xd5cad3b6, 0x0caec388, 0xf73001e1, 0x6c728aff, 0x71eae2a1, 0x1f9af36e, 0xcfcbd12f, 0xc1de8417,\n    0xac07be6b, 0xcb44a1d8, 0x8b9b0f56, 0x013988c3, 0xb1c52fca, 0xb4be31cd, 0xd8782806, 0x12a3a4e2,\n    0x6f7de532, 0x58fd7eb6, 0xd01ee900, 0x24adffc2, 0xf4990fc5, 0x9711aac5, 0x001d7b95, 0x82e5e7d2,\n    0x109873f6, 0x00613096, 0xc32d9521, 0xada121ff, 0x29908415, 0x7fbb977f, 0xaf9eb3db, 0x29c9ed2a,\n    0x5ce2a465, 0xa730f32c, 0xd0aa3fe8, 0x8a5cc091, 0xd49e2ce7, 0x0ce454a9, 0xd60acd86, 0x015f1919,\n    0x77079103, 0xdea03af6, 0x78a8565e, 0xdee356df, 0x21f05cbe, 0x8b75e387, 0xb3c50651, 0xb8a5c3ef,\n    0xd8eeb6d2, 0xe523be77, 0xc2154529, 0x2f69efdf, 0xafe67afb, 0xf470c4b2, 0xf3e0eb5b, 0xd6cc9876,\n    0x39e4460c, 0x1fda8538, 0x1987832f, 0xca007367, 0xa99144f8, 0x296b299e, 0x492fc295, 0x9266beab,\n    0xb5676e69, 0x9bd3ddda, 0xdf7e052f, 0xdb25701c, 0x1b5e51ee, 0xf65324e6, 0x6afce36c, 0x0316cc04,\n    0x8644213e, 0xb7dc59d0, 0x7965291f, 0xccd6fd43, 0x41823979, 0x932bcdf6, 0xb657c34d, 0x4edfd282,\n    0x7ae5290c, 0x3cb9536b, 0x851e20fe, 0x9833557e, 0x13ecf0b0, 0xd3ffb372, 0x3f85c5c1, 0x0aef7ed2);\n\n  sBox[4] = new Array(\n    0x7ec90c04, 0x2c6e74b9, 0x9b0e66df, 0xa6337911, 0xb86a7fff, 0x1dd358f5, 0x44dd9d44, 0x1731167f,\n    0x08fbf1fa, 0xe7f511cc, 0xd2051b00, 0x735aba00, 0x2ab722d8, 0x386381cb, 0xacf6243a, 0x69befd7a,\n    0xe6a2e77f, 0xf0c720cd, 0xc4494816, 0xccf5c180, 0x38851640, 0x15b0a848, 0xe68b18cb, 0x4caadeff,\n    0x5f480a01, 0x0412b2aa, 0x259814fc, 0x41d0efe2, 0x4e40b48d, 0x248eb6fb, 0x8dba1cfe, 0x41a99b02,\n    0x1a550a04, 0xba8f65cb, 0x7251f4e7, 0x95a51725, 0xc106ecd7, 0x97a5980a, 0xc539b9aa, 0x4d79fe6a,\n    0xf2f3f763, 0x68af8040, 0xed0c9e56, 0x11b4958b, 0xe1eb5a88, 0x8709e6b0, 0xd7e07156, 0x4e29fea7,\n    0x6366e52d, 0x02d1c000, 0xc4ac8e05, 0x9377f571, 0x0c05372a, 0x578535f2, 0x2261be02, 0xd642a0c9,\n    0xdf13a280, 0x74b55bd2, 0x682199c0, 0xd421e5ec, 0x53fb3ce8, 0xc8adedb3, 0x28a87fc9, 0x3d959981,\n    0x5c1ff900, 0xfe38d399, 0x0c4eff0b, 0x062407ea, 0xaa2f4fb1, 0x4fb96976, 0x90c79505, 0xb0a8a774,\n    0xef55a1ff, 0xe59ca2c2, 0xa6b62d27, 0xe66a4263, 0xdf65001f, 0x0ec50966, 0xdfdd55bc, 0x29de0655,\n    0x911e739a, 0x17af8975, 0x32c7911c, 0x89f89468, 0x0d01e980, 0x524755f4, 0x03b63cc9, 0x0cc844b2,\n    0xbcf3f0aa, 0x87ac36e9, 0xe53a7426, 0x01b3d82b, 0x1a9e7449, 0x64ee2d7e, 0xcddbb1da, 0x01c94910,\n    0xb868bf80, 0x0d26f3fd, 0x9342ede7, 0x04a5c284, 0x636737b6, 0x50f5b616, 0xf24766e3, 0x8eca36c1,\n    0x136e05db, 0xfef18391, 0xfb887a37, 0xd6e7f7d4, 0xc7fb7dc9, 0x3063fcdf, 0xb6f589de, 0xec2941da,\n    0x26e46695, 0xb7566419, 0xf654efc5, 0xd08d58b7, 0x48925401, 0xc1bacb7f, 0xe5ff550f, 0xb6083049,\n    0x5bb5d0e8, 0x87d72e5a, 0xab6a6ee1, 0x223a66ce, 0xc62bf3cd, 0x9e0885f9, 0x68cb3e47, 0x086c010f,\n    0xa21de820, 0xd18b69de, 0xf3f65777, 0xfa02c3f6, 0x407edac3, 0xcbb3d550, 0x1793084d, 0xb0d70eba,\n    0x0ab378d5, 0xd951fb0c, 0xded7da56, 0x4124bbe4, 0x94ca0b56, 0x0f5755d1, 0xe0e1e56e, 0x6184b5be,\n    0x580a249f, 0x94f74bc0, 0xe327888e, 0x9f7b5561, 0xc3dc0280, 0x05687715, 0x646c6bd7, 0x44904db3,\n    0x66b4f0a3, 0xc0f1648a, 0x697ed5af, 0x49e92ff6, 0x309e374f, 0x2cb6356a, 0x85808573, 0x4991f840,\n    0x76f0ae02, 0x083be84d, 0x28421c9a, 0x44489406, 0x736e4cb8, 0xc1092910, 0x8bc95fc6, 0x7d869cf4,\n    0x134f616f, 0x2e77118d, 0xb31b2be1, 0xaa90b472, 0x3ca5d717, 0x7d161bba, 0x9cad9010, 0xaf462ba2,\n    0x9fe459d2, 0x45d34559, 0xd9f2da13, 0xdbc65487, 0xf3e4f94e, 0x176d486f, 0x097c13ea, 0x631da5c7,\n    0x445f7382, 0x175683f4, 0xcdc66a97, 0x70be0288, 0xb3cdcf72, 0x6e5dd2f3, 0x20936079, 0x459b80a5,\n    0xbe60e2db, 0xa9c23101, 0xeba5315c, 0x224e42f2, 0x1c5c1572, 0xf6721b2c, 0x1ad2fff3, 0x8c25404e,\n    0x324ed72f, 0x4067b7fd, 0x0523138e, 0x5ca3bc78, 0xdc0fd66e, 0x75922283, 0x784d6b17, 0x58ebb16e,\n    0x44094f85, 0x3f481d87, 0xfcfeae7b, 0x77b5ff76, 0x8c2302bf, 0xaaf47556, 0x5f46b02a, 0x2b092801,\n    0x3d38f5f7, 0x0ca81f36, 0x52af4a8a, 0x66d5e7c0, 0xdf3b0874, 0x95055110, 0x1b5ad7a8, 0xf61ed5ad,\n    0x6cf6e479, 0x20758184, 0xd0cefa65, 0x88f7be58, 0x4a046826, 0x0ff6f8f3, 0xa09c7f70, 0x5346aba0,\n    0x5ce96c28, 0xe176eda3, 0x6bac307f, 0x376829d2, 0x85360fa9, 0x17e3fe2a, 0x24b79767, 0xf5a96b20,\n    0xd6cd2595, 0x68ff1ebf, 0x7555442c, 0xf19f06be, 0xf9e0659a, 0xeeb9491d, 0x34010718, 0xbb30cab8,\n    0xe822fe15, 0x88570983, 0x750e6249, 0xda627e55, 0x5e76ffa8, 0xb1534546, 0x6d47de08, 0xefe9e7d4);\n\n  sBox[5] = new Array(\n    0xf6fa8f9d, 0x2cac6ce1, 0x4ca34867, 0xe2337f7c, 0x95db08e7, 0x016843b4, 0xeced5cbc, 0x325553ac,\n    0xbf9f0960, 0xdfa1e2ed, 0x83f0579d, 0x63ed86b9, 0x1ab6a6b8, 0xde5ebe39, 0xf38ff732, 0x8989b138,\n    0x33f14961, 0xc01937bd, 0xf506c6da, 0xe4625e7e, 0xa308ea99, 0x4e23e33c, 0x79cbd7cc, 0x48a14367,\n    0xa3149619, 0xfec94bd5, 0xa114174a, 0xeaa01866, 0xa084db2d, 0x09a8486f, 0xa888614a, 0x2900af98,\n    0x01665991, 0xe1992863, 0xc8f30c60, 0x2e78ef3c, 0xd0d51932, 0xcf0fec14, 0xf7ca07d2, 0xd0a82072,\n    0xfd41197e, 0x9305a6b0, 0xe86be3da, 0x74bed3cd, 0x372da53c, 0x4c7f4448, 0xdab5d440, 0x6dba0ec3,\n    0x083919a7, 0x9fbaeed9, 0x49dbcfb0, 0x4e670c53, 0x5c3d9c01, 0x64bdb941, 0x2c0e636a, 0xba7dd9cd,\n    0xea6f7388, 0xe70bc762, 0x35f29adb, 0x5c4cdd8d, 0xf0d48d8c, 0xb88153e2, 0x08a19866, 0x1ae2eac8,\n    0x284caf89, 0xaa928223, 0x9334be53, 0x3b3a21bf, 0x16434be3, 0x9aea3906, 0xefe8c36e, 0xf890cdd9,\n    0x80226dae, 0xc340a4a3, 0xdf7e9c09, 0xa694a807, 0x5b7c5ecc, 0x221db3a6, 0x9a69a02f, 0x68818a54,\n    0xceb2296f, 0x53c0843a, 0xfe893655, 0x25bfe68a, 0xb4628abc, 0xcf222ebf, 0x25ac6f48, 0xa9a99387,\n    0x53bddb65, 0xe76ffbe7, 0xe967fd78, 0x0ba93563, 0x8e342bc1, 0xe8a11be9, 0x4980740d, 0xc8087dfc,\n    0x8de4bf99, 0xa11101a0, 0x7fd37975, 0xda5a26c0, 0xe81f994f, 0x9528cd89, 0xfd339fed, 0xb87834bf,\n    0x5f04456d, 0x22258698, 0xc9c4c83b, 0x2dc156be, 0x4f628daa, 0x57f55ec5, 0xe2220abe, 0xd2916ebf,\n    0x4ec75b95, 0x24f2c3c0, 0x42d15d99, 0xcd0d7fa0, 0x7b6e27ff, 0xa8dc8af0, 0x7345c106, 0xf41e232f,\n    0x35162386, 0xe6ea8926, 0x3333b094, 0x157ec6f2, 0x372b74af, 0x692573e4, 0xe9a9d848, 0xf3160289,\n    0x3a62ef1d, 0xa787e238, 0xf3a5f676, 0x74364853, 0x20951063, 0x4576698d, 0xb6fad407, 0x592af950,\n    0x36f73523, 0x4cfb6e87, 0x7da4cec0, 0x6c152daa, 0xcb0396a8, 0xc50dfe5d, 0xfcd707ab, 0x0921c42f,\n    0x89dff0bb, 0x5fe2be78, 0x448f4f33, 0x754613c9, 0x2b05d08d, 0x48b9d585, 0xdc049441, 0xc8098f9b,\n    0x7dede786, 0xc39a3373, 0x42410005, 0x6a091751, 0x0ef3c8a6, 0x890072d6, 0x28207682, 0xa9a9f7be,\n    0xbf32679d, 0xd45b5b75, 0xb353fd00, 0xcbb0e358, 0x830f220a, 0x1f8fb214, 0xd372cf08, 0xcc3c4a13,\n    0x8cf63166, 0x061c87be, 0x88c98f88, 0x6062e397, 0x47cf8e7a, 0xb6c85283, 0x3cc2acfb, 0x3fc06976,\n    0x4e8f0252, 0x64d8314d, 0xda3870e3, 0x1e665459, 0xc10908f0, 0x513021a5, 0x6c5b68b7, 0x822f8aa0,\n    0x3007cd3e, 0x74719eef, 0xdc872681, 0x073340d4, 0x7e432fd9, 0x0c5ec241, 0x8809286c, 0xf592d891,\n    0x08a930f6, 0x957ef305, 0xb7fbffbd, 0xc266e96f, 0x6fe4ac98, 0xb173ecc0, 0xbc60b42a, 0x953498da,\n    0xfba1ae12, 0x2d4bd736, 0x0f25faab, 0xa4f3fceb, 0xe2969123, 0x257f0c3d, 0x9348af49, 0x361400bc,\n    0xe8816f4a, 0x3814f200, 0xa3f94043, 0x9c7a54c2, 0xbc704f57, 0xda41e7f9, 0xc25ad33a, 0x54f4a084,\n    0xb17f5505, 0x59357cbe, 0xedbd15c8, 0x7f97c5ab, 0xba5ac7b5, 0xb6f6deaf, 0x3a479c3a, 0x5302da25,\n    0x653d7e6a, 0x54268d49, 0x51a477ea, 0x5017d55b, 0xd7d25d88, 0x44136c76, 0x0404a8c8, 0xb8e5a121,\n    0xb81a928a, 0x60ed5869, 0x97c55b96, 0xeaec991b, 0x29935913, 0x01fdb7f1, 0x088e8dfa, 0x9ab6f6f5,\n    0x3b4cbf9f, 0x4a5de3ab, 0xe6051d35, 0xa0e1d855, 0xd36b4cf1, 0xf544edeb, 0xb0e93524, 0xbebb8fbd,\n    0xa2d762cf, 0x49c92f54, 0x38b5f331, 0x7128a454, 0x48392905, 0xa65b1db8, 0x851c97bd, 0xd675cf2f);\n\n  sBox[6] = new Array(\n    0x85e04019, 0x332bf567, 0x662dbfff, 0xcfc65693, 0x2a8d7f6f, 0xab9bc912, 0xde6008a1, 0x2028da1f,\n    0x0227bce7, 0x4d642916, 0x18fac300, 0x50f18b82, 0x2cb2cb11, 0xb232e75c, 0x4b3695f2, 0xb28707de,\n    0xa05fbcf6, 0xcd4181e9, 0xe150210c, 0xe24ef1bd, 0xb168c381, 0xfde4e789, 0x5c79b0d8, 0x1e8bfd43,\n    0x4d495001, 0x38be4341, 0x913cee1d, 0x92a79c3f, 0x089766be, 0xbaeeadf4, 0x1286becf, 0xb6eacb19,\n    0x2660c200, 0x7565bde4, 0x64241f7a, 0x8248dca9, 0xc3b3ad66, 0x28136086, 0x0bd8dfa8, 0x356d1cf2,\n    0x107789be, 0xb3b2e9ce, 0x0502aa8f, 0x0bc0351e, 0x166bf52a, 0xeb12ff82, 0xe3486911, 0xd34d7516,\n    0x4e7b3aff, 0x5f43671b, 0x9cf6e037, 0x4981ac83, 0x334266ce, 0x8c9341b7, 0xd0d854c0, 0xcb3a6c88,\n    0x47bc2829, 0x4725ba37, 0xa66ad22b, 0x7ad61f1e, 0x0c5cbafa, 0x4437f107, 0xb6e79962, 0x42d2d816,\n    0x0a961288, 0xe1a5c06e, 0x13749e67, 0x72fc081a, 0xb1d139f7, 0xf9583745, 0xcf19df58, 0xbec3f756,\n    0xc06eba30, 0x07211b24, 0x45c28829, 0xc95e317f, 0xbc8ec511, 0x38bc46e9, 0xc6e6fa14, 0xbae8584a,\n    0xad4ebc46, 0x468f508b, 0x7829435f, 0xf124183b, 0x821dba9f, 0xaff60ff4, 0xea2c4e6d, 0x16e39264,\n    0x92544a8b, 0x009b4fc3, 0xaba68ced, 0x9ac96f78, 0x06a5b79a, 0xb2856e6e, 0x1aec3ca9, 0xbe838688,\n    0x0e0804e9, 0x55f1be56, 0xe7e5363b, 0xb3a1f25d, 0xf7debb85, 0x61fe033c, 0x16746233, 0x3c034c28,\n    0xda6d0c74, 0x79aac56c, 0x3ce4e1ad, 0x51f0c802, 0x98f8f35a, 0x1626a49f, 0xeed82b29, 0x1d382fe3,\n    0x0c4fb99a, 0xbb325778, 0x3ec6d97b, 0x6e77a6a9, 0xcb658b5c, 0xd45230c7, 0x2bd1408b, 0x60c03eb7,\n    0xb9068d78, 0xa33754f4, 0xf430c87d, 0xc8a71302, 0xb96d8c32, 0xebd4e7be, 0xbe8b9d2d, 0x7979fb06,\n    0xe7225308, 0x8b75cf77, 0x11ef8da4, 0xe083c858, 0x8d6b786f, 0x5a6317a6, 0xfa5cf7a0, 0x5dda0033,\n    0xf28ebfb0, 0xf5b9c310, 0xa0eac280, 0x08b9767a, 0xa3d9d2b0, 0x79d34217, 0x021a718d, 0x9ac6336a,\n    0x2711fd60, 0x438050e3, 0x069908a8, 0x3d7fedc4, 0x826d2bef, 0x4eeb8476, 0x488dcf25, 0x36c9d566,\n    0x28e74e41, 0xc2610aca, 0x3d49a9cf, 0xbae3b9df, 0xb65f8de6, 0x92aeaf64, 0x3ac7d5e6, 0x9ea80509,\n    0xf22b017d, 0xa4173f70, 0xdd1e16c3, 0x15e0d7f9, 0x50b1b887, 0x2b9f4fd5, 0x625aba82, 0x6a017962,\n    0x2ec01b9c, 0x15488aa9, 0xd716e740, 0x40055a2c, 0x93d29a22, 0xe32dbf9a, 0x058745b9, 0x3453dc1e,\n    0xd699296e, 0x496cff6f, 0x1c9f4986, 0xdfe2ed07, 0xb87242d1, 0x19de7eae, 0x053e561a, 0x15ad6f8c,\n    0x66626c1c, 0x7154c24c, 0xea082b2a, 0x93eb2939, 0x17dcb0f0, 0x58d4f2ae, 0x9ea294fb, 0x52cf564c,\n    0x9883fe66, 0x2ec40581, 0x763953c3, 0x01d6692e, 0xd3a0c108, 0xa1e7160e, 0xe4f2dfa6, 0x693ed285,\n    0x74904698, 0x4c2b0edd, 0x4f757656, 0x5d393378, 0xa132234f, 0x3d321c5d, 0xc3f5e194, 0x4b269301,\n    0xc79f022f, 0x3c997e7e, 0x5e4f9504, 0x3ffafbbd, 0x76f7ad0e, 0x296693f4, 0x3d1fce6f, 0xc61e45be,\n    0xd3b5ab34, 0xf72bf9b7, 0x1b0434c0, 0x4e72b567, 0x5592a33d, 0xb5229301, 0xcfd2a87f, 0x60aeb767,\n    0x1814386b, 0x30bcc33d, 0x38a0c07d, 0xfd1606f2, 0xc363519b, 0x589dd390, 0x5479f8e6, 0x1cb8d647,\n    0x97fd61a9, 0xea7759f4, 0x2d57539d, 0x569a58cf, 0xe84e63ad, 0x462e1b78, 0x6580f87e, 0xf3817914,\n    0x91da55f4, 0x40a230f3, 0xd1988f35, 0xb6e318d2, 0x3ffa50bc, 0x3d40f021, 0xc3c0bdae, 0x4958c24c,\n    0x518f36b2, 0x84b1d370, 0x0fedce83, 0x878ddada, 0xf2a279c7, 0x94e01be8, 0x90716f4b, 0x954b8aa3);\n\n  sBox[7] = new Array(\n    0xe216300d, 0xbbddfffc, 0xa7ebdabd, 0x35648095, 0x7789f8b7, 0xe6c1121b, 0x0e241600, 0x052ce8b5,\n    0x11a9cfb0, 0xe5952f11, 0xece7990a, 0x9386d174, 0x2a42931c, 0x76e38111, 0xb12def3a, 0x37ddddfc,\n    0xde9adeb1, 0x0a0cc32c, 0xbe197029, 0x84a00940, 0xbb243a0f, 0xb4d137cf, 0xb44e79f0, 0x049eedfd,\n    0x0b15a15d, 0x480d3168, 0x8bbbde5a, 0x669ded42, 0xc7ece831, 0x3f8f95e7, 0x72df191b, 0x7580330d,\n    0x94074251, 0x5c7dcdfa, 0xabbe6d63, 0xaa402164, 0xb301d40a, 0x02e7d1ca, 0x53571dae, 0x7a3182a2,\n    0x12a8ddec, 0xfdaa335d, 0x176f43e8, 0x71fb46d4, 0x38129022, 0xce949ad4, 0xb84769ad, 0x965bd862,\n    0x82f3d055, 0x66fb9767, 0x15b80b4e, 0x1d5b47a0, 0x4cfde06f, 0xc28ec4b8, 0x57e8726e, 0x647a78fc,\n    0x99865d44, 0x608bd593, 0x6c200e03, 0x39dc5ff6, 0x5d0b00a3, 0xae63aff2, 0x7e8bd632, 0x70108c0c,\n    0xbbd35049, 0x2998df04, 0x980cf42a, 0x9b6df491, 0x9e7edd53, 0x06918548, 0x58cb7e07, 0x3b74ef2e,\n    0x522fffb1, 0xd24708cc, 0x1c7e27cd, 0xa4eb215b, 0x3cf1d2e2, 0x19b47a38, 0x424f7618, 0x35856039,\n    0x9d17dee7, 0x27eb35e6, 0xc9aff67b, 0x36baf5b8, 0x09c467cd, 0xc18910b1, 0xe11dbf7b, 0x06cd1af8,\n    0x7170c608, 0x2d5e3354, 0xd4de495a, 0x64c6d006, 0xbcc0c62c, 0x3dd00db3, 0x708f8f34, 0x77d51b42,\n    0x264f620f, 0x24b8d2bf, 0x15c1b79e, 0x46a52564, 0xf8d7e54e, 0x3e378160, 0x7895cda5, 0x859c15a5,\n    0xe6459788, 0xc37bc75f, 0xdb07ba0c, 0x0676a3ab, 0x7f229b1e, 0x31842e7b, 0x24259fd7, 0xf8bef472,\n    0x835ffcb8, 0x6df4c1f2, 0x96f5b195, 0xfd0af0fc, 0xb0fe134c, 0xe2506d3d, 0x4f9b12ea, 0xf215f225,\n    0xa223736f, 0x9fb4c428, 0x25d04979, 0x34c713f8, 0xc4618187, 0xea7a6e98, 0x7cd16efc, 0x1436876c,\n    0xf1544107, 0xbedeee14, 0x56e9af27, 0xa04aa441, 0x3cf7c899, 0x92ecbae6, 0xdd67016d, 0x151682eb,\n    0xa842eedf, 0xfdba60b4, 0xf1907b75, 0x20e3030f, 0x24d8c29e, 0xe139673b, 0xefa63fb8, 0x71873054,\n    0xb6f2cf3b, 0x9f326442, 0xcb15a4cc, 0xb01a4504, 0xf1e47d8d, 0x844a1be5, 0xbae7dfdc, 0x42cbda70,\n    0xcd7dae0a, 0x57e85b7a, 0xd53f5af6, 0x20cf4d8c, 0xcea4d428, 0x79d130a4, 0x3486ebfb, 0x33d3cddc,\n    0x77853b53, 0x37effcb5, 0xc5068778, 0xe580b3e6, 0x4e68b8f4, 0xc5c8b37e, 0x0d809ea2, 0x398feb7c,\n    0x132a4f94, 0x43b7950e, 0x2fee7d1c, 0x223613bd, 0xdd06caa2, 0x37df932b, 0xc4248289, 0xacf3ebc3,\n    0x5715f6b7, 0xef3478dd, 0xf267616f, 0xc148cbe4, 0x9052815e, 0x5e410fab, 0xb48a2465, 0x2eda7fa4,\n    0xe87b40e4, 0xe98ea084, 0x5889e9e1, 0xefd390fc, 0xdd07d35b, 0xdb485694, 0x38d7e5b2, 0x57720101,\n    0x730edebc, 0x5b643113, 0x94917e4f, 0x503c2fba, 0x646f1282, 0x7523d24a, 0xe0779695, 0xf9c17a8f,\n    0x7a5b2121, 0xd187b896, 0x29263a4d, 0xba510cdf, 0x81f47c9f, 0xad1163ed, 0xea7b5965, 0x1a00726e,\n    0x11403092, 0x00da6d77, 0x4a0cdd61, 0xad1f4603, 0x605bdfb0, 0x9eedc364, 0x22ebe6a8, 0xcee7d28a,\n    0xa0e736a0, 0x5564a6b9, 0x10853209, 0xc7eb8f37, 0x2de705ca, 0x8951570f, 0xdf09822b, 0xbd691a6c,\n    0xaa12e4f2, 0x87451c0f, 0xe0f6a27a, 0x3ada4819, 0x4cf1764f, 0x0d771c2b, 0x67cdb156, 0x350d8384,\n    0x5938fa0f, 0x42399ef3, 0x36997b07, 0x0e84093d, 0x4aa93e61, 0x8360d87b, 0x1fa98b0c, 0x1149382c,\n    0xe97625a5, 0x0614d1b7, 0x0e25244b, 0x0c768347, 0x589e8d82, 0x0d2059d1, 0xa466bb1e, 0xf8da0a82,\n    0x04f19130, 0xba6e4ec0, 0x99265164, 0x1ee7230d, 0x50b2ad80, 0xeaee6801, 0x8db2a283, 0xea8bf59e);\n\n};\n\nvar util = require('../../util');\n\nfunction cast5(key) {\n  this.cast5 = new openpgp_symenc_cast5();\n  this.cast5.setKey(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.cast5.encrypt(block);\n  }\n}\n\nmodule.exports = cast5;\nmodule.exports.blockSize = cast5.prototype.blockSize = 8;\nmodule.exports.keySize = cast5.prototype.keySize = 16;\n","//Paul Tero, July 2001\n//http://www.tero.co.uk/des/\n//\n//Optimised for performance with large blocks by Michael Hayworth, November 2001\n//http://www.netdealing.com\n//\n// Modified by Recurity Labs GmbH\n\n//THIS SOFTWARE IS PROVIDED \"AS IS\" AND\n//ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE\n//IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE\n//ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE\n//FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL\n//DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS\n//OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)\n//HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT\n//LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY\n//OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF\n//SUCH DAMAGE.\n\n//des\n//this takes the key, the message, and whether to encrypt or decrypt\n\n\n\n\nfunction des(keys, message, encrypt, mode, iv, padding) {\n  //declaring this locally speeds things up a bit\n  var spfunction1 = new Array(0x1010400, 0, 0x10000, 0x1010404, 0x1010004, 0x10404, 0x4, 0x10000, 0x400, 0x1010400,\n    0x1010404, 0x400, 0x1000404, 0x1010004, 0x1000000, 0x4, 0x404, 0x1000400, 0x1000400, 0x10400, 0x10400, 0x1010000,\n    0x1010000, 0x1000404, 0x10004, 0x1000004, 0x1000004, 0x10004, 0, 0x404, 0x10404, 0x1000000, 0x10000, 0x1010404, 0x4,\n    0x1010000, 0x1010400, 0x1000000, 0x1000000, 0x400, 0x1010004, 0x10000, 0x10400, 0x1000004, 0x400, 0x4, 0x1000404,\n    0x10404, 0x1010404, 0x10004, 0x1010000, 0x1000404, 0x1000004, 0x404, 0x10404, 0x1010400, 0x404, 0x1000400,\n    0x1000400, 0, 0x10004, 0x10400, 0, 0x1010004);\n  var spfunction2 = new Array(-0x7fef7fe0, -0x7fff8000, 0x8000, 0x108020, 0x100000, 0x20, -0x7fefffe0, -0x7fff7fe0, -\n    0x7fffffe0, -0x7fef7fe0, -0x7fef8000, -0x80000000, -0x7fff8000, 0x100000, 0x20, -0x7fefffe0, 0x108000, 0x100020, -\n    0x7fff7fe0, 0, -0x80000000, 0x8000, 0x108020, -0x7ff00000, 0x100020, -0x7fffffe0, 0, 0x108000, 0x8020, -0x7fef8000, -\n    0x7ff00000, 0x8020, 0, 0x108020, -0x7fefffe0, 0x100000, -0x7fff7fe0, -0x7ff00000, -0x7fef8000, 0x8000, -0x7ff00000, -\n    0x7fff8000, 0x20, -0x7fef7fe0, 0x108020, 0x20, 0x8000, -0x80000000, 0x8020, -0x7fef8000, 0x100000, -0x7fffffe0,\n    0x100020, -0x7fff7fe0, -0x7fffffe0, 0x100020, 0x108000, 0, -0x7fff8000, 0x8020, -0x80000000, -0x7fefffe0, -\n    0x7fef7fe0, 0x108000);\n  var spfunction3 = new Array(0x208, 0x8020200, 0, 0x8020008, 0x8000200, 0, 0x20208, 0x8000200, 0x20008, 0x8000008,\n    0x8000008, 0x20000, 0x8020208, 0x20008, 0x8020000, 0x208, 0x8000000, 0x8, 0x8020200, 0x200, 0x20200, 0x8020000,\n    0x8020008, 0x20208, 0x8000208, 0x20200, 0x20000, 0x8000208, 0x8, 0x8020208, 0x200, 0x8000000, 0x8020200, 0x8000000,\n    0x20008, 0x208, 0x20000, 0x8020200, 0x8000200, 0, 0x200, 0x20008, 0x8020208, 0x8000200, 0x8000008, 0x200, 0,\n    0x8020008, 0x8000208, 0x20000, 0x8000000, 0x8020208, 0x8, 0x20208, 0x20200, 0x8000008, 0x8020000, 0x8000208, 0x208,\n    0x8020000, 0x20208, 0x8, 0x8020008, 0x20200);\n  var spfunction4 = new Array(0x802001, 0x2081, 0x2081, 0x80, 0x802080, 0x800081, 0x800001, 0x2001, 0, 0x802000,\n    0x802000, 0x802081, 0x81, 0, 0x800080, 0x800001, 0x1, 0x2000, 0x800000, 0x802001, 0x80, 0x800000, 0x2001, 0x2080,\n    0x800081, 0x1, 0x2080, 0x800080, 0x2000, 0x802080, 0x802081, 0x81, 0x800080, 0x800001, 0x802000, 0x802081, 0x81, 0,\n    0, 0x802000, 0x2080, 0x800080, 0x800081, 0x1, 0x802001, 0x2081, 0x2081, 0x80, 0x802081, 0x81, 0x1, 0x2000, 0x800001,\n    0x2001, 0x802080, 0x800081, 0x2001, 0x2080, 0x800000, 0x802001, 0x80, 0x800000, 0x2000, 0x802080);\n  var spfunction5 = new Array(0x100, 0x2080100, 0x2080000, 0x42000100, 0x80000, 0x100, 0x40000000, 0x2080000,\n    0x40080100, 0x80000, 0x2000100, 0x40080100, 0x42000100, 0x42080000, 0x80100, 0x40000000, 0x2000000, 0x40080000,\n    0x40080000, 0, 0x40000100, 0x42080100, 0x42080100, 0x2000100, 0x42080000, 0x40000100, 0, 0x42000000, 0x2080100,\n    0x2000000, 0x42000000, 0x80100, 0x80000, 0x42000100, 0x100, 0x2000000, 0x40000000, 0x2080000, 0x42000100,\n    0x40080100, 0x2000100, 0x40000000, 0x42080000, 0x2080100, 0x40080100, 0x100, 0x2000000, 0x42080000, 0x42080100,\n    0x80100, 0x42000000, 0x42080100, 0x2080000, 0, 0x40080000, 0x42000000, 0x80100, 0x2000100, 0x40000100, 0x80000, 0,\n    0x40080000, 0x2080100, 0x40000100);\n  var spfunction6 = new Array(0x20000010, 0x20400000, 0x4000, 0x20404010, 0x20400000, 0x10, 0x20404010, 0x400000,\n    0x20004000, 0x404010, 0x400000, 0x20000010, 0x400010, 0x20004000, 0x20000000, 0x4010, 0, 0x400010, 0x20004010,\n    0x4000, 0x404000, 0x20004010, 0x10, 0x20400010, 0x20400010, 0, 0x404010, 0x20404000, 0x4010, 0x404000, 0x20404000,\n    0x20000000, 0x20004000, 0x10, 0x20400010, 0x404000, 0x20404010, 0x400000, 0x4010, 0x20000010, 0x400000, 0x20004000,\n    0x20000000, 0x4010, 0x20000010, 0x20404010, 0x404000, 0x20400000, 0x404010, 0x20404000, 0, 0x20400010, 0x10, 0x4000,\n    0x20400000, 0x404010, 0x4000, 0x400010, 0x20004010, 0, 0x20404000, 0x20000000, 0x400010, 0x20004010);\n  var spfunction7 = new Array(0x200000, 0x4200002, 0x4000802, 0, 0x800, 0x4000802, 0x200802, 0x4200800, 0x4200802,\n    0x200000, 0, 0x4000002, 0x2, 0x4000000, 0x4200002, 0x802, 0x4000800, 0x200802, 0x200002, 0x4000800, 0x4000002,\n    0x4200000, 0x4200800, 0x200002, 0x4200000, 0x800, 0x802, 0x4200802, 0x200800, 0x2, 0x4000000, 0x200800, 0x4000000,\n    0x200800, 0x200000, 0x4000802, 0x4000802, 0x4200002, 0x4200002, 0x2, 0x200002, 0x4000000, 0x4000800, 0x200000,\n    0x4200800, 0x802, 0x200802, 0x4200800, 0x802, 0x4000002, 0x4200802, 0x4200000, 0x200800, 0, 0x2, 0x4200802, 0,\n    0x200802, 0x4200000, 0x800, 0x4000002, 0x4000800, 0x800, 0x200002);\n  var spfunction8 = new Array(0x10001040, 0x1000, 0x40000, 0x10041040, 0x10000000, 0x10001040, 0x40, 0x10000000,\n    0x40040, 0x10040000, 0x10041040, 0x41000, 0x10041000, 0x41040, 0x1000, 0x40, 0x10040000, 0x10000040, 0x10001000,\n    0x1040, 0x41000, 0x40040, 0x10040040, 0x10041000, 0x1040, 0, 0, 0x10040040, 0x10000040, 0x10001000, 0x41040,\n    0x40000, 0x41040, 0x40000, 0x10041000, 0x1000, 0x40, 0x10040040, 0x1000, 0x41040, 0x10001000, 0x40, 0x10000040,\n    0x10040000, 0x10040040, 0x10000000, 0x40000, 0x10001040, 0, 0x10041040, 0x40040, 0x10000040, 0x10040000, 0x10001000,\n    0x10001040, 0, 0x10041040, 0x41000, 0x41000, 0x1040, 0x1040, 0x40040, 0x10000000, 0x10041000);\n\n  //create the 16 or 48 subkeys we will need\n  var m = 0,\n    i, j, temp, temp2, right1, right2, left, right, looping;\n  var cbcleft, cbcleft2, cbcright, cbcright2\n  var endloop, loopinc;\n  var len = message.length;\n  var chunk = 0;\n  //set up the loops for single and triple des\n  var iterations = keys.length == 32 ? 3 : 9; //single or triple des\n  if (iterations == 3) {\n    looping = encrypt ? new Array(0, 32, 2) : new Array(30, -2, -2);\n  } else {\n    looping = encrypt ? new Array(0, 32, 2, 62, 30, -2, 64, 96, 2) : new Array(94, 62, -2, 32, 64, 2, 30, -2, -2);\n  }\n\n  //pad the message depending on the padding parameter\n  //only add padding if encrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (encrypt) {\n    message = des_addPadding(message, padding);\n    len = message.length;\n  }\n\n  //store the result here\n  result = \"\";\n  tempresult = \"\";\n\n  if (mode == 1) { //CBC mode\n    cbcleft = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    cbcright = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    m = 0;\n  }\n\n  //loop through each 64 bit chunk of the message\n  while (m < len) {\n    left = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) | message\n      .charCodeAt(m++);\n    right = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) |\n      message.charCodeAt(m++);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        left ^= cbcleft;\n        right ^= cbcright;\n      } else {\n        cbcleft2 = cbcleft;\n        cbcright2 = cbcright;\n        cbcleft = left;\n        cbcright = right;\n      }\n    }\n\n    //first each 64 but chunk of the message must be permuted according to IP\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    left = ((left << 1) | (left >>> 31));\n    right = ((right << 1) | (right >>> 31));\n\n    //do this either 1 or 3 times for each chunk of the message\n    for (j = 0; j < iterations; j += 3) {\n      endloop = looping[j + 1];\n      loopinc = looping[j + 2];\n      //now go through and perform the encryption or decryption  \n      for (i = looping[j]; i != endloop; i += loopinc) { //for efficiency\n        right1 = right ^ keys[i];\n        right2 = ((right >>> 4) | (right << 28)) ^ keys[i + 1];\n        //the result is attained by passing these bytes through the S selection functions\n        temp = left;\n        left = right;\n        right = temp ^ (spfunction2[(right1 >>> 24) & 0x3f] | spfunction4[(right1 >>> 16) & 0x3f] | spfunction6[(right1 >>>\n          8) & 0x3f] | spfunction8[right1 & 0x3f] | spfunction1[(right2 >>> 24) & 0x3f] | spfunction3[(right2 >>> 16) &\n          0x3f] | spfunction5[(right2 >>> 8) & 0x3f] | spfunction7[right2 & 0x3f]);\n      }\n      temp = left;\n      left = right;\n      right = temp; //unreverse left and right\n    } //for either 1 or 3 iterations\n\n    //move then each one bit to the right\n    left = ((left >>> 1) | (left << 31));\n    right = ((right >>> 1) | (right << 31));\n\n    //now perform IP-1, which is IP in the opposite direction\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        cbcleft = left;\n        cbcright = right;\n      } else {\n        left ^= cbcleft2;\n        right ^= cbcright2;\n      }\n    }\n    tempresult += String.fromCharCode((left >>> 24), ((left >>> 16) & 0xff), ((left >>> 8) & 0xff), (left & 0xff), (\n      right >>> 24), ((right >>> 16) & 0xff), ((right >>> 8) & 0xff), (right & 0xff));\n\n    chunk += 8;\n    if (chunk == 512) {\n      result += tempresult;\n      tempresult = \"\";\n      chunk = 0;\n    }\n  } //for every 8 characters, or 64 bits in the message\n\n  //return the result as an array\n  result += tempresult;\n\n  //only remove padding if decrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (!encrypt) {\n    result = des_removePadding(result, padding);\n  }\n\n  return result;\n} //end of des\n\n\n\n//des_createKeys\n//this takes as input a 64 bit key (even though only 56 bits are used)\n//as an array of 2 integers, and returns 16 48 bit keys\n\nfunction des_createKeys(key) {\n  //declaring this locally speeds things up a bit\n  pc2bytes0 = new Array(0, 0x4, 0x20000000, 0x20000004, 0x10000, 0x10004, 0x20010000, 0x20010004, 0x200, 0x204,\n    0x20000200, 0x20000204, 0x10200, 0x10204, 0x20010200, 0x20010204);\n  pc2bytes1 = new Array(0, 0x1, 0x100000, 0x100001, 0x4000000, 0x4000001, 0x4100000, 0x4100001, 0x100, 0x101, 0x100100,\n    0x100101, 0x4000100, 0x4000101, 0x4100100, 0x4100101);\n  pc2bytes2 = new Array(0, 0x8, 0x800, 0x808, 0x1000000, 0x1000008, 0x1000800, 0x1000808, 0, 0x8, 0x800, 0x808,\n    0x1000000, 0x1000008, 0x1000800, 0x1000808);\n  pc2bytes3 = new Array(0, 0x200000, 0x8000000, 0x8200000, 0x2000, 0x202000, 0x8002000, 0x8202000, 0x20000, 0x220000,\n    0x8020000, 0x8220000, 0x22000, 0x222000, 0x8022000, 0x8222000);\n  pc2bytes4 = new Array(0, 0x40000, 0x10, 0x40010, 0, 0x40000, 0x10, 0x40010, 0x1000, 0x41000, 0x1010, 0x41010, 0x1000,\n    0x41000, 0x1010, 0x41010);\n  pc2bytes5 = new Array(0, 0x400, 0x20, 0x420, 0, 0x400, 0x20, 0x420, 0x2000000, 0x2000400, 0x2000020, 0x2000420,\n    0x2000000, 0x2000400, 0x2000020, 0x2000420);\n  pc2bytes6 = new Array(0, 0x10000000, 0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002, 0, 0x10000000,\n    0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002);\n  pc2bytes7 = new Array(0, 0x10000, 0x800, 0x10800, 0x20000000, 0x20010000, 0x20000800, 0x20010800, 0x20000, 0x30000,\n    0x20800, 0x30800, 0x20020000, 0x20030000, 0x20020800, 0x20030800);\n  pc2bytes8 = new Array(0, 0x40000, 0, 0x40000, 0x2, 0x40002, 0x2, 0x40002, 0x2000000, 0x2040000, 0x2000000, 0x2040000,\n    0x2000002, 0x2040002, 0x2000002, 0x2040002);\n  pc2bytes9 = new Array(0, 0x10000000, 0x8, 0x10000008, 0, 0x10000000, 0x8, 0x10000008, 0x400, 0x10000400, 0x408,\n    0x10000408, 0x400, 0x10000400, 0x408, 0x10000408);\n  pc2bytes10 = new Array(0, 0x20, 0, 0x20, 0x100000, 0x100020, 0x100000, 0x100020, 0x2000, 0x2020, 0x2000, 0x2020,\n    0x102000, 0x102020, 0x102000, 0x102020);\n  pc2bytes11 = new Array(0, 0x1000000, 0x200, 0x1000200, 0x200000, 0x1200000, 0x200200, 0x1200200, 0x4000000, 0x5000000,\n    0x4000200, 0x5000200, 0x4200000, 0x5200000, 0x4200200, 0x5200200);\n  pc2bytes12 = new Array(0, 0x1000, 0x8000000, 0x8001000, 0x80000, 0x81000, 0x8080000, 0x8081000, 0x10, 0x1010,\n    0x8000010, 0x8001010, 0x80010, 0x81010, 0x8080010, 0x8081010);\n  pc2bytes13 = new Array(0, 0x4, 0x100, 0x104, 0, 0x4, 0x100, 0x104, 0x1, 0x5, 0x101, 0x105, 0x1, 0x5, 0x101, 0x105);\n\n  //how many iterations (1 for des, 3 for triple des)\n  var iterations = key.length > 8 ? 3 : 1; //changed by Paul 16/6/2007 to use Triple DES for 9+ byte keys\n  //stores the return keys\n  var keys = new Array(32 * iterations);\n  //now define the left shifts which need to be done\n  var shifts = new Array(0, 0, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 0);\n  //other variables\n  var lefttemp, righttemp, m = 0,\n    n = 0,\n    temp;\n\n  for (var j = 0; j < iterations; j++) { //either 1 or 3 iterations\n    left = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n    right = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 2) ^ right) & 0x33333333;\n    right ^= temp;\n    left ^= (temp << 2);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    //the right side needs to be shifted and to get the last four bits of the left side\n    temp = (left << 8) | ((right >>> 20) & 0x000000f0);\n    //left needs to be put upside down\n    left = (right << 24) | ((right << 8) & 0xff0000) | ((right >>> 8) & 0xff00) | ((right >>> 24) & 0xf0);\n    right = temp;\n\n    //now go through and perform these shifts on the left and right keys\n    for (i = 0; i < shifts.length; i++) {\n      //shift the keys either one or two bits to the left\n      if (shifts[i]) {\n        left = (left << 2) | (left >>> 26);\n        right = (right << 2) | (right >>> 26);\n      } else {\n        left = (left << 1) | (left >>> 27);\n        right = (right << 1) | (right >>> 27);\n      }\n      left &= -0xf;\n      right &= -0xf;\n\n      //now apply PC-2, in such a way that E is easier when encrypting or decrypting\n      //this conversion will look like PC-2 except only the last 6 bits of each byte are used\n      //rather than 48 consecutive bits and the order of lines will be according to \n      //how the S selection functions will be applied: S2, S4, S6, S8, S1, S3, S5, S7\n      lefttemp = pc2bytes0[left >>> 28] | pc2bytes1[(left >>> 24) & 0xf] | pc2bytes2[(left >>> 20) & 0xf] | pc2bytes3[(\n        left >>> 16) & 0xf] | pc2bytes4[(left >>> 12) & 0xf] | pc2bytes5[(left >>> 8) & 0xf] | pc2bytes6[(left >>> 4) &\n        0xf];\n      righttemp = pc2bytes7[right >>> 28] | pc2bytes8[(right >>> 24) & 0xf] | pc2bytes9[(right >>> 20) & 0xf] |\n        pc2bytes10[(right >>> 16) & 0xf] | pc2bytes11[(right >>> 12) & 0xf] | pc2bytes12[(right >>> 8) & 0xf] |\n        pc2bytes13[(right >>> 4) & 0xf];\n      temp = ((righttemp >>> 16) ^ lefttemp) & 0x0000ffff;\n      keys[n++] = lefttemp ^ temp;\n      keys[n++] = righttemp ^ (temp << 16);\n    }\n  } //for each iterations\n  //return the keys we've created\n  return keys;\n} //end of des_createKeys\n\n\nfunction des_addPadding(message, padding) {\n  var padLength = 8 - (message.length % 8);\n  if ((padding == 2) && (padLength < 8)) { //pad the message with spaces\n    message += \"        \".substr(0, padLength);\n  } else if (padding == 1) { //PKCS7 padding\n    message += String.fromCharCode(padLength, padLength, padLength, padLength, padLength, padLength, padLength,\n      padLength).substr(0, padLength);\n  } else if (!padding && (padLength < 8)) { //pad the message out with null bytes\n    message += \"\\0\\0\\0\\0\\0\\0\\0\\0\".substr(0, padLength);\n  }\n  return message;\n}\n\nfunction des_removePadding(message, padding) {\n  if (padding == 2) { // space padded\n    message = message.replace(/ *$/g, \"\");\n  } else if (padding == 1) { // PKCS7\n    var padCount = message.charCodeAt(message.length - 1);\n    message = message.substr(0, message.length - padCount);\n  } else if (!padding) { // null padding\n    message = message.replace(/\\0*$/g, \"\");\n  }\n  return message;\n}\n\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction Des(key) {\n  this.key = [];\n\n  for (var i = 0; i < 3; i++) {\n    this.key.push(key.substr(i * 8, 8));\n  }\n\n  this.encrypt = function(block) {\n    return util.str2bin(des(des_createKeys(this.key[2]),\n      des(des_createKeys(this.key[1]),\n      des(des_createKeys(this.key[0]),\n      util.bin2str(block), true, 0, null, null),\n      false, 0, null, null), true, 0, null, null));\n  }\n}\n\nDes.keySize = Des.prototype.keySize = 24;\nDes.blockSize = Des.prototype.blockSize = 8;\n\n// This is \"original\" DES - Des is actually Triple DES.\n// This is only exported so we can unit test.\n\nfunction OriginalDes(key) {\n  this.key = key;\n\n  this.encrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), true, 0, null, padding));\n  }\n\n  this.decrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), false, 0, null, padding));\n  }\n}\n\nmodule.exports = {\n  des: Des,\n  originalDes: OriginalDes\n}\n","var desModule = require('./des.js');\n\nmodule.exports = {\n  des: desModule['des'],\n  originalDes: desModule['originalDes'],\n  cast5: require('./cast5.js'),\n  twofish: require('./twofish.js'),\n  blowfish: require('./blowfish.js')\n}\n\nvar aes = require('./aes.js');\n\nfor (var i in aes) {\n  module.exports['aes' + i] = aes[i];\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Cipher.js\n * A block-cipher algorithm implementation on JavaScript\n * See Cipher.readme.txt for further information.\n *\n * Copyright(c) 2009 Atsushi Oka [ http://oka.nu/ ]\n * This script file is distributed under the LGPL\n *\n * ACKNOWLEDGMENT\n *\n *     The main subroutines are written by Michiel van Everdingen.\n * \n *     Michiel van Everdingen\n *     http://home.versatel.nl/MAvanEverdingen/index.html\n * \n *     All rights for these routines are reserved to Michiel van Everdingen.\n *\n */\n\n\n\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n//Math\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nvar MAXINT = 0xFFFFFFFF;\n\nfunction rotb(b, n) {\n  return (b << n | b >>> (8 - n)) & 0xFF;\n}\n\nfunction rotw(w, n) {\n  return (w << n | w >>> (32 - n)) & MAXINT;\n}\n\nfunction getW(a, i) {\n  return a[i] | a[i + 1] << 8 | a[i + 2] << 16 | a[i + 3] << 24;\n}\n\nfunction setW(a, i, w) {\n  a.splice(i, 4, w & 0xFF, (w >>> 8) & 0xFF, (w >>> 16) & 0xFF, (w >>> 24) & 0xFF);\n}\n\nfunction setWInv(a, i, w) {\n  a.splice(i, 4, (w >>> 24) & 0xFF, (w >>> 16) & 0xFF, (w >>> 8) & 0xFF, w & 0xFF);\n}\n\nfunction getB(x, n) {\n  return (x >>> (n * 8)) & 0xFF;\n}\n\nfunction getNrBits(i) {\n  var n = 0;\n  while (i > 0) {\n    n++;\n    i >>>= 1;\n  }\n  return n;\n}\n\nfunction getMask(n) {\n  return (1 << n) - 1;\n}\n\n//added 2008/11/13 XXX MUST USE ONE-WAY HASH FUNCTION FOR SECURITY REASON\n\nfunction randByte() {\n  return Math.floor(Math.random() * 256);\n}\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n// Twofish\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nfunction createTwofish() {\n  //\n  var keyBytes = null;\n  var dataBytes = null;\n  var dataOffset = -1;\n  // var dataLength = -1;\n  var algorithmName = null;\n  // var idx2 = -1;\n  //\n\n  algorithmName = \"twofish\";\n\n  var tfsKey = [];\n  var tfsM = [\n    [],\n    [],\n    [],\n    []\n  ];\n\n  function tfsInit(key) {\n    keyBytes = key;\n    var i, a, b, c, d, meKey = [],\n      moKey = [],\n      inKey = [];\n    var kLen;\n    var sKey = [];\n    var f01, f5b, fef;\n\n    var q0 = [\n      [8, 1, 7, 13, 6, 15, 3, 2, 0, 11, 5, 9, 14, 12, 10, 4],\n      [2, 8, 11, 13, 15, 7, 6, 14, 3, 1, 9, 4, 0, 10, 12, 5]\n    ];\n    var q1 = [\n      [14, 12, 11, 8, 1, 2, 3, 5, 15, 4, 10, 6, 7, 0, 9, 13],\n      [1, 14, 2, 11, 4, 12, 3, 7, 6, 13, 10, 5, 15, 9, 0, 8]\n    ];\n    var q2 = [\n      [11, 10, 5, 14, 6, 13, 9, 0, 12, 8, 15, 3, 2, 4, 7, 1],\n      [4, 12, 7, 5, 1, 6, 9, 10, 0, 14, 13, 8, 2, 11, 3, 15]\n    ];\n    var q3 = [\n      [13, 7, 15, 4, 1, 2, 6, 14, 9, 11, 3, 0, 8, 5, 12, 10],\n      [11, 9, 5, 1, 12, 3, 13, 14, 6, 4, 7, 15, 2, 0, 8, 10]\n    ];\n    var ror4 = [0, 8, 1, 9, 2, 10, 3, 11, 4, 12, 5, 13, 6, 14, 7, 15];\n    var ashx = [0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 5, 14, 7];\n    var q = [\n      [],\n      []\n    ];\n    var m = [\n      [],\n      [],\n      [],\n      []\n    ];\n\n    function ffm5b(x) {\n      return x ^ (x >> 2) ^ [0, 90, 180, 238][x & 3];\n    }\n\n    function ffmEf(x) {\n      return x ^ (x >> 1) ^ (x >> 2) ^ [0, 238, 180, 90][x & 3];\n    }\n\n    function mdsRem(p, q) {\n      var i, t, u;\n      for (i = 0; i < 8; i++) {\n        t = q >>> 24;\n        q = ((q << 8) & MAXINT) | p >>> 24;\n        p = (p << 8) & MAXINT;\n        u = t << 1;\n        if (t & 128) {\n          u ^= 333;\n        }\n        q ^= t ^ (u << 16);\n        u ^= t >>> 1;\n        if (t & 1) {\n          u ^= 166;\n        }\n        q ^= u << 24 | u << 8;\n      }\n      return q;\n    }\n\n    function qp(n, x) {\n      var a, b, c, d;\n      a = x >> 4;\n      b = x & 15;\n      c = q0[n][a ^ b];\n      d = q1[n][ror4[b] ^ ashx[a]];\n      return q3[n][ror4[d] ^ ashx[c]] << 4 | q2[n][c ^ d];\n    }\n\n    function hFun(x, key) {\n      var a = getB(x, 0),\n        b = getB(x, 1),\n        c = getB(x, 2),\n        d = getB(x, 3);\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(key[3], 0);\n          b = q[0][b] ^ getB(key[3], 1);\n          c = q[0][c] ^ getB(key[3], 2);\n          d = q[1][d] ^ getB(key[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(key[2], 0);\n          b = q[1][b] ^ getB(key[2], 1);\n          c = q[0][c] ^ getB(key[2], 2);\n          d = q[0][d] ^ getB(key[2], 3);\n        case 2:\n          a = q[0][q[0][a] ^ getB(key[1], 0)] ^ getB(key[0], 0);\n          b = q[0][q[1][b] ^ getB(key[1], 1)] ^ getB(key[0], 1);\n          c = q[1][q[0][c] ^ getB(key[1], 2)] ^ getB(key[0], 2);\n          d = q[1][q[1][d] ^ getB(key[1], 3)] ^ getB(key[0], 3);\n      }\n      return m[0][a] ^ m[1][b] ^ m[2][c] ^ m[3][d];\n    }\n\n    keyBytes = keyBytes.slice(0, 32);\n    i = keyBytes.length;\n    while (i != 16 && i != 24 && i != 32)\n      keyBytes[i++] = 0;\n\n    for (i = 0; i < keyBytes.length; i += 4) {\n      inKey[i >> 2] = getW(keyBytes, i);\n    }\n    for (i = 0; i < 256; i++) {\n      q[0][i] = qp(0, i);\n      q[1][i] = qp(1, i);\n    }\n    for (i = 0; i < 256; i++) {\n      f01 = q[1][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[0][i] = f01 + (f5b << 8) + (fef << 16) + (fef << 24);\n      m[2][i] = f5b + (fef << 8) + (f01 << 16) + (fef << 24);\n      f01 = q[0][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[1][i] = fef + (fef << 8) + (f5b << 16) + (f01 << 24);\n      m[3][i] = f5b + (f01 << 8) + (fef << 16) + (f5b << 24);\n    }\n\n    kLen = inKey.length / 2;\n    for (i = 0; i < kLen; i++) {\n      a = inKey[i + i];\n      meKey[i] = a;\n      b = inKey[i + i + 1];\n      moKey[i] = b;\n      sKey[kLen - i - 1] = mdsRem(a, b);\n    }\n    for (i = 0; i < 40; i += 2) {\n      a = 0x1010101 * i;\n      b = a + 0x1010101;\n      a = hFun(a, meKey);\n      b = rotw(hFun(b, moKey), 8);\n      tfsKey[i] = (a + b) & MAXINT;\n      tfsKey[i + 1] = rotw(a + 2 * b, 9);\n    }\n    for (i = 0; i < 256; i++) {\n      a = b = c = d = i;\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(sKey[3], 0);\n          b = q[0][b] ^ getB(sKey[3], 1);\n          c = q[0][c] ^ getB(sKey[3], 2);\n          d = q[1][d] ^ getB(sKey[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(sKey[2], 0);\n          b = q[1][b] ^ getB(sKey[2], 1);\n          c = q[0][c] ^ getB(sKey[2], 2);\n          d = q[0][d] ^ getB(sKey[2], 3);\n        case 2:\n          tfsM[0][i] = m[0][q[0][q[0][a] ^ getB(sKey[1], 0)] ^ getB(sKey[0], 0)];\n          tfsM[1][i] = m[1][q[0][q[1][b] ^ getB(sKey[1], 1)] ^ getB(sKey[0], 1)];\n          tfsM[2][i] = m[2][q[1][q[0][c] ^ getB(sKey[1], 2)] ^ getB(sKey[0], 2)];\n          tfsM[3][i] = m[3][q[1][q[1][d] ^ getB(sKey[1], 3)] ^ getB(sKey[0], 3)];\n      }\n    }\n  }\n\n  function tfsG0(x) {\n    return tfsM[0][getB(x, 0)] ^ tfsM[1][getB(x, 1)] ^ tfsM[2][getB(x, 2)] ^ tfsM[3][getB(x, 3)];\n  }\n\n  function tfsG1(x) {\n    return tfsM[0][getB(x, 3)] ^ tfsM[1][getB(x, 0)] ^ tfsM[2][getB(x, 1)] ^ tfsM[3][getB(x, 2)];\n  }\n\n  function tfsFrnd(r, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2] ^ (a + b + tfsKey[4 * r + 8]) & MAXINT, 31);\n    blk[3] = rotw(blk[3], 1) ^ (a + 2 * b + tfsKey[4 * r + 9]) & MAXINT;\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0] ^ (a + b + tfsKey[4 * r + 10]) & MAXINT, 31);\n    blk[1] = rotw(blk[1], 1) ^ (a + 2 * b + tfsKey[4 * r + 11]) & MAXINT;\n  }\n\n  function tfsIrnd(i, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2], 1) ^ (a + b + tfsKey[4 * i + 10]) & MAXINT;\n    blk[3] = rotw(blk[3] ^ (a + 2 * b + tfsKey[4 * i + 11]) & MAXINT, 31);\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0], 1) ^ (a + b + tfsKey[4 * i + 8]) & MAXINT;\n    blk[1] = rotw(blk[1] ^ (a + 2 * b + tfsKey[4 * i + 9]) & MAXINT, 31);\n  }\n\n  function tfsClose() {\n    tfsKey = [];\n    tfsM = [\n      [],\n      [],\n      [],\n      []\n    ];\n  }\n\n  function tfsEncrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[0],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[1],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[2],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[3]\n    ];\n    for (var j = 0; j < 8; j++) {\n      tfsFrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[4]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[5]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[6]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[7]);\n    dataOffset += 16;\n    return dataBytes;\n  }\n\n  function tfsDecrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[4],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[5],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[6],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[7]\n    ];\n    for (var j = 7; j >= 0; j--) {\n      tfsIrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[0]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[1]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[2]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[3]);\n    dataOffset += 16;\n  }\n\n  // added by Recurity Labs\n\n  function tfsFinal() {\n    return dataBytes;\n  }\n\n  return {\n    name: \"twofish\",\n    blocksize: 128 / 8,\n    open: tfsInit,\n    close: tfsClose,\n    encrypt: tfsEncrypt,\n    decrypt: tfsDecrypt,\n    // added by Recurity Labs\n    finalize: tfsFinal\n  };\n}\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction TFencrypt(block, key) {\n  var block_copy = [].concat(block);\n  var tf = createTwofish();\n  tf.open(util.str2bin(key), 0);\n  var result = tf.encrypt(block_copy, 0);\n  tf.close();\n  return result;\n}\n\nfunction TF(key) {\n  this.tf = createTwofish();\n  this.tf.open(util.str2bin(key), 0);\n\n  this.encrypt = function(block) {\n    return this.tf.encrypt([].concat(block), 0);\n  }\n}\n\n\nmodule.exports = TF;\nmodule.exports.keySize = TF.prototype.keySize = 32;\nmodule.exports.blockSize = TF.prototype.blockSize = 16;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar random = require('./random.js'),\n  cipher = require('./cipher'),\n  publicKey = require('./public_key'),\n  type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Encrypts data using the specified public key multiprecision integers \n   * and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi[]} if RSA an openpgp_type_mpi; \n   * if elgamal encryption an array of two openpgp_type_mpi is returned; otherwise null\n   */\n  publicKeyEncrypt: function(algo, publicMPIs, data) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          var rsa = new publicKey.rsa();\n          var n = publicMPIs[0].toBigInteger();\n          var e = publicMPIs[1].toBigInteger();\n          var m = data.toBigInteger();\n          return [rsa.encrypt(m, e, n)];\n\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var p = publicMPIs[0].toBigInteger();\n          var g = publicMPIs[1].toBigInteger();\n          var y = publicMPIs[2].toBigInteger();\n          var m = data.toBigInteger();\n          return elgamal.encrypt(m, g, p, y);\n\n        default:\n          return [];\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n  /**\n   * Decrypts data using the specified public key multiprecision integers of the private key,\n   * the specified secretMPIs of the private key and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers \n   * of the public key part of the private key\n   * @param {openpgp_type_mpi[]} secretMPIs Algorithm dependent multiprecision integers \n   * of the private key used\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi} returns a big integer containing the decrypted data; otherwise null\n   */\n\n  publicKeyDecrypt: function(algo, keyIntegers, dataIntegers) {\n    var bn = (function() {\n      switch (algo) {\n        case 'rsa_encrypt_sign':\n        case 'rsa_encrypt':\n          var rsa = new publicKey.rsa();\n          // 0 and 1 are the public key.\n          var d = keyIntegers[2].toBigInteger();\n          var p = keyIntegers[3].toBigInteger();\n          var q = keyIntegers[4].toBigInteger();\n          var u = keyIntegers[5].toBigInteger();\n          var m = dataIntegers[0].toBigInteger();\n          return rsa.decrypt(m, d, p, q, u);\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var x = keyIntegers[3].toBigInteger();\n          var c1 = dataIntegers[0].toBigInteger();\n          var c2 = dataIntegers[1].toBigInteger();\n          var p = keyIntegers[0].toBigInteger();\n          return elgamal.decrypt(c1, c2, p, x);\n        default:\n          return null;\n      }\n    })();\n\n    var result = new type_mpi();\n    result.fromBigInteger(bn);\n    return result;\n  },\n\n  /** Returns the number of integers comprising the private key of an algorithm\n   * @param {openpgp.publickey} algo The public key algorithm\n   * @return {Integer} The number of integers.\n   */\n  getPrivateMpiCount: function(algo) {\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        //   Algorithm-Specific Fields for RSA secret keys:\n        //   - multiprecision integer (MPI) of RSA secret exponent d.\n        //   - MPI of RSA secret prime value p.\n        //   - MPI of RSA secret prime value q (p < q).\n        //   - MPI of u, the multiplicative inverse of p, mod q.\n        return 4;\n      case 'elgamal':\n        // Algorithm-Specific Fields for Elgamal secret keys:\n        //   - MPI of Elgamal secret exponent x.\n        return 1;\n      case 'dsa':\n        // Algorithm-Specific Fields for DSA secret keys:\n        //   - MPI of DSA secret exponent x.\n        return 1;\n      default:\n        throw new Error('Unknown algorithm');\n    }\n  },\n\n  getPublicMpiCount: function(algo) {\n    // - A series of multiprecision integers comprising the key material:\n    //   Algorithm-Specific Fields for RSA public keys:\n    //       - a multiprecision integer (MPI) of RSA public modulus n;\n    //       - an MPI of RSA public encryption exponent e.\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        return 2;\n\n        //   Algorithm-Specific Fields for Elgamal public keys:\n        //     - MPI of Elgamal prime p;\n        //     - MPI of Elgamal group generator g;\n        //     - MPI of Elgamal public key value y (= g**x mod p where x  is secret).\n      case 'elgamal':\n        return 3;\n\n        //   Algorithm-Specific Fields for DSA public keys:\n        //       - MPI of DSA prime p;\n        //       - MPI of DSA group order q (q is a prime divisor of p-1);\n        //       - MPI of DSA group generator g;\n        //       - MPI of DSA public-key value y (= g**x mod p where x  is secret).\n      case 'dsa':\n        return 4;\n\n      default:\n        throw new Error('Unknown algorithm.');\n    }\n  },\n\n  generateMpi: function(algo, bits) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n        case 'rsa_sign':\n          //remember \"publicKey\" refers to the crypto/public_key dir\n          var rsa = new publicKey.rsa();\n          var keyObject = rsa.generate(bits, \"10001\");\n          var output = [];\n          output.push(keyObject.n);\n          output.push(keyObject.ee);\n          output.push(keyObject.d);\n          output.push(keyObject.p);\n          output.push(keyObject.q);\n          output.push(keyObject.u);\n          return output;\n        default:\n          throw new Error('Unsupported algorithm for key generation.');\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n\n  /**\n   * generate random byte prefix as string for the specified algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes with length equal to the block\n   * size of the cipher\n   */\n  getPrefixRandom: function(algo) {\n    return random.getRandomBytes(cipher[algo].blockSize);\n  },\n\n  /**\n   * Generating a session key for the specified symmetric algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes as a string to be used as a key\n   */\n  generateSessionKey: function(algo) {\n    return random.getRandomBytes(cipher[algo].keySize);\n  },\n\n};\n","var sha = require('./sha.js');\n\nmodule.exports = {\n  md5: require('./md5.js'),\n  sha1: sha.sha1,\n  sha256: sha.sha256,\n  sha224: sha.sha224,\n  sha384: sha.sha384,\n  sha512: sha.sha512,\n  ripemd: require('./ripe-md.js'),\n\n  /**\n   * Create a hash on the specified data using the specified algorithm\n   * @param {Integer} algo Hash algorithm type (see RFC4880 9.4)\n   * @param {String} data Data to be hashed\n   * @return {String} hash value\n   */\n  digest: function(algo, data) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return this.md5(data);\n      case 2:\n        // - SHA-1 [FIPS180]\n        return this.sha1(data);\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return this.ripemd(data);\n      case 8:\n        // - SHA256 [FIPS180]\n        return this.sha256(data);\n      case 9:\n        // - SHA384 [FIPS180]\n        return this.sha384(data);\n      case 10:\n        // - SHA512 [FIPS180]\n        return this.sha512(data);\n      case 11:\n        // - SHA224 [FIPS180]\n        return this.sha224(data);\n      default:\n        throw new Error('Invalid hash function.');\n    }\n  },\n\n  /**\n   * Returns the hash size in bytes of the specified hash algorithm type\n   * @param {Integer} algo Hash algorithm type (See RFC4880 9.4)\n   * @return {Integer} Size in bytes of the resulting hash\n   */\n  getHashByteLength: function(algo) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return 16;\n      case 2:\n        // - SHA-1 [FIPS180]\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return 20;\n      case 8:\n        // - SHA256 [FIPS180]\n        return 32;\n      case 9:\n        // - SHA384 [FIPS180]\n        return 48\n      case 10:\n        // - SHA512 [FIPS180]\n        return 64;\n      case 11:\n        // - SHA224 [FIPS180]\n        return 28;\n      default:\n        throw new Error('Invalid hash algorithm.');\n    }\n  }\n\n}\n","/**\n * A fast MD5 JavaScript implementation\n * Copyright (c) 2012 Joseph Myers\n * http://www.myersdaily.org/joseph/javascript/md5-text.html\n *\n * Permission to use, copy, modify, and distribute this software\n * and its documentation for any purposes and without\n * fee is hereby granted provided that this copyright notice\n * appears in all copies.\n *\n * Of course, this soft is provided \"as is\" without express or implied\n * warranty of any kind.\n */\n\nvar util = require('../../util/util.js');\n\nfunction MD5(entree) {\n  var hex = md5(entree);\n  var bin = util.hex2bin(hex);\n  return bin;\n}\n\nfunction md5cycle(x, k) {\n  var a = x[0],\n    b = x[1],\n    c = x[2],\n    d = x[3];\n\n  a = ff(a, b, c, d, k[0], 7, -680876936);\n  d = ff(d, a, b, c, k[1], 12, -389564586);\n  c = ff(c, d, a, b, k[2], 17, 606105819);\n  b = ff(b, c, d, a, k[3], 22, -1044525330);\n  a = ff(a, b, c, d, k[4], 7, -176418897);\n  d = ff(d, a, b, c, k[5], 12, 1200080426);\n  c = ff(c, d, a, b, k[6], 17, -1473231341);\n  b = ff(b, c, d, a, k[7], 22, -45705983);\n  a = ff(a, b, c, d, k[8], 7, 1770035416);\n  d = ff(d, a, b, c, k[9], 12, -1958414417);\n  c = ff(c, d, a, b, k[10], 17, -42063);\n  b = ff(b, c, d, a, k[11], 22, -1990404162);\n  a = ff(a, b, c, d, k[12], 7, 1804603682);\n  d = ff(d, a, b, c, k[13], 12, -40341101);\n  c = ff(c, d, a, b, k[14], 17, -1502002290);\n  b = ff(b, c, d, a, k[15], 22, 1236535329);\n\n  a = gg(a, b, c, d, k[1], 5, -165796510);\n  d = gg(d, a, b, c, k[6], 9, -1069501632);\n  c = gg(c, d, a, b, k[11], 14, 643717713);\n  b = gg(b, c, d, a, k[0], 20, -373897302);\n  a = gg(a, b, c, d, k[5], 5, -701558691);\n  d = gg(d, a, b, c, k[10], 9, 38016083);\n  c = gg(c, d, a, b, k[15], 14, -660478335);\n  b = gg(b, c, d, a, k[4], 20, -405537848);\n  a = gg(a, b, c, d, k[9], 5, 568446438);\n  d = gg(d, a, b, c, k[14], 9, -1019803690);\n  c = gg(c, d, a, b, k[3], 14, -187363961);\n  b = gg(b, c, d, a, k[8], 20, 1163531501);\n  a = gg(a, b, c, d, k[13], 5, -1444681467);\n  d = gg(d, a, b, c, k[2], 9, -51403784);\n  c = gg(c, d, a, b, k[7], 14, 1735328473);\n  b = gg(b, c, d, a, k[12], 20, -1926607734);\n\n  a = hh(a, b, c, d, k[5], 4, -378558);\n  d = hh(d, a, b, c, k[8], 11, -2022574463);\n  c = hh(c, d, a, b, k[11], 16, 1839030562);\n  b = hh(b, c, d, a, k[14], 23, -35309556);\n  a = hh(a, b, c, d, k[1], 4, -1530992060);\n  d = hh(d, a, b, c, k[4], 11, 1272893353);\n  c = hh(c, d, a, b, k[7], 16, -155497632);\n  b = hh(b, c, d, a, k[10], 23, -1094730640);\n  a = hh(a, b, c, d, k[13], 4, 681279174);\n  d = hh(d, a, b, c, k[0], 11, -358537222);\n  c = hh(c, d, a, b, k[3], 16, -722521979);\n  b = hh(b, c, d, a, k[6], 23, 76029189);\n  a = hh(a, b, c, d, k[9], 4, -640364487);\n  d = hh(d, a, b, c, k[12], 11, -421815835);\n  c = hh(c, d, a, b, k[15], 16, 530742520);\n  b = hh(b, c, d, a, k[2], 23, -995338651);\n\n  a = ii(a, b, c, d, k[0], 6, -198630844);\n  d = ii(d, a, b, c, k[7], 10, 1126891415);\n  c = ii(c, d, a, b, k[14], 15, -1416354905);\n  b = ii(b, c, d, a, k[5], 21, -57434055);\n  a = ii(a, b, c, d, k[12], 6, 1700485571);\n  d = ii(d, a, b, c, k[3], 10, -1894986606);\n  c = ii(c, d, a, b, k[10], 15, -1051523);\n  b = ii(b, c, d, a, k[1], 21, -2054922799);\n  a = ii(a, b, c, d, k[8], 6, 1873313359);\n  d = ii(d, a, b, c, k[15], 10, -30611744);\n  c = ii(c, d, a, b, k[6], 15, -1560198380);\n  b = ii(b, c, d, a, k[13], 21, 1309151649);\n  a = ii(a, b, c, d, k[4], 6, -145523070);\n  d = ii(d, a, b, c, k[11], 10, -1120210379);\n  c = ii(c, d, a, b, k[2], 15, 718787259);\n  b = ii(b, c, d, a, k[9], 21, -343485551);\n\n  x[0] = add32(a, x[0]);\n  x[1] = add32(b, x[1]);\n  x[2] = add32(c, x[2]);\n  x[3] = add32(d, x[3]);\n\n}\n\nfunction cmn(q, a, b, x, s, t) {\n  a = add32(add32(a, q), add32(x, t));\n  return add32((a << s) | (a >>> (32 - s)), b);\n}\n\nfunction ff(a, b, c, d, x, s, t) {\n  return cmn((b & c) | ((~b) & d), a, b, x, s, t);\n}\n\nfunction gg(a, b, c, d, x, s, t) {\n  return cmn((b & d) | (c & (~d)), a, b, x, s, t);\n}\n\nfunction hh(a, b, c, d, x, s, t) {\n  return cmn(b ^ c ^ d, a, b, x, s, t);\n}\n\nfunction ii(a, b, c, d, x, s, t) {\n  return cmn(c ^ (b | (~d)), a, b, x, s, t);\n}\n\nfunction md51(s) {\n  txt = '';\n  var n = s.length,\n    state = [1732584193, -271733879, -1732584194, 271733878],\n    i;\n  for (i = 64; i <= s.length; i += 64) {\n    md5cycle(state, md5blk(s.substring(i - 64, i)));\n  }\n  s = s.substring(i - 64);\n  var tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];\n  for (i = 0; i < s.length; i++)\n    tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3);\n  tail[i >> 2] |= 0x80 << ((i % 4) << 3);\n  if (i > 55) {\n    md5cycle(state, tail);\n    for (i = 0; i < 16; i++) tail[i] = 0;\n  }\n  tail[14] = n * 8;\n  md5cycle(state, tail);\n  return state;\n}\n\n/* there needs to be support for Unicode here,\n * unless we pretend that we can redefine the MD-5\n * algorithm for multi-byte characters (perhaps\n * by adding every four 16-bit characters and\n * shortening the sum to 32 bits). Otherwise\n * I suggest performing MD-5 as if every character\n * was two bytes--e.g., 0040 0025 = @%--but then\n * how will an ordinary MD-5 sum be matched?\n * There is no way to standardize text to something\n * like UTF-8 before transformation; speed cost is\n * utterly prohibitive. The JavaScript standard\n * itself needs to look at this: it should start\n * providing access to strings as preformed UTF-8\n * 8-bit unsigned value arrays.\n */\nfunction md5blk(s) { /* I figured global was faster.   */\n  var md5blks = [],\n    i; /* Andy King said do it this way. */\n  for (i = 0; i < 64; i += 4) {\n    md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) <<\n      24);\n  }\n  return md5blks;\n}\n\nvar hex_chr = '0123456789abcdef'.split('');\n\nfunction rhex(n) {\n  var s = '',\n    j = 0;\n  for (; j < 4; j++)\n    s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F];\n  return s;\n}\n\nfunction hex(x) {\n  for (var i = 0; i < x.length; i++)\n    x[i] = rhex(x[i]);\n  return x.join('');\n}\n\nfunction md5(s) {\n  return hex(md51(s));\n}\n\n/* this function is much faster,\nso if possible we use it. Some IEs\nare the only ones I know of that\nneed the idiotic second function,\ngenerated by an if clause.  */\n\nfunction add32(a, b) {\n  return (a + b) & 0xFFFFFFFF;\n}\n\nif (md5('hello') != '5d41402abc4b2a76b9719d911017c592') {\n  function add32(x, y) {\n    var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n      msw = (x >> 16) + (y >> 16) + (lsw >> 16);\n    return (msw << 16) | (lsw & 0xFFFF);\n  }\n}\n\nmodule.exports = MD5\n","/*\n * CryptoMX Tools\n * Copyright (C) 2004 - 2006 Derek Buitenhuis\n *\n * This program is free software; you can redistribute it and/or\n * modify it under the terms of the GNU General Public License\n * as published by the Free Software Foundation; either version 2\n * of the License, or (at your option) any later version.\n *\n * This program is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n * GNU General Public License for more details.\n *\n * You should have received a copy of the GNU General Public License\n * along with this program; if not, write to the Free Software\n * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.\n */\n\n/* Modified by Recurity Labs GmbH\n */\n\nvar RMDsize = 160;\nvar X = new Array();\n\nfunction ROL(x, n) {\n  return new Number((x << n) | (x >>> (32 - n)));\n}\n\nfunction F(x, y, z) {\n  return new Number(x ^ y ^ z);\n}\n\nfunction G(x, y, z) {\n  return new Number((x & y) | (~x & z));\n}\n\nfunction H(x, y, z) {\n  return new Number((x | ~y) ^ z);\n}\n\nfunction I(x, y, z) {\n  return new Number((x & z) | (y & ~z));\n}\n\nfunction J(x, y, z) {\n  return new Number(x ^ (y | ~z));\n}\n\nfunction mixOneRound(a, b, c, d, e, x, s, roundNumber) {\n  switch (roundNumber) {\n    case 0:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n    case 1:\n      a += G(b, c, d) + x + 0x5a827999;\n      break;\n    case 2:\n      a += H(b, c, d) + x + 0x6ed9eba1;\n      break;\n    case 3:\n      a += I(b, c, d) + x + 0x8f1bbcdc;\n      break;\n    case 4:\n      a += J(b, c, d) + x + 0xa953fd4e;\n      break;\n    case 5:\n      a += J(b, c, d) + x + 0x50a28be6;\n      break;\n    case 6:\n      a += I(b, c, d) + x + 0x5c4dd124;\n      break;\n    case 7:\n      a += H(b, c, d) + x + 0x6d703ef3;\n      break;\n    case 8:\n      a += G(b, c, d) + x + 0x7a6d76e9;\n      break;\n    case 9:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n\n    default:\n      document.write(\"Bogus round number\");\n      break;\n  }\n\n  a = ROL(a, s) + e;\n  c = ROL(c, 10);\n\n  a &= 0xffffffff;\n  b &= 0xffffffff;\n  c &= 0xffffffff;\n  d &= 0xffffffff;\n  e &= 0xffffffff;\n\n  var retBlock = new Array();\n  retBlock[0] = a;\n  retBlock[1] = b;\n  retBlock[2] = c;\n  retBlock[3] = d;\n  retBlock[4] = e;\n  retBlock[5] = x;\n  retBlock[6] = s;\n\n  return retBlock;\n}\n\nfunction MDinit(MDbuf) {\n  MDbuf[0] = 0x67452301;\n  MDbuf[1] = 0xefcdab89;\n  MDbuf[2] = 0x98badcfe;\n  MDbuf[3] = 0x10325476;\n  MDbuf[4] = 0xc3d2e1f0;\n}\n\nvar ROLs = [\n  [11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8],\n  [7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12],\n  [11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5],\n  [11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12],\n  [9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6],\n  [8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6],\n  [9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11],\n  [9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5],\n  [15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8],\n  [8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11]\n];\n\nvar indexes = [\n  [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15],\n  [7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8],\n  [3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12],\n  [1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2],\n  [4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13],\n  [5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12],\n  [6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2],\n  [15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13],\n  [8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14],\n  [12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11]\n];\n\nfunction compress(MDbuf, X) {\n  blockA = new Array();\n  blockB = new Array();\n\n  var retBlock;\n\n  for (var i = 0; i < 5; i++) {\n    blockA[i] = new Number(MDbuf[i]);\n    blockB[i] = new Number(MDbuf[i]);\n  }\n\n  var step = 0;\n  for (var j = 0; j < 5; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockA[(step + 0) % 5],\n        blockA[(step + 1) % 5],\n        blockA[(step + 2) % 5],\n        blockA[(step + 3) % 5],\n        blockA[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockA[(step + 0) % 5] = retBlock[0];\n      blockA[(step + 1) % 5] = retBlock[1];\n      blockA[(step + 2) % 5] = retBlock[2];\n      blockA[(step + 3) % 5] = retBlock[3];\n      blockA[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  step = 0;\n  for (var j = 5; j < 10; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockB[(step + 0) % 5],\n        blockB[(step + 1) % 5],\n        blockB[(step + 2) % 5],\n        blockB[(step + 3) % 5],\n        blockB[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockB[(step + 0) % 5] = retBlock[0];\n      blockB[(step + 1) % 5] = retBlock[1];\n      blockB[(step + 2) % 5] = retBlock[2];\n      blockB[(step + 3) % 5] = retBlock[3];\n      blockB[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  blockB[3] += blockA[2] + MDbuf[1];\n  MDbuf[1] = MDbuf[2] + blockA[3] + blockB[4];\n  MDbuf[2] = MDbuf[3] + blockA[4] + blockB[0];\n  MDbuf[3] = MDbuf[4] + blockA[0] + blockB[1];\n  MDbuf[4] = MDbuf[0] + blockA[1] + blockB[2];\n  MDbuf[0] = blockB[3];\n}\n\nfunction zeroX(X) {\n  for (var i = 0; i < 16; i++) {\n    X[i] = 0;\n  }\n}\n\nfunction MDfinish(MDbuf, strptr, lswlen, mswlen) {\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var i = 0; i < (lswlen & 63); i++) {\n    X[i >>> 2] ^= (strptr.charCodeAt(j++) & 255) << (8 * (i & 3));\n  }\n\n  X[(lswlen >>> 2) & 15] ^= 1 << (8 * (lswlen & 3) + 7);\n\n  if ((lswlen & 63) > 55) {\n    compress(MDbuf, X);\n    var X = new Array(16);\n    zeroX(X);\n  }\n\n  X[14] = lswlen << 3;\n  X[15] = (lswlen >>> 29) | (mswlen << 3);\n\n  compress(MDbuf, X);\n}\n\nfunction BYTES_TO_DWORD(fourChars) {\n  var tmp = (fourChars.charCodeAt(3) & 255) << 24;\n  tmp |= (fourChars.charCodeAt(2) & 255) << 16;\n  tmp |= (fourChars.charCodeAt(1) & 255) << 8;\n  tmp |= (fourChars.charCodeAt(0) & 255);\n\n  return tmp;\n}\n\nfunction RMD(message) {\n  var MDbuf = new Array(RMDsize / 32);\n  var hashcode = new Array(RMDsize / 8);\n  var length;\n  var nbytes;\n\n  MDinit(MDbuf);\n  length = message.length;\n\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var nbytes = length; nbytes > 63; nbytes -= 64) {\n    for (var i = 0; i < 16; i++) {\n      X[i] = BYTES_TO_DWORD(message.substr(j, 4));\n      j += 4;\n    }\n    compress(MDbuf, X);\n  }\n\n  MDfinish(MDbuf, message.substr(j), length, 0);\n\n  for (var i = 0; i < RMDsize / 8; i += 4) {\n    hashcode[i] = MDbuf[i >>> 2] & 255;\n    hashcode[i + 1] = (MDbuf[i >>> 2] >>> 8) & 255;\n    hashcode[i + 2] = (MDbuf[i >>> 2] >>> 16) & 255;\n    hashcode[i + 3] = (MDbuf[i >>> 2] >>> 24) & 255;\n  }\n\n  return hashcode;\n}\n\n\nfunction RMDstring(message) {\n  var hashcode = RMD(message);\n  var retString = \"\";\n\n  for (var i = 0; i < RMDsize / 8; i++) {\n    retString += String.fromCharCode(hashcode[i]);\n  }\n\n  return retString;\n}\n\nmodule.exports = RMDstring;\n","/* A JavaScript implementation of the SHA family of hashes, as defined in FIPS \n * PUB 180-2 as well as the corresponding HMAC implementation as defined in\n * FIPS PUB 198a\n *\n * Version 1.3 Copyright Brian Turek 2008-2010\n * Distributed under the BSD License\n * See http://jssha.sourceforge.net/ for more information\n *\n * Several functions taken from Paul Johnson\n */\n\n/* Modified by Recurity Labs GmbH\n * \n * This code has been slightly modified direct string output:\n * - bin2bstr has been added\n * - following wrappers of this library have been added:\n *   - str_sha1\n *   - str_sha256\n *   - str_sha224\n *   - str_sha384\n *   - str_sha512\n */\n\nvar jsSHA = (function() {\n\n  /*\n   * Configurable variables. Defaults typically work\n   */\n  /* Number of Bits Per character (8 for ASCII, 16 for Unicode) */\n  var charSize = 8,\n    /* base-64 pad character. \"=\" for strict RFC compliance */\n    b64pad = \"\",\n    /* hex output format. 0 - lowercase; 1 - uppercase */\n    hexCase = 0,\n\n    /*\n     * Int_64 is a object for 2 32-bit numbers emulating a 64-bit number\n     *\n     * @constructor\n     * @param {Number} msint_32 The most significant 32-bits of a 64-bit number\n     * @param {Number} lsint_32 The least significant 32-bits of a 64-bit number\n     */\n    Int_64 = function(msint_32, lsint_32) {\n      this.highOrder = msint_32;\n      this.lowOrder = lsint_32;\n    },\n\n    /*\n     * Convert a string to an array of big-endian words\n     * If charSize is ASCII, characters >255 have their hi-byte silently\n     * ignored.\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    str2binb = function(str) {\n      var bin = [],\n        mask = (1 << charSize) - 1,\n        length = str.length * charSize,\n        i;\n\n      for (i = 0; i < length; i += charSize) {\n        bin[i >> 5] |= (str.charCodeAt(i / charSize) & mask) <<\n          (32 - charSize - (i % 32));\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert a hex string to an array of big-endian words\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    hex2binb = function(str) {\n      var bin = [],\n        length = str.length,\n        i, num;\n\n      for (i = 0; i < length; i += 2) {\n        num = parseInt(str.substr(i, 2), 16);\n        if (!isNaN(num)) {\n          bin[i >> 3] |= num << (24 - (4 * (i % 8)));\n        } else {\n          return \"INVALID HEX STRING\";\n        }\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert an array of big-endian words to a hex string.\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to hexidecimal\n     *\t representation\n     * @return Hexidecimal representation of the parameter in String form\n     */\n    binb2hex = function(binarray) {\n      var hex_tab = (hexCase) ? \"0123456789ABCDEF\" : \"0123456789abcdef\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, srcByte;\n\n      for (i = 0; i < length; i += 1) {\n        srcByte = binarray[i >> 2] >> ((3 - (i % 4)) * 8);\n        str += hex_tab.charAt((srcByte >> 4) & 0xF) +\n          hex_tab.charAt(srcByte & 0xF);\n      }\n\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a base-64 string\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to base-64\n     *\t representation\n     * @return Base-64 encoded representation of the parameter in String form\n     */\n    binb2b64 = function(binarray) {\n      var tab = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\" +\n        \"0123456789+/\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, j,\n        triplet;\n\n      for (i = 0; i < length; i += 3) {\n        triplet = (((binarray[i >> 2] >> 8 * (3 - i % 4)) & 0xFF) << 16) |\n          (((binarray[i + 1 >> 2] >> 8 * (3 - (i + 1) % 4)) & 0xFF) << 8) |\n          ((binarray[i + 2 >> 2] >> 8 * (3 - (i + 2) % 4)) & 0xFF);\n        for (j = 0; j < 4; j += 1) {\n          if (i * 8 + j * 6 <= binarray.length * 32) {\n            str += tab.charAt((triplet >> 6 * (3 - j)) & 0x3F);\n          } else {\n            str += b64pad;\n          }\n        }\n      }\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a string\n     */\n    binb2str = function(bin) {\n      var str = \"\";\n      var mask = (1 << 8) - 1;\n      for (var i = 0; i < bin.length * 32; i += 8)\n        str += String.fromCharCode((bin[i >> 5] >>> (24 - i % 32)) & mask);\n      return str;\n    },\n    /*\n     * The 32-bit implementation of circular rotate left\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotl_32 = function(x, n) {\n      return (x << n) | (x >>> (32 - n));\n    },\n\n    /*\n     * The 32-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_32 = function(x, n) {\n      return (x >>> n) | (x << (32 - n));\n    },\n\n    /*\n     * The 64-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n        (x.highOrder >>> n) | (x.lowOrder << (32 - n)), (x.lowOrder >>> n) | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n        (x.lowOrder >>> n) | (x.highOrder << (32 - n)), (x.highOrder >>> n) | (x.lowOrder << (32 - n)));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of shift right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_32 = function(x, n) {\n      return x >>> n;\n    },\n\n    /*\n     * The 64-bit implementation of shift right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n          x.highOrder >>> n,\n          x.lowOrder >>> n | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n          0,\n          x.highOrder << (32 - n));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Parity function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    parity_32 = function(x, y, z) {\n      return x ^ y ^ z;\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_32 = function(x, y, z) {\n      return (x & y) ^ (~x & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^ (~x.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^ (~x.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_32 = function(x, y, z) {\n      return (x & y) ^ (x & z) ^ (y & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^\n        (x.highOrder & z.highOrder) ^\n        (y.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^\n        (x.lowOrder & z.lowOrder) ^\n        (y.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_32 = function(x) {\n      return rotr_32(x, 2) ^ rotr_32(x, 13) ^ rotr_32(x, 22);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_64 = function(x) {\n      var rotr28 = rotr_64(x, 28),\n        rotr34 = rotr_64(x, 34),\n        rotr39 = rotr_64(x, 39);\n\n      return new Int_64(\n        rotr28.highOrder ^ rotr34.highOrder ^ rotr39.highOrder,\n        rotr28.lowOrder ^ rotr34.lowOrder ^ rotr39.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_32 = function(x) {\n      return rotr_32(x, 6) ^ rotr_32(x, 11) ^ rotr_32(x, 25);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_64 = function(x) {\n      var rotr14 = rotr_64(x, 14),\n        rotr18 = rotr_64(x, 18),\n        rotr41 = rotr_64(x, 41);\n\n      return new Int_64(\n        rotr14.highOrder ^ rotr18.highOrder ^ rotr41.highOrder,\n        rotr14.lowOrder ^ rotr18.lowOrder ^ rotr41.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_32 = function(x) {\n      return rotr_32(x, 7) ^ rotr_32(x, 18) ^ shr_32(x, 3);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_64 = function(x) {\n      var rotr1 = rotr_64(x, 1),\n        rotr8 = rotr_64(x, 8),\n        shr7 = shr_64(x, 7);\n\n      return new Int_64(\n        rotr1.highOrder ^ rotr8.highOrder ^ shr7.highOrder,\n        rotr1.lowOrder ^ rotr8.lowOrder ^ shr7.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_32 = function(x) {\n      return rotr_32(x, 17) ^ rotr_32(x, 19) ^ shr_32(x, 10);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_64 = function(x) {\n      var rotr19 = rotr_64(x, 19),\n        rotr61 = rotr_64(x, 61),\n        shr6 = shr_64(x, 6);\n\n      return new Int_64(\n        rotr19.highOrder ^ rotr61.highOrder ^ shr6.highOrder,\n        rotr19.lowOrder ^ rotr61.lowOrder ^ shr6.lowOrder);\n    },\n\n    /*\n     * Add two 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument to be added\n     * @param {Number} y The second 32-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_32_2 = function(x, y) {\n      var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n        msw = (x >>> 16) + (y >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add four 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_32_4 = function(a, b, c, d) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add five 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @param {Number} e The fifth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_32_5 = function(a, b, c, d, e) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF) +\n        (e & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (e >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add two 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument to be added\n     * @param {Int_64} y The second 64-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_64_2 = function(x, y) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (x.lowOrder & 0xFFFF) + (y.lowOrder & 0xFFFF);\n      msw = (x.lowOrder >>> 16) + (y.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (x.highOrder & 0xFFFF) + (y.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (x.highOrder >>> 16) + (y.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add four 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_64_4 = function(a, b, c, d) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add five 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @param {Int_64} e The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_64_5 = function(a, b, c, d, e) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF) +\n        (e.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (e.lowOrder >>> 16) +\n        (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) +\n        (e.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) +\n        (e.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Calculates the SHA-1 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} message The binary array representation of the string to\n     *\t hash\n     * @param {Number} messageLen The number of bits in the message\n     * @return The array of integers representing the SHA-1 hash of message\n     */\n    coreSHA1 = function(message, messageLen) {\n      var W = [],\n        a, b, c, d, e, T, ch = ch_32,\n        parity = parity_32,\n        maj = maj_32,\n        rotl = rotl_32,\n        safeAdd_2 = safeAdd_32_2,\n        i, t,\n        safeAdd_5 = safeAdd_32_5,\n        appendedMessageLength,\n        H = [\n            0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0\n        ],\n        K = [\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6\n        ];\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - (messageLen % 32));\n      /* Append length of binary string in the position such that the new\n\t\tlength is a multiple of 512.  Logic does not work for even multiples\n\t\tof 512 but there can never be even multiples of 512 */\n      message[(((messageLen + 65) >> 9) << 4) + 15] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += 16) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n\n        for (t = 0; t < 80; t += 1) {\n          if (t < 16) {\n            W[t] = message[t + i];\n          } else {\n            W[t] = rotl(W[t - 3] ^ W[t - 8] ^ W[t - 14] ^ W[t - 16], 1);\n          }\n\n          if (t < 20) {\n            T = safeAdd_5(rotl(a, 5), ch(b, c, d), e, K[t], W[t]);\n          } else if (t < 40) {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          } else if (t < 60) {\n            T = safeAdd_5(rotl(a, 5), maj(b, c, d), e, K[t], W[t]);\n          } else {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          }\n\n          e = d;\n          d = c;\n          c = rotl(b, 30);\n          b = a;\n          a = T;\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n      }\n\n      return H;\n    },\n\n    /*\n     * Calculates the desired SHA-2 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} The binary array representation of the string to hash\n     * @param {Number} The number of bits in message\n     * @param {String} variant The desired SHA-2 variant\n     * @return The array of integers representing the SHA-2 hash of message\n     */\n    coreSHA2 = function(message, messageLen, variant) {\n      var a, b, c, d, e, f, g, h, T1, T2, H, numRounds, lengthPosition, i, t,\n        binaryStringInc, binaryStringMult, safeAdd_2, safeAdd_4, safeAdd_5,\n        gamma0, gamma1, sigma0, sigma1, ch, maj, Int, K, W = [],\n        appendedMessageLength;\n\n      /* Set up the various function handles and variable for the specific \n       * variant */\n      if (variant === \"SHA-224\" || variant === \"SHA-256\") {\n        /* 32-bit variant */\n        numRounds = 64;\n        lengthPosition = (((messageLen + 65) >> 9) << 4) + 15;\n        binaryStringInc = 16;\n        binaryStringMult = 1;\n        Int = Number;\n        safeAdd_2 = safeAdd_32_2;\n        safeAdd_4 = safeAdd_32_4;\n        safeAdd_5 = safeAdd_32_5;\n        gamma0 = gamma0_32;\n        gamma1 = gamma1_32;\n        sigma0 = sigma0_32;\n        sigma1 = sigma1_32;\n        maj = maj_32;\n        ch = ch_32;\n        K = [\n            0x428A2F98, 0x71374491, 0xB5C0FBCF, 0xE9B5DBA5,\n            0x3956C25B, 0x59F111F1, 0x923F82A4, 0xAB1C5ED5,\n            0xD807AA98, 0x12835B01, 0x243185BE, 0x550C7DC3,\n            0x72BE5D74, 0x80DEB1FE, 0x9BDC06A7, 0xC19BF174,\n            0xE49B69C1, 0xEFBE4786, 0x0FC19DC6, 0x240CA1CC,\n            0x2DE92C6F, 0x4A7484AA, 0x5CB0A9DC, 0x76F988DA,\n            0x983E5152, 0xA831C66D, 0xB00327C8, 0xBF597FC7,\n            0xC6E00BF3, 0xD5A79147, 0x06CA6351, 0x14292967,\n            0x27B70A85, 0x2E1B2138, 0x4D2C6DFC, 0x53380D13,\n            0x650A7354, 0x766A0ABB, 0x81C2C92E, 0x92722C85,\n            0xA2BFE8A1, 0xA81A664B, 0xC24B8B70, 0xC76C51A3,\n            0xD192E819, 0xD6990624, 0xF40E3585, 0x106AA070,\n            0x19A4C116, 0x1E376C08, 0x2748774C, 0x34B0BCB5,\n            0x391C0CB3, 0x4ED8AA4A, 0x5B9CCA4F, 0x682E6FF3,\n            0x748F82EE, 0x78A5636F, 0x84C87814, 0x8CC70208,\n            0x90BEFFFA, 0xA4506CEB, 0xBEF9A3F7, 0xC67178F2\n        ];\n\n        if (variant === \"SHA-224\") {\n          H = [\n              0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939,\n              0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4\n          ];\n        } else {\n          H = [\n              0x6A09E667, 0xBB67AE85, 0x3C6EF372, 0xA54FF53A,\n              0x510E527F, 0x9B05688C, 0x1F83D9AB, 0x5BE0CD19\n          ];\n        }\n      } else if (variant === \"SHA-384\" || variant === \"SHA-512\") {\n        /* 64-bit variant */\n        numRounds = 80;\n        lengthPosition = (((messageLen + 128) >> 10) << 5) + 31;\n        binaryStringInc = 32;\n        binaryStringMult = 2;\n        Int = Int_64;\n        safeAdd_2 = safeAdd_64_2;\n        safeAdd_4 = safeAdd_64_4;\n        safeAdd_5 = safeAdd_64_5;\n        gamma0 = gamma0_64;\n        gamma1 = gamma1_64;\n        sigma0 = sigma0_64;\n        sigma1 = sigma1_64;\n        maj = maj_64;\n        ch = ch_64;\n\n        K = [\n            new Int(0x428a2f98, 0xd728ae22), new Int(0x71374491, 0x23ef65cd),\n            new Int(0xb5c0fbcf, 0xec4d3b2f), new Int(0xe9b5dba5, 0x8189dbbc),\n            new Int(0x3956c25b, 0xf348b538), new Int(0x59f111f1, 0xb605d019),\n            new Int(0x923f82a4, 0xaf194f9b), new Int(0xab1c5ed5, 0xda6d8118),\n            new Int(0xd807aa98, 0xa3030242), new Int(0x12835b01, 0x45706fbe),\n            new Int(0x243185be, 0x4ee4b28c), new Int(0x550c7dc3, 0xd5ffb4e2),\n            new Int(0x72be5d74, 0xf27b896f), new Int(0x80deb1fe, 0x3b1696b1),\n            new Int(0x9bdc06a7, 0x25c71235), new Int(0xc19bf174, 0xcf692694),\n            new Int(0xe49b69c1, 0x9ef14ad2), new Int(0xefbe4786, 0x384f25e3),\n            new Int(0x0fc19dc6, 0x8b8cd5b5), new Int(0x240ca1cc, 0x77ac9c65),\n            new Int(0x2de92c6f, 0x592b0275), new Int(0x4a7484aa, 0x6ea6e483),\n            new Int(0x5cb0a9dc, 0xbd41fbd4), new Int(0x76f988da, 0x831153b5),\n            new Int(0x983e5152, 0xee66dfab), new Int(0xa831c66d, 0x2db43210),\n            new Int(0xb00327c8, 0x98fb213f), new Int(0xbf597fc7, 0xbeef0ee4),\n            new Int(0xc6e00bf3, 0x3da88fc2), new Int(0xd5a79147, 0x930aa725),\n            new Int(0x06ca6351, 0xe003826f), new Int(0x14292967, 0x0a0e6e70),\n            new Int(0x27b70a85, 0x46d22ffc), new Int(0x2e1b2138, 0x5c26c926),\n            new Int(0x4d2c6dfc, 0x5ac42aed), new Int(0x53380d13, 0x9d95b3df),\n            new Int(0x650a7354, 0x8baf63de), new Int(0x766a0abb, 0x3c77b2a8),\n            new Int(0x81c2c92e, 0x47edaee6), new Int(0x92722c85, 0x1482353b),\n            new Int(0xa2bfe8a1, 0x4cf10364), new Int(0xa81a664b, 0xbc423001),\n            new Int(0xc24b8b70, 0xd0f89791), new Int(0xc76c51a3, 0x0654be30),\n            new Int(0xd192e819, 0xd6ef5218), new Int(0xd6990624, 0x5565a910),\n            new Int(0xf40e3585, 0x5771202a), new Int(0x106aa070, 0x32bbd1b8),\n            new Int(0x19a4c116, 0xb8d2d0c8), new Int(0x1e376c08, 0x5141ab53),\n            new Int(0x2748774c, 0xdf8eeb99), new Int(0x34b0bcb5, 0xe19b48a8),\n            new Int(0x391c0cb3, 0xc5c95a63), new Int(0x4ed8aa4a, 0xe3418acb),\n            new Int(0x5b9cca4f, 0x7763e373), new Int(0x682e6ff3, 0xd6b2b8a3),\n            new Int(0x748f82ee, 0x5defb2fc), new Int(0x78a5636f, 0x43172f60),\n            new Int(0x84c87814, 0xa1f0ab72), new Int(0x8cc70208, 0x1a6439ec),\n            new Int(0x90befffa, 0x23631e28), new Int(0xa4506ceb, 0xde82bde9),\n            new Int(0xbef9a3f7, 0xb2c67915), new Int(0xc67178f2, 0xe372532b),\n            new Int(0xca273ece, 0xea26619c), new Int(0xd186b8c7, 0x21c0c207),\n            new Int(0xeada7dd6, 0xcde0eb1e), new Int(0xf57d4f7f, 0xee6ed178),\n            new Int(0x06f067aa, 0x72176fba), new Int(0x0a637dc5, 0xa2c898a6),\n            new Int(0x113f9804, 0xbef90dae), new Int(0x1b710b35, 0x131c471b),\n            new Int(0x28db77f5, 0x23047d84), new Int(0x32caab7b, 0x40c72493),\n            new Int(0x3c9ebe0a, 0x15c9bebc), new Int(0x431d67c4, 0x9c100d4c),\n            new Int(0x4cc5d4be, 0xcb3e42b6), new Int(0x597f299c, 0xfc657e2a),\n            new Int(0x5fcb6fab, 0x3ad6faec), new Int(0x6c44198c, 0x4a475817)\n        ];\n\n        if (variant === \"SHA-384\") {\n          H = [\n              new Int(0xcbbb9d5d, 0xc1059ed8), new Int(0x0629a292a, 0x367cd507),\n              new Int(0x9159015a, 0x3070dd17), new Int(0x0152fecd8, 0xf70e5939),\n              new Int(0x67332667, 0xffc00b31), new Int(0x98eb44a87, 0x68581511),\n              new Int(0xdb0c2e0d, 0x64f98fa7), new Int(0x047b5481d, 0xbefa4fa4)\n          ];\n        } else {\n          H = [\n              new Int(0x6a09e667, 0xf3bcc908), new Int(0xbb67ae85, 0x84caa73b),\n              new Int(0x3c6ef372, 0xfe94f82b), new Int(0xa54ff53a, 0x5f1d36f1),\n              new Int(0x510e527f, 0xade682d1), new Int(0x9b05688c, 0x2b3e6c1f),\n              new Int(0x1f83d9ab, 0xfb41bd6b), new Int(0x5be0cd19, 0x137e2179)\n          ];\n        }\n      }\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - messageLen % 32);\n      /* Append length of binary string in the position such that the new\n       * length is correct */\n      message[lengthPosition] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += binaryStringInc) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n        f = H[5];\n        g = H[6];\n        h = H[7];\n\n        for (t = 0; t < numRounds; t += 1) {\n          if (t < 16) {\n            /* Bit of a hack - for 32-bit, the second term is ignored */\n            W[t] = new Int(message[t * binaryStringMult + i],\n              message[t * binaryStringMult + i + 1]);\n          } else {\n            W[t] = safeAdd_4(\n              gamma1(W[t - 2]), W[t - 7],\n              gamma0(W[t - 15]), W[t - 16]);\n          }\n\n          T1 = safeAdd_5(h, sigma1(e), ch(e, f, g), K[t], W[t]);\n          T2 = safeAdd_2(sigma0(a), maj(a, b, c));\n          h = g;\n          g = f;\n          f = e;\n          e = safeAdd_2(d, T1);\n          d = c;\n          c = b;\n          b = a;\n          a = safeAdd_2(T1, T2);\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n        H[5] = safeAdd_2(f, H[5]);\n        H[6] = safeAdd_2(g, H[6]);\n        H[7] = safeAdd_2(h, H[7]);\n      }\n\n      switch (variant) {\n        case \"SHA-224\":\n          return [\n            H[0], H[1], H[2], H[3],\n            H[4], H[5], H[6]];\n        case \"SHA-256\":\n          return H;\n        case \"SHA-384\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder];\n        case \"SHA-512\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder,\n            H[6].highOrder, H[6].lowOrder,\n            H[7].highOrder, H[7].lowOrder];\n        default:\n          /* This should never be reached */\n          return [];\n      }\n    },\n\n    /*\n     * jsSHA is the workhorse of the library.  Instantiate it with the string to\n     * be hashed as the parameter\n     *\n     * @constructor\n     * @param {String} srcString The string to be hashed\n     * @param {String} inputFormat The format of srcString, ASCII or HEX\n     */\n    jsSHA = function(srcString, inputFormat) {\n\n      this.sha1 = null;\n      this.sha224 = null;\n      this.sha256 = null;\n      this.sha384 = null;\n      this.sha512 = null;\n\n      this.strBinLen = null;\n      this.strToHash = null;\n\n      /* Convert the input string into the correct type */\n      if (\"HEX\" === inputFormat) {\n        if (0 !== (srcString.length % 2)) {\n          return \"TEXT MUST BE IN BYTE INCREMENTS\";\n        }\n        this.strBinLen = srcString.length * 4;\n        this.strToHash = hex2binb(srcString);\n      } else if ((\"ASCII\" === inputFormat) ||\n        ('undefined' === typeof(inputFormat))) {\n        this.strBinLen = srcString.length * charSize;\n        this.strToHash = str2binb(srcString);\n      } else {\n        return \"UNKNOWN TEXT INPUT TYPE\";\n      }\n    };\n\n  jsSHA.prototype = {\n    /*\n     * Returns the desired SHA hash of the string specified at instantiation\n     * using the specified parameters\n     *\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} format The desired output formatting (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHash: function(variant, format) {\n      var formatFunc = null,\n        message = this.strToHash.slice();\n\n      switch (format) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      switch (variant) {\n        case \"SHA-1\":\n          if (null === this.sha1) {\n            this.sha1 = coreSHA1(message, this.strBinLen);\n          }\n          return formatFunc(this.sha1);\n        case \"SHA-224\":\n          if (null === this.sha224) {\n            this.sha224 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha224);\n        case \"SHA-256\":\n          if (null === this.sha256) {\n            this.sha256 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha256);\n        case \"SHA-384\":\n          if (null === this.sha384) {\n            this.sha384 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha384);\n        case \"SHA-512\":\n          if (null === this.sha512) {\n            this.sha512 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha512);\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n    },\n\n    /*\n     * Returns the desired HMAC of the string specified at instantiation\n     * using the key and variant param.\n     *\n     * @param {String} key The key used to calculate the HMAC\n     * @param {String} inputFormat The format of key, ASCII or HEX\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} outputFormat The desired output formatting\n     *\t (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHMAC: function(key, inputFormat, variant, outputFormat) {\n      var formatFunc, keyToUse, blockByteSize, blockBitSize, i,\n        retVal, lastArrayIndex, keyBinLen, hashBitSize,\n        keyWithIPad = [],\n        keyWithOPad = [];\n\n      /* Validate the output format selection */\n      switch (outputFormat) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      /* Validate the hash variant selection and set needed variables */\n      switch (variant) {\n        case \"SHA-1\":\n          blockByteSize = 64;\n          hashBitSize = 160;\n          break;\n        case \"SHA-224\":\n          blockByteSize = 64;\n          hashBitSize = 224;\n          break;\n        case \"SHA-256\":\n          blockByteSize = 64;\n          hashBitSize = 256;\n          break;\n        case \"SHA-384\":\n          blockByteSize = 128;\n          hashBitSize = 384;\n          break;\n        case \"SHA-512\":\n          blockByteSize = 128;\n          hashBitSize = 512;\n          break;\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n\n      /* Validate input format selection */\n      if (\"HEX\" === inputFormat) {\n        /* Nibbles must come in pairs */\n        if (0 !== (key.length % 2)) {\n          return \"KEY MUST BE IN BYTE INCREMENTS\";\n        }\n        keyToUse = hex2binb(key);\n        keyBinLen = key.length * 4;\n      } else if (\"ASCII\" === inputFormat) {\n        keyToUse = str2binb(key);\n        keyBinLen = key.length * charSize;\n      } else {\n        return \"UNKNOWN KEY INPUT TYPE\";\n      }\n\n      /* These are used multiple times, calculate and store them */\n      blockBitSize = blockByteSize * 8;\n      lastArrayIndex = (blockByteSize / 4) - 1;\n\n      /* Figure out what to do with the key based on its size relative to\n       * the hash's block size */\n      if (blockByteSize < (keyBinLen / 8)) {\n        if (\"SHA-1\" === variant) {\n          keyToUse = coreSHA1(keyToUse, keyBinLen);\n        } else {\n          keyToUse = coreSHA2(keyToUse, keyBinLen, variant);\n        }\n        /* For all variants, the block size is bigger than the output\n         * size so there will never be a useful byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      } else if (blockByteSize > (keyBinLen / 8)) {\n        /* If the blockByteSize is greater than the key length, there\n         * will always be at LEAST one \"useless\" byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      }\n\n      /* Create ipad and opad */\n      for (i = 0; i <= lastArrayIndex; i += 1) {\n        keyWithIPad[i] = keyToUse[i] ^ 0x36363636;\n        keyWithOPad[i] = keyToUse[i] ^ 0x5C5C5C5C;\n      }\n\n      /* Calculate the HMAC */\n      if (\"SHA-1\" === variant) {\n        retVal = coreSHA1(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen);\n        retVal = coreSHA1(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize);\n      } else {\n        retVal = coreSHA2(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen, variant);\n        retVal = coreSHA2(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize, variant);\n      }\n\n      return (formatFunc(retVal));\n    }\n  };\n\n  return jsSHA;\n}());\n\nmodule.exports = {\n  sha1: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-1\", \"ASCII\");\n  },\n  sha224: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-224\", \"ASCII\");\n  },\n  sha256: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-256\", \"ASCII\");\n  },\n  sha384: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-384\", \"ASCII\");\n\n  },\n  sha512: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-512\", \"ASCII\");\n  }\n}\n","module.exports = {\n  cipher: require('./cipher'),\n  hash: require('./hash'),\n  cfb: require('./cfb.js'),\n  publicKey: require('./public_key'),\n  signature: require('./signature.js'),\n  random: require('./random.js'),\n  pkcs1: require('./pkcs1.js')\n\n}\n\nvar crypto = require('./crypto.js');\n\nfor (var i in crypto)\n  module.exports[i] = crypto[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * ASN1 object identifiers for hashes (See RFC4880 5.2.2)\n */\nhash_headers = new Array();\nhash_headers[1] = [0x30, 0x20, 0x30, 0x0c, 0x06, 0x08, 0x2a, 0x86, 0x48, 0x86, 0xf7, 0x0d, 0x02, 0x05, 0x05, 0x00, 0x04,\n    0x10\n];\nhash_headers[3] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2B, 0x24, 0x03, 0x02, 0x01, 0x05, 0x00, 0x04, 0x14];\nhash_headers[2] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2b, 0x0e, 0x03, 0x02, 0x1a, 0x05, 0x00, 0x04, 0x14];\nhash_headers[8] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01, 0x05, 0x00,\n    0x04, 0x20\n];\nhash_headers[9] = [0x30, 0x41, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x02, 0x05, 0x00,\n    0x04, 0x30\n];\nhash_headers[10] = [0x30, 0x51, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x03, 0x05,\n    0x00, 0x04, 0x40\n];\nhash_headers[11] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x04, 0x05,\n    0x00, 0x04, 0x1C\n];\n\n\nvar crypto = require('./crypto.js'),\n  random = require('./random.js'),\n  util = require('../util'),\n  BigInteger = require('./public_key/jsbn.js'),\n  hash = require('./hash');\n\nmodule.exports = {\n  eme: {\n    /**\n     * create a EME-PKCS1-v1_5 padding (See RFC4880 13.1.1)\n     * @param {String} message message to be padded\n     * @param {Integer} length Length to the resulting message\n     * @return {String} EME-PKCS1 padded message\n     */\n    encode: function(message, length) {\n      if (message.length > length - 11)\n        return -1;\n      var result = \"\";\n      result += String.fromCharCode(0);\n      result += String.fromCharCode(2);\n      for (var i = 0; i < length - message.length - 3; i++) {\n        result += String.fromCharCode(random.getPseudoRandom(1, 255));\n      }\n      result += String.fromCharCode(0);\n      result += message;\n      return result;\n    },\n\n    /**\n     * decodes a EME-PKCS1-v1_5 padding (See RFC4880 13.1.2)\n     * @param {String} message EME-PKCS1 padded message\n     * @return {String} decoded message \n     */\n    decode: function(message, len) {\n      if (message.length < len)\n        message = String.fromCharCode(0) + message;\n      if (message.length < 12 || message.charCodeAt(0) != 0 || message.charCodeAt(1) != 2)\n        return -1;\n      var i = 2;\n      while (message.charCodeAt(i) != 0 && message.length > i)\n        i++;\n      return message.substring(i + 1, message.length);\n    },\n  },\n\n  emsa: {\n\n    /**\n     * create a EMSA-PKCS1-v1_5 padding (See RFC4880 13.1.3)\n     * @param {Integer} algo Hash algorithm type used\n     * @param {String} data Data to be hashed\n     * @param {Integer} keylength Key size of the public mpi in bytes\n     * @returns {String} Hashcode with pkcs1padding as string\n     */\n    encode: function(algo, data, keylength) {\n      var data2 = \"\";\n      data2 += String.fromCharCode(0x00);\n      data2 += String.fromCharCode(0x01);\n      for (var i = 0; i < (keylength - hash_headers[algo].length - 3 -\n        hash.getHashByteLength(algo)); i++)\n\n        data2 += String.fromCharCode(0xff);\n\n      data2 += String.fromCharCode(0x00);\n\n      for (var i = 0; i < hash_headers[algo].length; i++)\n        data2 += String.fromCharCode(hash_headers[algo][i]);\n\n      data2 += hash.digest(algo, data);\n      return new BigInteger(util.hexstrdump(data2), 16);\n    },\n\n    /**\n     * extract the hash out of an EMSA-PKCS1-v1.5 padding (See RFC4880 13.1.3) \n     * @param {String} data Hash in pkcs1 encoding\n     * @returns {String} The hash as string\n     */\n    decode: function(algo, data) {\n      var i = 0;\n      if (data.charCodeAt(0) == 0) i++;\n      else if (data.charCodeAt(0) != 1) return -1;\n      else i++;\n\n      while (data.charCodeAt(i) == 0xFF) i++;\n      if (data.charCodeAt(i++) != 0) return -1;\n      var j = 0;\n      for (j = 0; j < hash_headers[algo].length && j + i < data.length; j++) {\n        if (data.charCodeAt(j + i) != hash_headers[algo][j]) return -1;\n      }\n      i += j;\n      if (data.substring(i).length < hash.getHashByteLength(algo)) return -1;\n      return data.substring(i);\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// A Digital signature algorithm implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  hashModule = require('../hash'),\n  util = require('../../util');\n\nfunction DSA() {\n  // s1 = ((g**s) mod p) mod q\n  // s1 = ((s**-1)*(sha-1(m)+(s1*x) mod q)\n  function sign(hashalgo, m, g, p, q, x) {\n    // If the output size of the chosen hash is larger than the number of\n    // bits of q, the hash result is truncated to fit by taking the number\n    // of leftmost bits equal to the number of bits of q.  This (possibly\n    // truncated) hash function result is treated as a number and used\n    // directly in the DSA signature algorithm.\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    var k = random.getRandomBigIntegerInRange(BigInteger.ONE.add(BigInteger.ONE), q.subtract(BigInteger.ONE));\n    var s1 = (g.modPow(k, p)).mod(q);\n    var s2 = (k.modInverse(q).multiply(hash.add(x.multiply(s1)))).mod(q);\n    var result = new Array();\n    result[0] = s1.toMPI();\n    result[1] = s2.toMPI();\n    return result;\n  }\n\n  function select_hash_algorithm(q) {\n    var usersetting = openpgp.config.config.prefer_hash_algorithm;\n    /*\n     * 1024-bit key, 160-bit q, SHA-1, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 224-bit q, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     * 3072-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     */\n    switch (Math.round(q.bitLength() / 8)) {\n      case 20:\n        // 1024 bit\n        if (usersetting != 2 &&\n          usersetting > 11 &&\n          usersetting != 10 &&\n          usersetting < 8)\n          return 2; // prefer sha1\n        return usersetting;\n      case 28:\n        // 2048 bit\n        if (usersetting > 11 &&\n          usersetting < 8)\n          return 11;\n        return usersetting;\n      case 32:\n        // 4096 bit // prefer sha224\n        if (usersetting > 10 &&\n          usersetting < 8)\n          return 8; // prefer sha256\n        return usersetting;\n      default:\n        util.print_debug(\"DSA select hash algorithm: returning null for an unknown length of q\");\n        return null;\n\n    }\n  }\n  this.select_hash_algorithm = select_hash_algorithm;\n\n  function verify(hashalgo, s1, s2, m, p, q, g, y) {\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    if (BigInteger.ZERO.compareTo(s1) > 0 ||\n      s1.compareTo(q) > 0 ||\n      BigInteger.ZERO.compareTo(s2) > 0 ||\n      s2.compareTo(q) > 0) {\n      util.print_error(\"invalid DSA Signature\");\n      return null;\n    }\n    var w = s2.modInverse(q);\n    var u1 = hash.multiply(w).mod(q);\n    var u2 = s1.multiply(w).mod(q);\n    return g.modPow(u1, p).multiply(y.modPow(u2, p)).mod(p).mod(q);\n  }\n\n  /*\n\t * unused code. This can be used as a start to write a key generator\n\t * function.\n\t\n\tfunction generateKey(bitcount) {\n\t    var qi = new BigInteger(bitcount, primeCenterie);\n\t    var pi = generateP(q, 512);\n\t    var gi = generateG(p, q, bitcount);\n\t    var xi;\n\t    do {\n\t        xi = new BigInteger(q.bitCount(), rand);\n\t    } while (x.compareTo(BigInteger.ZERO) != 1 && x.compareTo(q) != -1);\n\t    var yi = g.modPow(x, p);\n\t    return {x: xi, q: qi, p: pi, g: gi, y: yi};\n\t}\n\n\tfunction generateP(q, bitlength, randomfn) {\n\t    if (bitlength % 64 != 0) {\n\t    \treturn false;\n\t    }\n\t    var pTemp;\n\t    var pTemp2;\n\t    do {\n\t        pTemp = randomfn(bitcount, true);\n\t        pTemp2 = pTemp.subtract(BigInteger.ONE);\n\t        pTemp = pTemp.subtract(pTemp2.remainder(q));\n\t    } while (!pTemp.isProbablePrime(primeCenterie) || pTemp.bitLength() != l);\n\t    return pTemp;\n\t}\n\t\n\tfunction generateG(p, q, bitlength, randomfn) {\n\t    var aux = p.subtract(BigInteger.ONE);\n\t    var pow = aux.divide(q);\n\t    var gTemp;\n\t    do {\n\t        gTemp = randomfn(bitlength);\n\t    } while (gTemp.compareTo(aux) != -1 && gTemp.compareTo(BigInteger.ONE) != 1);\n\t    return gTemp.modPow(pow, p);\n\t}\n\n\tfunction generateK(q, bitlength, randomfn) {\n\t    var tempK;\n\t    do {\n\t        tempK = randomfn(bitlength, false);\n\t    } while (tempK.compareTo(q) != -1 && tempK.compareTo(BigInteger.ZERO) != 1);\n\t    return tempK;\n\t}\n\n\tfunction generateR(q,p) {\n\t    k = generateK(q);\n\t    var r = g.modPow(k, p).mod(q);\n\t    return r;\n\t}\n\n\tfunction generateS(hashfn,k,r,m,q,x) {\n        var hash = hashfn(m);\n        s = (k.modInverse(q).multiply(hash.add(x.multiply(r)))).mod(q);\n\t    return s;\n\t} */\n  this.sign = sign;\n  this.verify = verify;\n  // this.generate = generateKey;\n}\n\nmodule.exports = DSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// ElGamal implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  util = require('../../util');\n\nfunction Elgamal() {\n\n  function encrypt(m, g, p, y) {\n    //  choose k in {2,...,p-2}\n    var two = BigInteger.ONE.add(BigInteger.ONE);\n    var pMinus2 = p.subtract(two);\n    var k = random.getRandomBigIntegerInRange(two, pMinus2);\n    k = k.mod(pMinus2).add(BigInteger.ONE);\n    var c = [];\n    c[0] = g.modPow(k, p);\n    c[1] = y.modPow(k, p).multiply(m).mod(p);\n    return c;\n  }\n\n  function decrypt(c1, c2, p, x) {\n    util.print_debug(\"Elgamal Decrypt:\\nc1:\" + util.hexstrdump(c1.toMPI()) + \"\\n\" +\n      \"c2:\" + util.hexstrdump(c2.toMPI()) + \"\\n\" +\n      \"p:\" + util.hexstrdump(p.toMPI()) + \"\\n\" +\n      \"x:\" + util.hexstrdump(x.toMPI()));\n    return (c1.modPow(x, p).modInverse(p)).multiply(c2).mod(p);\n    //var c = c1.pow(x).modInverse(p); // c0^-a mod p\n    //return c.multiply(c2).mod(p);\n  }\n\n  // signing and signature verification using Elgamal is not required by OpenPGP.\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n}\n\nmodule.exports = Elgamal;\n","module.exports = {\n  rsa: require('./rsa.js'),\n  elgamal: require('./elgamal.js'),\n  dsa: require('./dsa.js')\n}\n","/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH \n * \n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\nvar util = require('../../util');\n\n// Basic JavaScript BN library - subset useful for RSA encryption.\n\n// Bits per digit\nvar dbits;\n\n// JavaScript engine analysis\nvar canary = 0xdeadbeefcafe;\nvar j_lm = ((canary & 0xffffff) == 0xefcafe);\n\n// (public) Constructor\n\nfunction BigInteger(a, b, c) {\n  if (a != null)\n    if (\"number\" == typeof a) this.fromNumber(a, b, c);\n    else if (b == null && \"string\" != typeof a) this.fromString(a, 256);\n  else this.fromString(a, b);\n}\n\n// return new, unset BigInteger\n\nfunction nbi() {\n  return new BigInteger(null);\n}\n\n// am: Compute w_j += (x*this_i), propagate carries,\n// c is initial carry, returns final carry.\n// c < 3*dvalue, x < 2*dvalue, this_i < dvalue\n// We need to select the fastest one that works in this environment.\n\n// am1: use a single mult and divide to get the high bits,\n// max digit bits should be 26 because\n// max internal value = 2*dvalue^2-2*dvalue (< 2^53)\n\nfunction am1(i, x, w, j, c, n) {\n  while (--n >= 0) {\n    var v = x * this[i++] + w[j] + c;\n    c = Math.floor(v / 0x4000000);\n    w[j++] = v & 0x3ffffff;\n  }\n  return c;\n}\n// am2 avoids a big mult-and-extract completely.\n// Max digit bits should be <= 30 because we do bitwise ops\n// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)\n\nfunction am2(i, x, w, j, c, n) {\n  var xl = x & 0x7fff,\n    xh = x >> 15;\n  while (--n >= 0) {\n    var l = this[i] & 0x7fff;\n    var h = this[i++] >> 15;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x7fff) << 15) + w[j] + (c & 0x3fffffff);\n    c = (l >>> 30) + (m >>> 15) + xh * h + (c >>> 30);\n    w[j++] = l & 0x3fffffff;\n  }\n  return c;\n}\n// Alternately, set max digit bits to 28 since some\n// browsers slow down when dealing with 32-bit numbers.\n\nfunction am3(i, x, w, j, c, n) {\n  var xl = x & 0x3fff,\n    xh = x >> 14;\n  while (--n >= 0) {\n    var l = this[i] & 0x3fff;\n    var h = this[i++] >> 14;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x3fff) << 14) + w[j] + c;\n    c = (l >> 28) + (m >> 14) + xh * h;\n    w[j++] = l & 0xfffffff;\n  }\n  return c;\n}\n/*if(j_lm && (navigator != undefined && \n\tnavigator.appName == \"Microsoft Internet Explorer\")) {\n  BigInteger.prototype.am = am2;\n  dbits = 30;\n}\nelse if(j_lm && (navigator != undefined && navigator.appName != \"Netscape\")) {*/\nBigInteger.prototype.am = am1;\ndbits = 26;\n/*}\nelse { // Mozilla/Netscape seems to prefer am3\n  BigInteger.prototype.am = am3;\n  dbits = 28;\n}*/\n\nBigInteger.prototype.DB = dbits;\nBigInteger.prototype.DM = ((1 << dbits) - 1);\nBigInteger.prototype.DV = (1 << dbits);\n\nvar BI_FP = 52;\nBigInteger.prototype.FV = Math.pow(2, BI_FP);\nBigInteger.prototype.F1 = BI_FP - dbits;\nBigInteger.prototype.F2 = 2 * dbits - BI_FP;\n\n// Digit conversions\nvar BI_RM = \"0123456789abcdefghijklmnopqrstuvwxyz\";\nvar BI_RC = new Array();\nvar rr, vv;\nrr = \"0\".charCodeAt(0);\nfor (vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;\nrr = \"a\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\nrr = \"A\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\n\nfunction int2char(n) {\n  return BI_RM.charAt(n);\n}\n\nfunction intAt(s, i) {\n  var c = BI_RC[s.charCodeAt(i)];\n  return (c == null) ? -1 : c;\n}\n\n// (protected) copy this to r\n\nfunction bnpCopyTo(r) {\n  for (var i = this.t - 1; i >= 0; --i) r[i] = this[i];\n  r.t = this.t;\n  r.s = this.s;\n}\n\n// (protected) set from integer value x, -DV <= x < DV\n\nfunction bnpFromInt(x) {\n  this.t = 1;\n  this.s = (x < 0) ? -1 : 0;\n  if (x > 0) this[0] = x;\n  else if (x < -1) this[0] = x + DV;\n  else this.t = 0;\n}\n\n// return bigint initialized to value\n\nfunction nbv(i) {\n  var r = nbi();\n  r.fromInt(i);\n  return r;\n}\n\n// (protected) set from string and radix\n\nfunction bnpFromString(s, b) {\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 256) k = 8; // byte array\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else {\n    this.fromRadix(s, b);\n    return;\n  }\n  this.t = 0;\n  this.s = 0;\n  var i = s.length,\n    mi = false,\n    sh = 0;\n  while (--i >= 0) {\n    var x = (k == 8) ? s[i] & 0xff : intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\") mi = true;\n      continue;\n    }\n    mi = false;\n    if (sh == 0)\n      this[this.t++] = x;\n    else if (sh + k > this.DB) {\n      this[this.t - 1] |= (x & ((1 << (this.DB - sh)) - 1)) << sh;\n      this[this.t++] = (x >> (this.DB - sh));\n    } else\n      this[this.t - 1] |= x << sh;\n    sh += k;\n    if (sh >= this.DB) sh -= this.DB;\n  }\n  if (k == 8 && (s[0] & 0x80) != 0) {\n    this.s = -1;\n    if (sh > 0) this[this.t - 1] |= ((1 << (this.DB - sh)) - 1) << sh;\n  }\n  this.clamp();\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) clamp off excess high words\n\nfunction bnpClamp() {\n  var c = this.s & this.DM;\n  while (this.t > 0 && this[this.t - 1] == c)--this.t;\n}\n\n// (public) return string representation in given radix\n\nfunction bnToString(b) {\n  if (this.s < 0) return \"-\" + this.negate().toString(b);\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else return this.toRadix(b);\n  var km = (1 << k) - 1,\n    d, m = false,\n    r = \"\",\n    i = this.t;\n  var p = this.DB - (i * this.DB) % k;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) > 0) {\n      m = true;\n      r = int2char(d);\n    }\n    while (i >= 0) {\n      if (p < k) {\n        d = (this[i] & ((1 << p) - 1)) << (k - p);\n        d |= this[--i] >> (p += this.DB - k);\n      } else {\n        d = (this[i] >> (p -= k)) & km;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      if (d > 0) m = true;\n      if (m) r += int2char(d);\n    }\n  }\n  return m ? r : \"0\";\n}\n\n// (public) -this\n\nfunction bnNegate() {\n  var r = nbi();\n  BigInteger.ZERO.subTo(this, r);\n  return r;\n}\n\n// (public) |this|\n\nfunction bnAbs() {\n  return (this.s < 0) ? this.negate() : this;\n}\n\n// (public) return + if this > a, - if this < a, 0 if equal\n\nfunction bnCompareTo(a) {\n  var r = this.s - a.s;\n  if (r != 0) return r;\n  var i = this.t;\n  r = i - a.t;\n  if (r != 0) return r;\n  while (--i >= 0) if ((r = this[i] - a[i]) != 0) return r;\n  return 0;\n}\n\n// returns bit length of the integer x\n\nfunction nbits(x) {\n  var r = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    r += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    r += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    r += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    r += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    r += 1;\n  }\n  return r;\n}\n\n// (public) return the number of bits in \"this\"\n\nfunction bnBitLength() {\n  if (this.t <= 0) return 0;\n  return this.DB * (this.t - 1) + nbits(this[this.t - 1] ^ (this.s & this.DM));\n}\n\n// (protected) r = this << n*DB\n\nfunction bnpDLShiftTo(n, r) {\n  var i;\n  for (i = this.t - 1; i >= 0; --i) r[i + n] = this[i];\n  for (i = n - 1; i >= 0; --i) r[i] = 0;\n  r.t = this.t + n;\n  r.s = this.s;\n}\n\n// (protected) r = this >> n*DB\n\nfunction bnpDRShiftTo(n, r) {\n  for (var i = n; i < this.t; ++i) r[i - n] = this[i];\n  r.t = Math.max(this.t - n, 0);\n  r.s = this.s;\n}\n\n// (protected) r = this << n\n\nfunction bnpLShiftTo(n, r) {\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << cbs) - 1;\n  var ds = Math.floor(n / this.DB),\n    c = (this.s << bs) & this.DM,\n    i;\n  for (i = this.t - 1; i >= 0; --i) {\n    r[i + ds + 1] = (this[i] >> cbs) | c;\n    c = (this[i] & bm) << bs;\n  }\n  for (i = ds - 1; i >= 0; --i) r[i] = 0;\n  r[ds] = c;\n  r.t = this.t + ds + 1;\n  r.s = this.s;\n  r.clamp();\n}\n\n// (protected) r = this >> n\n\nfunction bnpRShiftTo(n, r) {\n  r.s = this.s;\n  var ds = Math.floor(n / this.DB);\n  if (ds >= this.t) {\n    r.t = 0;\n    return;\n  }\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << bs) - 1;\n  r[0] = this[ds] >> bs;\n  for (var i = ds + 1; i < this.t; ++i) {\n    r[i - ds - 1] |= (this[i] & bm) << cbs;\n    r[i - ds] = this[i] >> bs;\n  }\n  if (bs > 0) r[this.t - ds - 1] |= (this.s & bm) << cbs;\n  r.t = this.t - ds;\n  r.clamp();\n}\n\n// (protected) r = this - a\n\nfunction bnpSubTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] - a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c -= a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c -= a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c -= a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c < -1) r[i++] = this.DV + c;\n  else if (c > 0) r[i++] = c;\n  r.t = i;\n  r.clamp();\n}\n\n// (protected) r = this * a, r != this,a (HAC 14.12)\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyTo(a, r) {\n  var x = this.abs(),\n    y = a.abs();\n  var i = x.t;\n  r.t = i + y.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < y.t; ++i) r[i + x.t] = x.am(0, y[i], r, i, 0, x.t);\n  r.s = 0;\n  r.clamp();\n  if (this.s != a.s) BigInteger.ZERO.subTo(r, r);\n}\n\n// (protected) r = this^2, r != this (HAC 14.16)\n\nfunction bnpSquareTo(r) {\n  var x = this.abs();\n  var i = r.t = 2 * x.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < x.t - 1; ++i) {\n    var c = x.am(i, x[i], r, 2 * i, 0, 1);\n    if ((r[i + x.t] += x.am(i + 1, 2 * x[i], r, 2 * i + 1, c, x.t - i - 1)) >= x.DV) {\n      r[i + x.t] -= x.DV;\n      r[i + x.t + 1] = 1;\n    }\n  }\n  if (r.t > 0) r[r.t - 1] += x.am(i, x[i], r, 2 * i, 0, 1);\n  r.s = 0;\n  r.clamp();\n}\n\n// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)\n// r != q, this != m.  q or r may be null.\n\nfunction bnpDivRemTo(m, q, r) {\n  var pm = m.abs();\n  if (pm.t <= 0) return;\n  var pt = this.abs();\n  if (pt.t < pm.t) {\n    if (q != null) q.fromInt(0);\n    if (r != null) this.copyTo(r);\n    return;\n  }\n  if (r == null) r = nbi();\n  var y = nbi(),\n    ts = this.s,\n    ms = m.s;\n  var nsh = this.DB - nbits(pm[pm.t - 1]); // normalize modulus\n  if (nsh > 0) {\n    pm.lShiftTo(nsh, y);\n    pt.lShiftTo(nsh, r);\n  } else {\n    pm.copyTo(y);\n    pt.copyTo(r);\n  }\n  var ys = y.t;\n  var y0 = y[ys - 1];\n  if (y0 == 0) return;\n  var yt = y0 * (1 << this.F1) + ((ys > 1) ? y[ys - 2] >> this.F2 : 0);\n  var d1 = this.FV / yt,\n    d2 = (1 << this.F1) / yt,\n    e = 1 << this.F2;\n  var i = r.t,\n    j = i - ys,\n    t = (q == null) ? nbi() : q;\n  y.dlShiftTo(j, t);\n  if (r.compareTo(t) >= 0) {\n    r[r.t++] = 1;\n    r.subTo(t, r);\n  }\n  BigInteger.ONE.dlShiftTo(ys, t);\n  t.subTo(y, y); // \"negative\" y so we can replace sub with am later\n  while (y.t < ys) y[y.t++] = 0;\n  while (--j >= 0) {\n    // Estimate quotient digit\n    var qd = (r[--i] == y0) ? this.DM : Math.floor(r[i] * d1 + (r[i - 1] + e) * d2);\n    if ((r[i] += y.am(0, qd, r, j, 0, ys)) < qd) { // Try it out\n      y.dlShiftTo(j, t);\n      r.subTo(t, r);\n      while (r[i] < --qd) r.subTo(t, r);\n    }\n  }\n  if (q != null) {\n    r.drShiftTo(ys, q);\n    if (ts != ms) BigInteger.ZERO.subTo(q, q);\n  }\n  r.t = ys;\n  r.clamp();\n  if (nsh > 0) r.rShiftTo(nsh, r); // Denormalize remainder\n  if (ts < 0) BigInteger.ZERO.subTo(r, r);\n}\n\n// (public) this mod a\n\nfunction bnMod(a) {\n  var r = nbi();\n  this.abs().divRemTo(a, null, r);\n  if (this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r, r);\n  return r;\n}\n\n// Modular reduction using \"classic\" algorithm\n\nfunction Classic(m) {\n  this.m = m;\n}\n\nfunction cConvert(x) {\n  if (x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);\n  else return x;\n}\n\nfunction cRevert(x) {\n  return x;\n}\n\nfunction cReduce(x) {\n  x.divRemTo(this.m, null, x);\n}\n\nfunction cMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nfunction cSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\nClassic.prototype.convert = cConvert;\nClassic.prototype.revert = cRevert;\nClassic.prototype.reduce = cReduce;\nClassic.prototype.mulTo = cMulTo;\nClassic.prototype.sqrTo = cSqrTo;\n\n// (protected) return \"-1/this % 2^DB\"; useful for Mont. reduction\n// justification:\n//         xy == 1 (mod m)\n//         xy =  1+km\n//   xy(2-xy) = (1+km)(1-km)\n// x[y(2-xy)] = 1-k^2m^2\n// x[y(2-xy)] == 1 (mod m^2)\n// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2\n// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.\n// JS multiply \"overflows\" differently from C/C++, so care is needed here.\n\nfunction bnpInvDigit() {\n  if (this.t < 1) return 0;\n  var x = this[0];\n  if ((x & 1) == 0) return 0;\n  var y = x & 3; // y == 1/x mod 2^2\n  y = (y * (2 - (x & 0xf) * y)) & 0xf; // y == 1/x mod 2^4\n  y = (y * (2 - (x & 0xff) * y)) & 0xff; // y == 1/x mod 2^8\n  y = (y * (2 - (((x & 0xffff) * y) & 0xffff))) & 0xffff; // y == 1/x mod 2^16\n  // last step - calculate inverse mod DV directly;\n  // assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints\n  y = (y * (2 - x * y % this.DV)) % this.DV; // y == 1/x mod 2^dbits\n  // we really want the negative inverse, and -DV < y < DV\n  return (y > 0) ? this.DV - y : -y;\n}\n\n// Montgomery reduction\n\nfunction Montgomery(m) {\n  this.m = m;\n  this.mp = m.invDigit();\n  this.mpl = this.mp & 0x7fff;\n  this.mph = this.mp >> 15;\n  this.um = (1 << (m.DB - 15)) - 1;\n  this.mt2 = 2 * m.t;\n}\n\n// xR mod m\n\nfunction montConvert(x) {\n  var r = nbi();\n  x.abs().dlShiftTo(this.m.t, r);\n  r.divRemTo(this.m, null, r);\n  if (x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r, r);\n  return r;\n}\n\n// x/R mod m\n\nfunction montRevert(x) {\n  var r = nbi();\n  x.copyTo(r);\n  this.reduce(r);\n  return r;\n}\n\n// x = x/R mod m (HAC 14.32)\n\nfunction montReduce(x) {\n  while (x.t <= this.mt2) // pad x so am has enough room later\n    x[x.t++] = 0;\n  for (var i = 0; i < this.m.t; ++i) {\n    // faster way of calculating u0 = x[i]*mp mod DV\n    var j = x[i] & 0x7fff;\n    var u0 = (j * this.mpl + (((j * this.mph + (x[i] >> 15) * this.mpl) & this.um) << 15)) & x.DM;\n    // use am to combine the multiply-shift-add into one call\n    j = i + this.m.t;\n    x[j] += this.m.am(0, u0, x, i, 0, this.m.t);\n    // propagate carry\n    while (x[j] >= x.DV) {\n      x[j] -= x.DV;\n      x[++j]++;\n    }\n  }\n  x.clamp();\n  x.drShiftTo(this.m.t, x);\n  if (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = \"x^2/R mod m\"; x != r\n\nfunction montSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = \"xy/R mod m\"; x,y != r\n\nfunction montMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nMontgomery.prototype.convert = montConvert;\nMontgomery.prototype.revert = montRevert;\nMontgomery.prototype.reduce = montReduce;\nMontgomery.prototype.mulTo = montMulTo;\nMontgomery.prototype.sqrTo = montSqrTo;\n\n// (protected) true iff this is even\n\nfunction bnpIsEven() {\n  return ((this.t > 0) ? (this[0] & 1) : this.s) == 0;\n}\n\n// (protected) this^e, e < 2^32, doing sqr and mul with \"r\" (HAC 14.79)\n\nfunction bnpExp(e, z) {\n  if (e > 0xffffffff || e < 1) return BigInteger.ONE;\n  var r = nbi(),\n    r2 = nbi(),\n    g = z.convert(this),\n    i = nbits(e) - 1;\n  g.copyTo(r);\n  while (--i >= 0) {\n    z.sqrTo(r, r2);\n    if ((e & (1 << i)) > 0) z.mulTo(r2, g, r);\n    else {\n      var t = r;\n      r = r2;\n      r2 = t;\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) this^e % m, 0 <= e < 2^32\n\nfunction bnModPowInt(e, m) {\n  var z;\n  if (e < 256 || m.isEven()) z = new Classic(m);\n  else z = new Montgomery(m);\n  return this.exp(e, z);\n}\n\n// protected\nBigInteger.prototype.copyTo = bnpCopyTo;\nBigInteger.prototype.fromInt = bnpFromInt;\nBigInteger.prototype.fromString = bnpFromString;\nBigInteger.prototype.clamp = bnpClamp;\nBigInteger.prototype.dlShiftTo = bnpDLShiftTo;\nBigInteger.prototype.drShiftTo = bnpDRShiftTo;\nBigInteger.prototype.lShiftTo = bnpLShiftTo;\nBigInteger.prototype.rShiftTo = bnpRShiftTo;\nBigInteger.prototype.subTo = bnpSubTo;\nBigInteger.prototype.multiplyTo = bnpMultiplyTo;\nBigInteger.prototype.squareTo = bnpSquareTo;\nBigInteger.prototype.divRemTo = bnpDivRemTo;\nBigInteger.prototype.invDigit = bnpInvDigit;\nBigInteger.prototype.isEven = bnpIsEven;\nBigInteger.prototype.exp = bnpExp;\n\n// public\nBigInteger.prototype.toString = bnToString;\nBigInteger.prototype.negate = bnNegate;\nBigInteger.prototype.abs = bnAbs;\nBigInteger.prototype.compareTo = bnCompareTo;\nBigInteger.prototype.bitLength = bnBitLength;\nBigInteger.prototype.mod = bnMod;\nBigInteger.prototype.modPowInt = bnModPowInt;\n\n// \"constants\"\nBigInteger.ZERO = nbv(0);\nBigInteger.ONE = nbv(1);\n\nmodule.exports = BigInteger;\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH\n *\n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\n// Extended JavaScript BN functions, required for RSA private ops.\n\n// Version 1.1: new BigInteger(\"0\", 10) returns \"proper\" zero\n// Version 1.2: square() API, isProbablePrime fix\n\n// (public)\nfunction bnClone() {\n  var r = nbi();\n  this.copyTo(r);\n  return r;\n}\n\n// (public) return value as integer\n\nfunction bnIntValue() {\n  if (this.s < 0) {\n    if (this.t == 1) return this[0] - this.DV;\n    else if (this.t == 0) return -1;\n  } else if (this.t == 1) return this[0];\n  else if (this.t == 0) return 0;\n  // assumes 16 < DB < 32\n  return ((this[1] & ((1 << (32 - this.DB)) - 1)) << this.DB) | this[0];\n}\n\n// (public) return value as byte\n\nfunction bnByteValue() {\n  return (this.t == 0) ? this.s : (this[0] << 24) >> 24;\n}\n\n// (public) return value as short (assumes DB>=16)\n\nfunction bnShortValue() {\n  return (this.t == 0) ? this.s : (this[0] << 16) >> 16;\n}\n\n// (protected) return x s.t. r^x < DV\n\nfunction bnpChunkSize(r) {\n  return Math.floor(Math.LN2 * this.DB / Math.log(r));\n}\n\n// (public) 0 if this == 0, 1 if this > 0\n\nfunction bnSigNum() {\n  if (this.s < 0) return -1;\n  else if (this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;\n  else return 1;\n}\n\n// (protected) convert to radix string\n\nfunction bnpToRadix(b) {\n  if (b == null) b = 10;\n  if (this.signum() == 0 || b < 2 || b > 36) return \"0\";\n  var cs = this.chunkSize(b);\n  var a = Math.pow(b, cs);\n  var d = nbv(a),\n    y = nbi(),\n    z = nbi(),\n    r = \"\";\n  this.divRemTo(d, y, z);\n  while (y.signum() > 0) {\n    r = (a + z.intValue()).toString(b).substr(1) + r;\n    y.divRemTo(d, y, z);\n  }\n  return z.intValue().toString(b) + r;\n}\n\n// (protected) convert from radix string\n\nfunction bnpFromRadix(s, b) {\n  this.fromInt(0);\n  if (b == null) b = 10;\n  var cs = this.chunkSize(b);\n  var d = Math.pow(b, cs),\n    mi = false,\n    j = 0,\n    w = 0;\n  for (var i = 0; i < s.length; ++i) {\n    var x = intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\" && this.signum() == 0) mi = true;\n      continue;\n    }\n    w = b * w + x;\n    if (++j >= cs) {\n      this.dMultiply(d);\n      this.dAddOffset(w, 0);\n      j = 0;\n      w = 0;\n    }\n  }\n  if (j > 0) {\n    this.dMultiply(Math.pow(b, j));\n    this.dAddOffset(w, 0);\n  }\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) alternate constructor\n\nfunction bnpFromNumber(a, b, c) {\n  if (\"number\" == typeof b) {\n    // new BigInteger(int,int,RNG)\n    if (a < 2) this.fromInt(1);\n    else {\n      this.fromNumber(a, c);\n      if (!this.testBit(a - 1)) // force MSB set\n        this.bitwiseTo(BigInteger.ONE.shiftLeft(a - 1), op_or, this);\n      if (this.isEven()) this.dAddOffset(1, 0); // force odd\n      while (!this.isProbablePrime(b)) {\n        this.dAddOffset(2, 0);\n        if (this.bitLength() > a) this.subTo(BigInteger.ONE.shiftLeft(a - 1), this);\n      }\n    }\n  } else {\n    // new BigInteger(int,RNG)\n    var x = new Array(),\n      t = a & 7;\n    x.length = (a >> 3) + 1;\n    b.nextBytes(x);\n    if (t > 0) x[0] &= ((1 << t) - 1);\n    else x[0] = 0;\n    this.fromString(x, 256);\n  }\n}\n\n// (public) convert to bigendian byte array\n\nfunction bnToByteArray() {\n  var i = this.t,\n    r = new Array();\n  r[0] = this.s;\n  var p = this.DB - (i * this.DB) % 8,\n    d, k = 0;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) != (this.s & this.DM) >> p)\n      r[k++] = d | (this.s << (this.DB - p));\n    while (i >= 0) {\n      if (p < 8) {\n        d = (this[i] & ((1 << p) - 1)) << (8 - p);\n        d |= this[--i] >> (p += this.DB - 8);\n      } else {\n        d = (this[i] >> (p -= 8)) & 0xff;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      //if((d&0x80) != 0) d |= -256;\n      //if(k == 0 && (this.s&0x80) != (d&0x80)) ++k;\n      if (k > 0 || d != this.s) r[k++] = d;\n    }\n  }\n  return r;\n}\n\nfunction bnEquals(a) {\n  return (this.compareTo(a) == 0);\n}\n\nfunction bnMin(a) {\n  return (this.compareTo(a) < 0) ? this : a;\n}\n\nfunction bnMax(a) {\n  return (this.compareTo(a) > 0) ? this : a;\n}\n\n// (protected) r = this op a (bitwise)\n\nfunction bnpBitwiseTo(a, op, r) {\n  var i, f, m = Math.min(a.t, this.t);\n  for (i = 0; i < m; ++i) r[i] = op(this[i], a[i]);\n  if (a.t < this.t) {\n    f = a.s & this.DM;\n    for (i = m; i < this.t; ++i) r[i] = op(this[i], f);\n    r.t = this.t;\n  } else {\n    f = this.s & this.DM;\n    for (i = m; i < a.t; ++i) r[i] = op(f, a[i]);\n    r.t = a.t;\n  }\n  r.s = op(this.s, a.s);\n  r.clamp();\n}\n\n// (public) this & a\n\nfunction op_and(x, y) {\n  return x & y;\n}\n\nfunction bnAnd(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_and, r);\n  return r;\n}\n\n// (public) this | a\n\nfunction op_or(x, y) {\n  return x | y;\n}\n\nfunction bnOr(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_or, r);\n  return r;\n}\n\n// (public) this ^ a\n\nfunction op_xor(x, y) {\n  return x ^ y;\n}\n\nfunction bnXor(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_xor, r);\n  return r;\n}\n\n// (public) this & ~a\n\nfunction op_andnot(x, y) {\n  return x & ~y;\n}\n\nfunction bnAndNot(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_andnot, r);\n  return r;\n}\n\n// (public) ~this\n\nfunction bnNot() {\n  var r = nbi();\n  for (var i = 0; i < this.t; ++i) r[i] = this.DM & ~this[i];\n  r.t = this.t;\n  r.s = ~this.s;\n  return r;\n}\n\n// (public) this << n\n\nfunction bnShiftLeft(n) {\n  var r = nbi();\n  if (n < 0) this.rShiftTo(-n, r);\n  else this.lShiftTo(n, r);\n  return r;\n}\n\n// (public) this >> n\n\nfunction bnShiftRight(n) {\n  var r = nbi();\n  if (n < 0) this.lShiftTo(-n, r);\n  else this.rShiftTo(n, r);\n  return r;\n}\n\n// return index of lowest 1-bit in x, x < 2^31\n\nfunction lbit(x) {\n  if (x == 0) return -1;\n  var r = 0;\n  if ((x & 0xffff) == 0) {\n    x >>= 16;\n    r += 16;\n  }\n  if ((x & 0xff) == 0) {\n    x >>= 8;\n    r += 8;\n  }\n  if ((x & 0xf) == 0) {\n    x >>= 4;\n    r += 4;\n  }\n  if ((x & 3) == 0) {\n    x >>= 2;\n    r += 2;\n  }\n  if ((x & 1) == 0)++r;\n  return r;\n}\n\n// (public) returns index of lowest 1-bit (or -1 if none)\n\nfunction bnGetLowestSetBit() {\n  for (var i = 0; i < this.t; ++i)\n    if (this[i] != 0) return i * this.DB + lbit(this[i]);\n  if (this.s < 0) return this.t * this.DB;\n  return -1;\n}\n\n// return number of 1 bits in x\n\nfunction cbit(x) {\n  var r = 0;\n  while (x != 0) {\n    x &= x - 1;\n    ++r;\n  }\n  return r;\n}\n\n// (public) return number of set bits\n\nfunction bnBitCount() {\n  var r = 0,\n    x = this.s & this.DM;\n  for (var i = 0; i < this.t; ++i) r += cbit(this[i] ^ x);\n  return r;\n}\n\n// (public) true iff nth bit is set\n\nfunction bnTestBit(n) {\n  var j = Math.floor(n / this.DB);\n  if (j >= this.t) return (this.s != 0);\n  return ((this[j] & (1 << (n % this.DB))) != 0);\n}\n\n// (protected) this op (1<<n)\n\nfunction bnpChangeBit(n, op) {\n  var r = BigInteger.ONE.shiftLeft(n);\n  this.bitwiseTo(r, op, r);\n  return r;\n}\n\n// (public) this | (1<<n)\n\nfunction bnSetBit(n) {\n  return this.changeBit(n, op_or);\n}\n\n// (public) this & ~(1<<n)\n\nfunction bnClearBit(n) {\n  return this.changeBit(n, op_andnot);\n}\n\n// (public) this ^ (1<<n)\n\nfunction bnFlipBit(n) {\n  return this.changeBit(n, op_xor);\n}\n\n// (protected) r = this + a\n\nfunction bnpAddTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] + a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c += a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c += a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c > 0) r[i++] = c;\n  else if (c < -1) r[i++] = this.DV + c;\n  r.t = i;\n  r.clamp();\n}\n\n// (public) this + a\n\nfunction bnAdd(a) {\n  var r = nbi();\n  this.addTo(a, r);\n  return r;\n}\n\n// (public) this - a\n\nfunction bnSubtract(a) {\n  var r = nbi();\n  this.subTo(a, r);\n  return r;\n}\n\n// (public) this * a\n\nfunction bnMultiply(a) {\n  var r = nbi();\n  this.multiplyTo(a, r);\n  return r;\n}\n\n// (public) this^2\n\nfunction bnSquare() {\n  var r = nbi();\n  this.squareTo(r);\n  return r;\n}\n\n// (public) this / a\n\nfunction bnDivide(a) {\n  var r = nbi();\n  this.divRemTo(a, r, null);\n  return r;\n}\n\n// (public) this % a\n\nfunction bnRemainder(a) {\n  var r = nbi();\n  this.divRemTo(a, null, r);\n  return r;\n}\n\n// (public) [this/a,this%a]\n\nfunction bnDivideAndRemainder(a) {\n  var q = nbi(),\n    r = nbi();\n  this.divRemTo(a, q, r);\n  return new Array(q, r);\n}\n\n// (protected) this *= n, this >= 0, 1 < n < DV\n\nfunction bnpDMultiply(n) {\n  this[this.t] = this.am(0, n - 1, this, 0, 0, this.t);\n  ++this.t;\n  this.clamp();\n}\n\n// (protected) this += n << w words, this >= 0\n\nfunction bnpDAddOffset(n, w) {\n  if (n == 0) return;\n  while (this.t <= w) this[this.t++] = 0;\n  this[w] += n;\n  while (this[w] >= this.DV) {\n    this[w] -= this.DV;\n    if (++w >= this.t) this[this.t++] = 0;\n    ++this[w];\n  }\n}\n\n// A \"null\" reducer\n\nfunction NullExp() {}\n\nfunction nNop(x) {\n  return x;\n}\n\nfunction nMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n}\n\nfunction nSqrTo(x, r) {\n  x.squareTo(r);\n}\n\nNullExp.prototype.convert = nNop;\nNullExp.prototype.revert = nNop;\nNullExp.prototype.mulTo = nMulTo;\nNullExp.prototype.sqrTo = nSqrTo;\n\n// (public) this^e\n\nfunction bnPow(e) {\n  return this.exp(e, new NullExp());\n}\n\n// (protected) r = lower n words of \"this * a\", a.t <= n\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyLowerTo(a, n, r) {\n  var i = Math.min(this.t + a.t, n);\n  r.s = 0; // assumes a,this >= 0\n  r.t = i;\n  while (i > 0) r[--i] = 0;\n  var j;\n  for (j = r.t - this.t; i < j; ++i) r[i + this.t] = this.am(0, a[i], r, i, 0, this.t);\n  for (j = Math.min(a.t, n); i < j; ++i) this.am(0, a[i], r, i, 0, n - i);\n  r.clamp();\n}\n\n// (protected) r = \"this * a\" without lower n words, n > 0\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyUpperTo(a, n, r) {\n  --n;\n  var i = r.t = this.t + a.t - n;\n  r.s = 0; // assumes a,this >= 0\n  while (--i >= 0) r[i] = 0;\n  for (i = Math.max(n - this.t, 0); i < a.t; ++i)\n    r[this.t + i - n] = this.am(n - i, a[i], r, 0, 0, this.t + i - n);\n  r.clamp();\n  r.drShiftTo(1, r);\n}\n\n// Barrett modular reduction\n\nfunction Barrett(m) {\n  // setup Barrett\n  this.r2 = nbi();\n  this.q3 = nbi();\n  BigInteger.ONE.dlShiftTo(2 * m.t, this.r2);\n  this.mu = this.r2.divide(m);\n  this.m = m;\n}\n\nfunction barrettConvert(x) {\n  if (x.s < 0 || x.t > 2 * this.m.t) return x.mod(this.m);\n  else if (x.compareTo(this.m) < 0) return x;\n  else {\n    var r = nbi();\n    x.copyTo(r);\n    this.reduce(r);\n    return r;\n  }\n}\n\nfunction barrettRevert(x) {\n  return x;\n}\n\n// x = x mod m (HAC 14.42)\n\nfunction barrettReduce(x) {\n  x.drShiftTo(this.m.t - 1, this.r2);\n  if (x.t > this.m.t + 1) {\n    x.t = this.m.t + 1;\n    x.clamp();\n  }\n  this.mu.multiplyUpperTo(this.r2, this.m.t + 1, this.q3);\n  this.m.multiplyLowerTo(this.q3, this.m.t + 1, this.r2);\n  while (x.compareTo(this.r2) < 0) x.dAddOffset(1, this.m.t + 1);\n  x.subTo(this.r2, x);\n  while (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = x^2 mod m; x != r\n\nfunction barrettSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = x*y mod m; x,y != r\n\nfunction barrettMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nBarrett.prototype.convert = barrettConvert;\nBarrett.prototype.revert = barrettRevert;\nBarrett.prototype.reduce = barrettReduce;\nBarrett.prototype.mulTo = barrettMulTo;\nBarrett.prototype.sqrTo = barrettSqrTo;\n\n// (public) this^e % m (HAC 14.85)\n\nfunction bnModPow(e, m) {\n  var i = e.bitLength(),\n    k, r = nbv(1),\n    z;\n  if (i <= 0) return r;\n  else if (i < 18) k = 1;\n  else if (i < 48) k = 3;\n  else if (i < 144) k = 4;\n  else if (i < 768) k = 5;\n  else k = 6;\n  if (i < 8)\n    z = new Classic(m);\n  else if (m.isEven())\n    z = new Barrett(m);\n  else\n    z = new Montgomery(m);\n\n  // precomputation\n  var g = new Array(),\n    n = 3,\n    k1 = k - 1,\n    km = (1 << k) - 1;\n  g[1] = z.convert(this);\n  if (k > 1) {\n    var g2 = nbi();\n    z.sqrTo(g[1], g2);\n    while (n <= km) {\n      g[n] = nbi();\n      z.mulTo(g2, g[n - 2], g[n]);\n      n += 2;\n    }\n  }\n\n  var j = e.t - 1,\n    w, is1 = true,\n    r2 = nbi(),\n    t;\n  i = nbits(e[j]) - 1;\n  while (j >= 0) {\n    if (i >= k1) w = (e[j] >> (i - k1)) & km;\n    else {\n      w = (e[j] & ((1 << (i + 1)) - 1)) << (k1 - i);\n      if (j > 0) w |= e[j - 1] >> (this.DB + i - k1);\n    }\n\n    n = k;\n    while ((w & 1) == 0) {\n      w >>= 1;\n      --n;\n    }\n    if ((i -= n) < 0) {\n      i += this.DB;\n      --j;\n    }\n    if (is1) { // ret == 1, don't bother squaring or multiplying it\n      g[w].copyTo(r);\n      is1 = false;\n    } else {\n      while (n > 1) {\n        z.sqrTo(r, r2);\n        z.sqrTo(r2, r);\n        n -= 2;\n      }\n      if (n > 0) z.sqrTo(r, r2);\n      else {\n        t = r;\n        r = r2;\n        r2 = t;\n      }\n      z.mulTo(r2, g[w], r);\n    }\n\n    while (j >= 0 && (e[j] & (1 << i)) == 0) {\n      z.sqrTo(r, r2);\n      t = r;\n      r = r2;\n      r2 = t;\n      if (--i < 0) {\n        i = this.DB - 1;\n        --j;\n      }\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) gcd(this,a) (HAC 14.54)\n\nfunction bnGCD(a) {\n  var x = (this.s < 0) ? this.negate() : this.clone();\n  var y = (a.s < 0) ? a.negate() : a.clone();\n  if (x.compareTo(y) < 0) {\n    var t = x;\n    x = y;\n    y = t;\n  }\n  var i = x.getLowestSetBit(),\n    g = y.getLowestSetBit();\n  if (g < 0) return x;\n  if (i < g) g = i;\n  if (g > 0) {\n    x.rShiftTo(g, x);\n    y.rShiftTo(g, y);\n  }\n  while (x.signum() > 0) {\n    if ((i = x.getLowestSetBit()) > 0) x.rShiftTo(i, x);\n    if ((i = y.getLowestSetBit()) > 0) y.rShiftTo(i, y);\n    if (x.compareTo(y) >= 0) {\n      x.subTo(y, x);\n      x.rShiftTo(1, x);\n    } else {\n      y.subTo(x, y);\n      y.rShiftTo(1, y);\n    }\n  }\n  if (g > 0) y.lShiftTo(g, y);\n  return y;\n}\n\n// (protected) this % n, n < 2^26\n\nfunction bnpModInt(n) {\n  if (n <= 0) return 0;\n  var d = this.DV % n,\n    r = (this.s < 0) ? n - 1 : 0;\n  if (this.t > 0)\n    if (d == 0) r = this[0] % n;\n    else for (var i = this.t - 1; i >= 0; --i) r = (d * r + this[i]) % n;\n  return r;\n}\n\n// (public) 1/this % m (HAC 14.61)\n\nfunction bnModInverse(m) {\n  var ac = m.isEven();\n  if ((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;\n  var u = m.clone(),\n    v = this.clone();\n  var a = nbv(1),\n    b = nbv(0),\n    c = nbv(0),\n    d = nbv(1);\n  while (u.signum() != 0) {\n    while (u.isEven()) {\n      u.rShiftTo(1, u);\n      if (ac) {\n        if (!a.isEven() || !b.isEven()) {\n          a.addTo(this, a);\n          b.subTo(m, b);\n        }\n        a.rShiftTo(1, a);\n      } else if (!b.isEven()) b.subTo(m, b);\n      b.rShiftTo(1, b);\n    }\n    while (v.isEven()) {\n      v.rShiftTo(1, v);\n      if (ac) {\n        if (!c.isEven() || !d.isEven()) {\n          c.addTo(this, c);\n          d.subTo(m, d);\n        }\n        c.rShiftTo(1, c);\n      } else if (!d.isEven()) d.subTo(m, d);\n      d.rShiftTo(1, d);\n    }\n    if (u.compareTo(v) >= 0) {\n      u.subTo(v, u);\n      if (ac) a.subTo(c, a);\n      b.subTo(d, b);\n    } else {\n      v.subTo(u, v);\n      if (ac) c.subTo(a, c);\n      d.subTo(b, d);\n    }\n  }\n  if (v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;\n  if (d.compareTo(m) >= 0) return d.subtract(m);\n  if (d.signum() < 0) d.addTo(m, d);\n  else return d;\n  if (d.signum() < 0) return d.add(m);\n  else return d;\n}\n\nvar lowprimes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97, 101,\n    103, 107, 109, 113, 127, 131, 137, 139, 149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223, 227,\n    229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293, 307, 311, 313, 317, 331, 337, 347, 349, 353, 359,\n    367, 373, 379, 383, 389, 397, 401, 409, 419, 421, 431, 433, 439, 443, 449, 457, 461, 463, 467, 479, 487, 491, 499,\n    503, 509, 521, 523, 541, 547, 557, 563, 569, 571, 577, 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, 643, 647,\n    653, 659, 661, 673, 677, 683, 691, 701, 709, 719, 727, 733, 739, 743, 751, 757, 761, 769, 773, 787, 797, 809, 811,\n    821, 823, 827, 829, 839, 853, 857, 859, 863, 877, 881, 883, 887, 907, 911, 919, 929, 937, 941, 947, 953, 967, 971,\n    977, 983, 991, 997\n];\nvar lplim = (1 << 26) / lowprimes[lowprimes.length - 1];\n\n// (public) test primality with certainty >= 1-.5^t\n\nfunction bnIsProbablePrime(t) {\n  var i, x = this.abs();\n  if (x.t == 1 && x[0] <= lowprimes[lowprimes.length - 1]) {\n    for (i = 0; i < lowprimes.length; ++i)\n      if (x[0] == lowprimes[i]) return true;\n    return false;\n  }\n  if (x.isEven()) return false;\n  i = 1;\n  while (i < lowprimes.length) {\n    var m = lowprimes[i],\n      j = i + 1;\n    while (j < lowprimes.length && m < lplim) m *= lowprimes[j++];\n    m = x.modInt(m);\n    while (i < j) if (m % lowprimes[i++] == 0) return false;\n  }\n  return x.millerRabin(t);\n}\n\n/* added by Recurity Labs */\n\nfunction nbits(x) {\n  var n = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    n += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    n += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    n += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    n += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    n += 1;\n  }\n  return n;\n}\n\nfunction bnToMPI() {\n  var ba = this.toByteArray();\n  var size = (ba.length - 1) * 8 + nbits(ba[0]);\n  var result = \"\";\n  result += String.fromCharCode((size & 0xFF00) >> 8);\n  result += String.fromCharCode(size & 0xFF);\n  result += util.bin2str(ba);\n  return result;\n}\n/* END of addition */\n\n// (protected) true if probably prime (HAC 4.24, Miller-Rabin)\nfunction bnpMillerRabin(t) {\n  var n1 = this.subtract(BigInteger.ONE);\n  var k = n1.getLowestSetBit();\n  if (k <= 0) return false;\n  var r = n1.shiftRight(k);\n  t = (t + 1) >> 1;\n  if (t > lowprimes.length) t = lowprimes.length;\n  var a = nbi();\n  var j, bases = [];\n  for (var i = 0; i < t; ++i) {\n    //Pick bases at random, instead of starting at 2\n    for (;;) {\n      j = lowprimes[Math.floor(Math.random() * lowprimes.length)];\n      if (bases.indexOf(j) == -1) break;\n    }\n    bases.push(j);\n    a.fromInt(j);\n    var y = a.modPow(r, this);\n    if (y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {\n      var j = 1;\n      while (j++ < k && y.compareTo(n1) != 0) {\n        y = y.modPowInt(2, this);\n        if (y.compareTo(BigInteger.ONE) == 0) return false;\n      }\n      if (y.compareTo(n1) != 0) return false;\n    }\n  }\n  return true;\n}\n\nvar BigInteger = require('./jsbn.js');\n\n// protected\nBigInteger.prototype.chunkSize = bnpChunkSize;\nBigInteger.prototype.toRadix = bnpToRadix;\nBigInteger.prototype.fromRadix = bnpFromRadix;\nBigInteger.prototype.fromNumber = bnpFromNumber;\nBigInteger.prototype.bitwiseTo = bnpBitwiseTo;\nBigInteger.prototype.changeBit = bnpChangeBit;\nBigInteger.prototype.addTo = bnpAddTo;\nBigInteger.prototype.dMultiply = bnpDMultiply;\nBigInteger.prototype.dAddOffset = bnpDAddOffset;\nBigInteger.prototype.multiplyLowerTo = bnpMultiplyLowerTo;\nBigInteger.prototype.multiplyUpperTo = bnpMultiplyUpperTo;\nBigInteger.prototype.modInt = bnpModInt;\nBigInteger.prototype.millerRabin = bnpMillerRabin;\n\n// public\nBigInteger.prototype.clone = bnClone;\nBigInteger.prototype.intValue = bnIntValue;\nBigInteger.prototype.byteValue = bnByteValue;\nBigInteger.prototype.shortValue = bnShortValue;\nBigInteger.prototype.signum = bnSigNum;\nBigInteger.prototype.toByteArray = bnToByteArray;\nBigInteger.prototype.equals = bnEquals;\nBigInteger.prototype.min = bnMin;\nBigInteger.prototype.max = bnMax;\nBigInteger.prototype.and = bnAnd;\nBigInteger.prototype.or = bnOr;\nBigInteger.prototype.xor = bnXor;\nBigInteger.prototype.andNot = bnAndNot;\nBigInteger.prototype.not = bnNot;\nBigInteger.prototype.shiftLeft = bnShiftLeft;\nBigInteger.prototype.shiftRight = bnShiftRight;\nBigInteger.prototype.getLowestSetBit = bnGetLowestSetBit;\nBigInteger.prototype.bitCount = bnBitCount;\nBigInteger.prototype.testBit = bnTestBit;\nBigInteger.prototype.setBit = bnSetBit;\nBigInteger.prototype.clearBit = bnClearBit;\nBigInteger.prototype.flipBit = bnFlipBit;\nBigInteger.prototype.add = bnAdd;\nBigInteger.prototype.subtract = bnSubtract;\nBigInteger.prototype.multiply = bnMultiply;\nBigInteger.prototype.divide = bnDivide;\nBigInteger.prototype.remainder = bnRemainder;\nBigInteger.prototype.divideAndRemainder = bnDivideAndRemainder;\nBigInteger.prototype.modPow = bnModPow;\nBigInteger.prototype.modInverse = bnModInverse;\nBigInteger.prototype.pow = bnPow;\nBigInteger.prototype.gcd = bnGCD;\nBigInteger.prototype.isProbablePrime = bnIsProbablePrime;\nBigInteger.prototype.toMPI = bnToMPI;\n\n// JSBN-specific extension\nBigInteger.prototype.square = bnSquare;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// RSA implementation\n\nvar BigInteger = require('./jsbn.js'),\n  util = require('../../util'),\n  random = require('../random.js');\n\nfunction SecureRandom() {\n  function nextBytes(byteArray) {\n    for (var n = 0; n < byteArray.length; n++) {\n      byteArray[n] = random.getSecureRandomOctet();\n    }\n  }\n  this.nextBytes = nextBytes;\n}\n\nfunction RSA() {\n  /**\n   * This function uses jsbn Big Num library to decrypt RSA\n   * @param m\n   *            message\n   * @param d\n   *            RSA d as BigInteger\n   * @param p\n   *            RSA p as BigInteger\n   * @param q\n   *            RSA q as BigInteger\n   * @param u\n   *            RSA u as BigInteger\n   * @return {BigInteger} The decrypted value of the message\n   */\n  function decrypt(m, d, p, q, u) {\n    var xp = m.mod(p).modPow(d.mod(p.subtract(BigInteger.ONE)), p);\n    var xq = m.mod(q).modPow(d.mod(q.subtract(BigInteger.ONE)), q);\n    util.print_debug(\"rsa.js decrypt\\nxpn:\" + util.hexstrdump(xp.toMPI()) + \"\\nxqn:\" + util.hexstrdump(xq.toMPI()));\n\n    var t = xq.subtract(xp);\n    if (t[0] == 0) {\n      t = xp.subtract(xq);\n      t = t.multiply(u).mod(q);\n      t = q.subtract(t);\n    } else {\n      t = t.multiply(u).mod(q);\n    }\n    return t.multiply(p).add(xp);\n  }\n\n  /**\n   * encrypt message\n   * @param m message as BigInteger\n   * @param e public MPI part as BigInteger\n   * @param n public MPI part as BigInteger\n   * @return BigInteger\n   */\n  function encrypt(m, e, n) {\n    return m.modPowInt(e, n);\n  }\n\n  /* Sign and Verify */\n  function sign(m, d, n) {\n    return m.modPow(d, n);\n  }\n\n  function verify(x, e, n) {\n    return x.modPowInt(e, n);\n  }\n\n  // \"empty\" RSA key constructor\n\n  function keyObject() {\n    this.n = null;\n    this.e = 0;\n    this.ee = null;\n    this.d = null;\n    this.p = null;\n    this.q = null;\n    this.dmp1 = null;\n    this.dmq1 = null;\n    this.u = null;\n  }\n\n  // Generate a new random private key B bits long, using public expt E\n\n  function generate(B, E) {\n    var key = new keyObject();\n    var rng = new SecureRandom();\n    var qs = B >> 1;\n    key.e = parseInt(E, 16);\n    key.ee = new BigInteger(E, 16);\n    for (;;) {\n      for (;;) {\n        key.p = new BigInteger(B - qs, 1, rng);\n        if (key.p.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.p.isProbablePrime(10))\n          break;\n      }\n      for (;;) {\n        key.q = new BigInteger(qs, 1, rng);\n        if (key.q.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.q.isProbablePrime(10))\n          break;\n      }\n      if (key.p.compareTo(key.q) <= 0) {\n        var t = key.p;\n        key.p = key.q;\n        key.q = t;\n      }\n      var p1 = key.p.subtract(BigInteger.ONE);\n      var q1 = key.q.subtract(BigInteger.ONE);\n      var phi = p1.multiply(q1);\n      if (phi.gcd(key.ee).compareTo(BigInteger.ONE) == 0) {\n        key.n = key.p.multiply(key.q);\n        key.d = key.ee.modInverse(phi);\n        key.dmp1 = key.d.mod(p1);\n        key.dmq1 = key.d.mod(q1);\n        key.u = key.p.modInverse(key.q);\n        break;\n      }\n    }\n    return key;\n  }\n\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n  this.verify = verify;\n  this.sign = sign;\n  this.generate = generate;\n  this.keyObject = keyObject;\n}\n\nmodule.exports = RSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Retrieve secure random byte string of the specified length\n   * @param {Integer} length Length in bytes to generate\n   * @return {String} Random byte string\n   */\n  getRandomBytes: function(length) {\n    var result = '';\n    for (var i = 0; i < length; i++) {\n      result += String.fromCharCode(this.getSecureRandomOctet());\n    }\n    return result;\n  },\n\n  /**\n   * Return a pseudo-random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A pseudo random number\n   */\n  getPseudoRandom: function(from, to) {\n    return Math.round(Math.random() * (to - from)) + from;\n  },\n\n  /**\n   * Return a secure random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A secure random number\n   */\n  getSecureRandom: function(from, to) {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    var bits = ((to - from)).toString(2).length;\n    while ((buf[0] & (Math.pow(2, bits) - 1)) > (to - from))\n      window.crypto.getRandomValues(buf);\n    return from + (Math.abs(buf[0] & (Math.pow(2, bits) - 1)));\n  },\n\n  getSecureRandomOctet: function() {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    return buf[0] & 0xFF;\n  },\n\n  /**\n   * Create a secure random big integer of bits length\n   * @param {Integer} bits Bit length of the MPI to create\n   * @return {BigInteger} Resulting big integer\n   */\n  getRandomBigInteger: function(bits) {\n    if (bits < 0) {\n      return null;\n    }\n    var numBytes = Math.floor((bits + 7) / 8);\n\n    var randomBits = this.getRandomBytes(numBytes);\n    if (bits % 8 > 0) {\n\n      randomBits = String.fromCharCode(\n      (Math.pow(2, bits % 8) - 1) &\n        randomBits.charCodeAt(0)) +\n        randomBits.substring(1);\n    }\n    var mpi = new type_mpi();\n    mpi.fromBytes(randomBits);\n    return mpi.toBigInteger();\n  },\n\n  getRandomBigIntegerInRange: function(min, max) {\n    if (max.compareTo(min) <= 0) {\n      return;\n    }\n\n    var range = max.subtract(min);\n    var r = this.getRandomBigInteger(range.bitLength());\n    while (r > range) {\n      r = this.getRandomBigInteger(range.bitLength());\n    }\n    return min.add(r);\n  }\n\n};\n","var publicKey = require('./public_key'),\n  pkcs1 = require('./pkcs1.js'),\n  hashModule = require('./hash');\n\nmodule.exports = {\n  /**\n   * \n   * @param {Integer} algo public Key algorithm\n   * @param {Integer} hash_algo Hash algorithm\n   * @param {openpgp_type_mpi[]} msg_MPIs Signature multiprecision integers\n   * @param {openpgp_type_mpi[]} publickey_MPIs Public key multiprecision integers \n   * @param {String} data Data on where the signature was computed on.\n   * @return {Boolean} true if signature (sig_data was equal to data over hash)\n   */\n  verify: function(algo, hash_algo, msg_MPIs, publickey_MPIs, data) {\n    var calc_hash = hashModule.digest(hash_algo, data);\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var n = publickey_MPIs[0].toBigInteger();\n        var e = publickey_MPIs[1].toBigInteger();\n        var x = msg_MPIs[0].toBigInteger();\n        var dopublic = rsa.verify(x, e, n);\n        var hash = pkcs1.emsa.decode(hash_algo, dopublic.toMPI().substring(2));\n        if (hash == -1) {\n          throw new Error('PKCS1 padding in message or key incorrect. Aborting...');\n        }\n        return hash == calc_hash;\n\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error(\"signing with Elgamal is not defined in the OpenPGP standard.\");\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n        var s1 = msg_MPIs[0].toBigInteger();\n        var s2 = msg_MPIs[1].toBigInteger();\n        var p = publickey_MPIs[0].toBigInteger();\n        var q = publickey_MPIs[1].toBigInteger();\n        var g = publickey_MPIs[2].toBigInteger();\n        var y = publickey_MPIs[3].toBigInteger();\n        var m = data;\n        var dopublic = dsa.verify(hash_algo, s1, s2, m, p, q, g, y);\n        return dopublic.compareTo(s1) == 0;\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n\n  },\n\n  /**\n   * Create a signature on data using the specified algorithm\n   * @param {Integer} hash_algo hash Algorithm to use (See RFC4880 9.4)\n   * @param {Integer} algo Asymmetric cipher algorithm to use (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Public key multiprecision integers \n   * of the private key \n   * @param {openpgp_type_mpi[]} secretMPIs Private key multiprecision \n   * integers which is used to sign the data\n   * @param {String} data Data to be signed\n   * @return {openpgp_type_mpi[]}\n   */\n  sign: function(hash_algo, algo, keyIntegers, data) {\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var d = keyIntegers[2].toBigInteger();\n        var n = keyIntegers[0].toBigInteger();\n        var m = pkcs1.emsa.encode(hash_algo,\n          data, keyIntegers[0].byteLength());\n\n        return rsa.sign(m, d, n).toMPI();\n\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n\n        var p = keyIntegers[0].toBigInteger();\n        var q = keyIntegers[1].toBigInteger();\n        var g = keyIntegers[2].toBigInteger();\n        var y = keyIntegers[3].toBigInteger();\n        var x = keyIntegers[4].toBigInteger();\n        var m = data;\n        var result = dsa.sign(hash_algo, m, g, p, q, x);\n\n        return result[0].toString() + result[1].toString();\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error('Signing with Elgamal is not defined in the OpenPGP standard.');\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar base64 = require('./base64.js');\nvar enums = require('../enums.js');\n\n/**\n * Finds out which Ascii Armoring type is used. This is an internal function\n * @param {String} text [String] ascii armored text\n * @returns {Integer} 0 = MESSAGE PART n of m\n *         1 = MESSAGE PART n\n *         2 = SIGNED MESSAGE\n *         3 = PGP MESSAGE\n *         4 = PUBLIC KEY BLOCK\n *         5 = PRIVATE KEY BLOCK\n *         null = unknown\n */\nfunction get_type(text) {\n  var splittedtext = text.split('-----');\n  // BEGIN PGP MESSAGE, PART X/Y\n  // Used for multi-part messages, where the armor is split amongst Y\n  // parts, and this is the Xth part out of Y.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+\\/\\d+/)) {\n    return enums.armor.multipart_section;\n  } else\n  // BEGIN PGP MESSAGE, PART X\n  // Used for multi-part messages, where this is the Xth part of an\n  // unspecified number of parts. Requires the MESSAGE-ID Armor\n  // Header to be used.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+/)) {\n    return enums.armor.multipart_last;\n\n  } else\n  // BEGIN PGP SIGNATURE\n  // Used for detached signatures, OpenPGP/MIME signatures, and\n  // cleartext signatures. Note that PGP 2.x uses BEGIN PGP MESSAGE\n  // for detached signatures.\n  if (splittedtext[1].match(/BEGIN PGP SIGNED MESSAGE/)) {\n    return enums.armor.signed;\n\n  } else\n  // BEGIN PGP MESSAGE\n  // Used for signed, encrypted, or compressed files.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE/)) {\n    return enums.armor.message;\n\n  } else\n  // BEGIN PGP PUBLIC KEY BLOCK\n  // Used for armoring public keys.\n  if (splittedtext[1].match(/BEGIN PGP PUBLIC KEY BLOCK/)) {\n    return enums.armor.public_key;\n\n  } else\n  // BEGIN PGP PRIVATE KEY BLOCK\n  // Used for armoring private keys.\n  if (splittedtext[1].match(/BEGIN PGP PRIVATE KEY BLOCK/)) {\n    return enums.armor.private_key;\n  }\n}\n\n/**\n * Add additional information to the armor version of an OpenPGP binary\n * packet block.\n * @author  Alex\n * @version 2011-12-16\n * @returns {String} The header information\n */\nfunction armor_addheader(options) {\n  var result = \"\";\n  if (options.show_version) {\n    result += \"Version: \" + options.versionstring + '\\r\\n';\n  }\n  if (options.show_comment) {\n    result += \"Comment: \" + options.commentstring + '\\r\\n';\n  }\n  result += '\\r\\n';\n  return result;\n}\n\n\n\n/**\n * Calculates a checksum over the given data and returns it base64 encoded\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {String} Base64 encoded checksum\n */\nfunction getCheckSum(data) {\n  var c = createcrc24(data);\n  var str = \"\" + String.fromCharCode(c >> 16) +\n    String.fromCharCode((c >> 8) & 0xFF) +\n    String.fromCharCode(c & 0xFF);\n  return base64.encode(str);\n}\n\n/**\n * Calculates the checksum over the given data and compares it with the \n * given base64 encoded checksum\n * @param {String} data Data to create a CRC-24 checksum for\n * @param {String} checksum Base64 encoded checksum\n * @return {Boolean} True if the given checksum is correct; otherwise false\n */\nfunction verifyCheckSum(data, checksum) {\n  var c = getCheckSum(data);\n  var d = checksum;\n  return c[0] == d[0] && c[1] == d[1] && c[2] == d[2];\n}\n/**\n * Internal function to calculate a CRC-24 checksum over a given string (data)\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {Integer} The CRC-24 checksum as number\n */\nvar crc_table = [\n    0x00000000, 0x00864cfb, 0x018ad50d, 0x010c99f6, 0x0393e6e1, 0x0315aa1a, 0x021933ec, 0x029f7f17, 0x07a18139,\n    0x0727cdc2, 0x062b5434, 0x06ad18cf, 0x043267d8, 0x04b42b23, 0x05b8b2d5, 0x053efe2e, 0x0fc54e89, 0x0f430272,\n    0x0e4f9b84, 0x0ec9d77f, 0x0c56a868, 0x0cd0e493, 0x0ddc7d65, 0x0d5a319e, 0x0864cfb0, 0x08e2834b, 0x09ee1abd,\n    0x09685646, 0x0bf72951, 0x0b7165aa, 0x0a7dfc5c, 0x0afbb0a7, 0x1f0cd1e9, 0x1f8a9d12, 0x1e8604e4, 0x1e00481f,\n    0x1c9f3708, 0x1c197bf3, 0x1d15e205, 0x1d93aefe, 0x18ad50d0, 0x182b1c2b, 0x192785dd, 0x19a1c926, 0x1b3eb631,\n    0x1bb8faca, 0x1ab4633c, 0x1a322fc7, 0x10c99f60, 0x104fd39b, 0x11434a6d, 0x11c50696, 0x135a7981, 0x13dc357a,\n    0x12d0ac8c, 0x1256e077, 0x17681e59, 0x17ee52a2, 0x16e2cb54, 0x166487af, 0x14fbf8b8, 0x147db443, 0x15712db5,\n    0x15f7614e, 0x3e19a3d2, 0x3e9fef29, 0x3f9376df, 0x3f153a24, 0x3d8a4533, 0x3d0c09c8, 0x3c00903e, 0x3c86dcc5,\n    0x39b822eb, 0x393e6e10, 0x3832f7e6, 0x38b4bb1d, 0x3a2bc40a, 0x3aad88f1, 0x3ba11107, 0x3b275dfc, 0x31dced5b,\n    0x315aa1a0,\n    0x30563856, 0x30d074ad, 0x324f0bba, 0x32c94741, 0x33c5deb7, 0x3343924c, 0x367d6c62, 0x36fb2099, 0x37f7b96f,\n    0x3771f594, 0x35ee8a83, 0x3568c678, 0x34645f8e, 0x34e21375, 0x2115723b, 0x21933ec0, 0x209fa736, 0x2019ebcd,\n    0x228694da, 0x2200d821, 0x230c41d7, 0x238a0d2c, 0x26b4f302, 0x2632bff9, 0x273e260f, 0x27b86af4, 0x252715e3,\n    0x25a15918, 0x24adc0ee, 0x242b8c15, 0x2ed03cb2, 0x2e567049, 0x2f5ae9bf, 0x2fdca544, 0x2d43da53, 0x2dc596a8,\n    0x2cc90f5e, 0x2c4f43a5, 0x2971bd8b, 0x29f7f170, 0x28fb6886, 0x287d247d, 0x2ae25b6a, 0x2a641791, 0x2b688e67,\n    0x2beec29c, 0x7c3347a4, 0x7cb50b5f, 0x7db992a9, 0x7d3fde52, 0x7fa0a145, 0x7f26edbe, 0x7e2a7448, 0x7eac38b3,\n    0x7b92c69d, 0x7b148a66, 0x7a181390, 0x7a9e5f6b, 0x7801207c, 0x78876c87, 0x798bf571, 0x790db98a, 0x73f6092d,\n    0x737045d6, 0x727cdc20, 0x72fa90db, 0x7065efcc, 0x70e3a337, 0x71ef3ac1, 0x7169763a, 0x74578814, 0x74d1c4ef,\n    0x75dd5d19, 0x755b11e2, 0x77c46ef5, 0x7742220e, 0x764ebbf8, 0x76c8f703, 0x633f964d, 0x63b9dab6, 0x62b54340,\n    0x62330fbb,\n    0x60ac70ac, 0x602a3c57, 0x6126a5a1, 0x61a0e95a, 0x649e1774, 0x64185b8f, 0x6514c279, 0x65928e82, 0x670df195,\n    0x678bbd6e, 0x66872498, 0x66016863, 0x6cfad8c4, 0x6c7c943f, 0x6d700dc9, 0x6df64132, 0x6f693e25, 0x6fef72de,\n    0x6ee3eb28, 0x6e65a7d3, 0x6b5b59fd, 0x6bdd1506, 0x6ad18cf0, 0x6a57c00b, 0x68c8bf1c, 0x684ef3e7, 0x69426a11,\n    0x69c426ea, 0x422ae476, 0x42aca88d, 0x43a0317b, 0x43267d80, 0x41b90297, 0x413f4e6c, 0x4033d79a, 0x40b59b61,\n    0x458b654f, 0x450d29b4, 0x4401b042, 0x4487fcb9, 0x461883ae, 0x469ecf55, 0x479256a3, 0x47141a58, 0x4defaaff,\n    0x4d69e604, 0x4c657ff2, 0x4ce33309, 0x4e7c4c1e, 0x4efa00e5, 0x4ff69913, 0x4f70d5e8, 0x4a4e2bc6, 0x4ac8673d,\n    0x4bc4fecb, 0x4b42b230, 0x49ddcd27, 0x495b81dc, 0x4857182a, 0x48d154d1, 0x5d26359f, 0x5da07964, 0x5cace092,\n    0x5c2aac69, 0x5eb5d37e, 0x5e339f85, 0x5f3f0673, 0x5fb94a88, 0x5a87b4a6, 0x5a01f85d, 0x5b0d61ab, 0x5b8b2d50,\n    0x59145247, 0x59921ebc, 0x589e874a, 0x5818cbb1, 0x52e37b16, 0x526537ed, 0x5369ae1b, 0x53efe2e0, 0x51709df7,\n    0x51f6d10c,\n    0x50fa48fa, 0x507c0401, 0x5542fa2f, 0x55c4b6d4, 0x54c82f22, 0x544e63d9, 0x56d11cce, 0x56575035, 0x575bc9c3,\n    0x57dd8538\n];\n\nfunction createcrc24(input) {\n  var crc = 0xB704CE;\n  var index = 0;\n\n  while ((input.length - index) > 16) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 1)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 2)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 3)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 4)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 5)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 6)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 7)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 8)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 9)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 10)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 11)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 12)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 13)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 14)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 15)) & 0xff];\n    index += 16;\n  }\n\n  for (var j = index; j < input.length; j++) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index++)) & 0xff];\n  }\n  return crc & 0xffffff;\n}\n\n/**\n * DeArmor an OpenPGP armored message; verify the checksum and return \n * the encoded bytes\n * @param {String} text OpenPGP armored message\n * @returns {(Boolean|Object)} Either false in case of an error \n * or an object with attribute \"text\" containing the message text\n * and an attribute \"openpgp\" containing the bytes.\n */\nfunction dearmor(text) {\n  text = text.replace(/\\r/g, '');\n\n  var type = get_type(text);\n\n  if (type != 2) {\n    var splittedtext = text.split('-----');\n\n    var data = {\n      openpgp: base64.decode(\n        splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[0]\n        .replace(/\\n- /g, \"\\n\")),\n      type: type\n    };\n\n    if (verifyCheckSum(data.openpgp,\n      splittedtext[2]\n      .split('\\n\\n')[1]\n      .split(\"\\n=\")[1]\n      .split('\\n')[0]))\n\n      return data;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed: '\" + splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[1]\n        .split('\\n')[0] + \"' should be '\" + getCheckSum(data)) + \"'\";\n      return false;\n    }\n  } else {\n    var splittedtext = text.split('-----');\n\n    var result = {\n      text: splittedtext[2]\n        .replace(/\\n- /g, \"\\n\")\n        .split(\"\\n\\n\")[1],\n      openpgp: base64_decode(splittedtext[4]\n        .split(\"\\n\\n\")[1]\n        .split(\"\\n=\")[0]),\n      type: type\n    };\n\n    if (verifyCheckSum(result.openpgp, splittedtext[4]\n      .split(\"\\n\\n\")[1]\n      .split(\"\\n=\")[1]))\n\n      return result;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed\");\n      return false;\n    }\n  }\n}\n\n\n/**\n * Armor an OpenPGP binary packet block\n * @param {Integer} messagetype type of the message\n * @param data\n * @param {Integer} partindex\n * @param {Integer} parttotal\n * @returns {String} Armored text\n */\nfunction armor(messagetype, data, options, partindex, parttotal) {\n  var result = \"\";\n  switch (messagetype) {\n    case enums.armor.multipart_section:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      break;\n    case enums.armor.mutlipart_last:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      break;\n    case enums.armor.signed:\n      result += \"\\r\\n-----BEGIN PGP SIGNED MESSAGE-----\\r\\nHash: \" + data.hash + \"\\r\\n\\r\\n\";\n      result += data.text.replace(/\\n-/g, \"\\n- -\");\n      result += \"\\r\\n-----BEGIN PGP SIGNATURE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data.openpgp);\n      result += \"\\r\\n=\" + getCheckSum(data.openpgp) + \"\\r\\n\";\n      result += \"-----END PGP SIGNATURE-----\\r\\n\";\n      break;\n    case enums.armor.message:\n      result += \"-----BEGIN PGP MESSAGE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE-----\\r\\n\";\n      break;\n    case enums.armor.public_key:\n      result += \"-----BEGIN PGP PUBLIC KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PUBLIC KEY BLOCK-----\\r\\n\\r\\n\";\n      break;\n    case enums.armor.private_key:\n      result += \"-----BEGIN PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      break;\n  }\n\n  return result;\n}\n\nmodule.exports = {\n  encode: armor,\n  decode: dearmor\n};\n","/* OpenPGP radix-64/base64 string encoding/decoding\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.0, check www.haneWIN.de for the latest version\n *\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other materials\n * provided with the application or distribution.\n */\n\nvar b64s = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\n\nfunction s2r(t) {\n  var a, c, n;\n  var r = '',\n    l = 0,\n    s = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = t.charCodeAt(n);\n    if (s == 0) {\n      r += b64s.charAt((c >> 2) & 63);\n      a = (c & 3) << 4;\n    } else if (s == 1) {\n      r += b64s.charAt((a | (c >> 4) & 15));\n      a = (c & 15) << 2;\n    } else if (s == 2) {\n      r += b64s.charAt(a | ((c >> 6) & 3));\n      l += 1;\n      if ((l % 60) == 0)\n        r += \"\\n\";\n      r += b64s.charAt(c & 63);\n    }\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n\n    s += 1;\n    if (s == 3)\n      s = 0;\n  }\n  if (s > 0) {\n    r += b64s.charAt(a);\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n    l += 1;\n  }\n  if (s == 1) {\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n  }\n\n  return r;\n}\n\nfunction r2s(t) {\n  var c, n;\n  var r = '',\n    s = 0,\n    a = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = b64s.indexOf(t.charAt(n));\n    if (c >= 0) {\n      if (s)\n        r += String.fromCharCode(a | (c >> (6 - s)) & 255);\n      s = (s + 2) & 7;\n      a = (c << s) & 255;\n    }\n  }\n  return r;\n}\n\nmodule.exports = {\n  encode: s2r,\n  decode: r2s\n}\n","var enums = {\n\n  /** A string to key specifier type\n   * @enum {Integer}\n   */\n  s2k: {\n    simple: 0,\n    salted: 1,\n    iterated: 3,\n    gnu: 101\n  },\n\n  /** RFC4880, section 9.1 \n   * @enum {String}\n   */\n  publicKey: {\n    rsa_encrypt_sign: 1,\n    rsa_encrypt: 2,\n    rsa_sign: 3,\n    elgamal: 16,\n    dsa: 17\n  },\n\n  /** RFC4880, section 9.2 \n   * @enum {String}\n   */\n  symmetric: {\n    plaintext: 0,\n    /** Not implemented! */\n    idea: 1,\n    tripledes: 2,\n    cast5: 3,\n    blowfish: 4,\n    aes128: 7,\n    aes192: 8,\n    aes256: 9,\n    twofish: 10\n  },\n\n  /** RFC4880, section 9.3\n   * @enum {String}\n   */\n  compression: {\n    uncompressed: 0,\n    /** RFC1951 */\n    zip: 1,\n    /** RFC1950 */\n    zlib: 2,\n    bzip2: 3\n  },\n\n  /** RFC4880, section 9.4\n   * @enum {String}\n   */\n  hash: {\n    md5: 1,\n    sha1: 2,\n    ripemd: 3,\n    sha256: 8,\n    sha384: 9,\n    sha512: 10,\n    sha224: 11\n  },\n\n\n  /**\n   * @enum {String}\n   * A list of packet types and numeric tags associated with them.\n   */\n  packet: {\n    public_key_encrypted_session_key: 1,\n    signature: 2,\n    sym_encrypted_session_key: 3,\n    one_pass_signature: 4,\n    secret_key: 5,\n    public_key: 6,\n    secret_subkey: 7,\n    compressed: 8,\n    symmetrically_encrypted: 9,\n    marker: 10,\n    literal: 11,\n    trust: 12,\n    userid: 13,\n    public_subkey: 14,\n    user_attribute: 17,\n    sym_encrypted_integrity_protected: 18,\n    modification_detection_code: 19\n  },\n\n\n  /**\n   * Data types in the literal packet\n   * @readonly\n   * @enum {String}\n   */\n  literal: {\n    /** Binary data */\n    binary: 'b'.charCodeAt(),\n    /** Text data */\n    text: 't'.charCodeAt(),\n    /** Utf8 data */\n    utf8: 'u'.charCodeAt()\n  },\n\n\n  /** One pass signature packet type\n   * @enum {String} */\n  signature: {\n    /** 0x00: Signature of a binary document. */\n    binary: 0,\n    /** 0x01: Signature of a canonical text document.\n     * Canonicalyzing the document by converting line endings. */\n    text: 1,\n    /** 0x02: Standalone signature.\n     * This signature is a signature of only its own subpacket contents.\n     * It is calculated identically to a signature over a zero-lengh\n     * binary document.  Note that it doesn't make sense to have a V3\n     * standalone signature. */\n    standalone: 2,\n    /** 0x10: Generic certification of a User ID and Public-Key packet.\n     * The issuer of this certification does not make any particular\n     * assertion as to how well the certifier has checked that the owner\n     * of the key is in fact the person described by the User ID. */\n    cert_generic: 16,\n    /** 0x11: Persona certification of a User ID and Public-Key packet.\n     * The issuer of this certification has not done any verification of\n     * the claim that the owner of this key is the User ID specified. */\n    cert_persona: 17,\n    /** 0x12: Casual certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done some casual\n     * verification of the claim of identity. */\n    cert_casual: 18,\n    /** 0x13: Positive certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done substantial\n     * verification of the claim of identity.\n     * \n     * Most OpenPGP implementations make their \"key signatures\" as 0x10\n     * certifications.  Some implementations can issue 0x11-0x13\n     * certifications, but few differentiate between the types. */\n    cert_positive: 19,\n    /** 0x30: Certification revocation signature\n     * This signature revokes an earlier User ID certification signature\n     * (signature class 0x10 through 0x13) or direct-key signature\n     * (0x1F).  It should be issued by the same key that issued the\n     * revoked signature or an authorized revocation key.  The signature\n     * is computed over the same data as the certificate that it\n     * revokes, and should have a later creation date than that\n     * certificate. */\n    cert_revocation: 48,\n    /** 0x18: Subkey Binding Signature\n     * This signature is a statement by the top-level signing key that\n     * indicates that it owns the subkey.  This signature is calculated\n     * directly on the primary key and subkey, and not on any User ID or\n     * other packets.  A signature that binds a signing subkey MUST have\n     * an Embedded Signature subpacket in this binding signature that\n     * contains a 0x19 signature made by the signing subkey on the\n     * primary key and subkey. */\n    subkey_binding: 24,\n    /** 0x19: Primary Key Binding Signature\n\t\t* This signature is a statement by a signing subkey, indicating\n\t\t* that it is owned by the primary key and subkey.  This signature\n\t\t* is calculated the same way as a 0x18 signature: directly on the\n\t\t* primary key and subkey, and not on any User ID or other packets.\n\t\t\n\t\t* When a signature is made over a key, the hash data starts with the\n\t\t* octet 0x99, followed by a two-octet length of the key, and then body\n\t\t* of the key packet.  (Note that this is an old-style packet header for\n\t\t* a key packet with two-octet length.)  A subkey binding signature\n\t\t* (type 0x18) or primary key binding signature (type 0x19) then hashes\n\t\t* the subkey using the same format as the main key (also using 0x99 as\n\t\t* the first octet). */\n    key_binding: 25,\n    /** 0x1F: Signature directly on a key\n     * This signature is calculated directly on a key.  It binds the\n     * information in the Signature subpackets to the key, and is\n     * appropriate to be used for subpackets that provide information\n     * about the key, such as the Revocation Key subpacket.  It is also\n     * appropriate for statements that non-self certifiers want to make\n     * about the key itself, rather than the binding between a key and a\n     * name. */\n    key: 31,\n    /** 0x20: Key revocation signature\n     * The signature is calculated directly on the key being revoked.  A\n     * revoked key is not to be used.  Only revocation signatures by the\n     * key being revoked, or by an authorized revocation key, should be\n     * considered valid revocation signatures.a */\n    key_revocation: 32,\n    /** 0x28: Subkey revocation signature\n     * The signature is calculated directly on the subkey being revoked.\n     * A revoked subkey is not to be used.  Only revocation signatures\n     * by the top-level signature key that is bound to this subkey, or\n     * by an authorized revocation key, should be considered valid\n     * revocation signatures.\n     * Key revocation signatures (types 0x20 and 0x28)\n     * hash only the key being revoked. */\n    subkey_revocation: 40,\n    /** 0x40: Timestamp signature.\n     * This signature is only meaningful for the timestamp contained in\n     * it. */\n    timestamp: 64,\n    /**    0x50: Third-Party Confirmation signature.\n     * This signature is a signature over some other OpenPGP Signature\n     * packet(s).  It is analogous to a notary seal on the signed data.\n     * A third-party signature SHOULD include Signature Target\n     * subpacket(s) to give easy identification.  Note that we really do\n     * mean SHOULD.  There are plausible uses for this (such as a blind\n     * party that only sees the signature, not the key or source\n     * document) that cannot include a target subpacket. */\n    third_party: 80\n  },\n\n  signatureSubpacket: {\n    signature_creation_time: 2,\n    signature_expiration_time: 3,\n    exportable_certification: 4,\n    trust_signature: 5,\n    regular_expression: 6,\n    revocable: 7,\n    reserved: 8,\n    key_expiration_time: 9,\n    placeholder_backwards_compatibility: 10,\n    preferred_symmetric_algorithms: 11,\n    revocation_key: 12,\n    issuer: 16,\n    notification_data: 20,\n    preferred_hash_algorithms: 21,\n    preferred_compression_algorithms: 22,\n    key_server_preferences: 23,\n    preferred_key_server: 24,\n    primary_user_id: 25,\n    policy_uri: 26,\n    key_flags: 27,\n    signers_user_id: 28,\n    reason_for_revocation: 29,\n    features: 30,\n    signature_target: 31,\n    embedded_signature: 32\n  },\n\n  armor: {\n    multipart_section: 0,\n    multipart_last: 1,\n    signed: 2,\n    message: 3,\n    public_key: 4,\n    private_key: 5\n  },\n\n  // Asserts validity and converts from string/integer to integer.\n  write: function(type, e) {\n    if (typeof e == 'number') {\n      e = this.read(type, e);\n    }\n\n    if (type[e] !== undefined) {\n      return type[e];\n    } else throw new Error('Invalid enum value.');\n  },\n  // Converts from an integer to string.\n  read: function(type, e) {\n    for (var i in type)\n      if (type[i] == e) return i;\n\n    throw new Error('Invalid enum value.');\n  }\n}\n\nmodule.exports = enums;\n","\nmodule.exports = require('./openpgp.js');\nmodule.exports.key = require('./key.js');\nmodule.exports.message = require('./message.js');\nmodule.exports.util = require('./util');\nmodule.exports.packet = require('./packet');\nmodule.exports.mpi = require('./type/mpi.js');\nmodule.exports.s2k = require('./type/s2k.js');\nmodule.exports.keyid = require('./type/keyid.js');\nmodule.exports.armor = require('./encoding/armor.js');\nmodule.exports.enums = require('./enums.js');\nmodule.exports.config = require('./config');\nmodule.exports.crypto = require('./crypto');\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc Class that represents an OpenPGP key. Must contain a master key. \n * @param  {packetlist} packetlist [description]\n * Can contain additional subkeys, signatures,\n * user ids, user attributes.\n */\n\n function key(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n\n  /** Returns the primary key (secret or public)\n   * @returns {packet_secret_key|packet_public_key|null} */\n  this.getKey = function() {\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_key ||\n        this.packets[i].tag == enums.packet.secret_key) {\n        return this.packets[i];\n      }\n    }\n    return null;\n  };\n\n  /** Returns all the private and public subkeys \n   * @returns {[public_subkey|secret_subkey]} */\n  this.getSubkeys = function() {\n\n    var subkeys = [];\n\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_subkey ||\n        this.packets[i].tag == enums.packet.secret_subkey) {\n        subkeys.push(this.packets[i]);\n      }\n    }\n\n    return subkeys;\n  };\n\n  this.getAllKeys = function() {\n    return [this.getKey()].concat(this.getSubkeys());\n  };\n\n  this.getKeyids = function() {\n    var keyids = [];\n    var keys = this.getAllKeys();\n    for (var i = 0; i < keys.length; i++) {\n      keyids.push(keys[i].getKeyId());\n    }\n    return keyids;\n  };\n\n  this.getKeyById = function(keyid) {\n    var keys = this.getAllKeys();\n    for (var i = 0; i < keys.length; i++) {\n      if (keys[i].getKeyId().equals(keyid)) {\n        return keys[i];\n      }\n    }\n  }\n\n  this.getSigningKey = function() {\n\n    var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa];\n\n    signing = signing.map(function(s) {\n      return enums.read(enums.publicKey, s);\n    });\n\n    var keys = this.getAllKeys();\n\n    for (var i = 0; i < keys.length; i++) {\n      if (signing.indexOf(keys[i].algorithm) !== -1) {\n        return keys[i];\n      }\n    }\n\n    return null;\n  };\n\n  function getPreferredSignatureHashAlgorithm() {\n    var pkey = this.getSigningKey();\n    if (pkey === null) {\n      util.print_error(\"private key is for encryption only! Cannot create a signature.\");\n      return null;\n    }\n    if (pkey.publicKey.publicKeyAlgorithm == 17) {\n      var dsa = new DSA();\n      return dsa.select_hash_algorithm(pkey.publicKey.MPIs[1].toBigInteger()); // q\n    }\n    //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8\n    //separate private key preference from digest preferences\n    return openpgp.config.config.prefer_hash_algorithm;\n  }\n\n  /**\n   * Finds an encryption key for this key\n   * @returns null if no encryption key has been found\n   */\n  this.getEncryptionKey = function() {\n    // V4: by convention subkeys are prefered for encryption service\n    // V3: keys MUST NOT have subkeys\n    var isValidEncryptionKey = function(key) {\n      return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey,\n        enums.publicKey.rsa_sign);\n      //TODO verify key\n      //&& keys.verifyKey()\n    };\n\n    var subkeys = this.getSubkeys();\n\n    for (var j = 0; j < subkeys.length; j++) {\n      if (isValidEncryptionKey(subkeys[j])) {\n        return subkeys[j];\n      }\n    }\n    // if no valid subkey for encryption, use primary key\n    var primaryKey = this.getKey();\n    if (isValidEncryptionKey(primaryKey)) {\n      return primaryKey;\n    }\n    return null;\n  };\n\n  this.decrypt = function(passphrase) {\n    var keys = this.getAllKeys();\n\n    for (var i in keys)\n      if (keys[i].tag == enums.packet.secret_subkey ||\n        keys[i].tag == enums.packet.secret_key) {\n        keys[i].decrypt(passphrase);\n      }\n  };\n\n\n  // TODO need to implement this\n\n  function revoke() {\n\n  }\n\n};\n\n/**\n * reads an OpenPGP armored text and returns a key object\n * @param {String} armoredText text to be parsed\n * @return {key} new key object\n */\nkey.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newKey = new key(packetlist);\n  return newKey;\n}\n\nmodule.exports = key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc A generic message containing one or more literal packets.\n */\n\nfunction message(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n  /**\n   * Returns the key IDs of the public keys to which the session key is encrypted\n   * @return {[keyId]} keyId provided as string of hex number (lowercase)\n   */\n  this.getKeyIds = function() {\n    var keyIds = [];\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    pkESKeyPacketlist.forEach(function(packet) {\n      keyIds.push(packet.publicKeyId.toHex());\n    });\n    return keyIds;\n  }\n\n  function generic_decrypt(packets, passphrase) {\n    var sessionkey;\n\n    for (var i = 0; i < packets.length; i++) {\n      if (packets[i].tag == openpgp_packet.tags.public_key_encrypted_session_key) {\n        var key = openpgp.keyring.getKeyById(packets[i].public_key_id);\n\n      }\n    }\n\n  }\n\n  /**\n   * Decrypts a message and generates user interface message out of the found.\n   * MDC will be verified as well as message signatures\n   * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key)\n   * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message\n   * @return {} plaintext of the message or null on error\n   */\n  this.decrypt = function(key) {\n    var decryptedMessages = [];\n    var keyids = key.getKeyids();\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    outer: for (var i = 0; i < pkESKeyPacketlist.length; i++) {\n      var pkESKeyPacket = pkESKeyPacketlist[i];\n      for (var j = 0; j < keyids.length; j++) {\n        if (pkESKeyPacket.publicKeyId.equals(keyids[j])) {\n          pkESKeyPacket.decrypt(key.getKeyById(pkESKeyPacket.publicKeyId));\n          var symEncryptedPacketlist = this.packets.filter(function(packet) {\n            return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected;\n          });\n          for (var k = 0; k < symEncryptedPacketlist.length; k++) {\n            var symEncryptedPacket = symEncryptedPacketlist[k];\n            symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey);\n            for (var l = 0; l < symEncryptedPacket.packets.length; l++) {\n              var dataPacket = symEncryptedPacket.packets[l];\n              switch (dataPacket.tag) {\n                case enums.packet.literal:\n                  decryptedMessages.push(dataPacket.getBytes());\n                  break;\n                case enums.packet.compressed:\n                  //TODO\n                  break;\n                default:\n                  //TODO\n              }\n            }\n          }\n          break outer;\n        }\n      }\n    }\n    return decryptedMessages;\n  }\n\n  /**\n   * Decrypts a message and generates user interface message out of the found.\n   * MDC will be verified as well as message signatures\n   * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key)\n   * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {String} plaintext of the message or null on error\n   */\n  function decryptAndVerifySignature(private_key, sessionkey, pubkey) {\n    if (private_key == null || sessionkey == null || sessionkey == \"\")\n      return null;\n    var decrypted = sessionkey.decrypt(this, private_key.keymaterial);\n    if (decrypted == null)\n      return null;\n    var packet;\n    var position = 0;\n    var len = decrypted.length;\n    var validSignatures = new Array();\n    util.print_debug_hexstr_dump(\"openpgp.msg.messge decrypt:\\n\", decrypted);\n\n    var messages = openpgp.read_messages_dearmored({\n      text: decrypted,\n      openpgp: decrypted\n    });\n    for (var m in messages) {\n      if (messages[m].data) {\n        this.text = messages[m].data;\n      }\n      if (messages[m].signature) {\n        validSignatures.push(messages[m].verifySignature(pubkey));\n      }\n    }\n    return {\n      text: this.text,\n      validSignatures: validSignatures\n    };\n  }\n\n  /**\n   * Verifies a message signature. This function can be called after read_message if the message was signed only.\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {boolean} true if the signature was correct; otherwise false\n   */\n  function verifySignature(pubkey) {\n    var result = false;\n    if (this.signature.tagType == 2) {\n      if (!pubkey || pubkey.length == 0) {\n        var pubkey;\n        if (this.signature.version == 4) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.issuerKeyId);\n        } else if (this.signature.version == 3) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.keyId);\n        } else {\n          util.print_error(\"unknown signature type on message!\");\n          return false;\n        }\n      }\n      if (pubkey.length == 0)\n        util.print_warning(\"Unable to verify signature of issuer: \" + util.hexstrdump(this.signature.issuerKeyId) +\n          \". Public key not found in keyring.\");\n      else {\n        for (var i = 0; i < pubkey.length; i++) {\n          var tohash = this.text.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g, \"\\r\\n\");\n          if (this.signature.verify(tohash, pubkey[i])) {\n            util.print_info(\"Found Good Signature from \" + pubkey[i].obj.userIds[0].text + \" (0x\" + util.hexstrdump(\n              pubkey[i].obj.getKeyId()).substring(8) + \")\");\n            result = true;\n          } else {\n            util.print_error(\"Signature verification failed: Bad Signature from \" + pubkey[i].obj.userIds[0].text +\n              \" (0x\" + util.hexstrdump(pubkey[0].obj.getKeyId()).substring(8) + \")\");\n          }\n        }\n      }\n    }\n    return result;\n  }\n}\n\n/**\n * reads an OpenPGP armored message and returns a message object\n * @param {String} armoredText text to be parsed\n * @return {key} new message object\n */\nmessage.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newMessage = new message(packetlist);\n  return newMessage;\n}\n\nmodule.exports = message;","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @fileoverview The openpgp base class should provide all of the functionality \n * to consume the openpgp.js library. All additional classes are documented \n * for extending and developing on top of the base library.\n */\n\nvar armor = require('./encoding/armor.js');\nvar packet = require('./packet');\nvar util = require('./util');\nvar enums = require('./enums.js');\nvar crypto = require('./crypto');\nvar config = require('./config');\n\n/**\n * GPG4Browsers Core interface. A single instance is hold\n * from the beginning. To use this library call \"openpgp.init()\"\n * @alias openpgp\n * @class\n * @classdesc Main Openpgp.js class. Use this to initiate and make all calls to this library.\n */\nfunction _openpgp() {\n  this.tostring = \"\";\n\n  /**\n   * encrypts message with keys\n   * @param  {[key]} keys    array of keys, used to encrypt the message\n   * @param  {String} message text message as native JavaScript string\n   * @return {String}         encrypted ASCII armored message \n   */\n  function encryptMessage(keys, message) {\n\n    var messagePacketlist = new packet.list();\n\n    //TODO get preferred algo from signature\n    var sessionKey = crypto.generateSessionKey(enums.read(enums.symmetric, config.encryption_cipher));\n\n    keys.forEach(function(key) {\n      var encryptionKey = key.getEncryptionKey();\n      if (encryptionKey) {\n        var pkESKeyPacket = new packet.public_key_encrypted_session_key();\n        pkESKeyPacket.publicKeyId = encryptionKey.getKeyId();\n        pkESKeyPacket.publicKeyAlgorithm = encryptionKey.algorithm;\n        pkESKeyPacket.sessionKey = sessionKey;\n        //TODO get preferred algo from signature\n        pkESKeyPacket.sessionKeyAlgorithm = enums.read(enums.symmetric, config.encryption_cipher);\n        pkESKeyPacket.encrypt(encryptionKey);\n        messagePacketlist.push(pkESKeyPacket);\n      }\n    });\n\n    var literalDataPacket = new packet.literal();\n    literalDataPacket.set(message, 'utf8');\n    var literalDataPacketlist = new packet.list();\n    literalDataPacketlist.push(literalDataPacket);\n\n    var symEncryptedPacket;\n    if (config.integrity_protect) {\n      symEncryptedPacket = new packet.sym_encrypted_integrity_protected();\n    } else {\n      symEncryptedPacket = new packet.symmetrically_encrypted();\n    }\n    symEncryptedPacket.packets = literalDataPacketlist;\n    //TODO get preferred algo from signature\n    symEncryptedPacket.encrypt(enums.read(enums.symmetric, config.encryption_cipher), sessionKey);\n    messagePacketlist.push(symEncryptedPacket);\n\n    var armored = armor.encode(enums.armor.message, messagePacketlist.write(), config);\n    return armored;\n  }\n\n  function encryptAndSignMessage(publicKeys, privateKey, message) {\n\n  }\n\n  /**\n   * decrypts message\n   * @param  {[key]} decrypted privateKey\n   * @param  {message} message the message object with the encrypted data\n   * @return {String}         decrypted message as as native JavaScript string\n   */\n  function decryptMessage(privateKey, message) {\n    return message.decrypt(privateKey);\n  }\n\n  function decryptAndVerifyMessage(privateKey, publicKeys, messagePacketlist) {\n\n  }\n\n  function verifyMessage(publicKeys, messagePacketlist) {\n\n  }\n\n  function signMessage(privateKey, messagePacketlist) {\n\n  }\n\n  /**\n   * TODO: update this doc\n   * generates a new key pair for openpgp. Beta stage. Currently only \n   * supports RSA keys, and no subkeys.\n   * @param {Integer} keyType to indicate what type of key to make. \n   * RSA is 1. Follows algorithms outlined in OpenPGP.\n   * @param {Integer} numBits number of bits for the key creation. (should \n   * be 1024+, generally)\n   * @param {String} userId assumes already in form of \"User Name \n   * <username@email.com>\"\n   * @param {String} passphrase The passphrase used to encrypt the resulting private key\n   * @return {Object} {privateKey: [openpgp_msg_privatekey], \n   * privateKeyArmored: [string], publicKeyArmored: [string]}\n   */\n  function generateKeyPair(keyType, numBits, userId, passphrase) {\n    var packetlist = new packet.list();\n\n    var secretKeyPacket = new packet.secret_key();\n    secretKeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretKeyPacket.generate(numBits);\n    secretKeyPacket.encrypt(passphrase);\n\n    var userIdPacket = new packet.userid();\n    userIdPacket.read(userId);\n\n    var dataToSign = {};\n    dataToSign.userid = userIdPacket;\n    dataToSign.key = secretKeyPacket;\n    var signaturePacket = new packet.signature();\n    signaturePacket.issuerKeyId = secretKeyPacket.getKeyId().write();\n    signaturePacket.signatureType = enums.signature.cert_generic;\n    signaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    signaturePacket.hashAlgorithm = enums.hash.sha256;\n    signaturePacket.sign(secretKeyPacket, dataToSign);\n\n    var secretSubkeyPacket = new packet.secret_subkey();\n    secretSubkeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretSubkeyPacket.generate(numBits);\n    secretSubkeyPacket.encrypt(passphrase);\n\n    dataToSign = {};\n    dataToSign.key = secretKeyPacket;\n    dataToSign.bind = secretSubkeyPacket;\n    var subkeySignaturePacket = new packet.signature();\n    subkeySignaturePacket.issuerKeyId = secretSubkeyPacket.getKeyId().write();\n    subkeySignaturePacket.signatureType = enums.signature.subkey_binding;\n    subkeySignaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    subkeySignaturePacket.hashAlgorithm = enums.hash.sha256;\n    subkeySignaturePacket.sign(secretSubkeyPacket, dataToSign);\n\n    packetlist.push(secretKeyPacket);\n    packetlist.push(userIdPacket);\n    packetlist.push(signaturePacket);\n    packetlist.push(secretSubkeyPacket);\n    packetlist.push(subkeySignaturePacket);\n\n    var armored = armor.encode(enums.armor.private_key, packetlist.write(), this.config);\n    return armored;\n  }\n\n  /**\n   * creates a binary string representation of an encrypted and signed message.\n   * The message will be encrypted with the public keys specified and signed\n   * with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]} Private key \n   * to be used to sign the message\n   * @param {Object[]} publickeys An arraf of {obj: [openpgp_msg_publickey]}\n   * - public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt and sign\n   * @return {String} a binary string representation of the message which \n   * can be OpenPGP armored\n   */\n  function write_signed_and_encrypted_message(privatekey, publickeys, messagetext) {\n    var result = \"\";\n    var i;\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    for (i = 0; i < publickeys.length; i++) {\n      var onepasssignature = new openpgp_packet_onepasssignature();\n      var onepasssigstr = \"\";\n      if (i === 0)\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      else\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      util.print_debug_hexstr_dump(\"onepasssigstr: |\" + onepasssigstr + \"|\\n\", onepasssigstr);\n      var datasignature = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(\n        /\\n/g, \"\\r\\n\"), privatekey);\n      util.print_debug_hexstr_dump(\"datasignature: |\" + datasignature.openpgp + \"|\\n\", datasignature.openpgp);\n      if (i === 0) {\n        result = onepasssigstr + literal + datasignature.openpgp;\n      } else {\n        result = onepasssigstr + result + datasignature.openpgp;\n      }\n    }\n\n    util.print_debug_hexstr_dump(\"signed packet: |\" + result + \"|\\n\", result);\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n  /**\n   * creates a binary string representation of an encrypted message.\n   * The message will be encrypted with the public keys specified \n   * @param {Object[]} publickeys An array of {obj: [openpgp_msg_publickey]}\n   * -public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt\n   * @return {String} a binary string representation of the message\n   * which can be OpenPGP armored\n   */\n  function write_encrypted_message(publickeys, messagetext) {\n    var result = \"\";\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    result = literal;\n\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (var i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n\n  /**\n   * creates a binary string representation a signed message.\n   * The message will be signed with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]}\n   * - the private key to be used to sign the message \n   * @param {String} messagetext message text to sign\n   * @return {Object} {Object: text [String]}, openpgp: {String} a binary\n   *  string representation of the message which can be OpenPGP\n   *   armored(openpgp) and a text representation of the message (text). \n   * This can be directly used to OpenPGP armor the message\n   */\n  function write_signed_message(privatekey, messagetext) {\n    var sig = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/,\n      \"\\r\\n\"), privatekey);\n    var result = {\n      text: messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/, \"\\r\\n\"),\n      openpgp: sig.openpgp,\n      hash: sig.hash\n    };\n    return armor.encode(2, result, null, null);\n  }\n\n  this.generateKeyPair = generateKeyPair;\n  this.write_signed_message = write_signed_message;\n  this.write_signed_and_encrypted_message = write_signed_and_encrypted_message;\n  this.encryptMessage = encryptMessage;\n  this.decryptMessage = decryptMessage;\n\n}\n\nmodule.exports = new _openpgp();\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc The class that deals with storage of the keyring. Currently the only option is to use HTML5 local storage.\n */\nvar keyring = function() {\n  this.armoredPacketlists = [];\n  this.parsedPacketlists = [];\n\n  /**\n   * Initialization routine for the keyring. This method reads the \n   * keyring from HTML5 local storage and initializes this instance.\n   * This method is called by openpgp.init().\n   */\n  function init() {\n    var armoredPacketlists = JSON.parse(window.localStorage.getItem(\"armoredPacketlists\"));\n    if (armoredPacketlists === null || armoredPacketlists.length === 0) {\n      armoredPacketlists = [];\n    }\n    this.armoredPacketlists = armoredPacketlists;\n\n    var packetlist;\n    for (var i = 0; i < armoredPacketlists.length; i++) {\n      packetlist = new packet.list();\n      packetlist.read(armoredPacketlists[i]);\n      this.parsedPacketlists.push(packetlist);\n    }\n  }\n  this.init = init;\n\n  /**\n   * Saves the current state of the keyring to HTML5 local storage.\n   * The privateKeys array and publicKeys array gets Stringified using JSON\n   */\n  function store() {\n    window.localStorage.setItem(\"armoredPacketlists\", JSON.stringify(this.armoredPacketlists));\n  }\n  this.store = store;\n\n  function emailPacketCheck(packet, email) {\n    var emailMatch = false;\n    var packetEmail;\n    email = email.toLowerCase();\n    if (packet.tag == enums.packet.userid) {\n      packetEmail = packet.userid;\n      //we need to get just the email from the userid packet\n      packetEmail = packetEmail.split('<')[1].split('<')[0].trim.toLowerCase();\n      if (packetEmail == email) {\n        emailMatch = true;\n      }\n    }\n    return emailMatch;\n  }\n\n  function idPacketCheck(packet, id) {\n    if (packet.getKeyId && packet.getKeyId().write() == id) {\n      return true;\n    }\n    return false;\n  }\n\n  function helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist) {\n    var packet;\n    for (var l = 0; l < packetlist.length; l++) {\n      packet = packetlist[l];\n      identityMatch = identityFunction(packet, identityInput);\n      if (!packetType) {\n        packetMatch = true;\n      } else if (packet.tag == packetType) {\n        packetMatch = true;\n      }\n      if (packetMatch && identityMatch) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  function checkForIdentityAndPacketMatch(identityFunction, identityInput, packetType) {\n    var results = [];\n    var packetlist;\n    var identityMatch;\n    var packetMatch;\n    for (var p = 0; p < this.parsedPacketlists.length; p++) {\n      identityMatch = false;\n      packetMatch = false;\n      packetlist = this.parsedPacketlists[p];\n      if (helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist)) {\n        results.push(packetlist);\n      }\n    }\n    return results;\n  }\n  this.checkForIdentityAndPacketMatch = checkForIdentityAndPacketMatch;\n\n  /**\n   * searches all public keys in the keyring matching the address or address part of the user ids\n   * @param {String} email_address\n   * @return {openpgp_msg_publickey[]} The public keys associated with provided email address.\n   */\n  function getPublicKeyForAddress(email) {\n    return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.public_key);\n  }\n  this.getPublicKeyForAddress = getPublicKeyForAddress;\n\n  /**\n   * Searches the keyring for a private key containing the specified email address\n   * @param {String} email_address email address to search for\n   * @return {openpgp_msg_privatekey[]} private keys found\n   */\n  function getPrivateKeyForAddress(email_address) {\n    return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.secret_key);\n  }\n  this.getPrivateKeyForAddress = getPrivateKeyForAddress;\n\n  /**\n   * Searches the keyring for public keys having the specified key id\n   * @param {String} keyId provided as string of hex number (lowercase)\n   * @return {openpgp_msg_privatekey[]} public keys found\n   */\n  function getPacketlistsForKeyId(keyId) {\n    return this.checkForIdentityAndPacketMatch(idPacketCheck, keyId);\n  }\n  this.getPacketlistsForKeyId = getPacketlistsForKeyId;\n\n  /**\n   * Imports a packet list (public or private key block) from an ascii armored message \n   * @param {String} armored message to read the packets/key from\n   */\n  function importPacketlist(armored) {\n    this.armoredPacketlists.push(armored);\n\n    var dearmored = armor.decode(armored.replace(/\\r/g, '')).openpgp;\n\n    packetlist = new packet.list();\n    packetlist.read(dearmored);\n    this.parsedPacketlists.push(packetlist);\n\n    return true;\n  }\n  this.importPacketlist = importPacketlist;\n\n  /**\n   * TODO\n   * returns the openpgp_msg_privatekey representation of the public key at public key ring index  \n   * @param {Integer} index the index of the public key within the publicKeys array\n   * @return {openpgp_msg_privatekey} the public key object\n   */\n  function exportPublicKey(index) {\n    return this.publicKey[index];\n  }\n  this.exportPublicKey = exportPublicKey;\n\n  /**\n   * TODO\n   * Removes a public key from the public key keyring at the specified index \n   * @param {Integer} index the index of the public key within the publicKeys array\n   * @return {openpgp_msg_privatekey} The public key object which has been removed\n   */\n  function removePublicKey(index) {\n    var removed = this.publicKeys.splice(index, 1);\n    this.store();\n    return removed;\n  }\n  this.removePublicKey = removePublicKey;\n\n};\n\nmodule.exports = new keyring();\n","var enums = require('../enums.js');\n\n// This is pretty ugly, but browserify needs to have the requires explicitly written.\nmodule.exports = {\n  compressed: require('./compressed.js'),\n  sym_encrypted_integrity_protected: require('./sym_encrypted_integrity_protected.js'),\n  public_key_encrypted_session_key: require('./public_key_encrypted_session_key.js'),\n  sym_encrypted_session_key: require('./sym_encrypted_session_key.js'),\n  literal: require('./literal.js'),\n  public_key: require('./public_key.js'),\n  symmetrically_encrypted: require('./symmetrically_encrypted.js'),\n  marker: require('./marker.js'),\n  public_subkey: require('./public_subkey.js'),\n  user_attribute: require('./user_attribute.js'),\n  one_pass_signature: require('./one_pass_signature.js'),\n  secret_key: require('./secret_key.js'),\n  userid: require('./userid.js'),\n  secret_subkey: require('./secret_subkey.js'),\n  signature: require('./signature.js'),\n  trust: require('./trust.js')\n}\n\nfor (var i in enums.packet) {\n  var packetClass = module.exports[i];\n\n  if (packetClass != undefined)\n    packetClass.prototype.tag = enums.packet[i];\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  JXG = require('../compression/jxg.js'),\n  base64 = require('../encoding/base64.js');\n\n/**\n * @class\n * @classdesc Implementation of the Compressed Data Packet (Tag 8)\n * \n * RFC4880 5.6:\n * The Compressed Data packet contains compressed data.  Typically, this\n * packet is found as the contents of an encrypted packet, or following\n * a Signature or One-Pass Signature packet, and contains a literal data\n * packet.\n */\nmodule.exports = function packet_compressed() {\n  /** @type {packetlist} */\n  this.packets;\n  /** @type {compression} */\n  this.algorithm = 'uncompressed';\n\n  this.compressed = null;\n\n\n  /**\n   * Parsing function for the packet.\n   * @param {String} input Payload of a tag 8 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @parAM {iNTEGER} LEN lENGTH OF the packet or the remaining length of \n   * input at position\n   * @return {openpgp_packet_compressed} Object representation\n   */\n  this.read = function(bytes) {\n    // One octet that gives the algorithm used to compress the packet.\n    this.algorithm = enums.read(enums.compression, bytes.charCodeAt(0));\n\n    // Compressed data, which makes up the remainder of the packet.\n    this.compressed = bytes.substr(1);\n\n    this.decompress();\n  }\n\n\n\n  this.write = function() {\n    if (this.compressed == null)\n      this.compress();\n\n    return String.fromCharCode(enums.write(enums.compression, this.algorithm)) + this.compressed;\n  }\n\n\n  /**\n   * Decompression method for decompressing the compressed data\n   * read by read_packet\n   * @return {String} The decompressed data\n   */\n  this.decompress = function() {\n    var decompressed;\n\n    switch (this.algorithm) {\n      case 'uncompressed':\n        decompressed = this.compressed;\n        break;\n\n      case 'zip':\n        var compData = this.compressed;\n\n        var radix = base64.encode(compData).replace(/\\n/g, \"\");\n        // no header in this case, directly call deflate\n        var jxg_obj = new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(radix));\n\n        decompressed = unescape(jxg_obj.deflate()[0][0]);\n        break;\n\n      case 'zlib':\n        //RFC 1950. Bits 0-3 Compression Method\n        var compressionMethod = this.compressed.charCodeAt(0) % 0x10;\n\n        //Bits 4-7 RFC 1950 are LZ77 Window. Generally this value is 7 == 32k window size.\n        // 2nd Byte in RFC 1950 is for \"FLAGs\" Allows for a Dictionary \n        // (how is this defined). Basic checksum, and compression level.\n\n        if (compressionMethod == 8) { //CM 8 is for DEFLATE, RFC 1951\n          // remove 4 bytes ADLER32 checksum from the end\n          var compData = this.compressed.substring(0, this.compressed.length - 4);\n          var radix = base64.encode(compData).replace(/\\n/g, \"\");\n          //TODO check ADLER32 checksum\n          decompressed = JXG.decompress(radix);\n          break;\n\n        } else {\n          util.print_error(\"Compression algorithm ZLIB only supports \" +\n            \"DEFLATE compression method.\");\n        }\n        break;\n\n      case 'bzip2':\n        // TODO: need to implement this\n        throw new Error('Compression algorithm BZip2 [BZ2] is not implemented.');\n        break;\n\n      default:\n        throw new Error(\"Compression algorithm unknown :\" + this.alogrithm);\n        break;\n    }\n\n    this.packets.read(decompressed);\n  }\n\n  /**\n   * Compress the packet data (member decompressedData)\n   * @param {Integer} type Algorithm to be used // See RFC 4880 9.3\n   * @param {String} data Data to be compressed\n   * @return {String} The compressed data stored in attribute compressedData\n   */\n  this.compress = function() {\n    switch (this.algorithm) {\n\n      case 'uncompressed':\n        // - Uncompressed\n        this.compressed = this.packets.write();\n        break;\n\n      case 'zip':\n        // - ZIP [RFC1951]\n        util.print_error(\"Compression algorithm ZIP [RFC1951] is not implemented.\");\n        break;\n\n      case 'zlib':\n        // - ZLIB [RFC1950]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm ZLIB [RFC1950] is not implemented.\");\n        break;\n\n      case 'bzip2':\n        //  - BZip2 [BZ2]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm BZip2 [BZ2] is not implemented.\");\n        break;\n\n      default:\n        util.print_error(\"Compression algorithm unknown :\" + this.type);\n        break;\n    }\n  }\n};\n","var enums = require('../enums.js');\n\nmodule.exports = {\n  list: require('./packetlist.js'),\n};\n\nvar packets = require('./all_packets.js');\n\nfor (var i in packets)\n  module.exports[i] = packets[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc Implementation of the Literal Data Packet (Tag 11)\n * \n * RFC4880 5.9: A Literal Data packet contains the body of a message; data that\n * is not to be further interpreted.\n */\nmodule.exports = function packet_literal() {\n  this.format = 'utf8';\n  this.data = '';\n  this.date = new Date();\n\n\n  /**\n   * Set the packet data to a javascript native string or a squence of \n   * bytes. Conversion to a proper utf8 encoding takes place when the \n   * packet is written.\n   * @param {String} str Any native javascript string\n   * @param {openpgp_packet_literaldata.format} format \n   */\n  this.set = function(str, format) {\n    this.format = format;\n    this.data = str;\n  }\n\n  /**\n   * Set the packet data to value represented by the provided string\n   * of bytes together with the appropriate conversion format.\n   * @param {String} bytes The string of bytes\n   * @param {openpgp_packet_literaldata.format} format\n   */\n  this.setBytes = function(bytes, format) {\n    this.format = format;\n\n    if (format == 'utf8')\n      bytes = util.decode_utf8(bytes);\n\n    this.data = bytes;\n  }\n\n  /**\n   * Get the byte sequence representing the literal packet data\n   * @returns {String} A sequence of bytes\n   */\n  this.getBytes = function() {\n    if (this.format == 'utf8')\n      return util.encode_utf8(this.data);\n    else\n      return this.data;\n  }\n\n\n\n  /**\n   * Parsing function for a literal data packet (tag 11).\n   * \n   * @param {String} input Payload of a tag 11 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    // - A one-octet field that describes how the data is formatted.\n\n    var format = enums.read(enums.literal, bytes[0].charCodeAt());\n\n    var filename_len = bytes.charCodeAt(1);\n    this.filename = util.decode_utf8(bytes.substr(2, filename_len));\n\n    this.date = util.readDate(bytes.substr(2 + filename_len, 4));\n\n    var data = bytes.substring(6 + filename_len);\n\n    this.setBytes(data, format);\n  }\n\n  /**\n   * Creates a string representation of the packet\n   * \n   * @param {String} data The data to be inserted as body\n   * @return {String} string-representation of the packet\n   */\n  this.write = function() {\n    var filename = util.encode_utf8(\"msg.txt\");\n\n    var data = this.getBytes();\n\n    var result = '';\n    result += String.fromCharCode(enums.write(enums.literal, this.format));\n    result += String.fromCharCode(filename.length);\n    result += filename;\n    result += util.writeDate(this.date);\n    result += data;\n    return result;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the strange \"Marker packet\" (Tag 10)\n * \n * RFC4880 5.8: An experimental version of PGP used this packet as the Literal\n * packet, but no released version of PGP generated Literal packets with this\n * tag. With PGP 5.x, this packet has been reassigned and is reserved for use as\n * the Marker packet.\n * \n * Such a packet MUST be ignored when received.\n */\nfunction packet_marker() {\n  /**\n   * Parsing function for a literal data packet (tag 10).\n   * \n   * @param {String} input Payload of a tag 10 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    if (bytes[0].charCodeAt() == 0x50 && // P\n    bytes[1].charCodeAt() == 0x47 && // G\n    bytes[2].charCodeAt() == 0x50) // P\n      return true;\n    // marker packet does not contain \"PGP\"\n    return false;\n  }\n}\n\nmodule.exports = packet_marker;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the One-Pass Signature Packets (Tag 4)\n * \n * RFC4880 5.4:\n * The One-Pass Signature packet precedes the signed data and contains\n * enough information to allow the receiver to begin calculating any\n * hashes needed to verify the signature.  It allows the Signature\n * packet to be placed at the end of the message, so that the signer\n * can compute the entire signed message in one pass.\n */\n\nvar enums = require('../enums.js'),\n  type_keyid = require('../type/keyid.js');\n\nmodule.exports = function packet_one_pass_signature() {\n  this.version = null; // A one-octet version number.  The current version is 3.\n  this.type = null; // A one-octet signature type.  Signature types are described in RFC4880 Section 5.2.1.\n  this.hashAlgorithm = null; // A one-octet number describing the hash algorithm used. (See RFC4880 9.4)\n  this.publicKeyAlgorithm = null; // A one-octet number describing the public-key algorithm used. (See RFC4880 9.1)\n  this.signingKeyId = null; // An eight-octet number holding the Key ID of the signing key.\n  this.flags = null; //  A one-octet number holding a flag showing whether the signature is nested.  A zero value indicates that the next packet is another One-Pass Signature packet that describes another signature to be applied to the same message data.\n\n  /**\n   * parsing function for a one-pass signature packet (tag 4).\n   * @param {String} bytes payload of a tag 4 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var mypos = 0;\n    // A one-octet version number.  The current version is 3.\n    this.version = bytes.charCodeAt(mypos++);\n\n    // A one-octet signature type.  Signature types are described in\n    //   Section 5.2.1.\n    this.type = enums.read(enums.signature, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the hash algorithm used.\n    this.hashAlgorithm = enums.read(enums.hash, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the public-key algorithm used.\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes.charCodeAt(mypos++));\n\n    // An eight-octet number holding the Key ID of the signing key.\n    this.signingKeyId = new type_keyid();\n    this.signingKeyId.read(bytes.substr(mypos));\n    mypos += 8;\n\n    // A one-octet number holding a flag showing whether the signature\n    //   is nested.  A zero value indicates that the next packet is\n    //   another One-Pass Signature packet that describes another\n    //   signature to be applied to the same message data.\n    this.flags = bytes.charCodeAt(mypos++);\n    return this;\n  }\n\n  /**\n   * creates a string representation of a one-pass signature packet\n   * @param {Integer} type Signature types as described in RFC4880 Section 5.2.1.\n   * @param {Integer} hashalgorithm the hash algorithm used within the signature\n   * @param {openpgp_msg_privatekey} privatekey the private key used to generate the signature\n   * @param {Integer} length length of data to be signed\n   * @param {boolean} nested boolean showing whether the signature is nested. \n   *  \"true\" indicates that the next packet is another One-Pass Signature packet\n   *   that describes another signature to be applied to the same message data. \n   * @return {String} a string representation of a one-pass signature packet\n   */\n  this.write = function(type, hashalgorithm, privatekey, length, nested) {\n    var result = \"\";\n\n    result += String.fromCharCode(3);\n    result += String.fromCharCode(enums.write(enums.signature, type));\n    result += String.fromCharCode(enums.write(enums.hash, this.hashAlgorithm));\n    result += String.fromCharCode(enums.write(enums.publicKey, privatekey.algorithm));\n    result += privatekey.getKeyId().write();\n    if (nested)\n      result += String.fromCharCode(0);\n    else\n      result += String.fromCharCode(1);\n\n    return result;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util');\n\n\nmodule.exports = {\n  readSimpleLength: function(bytes) {\n    var len = 0,\n      offset,\n      type = bytes[0].charCodeAt();\n\n\n    if (type < 192) {\n      len = bytes[0].charCodeAt();\n      offset = 1;\n    } else if (type < 255) {\n      len = ((bytes[0].charCodeAt() - 192) << 8) + (bytes[1].charCodeAt()) + 192;\n      offset = 2;\n    } else if (type == 255) {\n      len = util.readNumber(bytes.substr(1, 4));\n      offset = 5;\n    }\n\n    return {\n      len: len,\n      offset: offset\n    };\n  },\n\n  /**\n   * Encodes a given integer of length to the openpgp length specifier to a\n   * string\n   * \n   * @param {Integer} length The length to encode\n   * @return {String} String with openpgp length representation\n   */\n  writeSimpleLength: function(length) {\n    var result = \"\";\n    if (length < 192) {\n      result += String.fromCharCode(length);\n    } else if (length > 191 && length < 8384) {\n      /*\n       * let a = (total data packet length) - 192 let bc = two octet\n       * representation of a let d = b + 192\n       */\n      result += String.fromCharCode(((length - 192) >> 8) + 192);\n      result += String.fromCharCode((length - 192) & 0xFF);\n    } else {\n      result += String.fromCharCode(255);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Writes a packet header version 4 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeHeader: function(tag_type, length) {\n    /* we're only generating v4 packet headers here */\n    var result = \"\";\n    result += String.fromCharCode(0xC0 | tag_type);\n    result += this.writeSimpleLength(length);\n    return result;\n  },\n\n  /**\n   * Writes a packet header Version 3 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeOldHeader: function(tag_type, length) {\n    var result = \"\";\n    if (length < 256) {\n      result += String.fromCharCode(0x80 | (tag_type << 2));\n      result += String.fromCharCode(length);\n    } else if (length < 65536) {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 1);\n      result += util.writeNumber(length, 2);\n    } else {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 2);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Generic static Packet Parser function\n   * \n   * @param {String} input Input stream as string\n   * @param {integer} position Position to start parsing\n   * @param {integer} len Length of the input from position on\n   * @return {Object} Returns a parsed openpgp_packet\n   */\n  read: function(input, position, len) {\n    // some sanity checks\n    if (input == null || input.length <= position || input.substring(position).length < 2 || (input[position].charCodeAt() &\n      0x80) == 0) {\n      util\n        .print_error(\"Error during parsing. This message / key is probably not containing a valid OpenPGP format.\");\n      return null;\n    }\n    var mypos = position;\n    var tag = -1;\n    var format = -1;\n    var packet_length;\n\n    format = 0; // 0 = old format; 1 = new format\n    if ((input[mypos].charCodeAt() & 0x40) != 0) {\n      format = 1;\n    }\n\n    var packet_length_type;\n    if (format) {\n      // new format header\n      tag = input[mypos].charCodeAt() & 0x3F; // bit 5-0\n    } else {\n      // old format header\n      tag = (input[mypos].charCodeAt() & 0x3F) >> 2; // bit 5-2\n      packet_length_type = input[mypos].charCodeAt() & 0x03; // bit 1-0\n    }\n\n    // header octet parsing done\n    mypos++;\n\n    // parsed length from length field\n    var bodydata = null;\n\n    // used for partial body lengths\n    var real_packet_length = -1;\n    if (!format) {\n      // 4.2.1. Old Format Packet Lengths\n      switch (packet_length_type) {\n        case 0:\n          // The packet has a one-octet length. The header is 2 octets\n          // long.\n          packet_length = input[mypos++].charCodeAt();\n          break;\n        case 1:\n          // The packet has a two-octet length. The header is 3 octets\n          // long.\n          packet_length = (input[mypos++].charCodeAt() << 8) | input[mypos++].charCodeAt();\n          break;\n        case 2:\n          // The packet has a four-octet length. The header is 5\n          // octets long.\n          packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n            8) | input[mypos++].charCodeAt();\n          break;\n        default:\n          // 3 - The packet is of indeterminate length. The header is 1\n          // octet long, and the implementation must determine how long\n          // the packet is. If the packet is in a file, this means that\n          // the packet extends until the end of the file. In general, \n          // an implementation SHOULD NOT use indeterminate-length \n          // packets except where the end of the data will be clear \n          // from the context, and even then it is better to use a \n          // definite length, or a new format header. The new format \n          // headers described below have a mechanism for precisely\n          // encoding data of indeterminate length.\n          packet_length = len;\n          break;\n      }\n\n    } else // 4.2.2. New Format Packet Lengths\n    {\n\n      // 4.2.2.1. One-Octet Lengths\n      if (input[mypos].charCodeAt() < 192) {\n        packet_length = input[mypos++].charCodeAt();\n        util.print_debug(\"1 byte length:\" + packet_length);\n        // 4.2.2.2. Two-Octet Lengths\n      } else if (input[mypos].charCodeAt() >= 192 && input[mypos].charCodeAt() < 224) {\n        packet_length = ((input[mypos++].charCodeAt() - 192) << 8) + (input[mypos++].charCodeAt()) + 192;\n        util.print_debug(\"2 byte length:\" + packet_length);\n        // 4.2.2.4. Partial Body Lengths\n      } else if (input[mypos].charCodeAt() > 223 && input[mypos].charCodeAt() < 255) {\n        packet_length = 1 << (input[mypos++].charCodeAt() & 0x1F);\n        util.print_debug(\"4 byte length:\" + packet_length);\n        // EEEK, we're reading the full data here...\n        var mypos2 = mypos + packet_length;\n        bodydata = input.substring(mypos, mypos + packet_length);\n        while (true) {\n          if (input[mypos2].charCodeAt() < 192) {\n            var tmplen = input[mypos2++].charCodeAt();\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() >= 192 && input[mypos2].charCodeAt() < 224) {\n            var tmplen = ((input[mypos2++].charCodeAt() - 192) << 8) + (input[mypos2++].charCodeAt()) + 192;\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() > 223 && input[mypos2].charCodeAt() < 255) {\n            var tmplen = 1 << (input[mypos2++].charCodeAt() & 0x1F);\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n          } else {\n            mypos2++;\n            var tmplen = (input[mypos2++].charCodeAt() << 24) | (input[mypos2++].charCodeAt() << 16) | (input[mypos2++]\n              .charCodeAt() << 8) | input[mypos2++].charCodeAt();\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            packet_length += tmplen;\n            mypos2 += tmplen;\n            break;\n          }\n        }\n        real_packet_length = mypos2;\n        // 4.2.2.3. Five-Octet Lengths\n      } else {\n        mypos++;\n        packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n          8) | input[mypos++].charCodeAt();\n      }\n    }\n\n    // if there was'nt a partial body length: use the specified\n    // packet_length\n    if (real_packet_length == -1) {\n      real_packet_length = packet_length;\n    }\n\n    if (bodydata == null) {\n      bodydata = input.substring(mypos, mypos + real_packet_length);\n    }\n\n    return {\n      tag: tag,\n      packet: bodydata,\n      offset: mypos + real_packet_length\n    };\n  }\n}\n","var packetParser = require('./packet.js'),\n  packets = require('./all_packets.js'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc This class represents a list of openpgp packets.\n * Take care when iterating over it - the packets themselves\n * are stored as numerical indices.\n */\nmodule.exports = function packetlist() {\n  /** The number of packets contained within the list.\n   * @readonly\n   * @type {Integer} */\n  this.length = 0;\n\n  /**\n   * Reads a stream of binary data and interprents it as a list of packets.\n   * @param {openpgp_bytearray} An array of bytes.\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    while (i < bytes.length) {\n      var parsed = packetParser.read(bytes, i, bytes.length - i);\n      i = parsed.offset;\n\n      var tag = enums.read(enums.packet, parsed.tag);\n      var packet = new packets[tag]();\n\n      this.push(packet);\n\n      packet.read(parsed.packet);\n    }\n  }\n\n  /**\n   * Creates a binary representation of openpgp objects contained within the\n   * class instance.\n   * @returns {openpgp_bytearray} An array of bytes containing valid openpgp packets.\n   */\n  this.write = function() {\n    var bytes = '';\n\n    for (var i = 0; i < this.length; i++) {\n      var packetbytes = this[i].write();\n      bytes += packetParser.writeHeader(this[i].tag, packetbytes.length);\n      bytes += packetbytes;\n    }\n\n    return bytes;\n  }\n\n  /**\n   * Adds a packet to the list. This is the only supported method of doing so;\n   * writing to packetlist[i] directly will result in an error.\n   */\n  this.push = function(packet) {\n    packet.packets = new packetlist();\n\n    this[this.length] = packet;\n    this.length++;\n  }\n\n    /**\n   * Creates a new packetList with all packets that pass the test implemented by the provided function.\n   */\n   this.filter = function(callback) {\n\n    var filtered = new packetlist();\n\n    for (var i = 0; i < this.length; i++) {\n      if (callback(this[i], i, this)) {\n        filtered.push(this[i]);\n      }\n    }\n\n    return filtered; \n   }\n\n   /**\n   * Creates a new packetList with all packets from the given type\n   */\n   this.filterByType = function(packetType) {\n     return this.filter(function(packet) {\n       return packet.tag == packetType;\n     });\n   } \n\n   /**\n    * Executes the provided callback once for each element\n    */\n   this.forEach = function(callback) {\n     for (var i = 0; i < this.length; i++) {\n      callback(this[i]);\n    }\n   }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  type_keyid = require('../type/keyid.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nmodule.exports = function packet_public_key() {\n  /** Key creation date.\n   * @type {Date} */\n  this.created = new Date();\n  /** A list of multiprecision integers\n   * @type {openpgp_type_mpi} */\n  this.mpi = [];\n  /** Public key algorithm\n   * @type {openpgp.publickey} */\n  this.algorithm = 'rsa_sign';\n\n\n  /**\n   * Internal Parser for public keys as specified in RFC 4880 section \n   * 5.5.2 Public-Key Packet Formats\n   * called by read_tag&lt;num&gt;\n   * @param {String} input Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of input\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.readPublicKey = this.read = function(bytes) {\n    // A one-octet version number (3 or 4).\n    var version = bytes[0].charCodeAt();\n\n    if (version == 4) {\n      // - A four-octet number denoting the time that the key was created.\n      this.created = util.readDate(bytes.substr(1, 4));\n\n      // - A one-octet number denoting the public-key algorithm of this key.\n      this.algorithm = enums.read(enums.publicKey, bytes[5].charCodeAt());\n\n      var mpicount = crypto.getPublicMpiCount(this.algorithm);\n      this.mpi = [];\n\n      var bmpi = bytes.substr(6);\n      var p = 0;\n\n      for (var i = 0; i < mpicount && p < bmpi.length; i++) {\n\n        this.mpi[i] = new type_mpi();\n\n        p += this.mpi[i].read(bmpi.substr(p))\n\n        if (p > bmpi.length)\n          util.print_error(\"openpgp.packet.keymaterial.js\\n\" + 'error reading MPI @:' + p);\n      }\n\n      return p + 6;\n    } else {\n      throw new Error('Version ' + version + ' of the key packet is unsupported.');\n    }\n  }\n\n  /*\n   * Same as write_private_key, but has less information because of \n   * public key.\n   * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n   * IE 1 corresponds to RSA.\n   * @param {RSA.keyObject} key\n   * @param timePacket\n   * @return {Object} {body: [string]OpenPGP packet body contents, \n   * header: [string] OpenPGP packet header, string: [string] header+body}\n   */\n  this.writePublicKey = this.write = function() {\n    // Version\n    var result = String.fromCharCode(4);\n    result += util.writeDate(this.created);\n    result += String.fromCharCode(enums.write(enums.publicKey, this.algorithm));\n\n    var mpicount = crypto.getPublicMpiCount(this.algorithm);\n\n    for (var i = 0; i < mpicount; i++) {\n      result += this.mpi[i].write();\n    }\n\n    return result;\n  }\n\n  // Write an old version packet - it's used by some of the internal routines.\n  this.writeOld = function() {\n    var bytes = this.writePublicKey();\n\n    return String.fromCharCode(0x99) +\n      util.writeNumber(bytes.length, 2) +\n      bytes;\n  }\n\n  /**\n   * Calculates the key id of the key \n   * @return {String} A 8 byte key id\n   */\n  this.getKeyId = function() {\n    var keyid = new type_keyid();\n    keyid.read(this.getFingerprint().substr(12, 8));\n    return keyid;\n  }\n\n  /**\n   * Calculates the fingerprint of the key\n   * @return {String} A string containing the fingerprint\n   */\n  this.getFingerprint = function() {\n    var toHash = this.writeOld();\n    return crypto.hash.sha1(toHash, toHash.length);\n  }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_keyid = require('../type/keyid.js'),\n  util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_public_key_encrypted_session_key() {\n  this.version = 3;\n\n  this.publicKeyId = new type_keyid();\n  this.publicKeyAlgorithm = 'rsa_encrypt';\n\n  this.sessionKey = null;\n  this.sessionKeyAlgorithm = 'aes256';\n\n  /** @type {openpgp_type_mpi[]} */\n  this.encrypted = [];\n\n  /**\n   * Parsing function for a publickey encrypted session key packet (tag 1).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n\n    this.version = bytes[0].charCodeAt();\n    this.publicKeyId.read(bytes.substr(1));\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes[9].charCodeAt());\n\n    var i = 10;\n\n    var integerCount = (function(algo) {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          return 1;\n\n        case 'elgamal':\n          return 2;\n\n        default:\n          throw new Error(\"Invalid algorithm.\");\n      }\n    })(this.publicKeyAlgorithm);\n\n    this.encrypted = [];\n\n    for (var j = 0; j < integerCount; j++) {\n      var mpi = new type_mpi();\n      i += mpi.read(bytes.substr(i));\n      this.encrypted.push(mpi);\n    }\n  }\n\n  /**\n   * Create a string representation of a tag 1 packet\n   * \n   * @param {String} publicKeyId\n   *             The public key id corresponding to publicMPIs key as string\n   * @param {openpgp_type_mpi[]} publicMPIs\n   *            Multiprecision integer objects describing the public key\n   * @param {Integer} pubalgo\n   *            The corresponding public key algorithm // See RFC4880 9.1\n   * @param {Integer} symmalgo\n   *            The symmetric cipher algorithm used to encrypt the data \n   *            within an encrypteddatapacket or encryptedintegrity-\n   *            protecteddatapacket \n   *            following this packet //See RFC4880 9.2\n   * @param {String} sessionkey\n   *            A string of randombytes representing the session key\n   * @return {String} The string representation\n   */\n  this.write = function() {\n\n    var result = String.fromCharCode(this.version);\n    result += this.publicKeyId.write();\n    result += String.fromCharCode(\n      enums.write(enums.publicKey, this.publicKeyAlgorithm));\n\n    for (var i = 0; i < this.encrypted.length; i++) {\n      result += this.encrypted[i].write()\n    }\n\n    return result;\n  }\n\n  this.encrypt = function(key) {\n    var data = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm));\n\n    data += this.sessionKey;\n    var checksum = util.calc_checksum(this.sessionKey);\n    data += util.writeNumber(checksum, 2);\n\n    var mpi = new type_mpi();\n    mpi.fromBytes(crypto.pkcs1.eme.encode(\n      data,\n      key.mpi[0].byteLength()));\n\n    this.encrypted = crypto.publicKeyEncrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      mpi);\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(key) {\n    var result = crypto.publicKeyDecrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      this.encrypted).toBytes();\n\n    var checksum = util.readNumber(result.substr(result.length - 2));\n\n    var decoded = crypto.pkcs1.eme.decode(\n      result,\n      key.mpi[0].byteLength());\n\n    var key = decoded.substring(1, decoded.length - 2);\n\n    if (checksum != util.calc_checksum(key)) {\n      throw new Error('Checksum mismatch');\n    } else {\n      this.sessionKey = key;\n      this.sessionKeyAlgorithm =\n        enums.read(enums.symmetric, decoded.charCodeAt(0));\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar public_key = require('./public_key.js');\n\nmodule.exports = function public_subkey() {\n  public_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar publicKey = require('./public_key.js'),\n  enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js'),\n  type_s2k = require('../type/s2k.js');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nfunction packet_secret_key() {\n  publicKey.call(this);\n\n  this.encrypted = null;\n\n\n  function get_hash_len(hash) {\n    if (hash == 'sha1')\n      return 20;\n    else\n      return 2;\n  }\n\n  function get_hash_fn(hash) {\n    if (hash == 'sha1')\n      return crypto.hash.sha1;\n    else\n      return function(c) {\n        return util.writeNumber(util.calc_checksum(c), 2);\n    };\n  }\n\n  // Helper function\n\n  function parse_cleartext_mpi(hash_algorithm, cleartext, algorithm) {\n    var hashlen = get_hash_len(hash_algorithm),\n      hashfn = get_hash_fn(hash_algorithm);\n\n    var hashtext = cleartext.substr(cleartext.length - hashlen);\n    cleartext = cleartext.substr(0, cleartext.length - hashlen);\n\n    var hash = hashfn(cleartext);\n\n    if (hash != hashtext)\n      throw new Error(\"Hash mismatch.\");\n\n    var mpis = crypto.getPrivateMpiCount(algorithm);\n\n    var j = 0;\n    var mpi = [];\n\n    for (var i = 0; i < mpis && j < cleartext.length; i++) {\n      mpi[i] = new type_mpi();\n      j += mpi[i].read(cleartext.substr(j));\n    }\n\n    return mpi;\n  }\n\n  function write_cleartext_mpi(hash_algorithm, algorithm, mpi) {\n    var bytes = '';\n    var discard = crypto.getPublicMpiCount(algorithm);\n\n    for (var i = discard; i < mpi.length; i++) {\n      bytes += mpi[i].write();\n    }\n\n\n    bytes += get_hash_fn(hash_algorithm)(bytes);\n\n    return bytes;\n  }\n\n\n  // 5.5.3.  Secret-Key Packet Formats\n\n  /**\n   * Internal parser for private keys as specified in RFC 4880 section 5.5.3\n   * @param {String} bytes Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of bytes\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.read = function(bytes) {\n    // - A Public-Key or Public-Subkey packet, as described above.\n    var len = this.readPublicKey(bytes);\n\n    bytes = bytes.substr(len);\n\n\n    // - One octet indicating string-to-key usage conventions.  Zero\n    //   indicates that the secret-key data is not encrypted.  255 or 254\n    //   indicates that a string-to-key specifier is being given.  Any\n    //   other value is a symmetric-key encryption algorithm identifier.\n    var isEncrypted = bytes[0].charCodeAt();\n\n    if (isEncrypted) {\n      this.encrypted = bytes;\n    } else {\n\n      // - Plain or encrypted multiprecision integers comprising the secret\n      //   key data.  These algorithm-specific fields are as described\n      //   below.\n\n      this.mpi = this.mpi.concat(parse_cleartext_mpi('mod', bytes.substr(1),\n        this.algorithm));\n    }\n\n  };\n\n  /*\n     * Creates an OpenPGP key packet for the given key. much \n\t * TODO in regards to s2k, subkeys.\n     * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n\t * IE 1 corresponds to RSA.\n     * @param {RSA.keyObject} key\n     * @param passphrase\n     * @param s2kHash\n     * @param symmetricEncryptionAlgorithm\n     * @param timePacket\n     * @return {Object} {body: [string]OpenPGP packet body contents, \n\t\theader: [string] OpenPGP packet header, string: [string] header+body}\n     */\n  this.write = function() {\n    var bytes = this.writePublicKey();\n\n    if (!this.encrypted) {\n      bytes += String.fromCharCode(0);\n\n      bytes += write_cleartext_mpi('mod', this.algorithm, this.mpi);\n    } else {\n      bytes += this.encrypted;\n    }\n\n    return bytes;\n  };\n\n\n\n\n  /** Encrypt the payload. By default, we use aes256 and iterated, salted string\n   * to key specifier\n   * @param {String} passphrase\n   */\n  this.encrypt = function(passphrase) {\n\n    var s2k = new type_s2k(),\n      symmetric = 'aes256',\n      cleartext = write_cleartext_mpi('sha1', this.algorithm, this.mpi),\n      key = produceEncryptionKey(s2k, passphrase, symmetric),\n      blockLen = crypto.cipher[symmetric].blockSize,\n      iv = crypto.random.getRandomBytes(blockLen);\n\n\n    this.encrypted = '';\n    this.encrypted += String.fromCharCode(254);\n    this.encrypted += String.fromCharCode(enums.write(enums.symmetric, symmetric));\n    this.encrypted += s2k.write();\n    this.encrypted += iv;\n\n    this.encrypted += crypto.cfb.normalEncrypt(symmetric, key, cleartext, iv);\n  };\n\n  function produceEncryptionKey(s2k, passphrase, algorithm) {\n    return s2k.produce_key(passphrase,\n      crypto.cipher[algorithm].keySize);\n  }\n\n  /**\n   * Decrypts the private key MPIs which are needed to use the key.\n   * openpgp_packet_keymaterial.hasUnencryptedSecretKeyData should be \n   * false otherwise\n   * a call to this function is not needed\n   * \n   * @param {String} str_passphrase The passphrase for this private key \n   * as string\n   * @return {Boolean} True if the passphrase was correct; false if not\n   */\n  this.decrypt = function(passphrase) {\n    if (!this.encrypted)\n      return;\n\n    var i = 0,\n      symmetric,\n      key;\n\n    var s2k_usage = this.encrypted[i++].charCodeAt();\n\n    // - [Optional] If string-to-key usage octet was 255 or 254, a one-\n    //   octet symmetric encryption algorithm.\n    if (s2k_usage == 255 || s2k_usage == 254) {\n      symmetric = this.encrypted[i++].charCodeAt();\n      symmetric = enums.read(enums.symmetric, symmetric);\n\n      // - [Optional] If string-to-key usage octet was 255 or 254, a\n      //   string-to-key specifier.  The length of the string-to-key\n      //   specifier is implied by its type, as described above.\n      var s2k = new type_s2k();\n      i += s2k.read(this.encrypted.substr(i));\n\n      key = produceEncryptionKey(s2k, passphrase, symmetric);\n    } else {\n      symmetric = s2k_usage;\n      symmetric = enums.read(enums.symmetric, symmetric);\n      key = crypto.hash.md5(passphrase);\n    }\n\n\n    // - [Optional] If secret data is encrypted (string-to-key usage octet\n    //   not zero), an Initial Vector (IV) of the same length as the\n    //   cipher's block size.\n    var iv = this.encrypted.substr(i,\n      crypto.cipher[symmetric].blockSize);\n\n    i += iv.length;\n\n    var cleartext,\n      ciphertext = this.encrypted.substr(i);\n\n    cleartext = crypto.cfb.normalDecrypt(symmetric, key, ciphertext, iv);\n\n    var hash = s2k_usage == 254 ?\n      'sha1' :\n      'mod';\n\n\n    this.mpi = this.mpi.concat(parse_cleartext_mpi(hash, cleartext,\n      this.algorithm));\n  };\n\n  this.generate = function(bits, passphrase) {\n    this.mpi = crypto.generateMpi(this.algorithm, bits);\n  };\n\n}\n\npacket_secret_key.prototype = new publicKey;\n\nmodule.exports = packet_secret_key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar secret_key = require('./secret_key.js');\n\nmodule.exports = function secret_subkey() {\n  secret_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  packet = require('./packet.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js');\n\n/**\n * @class\n * @classdesc Implementation of the Signature Packet (Tag 2)\n * \n * RFC4480 5.2:\n * A Signature packet describes a binding between some public key and\n * some data.  The most common signatures are a signature of a file or a\n * block of text, and a signature that is a certification of a User ID.\n */\nmodule.exports = function packet_signature() {\n\n  this.signatureType = null;\n  this.hashAlgorithm = null;\n  this.publicKeyAlgorithm = null;\n\n  this.signatureData = null;\n  this.signedHashValue = null;\n  this.mpi = null;\n\n  this.created = null;\n  this.signatureExpirationTime = null;\n  this.signatureNeverExpires = null;\n  this.exportable = null;\n  this.trustLevel = null;\n  this.trustAmount = null;\n  this.regularExpression = null;\n  this.revocable = null;\n  this.keyExpirationTime = null;\n  this.keyNeverExpires = null;\n  this.preferredSymmetricAlgorithms = null;\n  this.revocationKeyClass = null;\n  this.revocationKeyAlgorithm = null;\n  this.revocationKeyFingerprint = null;\n  this.issuerKeyId = null;\n  this.notation = {};\n  this.preferredHashAlgorithms = null;\n  this.preferredCompressionAlgorithms = null;\n  this.keyServerPreferences = null;\n  this.preferredKeyServer = null;\n  this.isPrimaryUserID = null;\n  this.policyURI = null;\n  this.keyFlags = null;\n  this.signersUserId = null;\n  this.reasonForRevocationFlag = null;\n  this.reasonForRevocationString = null;\n  this.signatureTargetPublicKeyAlgorithm = null;\n  this.signatureTargetHashAlgorithm = null;\n  this.signatureTargetHash = null;\n  this.embeddedSignature = null;\n\n  this.verified = false;\n\n  /**\n   * parsing function for a signature packet (tag 2).\n   * @param {String} bytes payload of a tag 2 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    var version = bytes[i++].charCodeAt();\n    // switch on version (3 and 4)\n    switch (version) {\n      case 3:\n        // One-octet length of following hashed material. MUST be 5.\n        if (bytes[i++].charCodeAt() != 5)\n          util.print_debug(\"openpgp.packet.signature.js\\n\" +\n            'invalid One-octet length of following hashed material.' +\n            'MUST be 5. @:' + (i - 1));\n\n        var sigpos = i;\n        // One-octet signature type.\n        this.signatureType = bytes[i++].charCodeAt();\n\n        // Four-octet creation time.\n        this.created = util.readDate(bytes.substr(i, 4));\n        i += 4;\n\n        // storing data appended to data which gets verified\n        this.signatureData = bytes.substring(position, i);\n\n        // Eight-octet Key ID of signer.\n        this.issuerKeyId = bytes.substring(i, i + 8);\n        i += 8;\n\n        // One-octet public-key algorithm.\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n\n        // One-octet hash algorithm.\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n        break;\n      case 4:\n        this.signatureType = bytes[i++].charCodeAt();\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n\n\n        function subpackets(bytes, signed) {\n          // Two-octet scalar octet count for following hashed subpacket\n          // data.\n          var subpacket_length = util.readNumber(\n            bytes.substr(0, 2));\n\n          var i = 2;\n\n          // Hashed subpacket data set (zero or more subpackets)\n          var subpacked_read = 0;\n          while (i < 2 + subpacket_length) {\n\n            var len = packet.readSimpleLength(bytes.substr(i));\n            i += len.offset;\n\n            // Since it is trivial to add data to the unhashed portion of \n            // the packet we simply ignore all unauthenticated data.\n            if (signed)\n              this.read_sub_packet(bytes.substr(i, len.len));\n\n            i += len.len;\n          }\n\n          return i;\n        }\n\n        i += subpackets.call(this, bytes.substr(i), true);\n\n        // A V4 signature hashes the packet body\n        // starting from its first field, the version number, through the end\n        // of the hashed subpacket data.  Thus, the fields hashed are the\n        // signature version, the signature type, the public-key algorithm, the\n        // hash algorithm, the hashed subpacket length, and the hashed\n        // subpacket body.\n        this.signatureData = bytes.substr(0, i);\n\n        i += subpackets.call(this, bytes.substr(i), false);\n\n        break;\n      default:\n        throw new Error('Version ' + version + ' of the signature is unsupported.');\n        break;\n    }\n\n    // Two-octet field holding left 16 bits of signed hash value.\n    this.signedHashValue = bytes.substr(i, 2);\n    i += 2;\n\n    this.signature = bytes.substr(i);\n  };\n\n  this.write = function() {\n    return this.signatureData +\n      util.writeNumber(0, 2) + // Number of unsigned subpackets.\n    this.signedHashValue +\n      this.signature;\n  };\n\n  /**\n   * Signs provided data. This needs to be done prior to serialization.\n   * @param {Object} data Contains packets to be signed.\n   * @param {openpgp_msg_privatekey} privatekey private key used to sign the message. \n   */\n  this.sign = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var result = String.fromCharCode(4);\n    result += String.fromCharCode(signatureType);\n    result += String.fromCharCode(publicKeyAlgorithm);\n    result += String.fromCharCode(hashAlgorithm);\n\n    //Calculate subpackets\n    var creationTimeSubpacket = write_sub_packet(enums.signatureSubpacket.signature_creation_time,\n      util.writeDate(new Date()));\n\n    var issuerSubpacket = write_sub_packet(enums.signatureSubpacket.issuer, key.getKeyId().write());\n\n    // Add subpackets here\n    result += util.writeNumber(creationTimeSubpacket.length + issuerSubpacket.length, 2);\n    result += creationTimeSubpacket + issuerSubpacket;\n\n    this.signatureData = result;\n\n    var trailer = this.calculateTrailer();\n\n    var toHash = this.toSign(signatureType, data) +\n      this.signatureData + trailer;\n\n    var hash = crypto.hash.digest(hashAlgorithm, toHash);\n\n    this.signedHashValue = hash.substr(0, 2);\n\n    this.signature = crypto.signature.sign(hashAlgorithm,\n      publicKeyAlgorithm, key.mpi, toHash);\n  };\n\n  /**\n   * creates a string representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   * @param {Integer} type subpacket signature type. Signature types as described \n   * in RFC4880 Section 5.2.3.2\n   * @param {String} data data to be included\n   * @return {String} a string-representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   */\n  function write_sub_packet(type, data) {\n    var result = \"\";\n    result += packet.writeSimpleLength(data.length + 1);\n    result += String.fromCharCode(type);\n    result += data;\n    return result;\n  }\n\n  // V4 signature sub packets\n\n  this.read_sub_packet = function(bytes) {\n    var mypos = 0;\n\n    function read_array(prop, bytes) {\n      this[prop] = [];\n\n      for (var i = 0; i < bytes.length; i++) {\n        this[prop].push(bytes[i].charCodeAt());\n      }\n    }\n\n    // The leftwost bit denotes a \"critical\" packet, but we ignore it.\n    var type = bytes[mypos++].charCodeAt() & 0x7F;\n\n    // subpacket type\n    switch (type) {\n      case 2:\n        // Signature Creation Time\n        this.created = util.readDate(bytes.substr(mypos));\n        break;\n      case 3:\n        // Signature Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.signatureNeverExpires = time.getTime() == 0;\n        this.signatureExpirationTime = time;\n\n        break;\n      case 4:\n        // Exportable Certification\n        this.exportable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 5:\n        // Trust Signature\n        this.trustLevel = bytes[mypos++].charCodeAt();\n        this.trustAmount = bytes[mypos++].charCodeAt();\n        break;\n      case 6:\n        // Regular Expression\n        this.regularExpression = bytes.substr(mypos);\n        break;\n      case 7:\n        // Revocable\n        this.revocable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 9:\n        // Key Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.keyExpirationTime = time;\n        this.keyNeverExpires = time.getTime() == 0;\n\n        break;\n      case 11:\n        // Preferred Symmetric Algorithms\n        this.preferredSymmetricAlgorithms = [];\n\n        while (mypos != bytes.length) {\n          this.preferredSymmetricAlgorithms.push(bytes[mypos++].charCodeAt());\n        }\n\n        break;\n      case 12:\n        // Revocation Key\n        // (1 octet of class, 1 octet of public-key algorithm ID, 20\n        // octets of\n        // fingerprint)\n        this.revocationKeyClass = bytes[mypos++].charCodeAt();\n        this.revocationKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.revocationKeyFingerprint = bytes.substr(mypos, 20);\n        break;\n\n      case 16:\n        // Issuer\n        this.issuerKeyId = bytes.substr(mypos, 8);\n        break;\n\n      case 20:\n        // Notation Data\n        // We don't know how to handle anything but a text flagged data.\n        if (bytes[mypos].charCodeAt() == 0x80) {\n\n          // We extract key/value tuple from the byte stream.\n          mypos += 4;\n          var m = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n          var n = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n\n          var name = bytes.substr(mypos, m),\n            value = bytes.substr(mypos + m, n);\n\n          this.notation[name] = value;\n        } else throw new Error(\"Unsupported notation flag.\");\n        break;\n      case 21:\n        // Preferred Hash Algorithms\n        read_array.call(this, 'preferredHashAlgorithms', bytes.substr(mypos));\n        break;\n      case 22:\n        // Preferred Compression Algorithms\n        read_array.call(this, 'preferredCompressionAlgorithms ', bytes.substr(mypos));\n        break;\n      case 23:\n        // Key Server Preferences\n        read_array.call(this, 'keyServerPreferencess', bytes.substr(mypos));\n        break;\n      case 24:\n        // Preferred Key Server\n        this.preferredKeyServer = bytes.substr(mypos);\n        break;\n      case 25:\n        // Primary User ID\n        this.isPrimaryUserID = bytes[mypos++] != 0;\n        break;\n      case 26:\n        // Policy URI\n        this.policyURI = bytes.substr(mypos);\n        break;\n      case 27:\n        // Key Flags\n        read_array.call(this, 'keyFlags', bytes.substr(mypos));\n        break;\n      case 28:\n        // Signer's User ID\n        this.signersUserId += bytes.substr(mypos);\n        break;\n      case 29:\n        // Reason for Revocation\n        this.reasonForRevocationFlag = bytes[mypos++].charCodeAt();\n        this.reasonForRevocationString = bytes.substr(mypos);\n        break;\n      case 30:\n        // Features\n        read_array.call(this, 'features', bytes.substr(mypos));\n        break;\n      case 31:\n        // Signature Target\n        // (1 octet public-key algorithm, 1 octet hash algorithm, N octets hash)\n        this.signatureTargetPublicKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.signatureTargetHashAlgorithm = bytes[mypos++].charCodeAt();\n\n        var len = crypto.getHashByteLength(this.signatureTargetHashAlgorithm);\n\n        this.signatureTargetHash = bytes.substr(mypos, len);\n        break;\n      case 32:\n        // Embedded Signature\n        this.embeddedSignature = new packet_signature();\n        this.embeddedSignature.read(bytes.substr(mypos));\n        break;\n      default:\n        util.print_error(\"openpgp.packet.signature.js\\n\" +\n          'unknown signature subpacket type ' + type + \" @:\" + mypos +\n          \" subplen:\" + subplen + \" len:\" + len);\n        break;\n    }\n  };\n\n  // Produces data to produce signature on\n  this.toSign = function(type, data) {\n    var t = enums.signature;\n\n    switch (type) {\n      case t.binary:\n        return data.getBytes();\n\n      case t.text:\n        return this.toSign(t.binary, data)\n          .replace(/\\r\\n/g, '\\n')\n          .replace(/\\n/g, '\\r\\n');\n\n      case t.standalone:\n        return '';\n\n      case t.cert_generic:\n      case t.cert_persona:\n      case t.cert_casual:\n      case t.cert_positive:\n      case t.cert_revocation:\n        {\n          var packet, tag;\n\n          if (data.userid !== undefined) {\n            tag = 0xB4;\n            packet = data.userid;\n          } else if (data.userattribute !== undefined) {\n            tag = 0xD1;\n            packet = data.userattribute;\n          } else throw new Error('Either a userid or userattribute packet needs to be ' +\n              'supplied for certification.');\n\n          var bytes = packet.write();\n\n          return this.toSign(t.key, data) +\n            String.fromCharCode(tag) +\n            util.writeNumber(bytes.length, 4) +\n            bytes;\n        }\n      case t.subkey_binding:\n      case t.key_binding:\n        {\n          return this.toSign(t.key, data) + this.toSign(t.key, {\n            key: data.bind\n          });\n        }\n      case t.key:\n        {\n          if (data.key == undefined)\n            throw new Error('Key packet is required for this sigtature.');\n\n          return data.key.writeOld();\n        }\n      case t.key_revocation:\n      case t.subkey_revocation:\n        return this.toSign(t.key, data);\n      case t.timestamp:\n        return '';\n      case t.thrid_party:\n        throw new Error('Not implemented');\n        break;\n      default:\n        throw new Error('Unknown signature type.')\n    }\n  }\n\n\n  this.calculateTrailer = function() {\n    // calculating the trailer\n    var trailer = '';\n    trailer += String.fromCharCode(4); // Version\n    trailer += String.fromCharCode(0xFF);\n    trailer += util.writeNumber(this.signatureData.length, 4);\n    return trailer\n  }\n\n\n  /**\n   * verifys the signature packet. Note: not signature types are implemented\n   * @param {String} data data which on the signature applies\n   * @param {openpgp_msg_privatekey} key the public key to verify the signature\n   * @return {boolean} True if message is verified, else false.\n   */\n  this.verify = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var bytes = this.toSign(signatureType, data),\n      trailer = this.calculateTrailer();\n\n\n    var mpicount = 0;\n    // Algorithm-Specific Fields for RSA signatures:\n    // \t    - multiprecision number (MPI) of RSA signature value m**d mod n.\n    if (publicKeyAlgorithm > 0 && publicKeyAlgorithm < 4)\n      mpicount = 1;\n    //    Algorithm-Specific Fields for DSA signatures:\n    //      - MPI of DSA value r.\n    //      - MPI of DSA value s.\n    else if (publicKeyAlgorithm == 17)\n      mpicount = 2;\n\n    var mpi = [],\n      i = 0;\n    for (var j = 0; j < mpicount; j++) {\n      mpi[j] = new type_mpi();\n      i += mpi[j].read(this.signature.substr(i));\n    }\n\n    this.verified = crypto.signature.verify(publicKeyAlgorithm,\n      hashAlgorithm, mpi, key.mpi,\n      bytes + this.signatureData + trailer);\n\n    return this.verified;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Sym. Encrypted Integrity Protected Data \n * Packet (Tag 18)\n * \n * RFC4880 5.13: The Symmetrically Encrypted Integrity Protected Data packet is\n * a variant of the Symmetrically Encrypted Data packet. It is a new feature\n * created for OpenPGP that addresses the problem of detecting a modification to\n * encrypted data. It is used in combination with a Modification Detection Code\n * packet.\n */\n\nmodule.exports = function packet_sym_encrypted_integrity_protected() {\n  /** The encrypted payload. */\n  this.encrypted = null; // string\n  /** @type {Boolean}\n   * If after decrypting the packet this is set to true,\n   * a modification has been detected and thus the contents\n   * should be discarded.\n   */\n  this.modification = false;\n  this.packets;\n\n\n  this.read = function(bytes) {\n    // - A one-octet version number. The only currently defined value is\n    // 1.\n    var version = bytes[0].charCodeAt();\n\n    if (version != 1) {\n      throw new Error('Invalid packet version.');\n    }\n\n    // - Encrypted data, the output of the selected symmetric-key cipher\n    //   operating in Cipher Feedback mode with shift amount equal to the\n    //   block size of the cipher (CFB-n where n is the block size).\n    this.encrypted = bytes.substr(1);\n  }\n\n  this.write = function() {\n\n    return String.fromCharCode(1) // Version\n    + this.encrypted;\n  }\n\n  this.encrypt = function(sessionKeyAlgorithm, key) {\n    var bytes = this.packets.write()\n\n    var prefixrandom = crypto.getPrefixRandom(sessionKeyAlgorithm);\n    var prefix = prefixrandom + prefixrandom.charAt(prefixrandom.length - 2) + prefixrandom.charAt(prefixrandom.length -\n      1)\n\n    var tohash = bytes;\n\n\n    // Modification detection code packet.\n    tohash += String.fromCharCode(0xD3);\n    tohash += String.fromCharCode(0x14);\n\n\n    tohash += crypto.hash.sha1(prefix + tohash);\n\n\n    this.encrypted = crypto.cfb.encrypt(prefixrandom,\n      sessionKeyAlgorithm, tohash, key, false).substring(0,\n      prefix.length + tohash.length);\n  }\n\n  /**\n   * Decrypts the encrypted data contained in this object read_packet must\n   * have been called before\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *            The selected symmetric encryption algorithm to be used\n   * @param {String} key The key of cipher blocksize length to be used\n   * @return {String} The decrypted data of this packet\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, false);\n\n\n    // there must be a modification detection code packet as the\n    // last packet and everything gets hashed except the hash itself\n    this.hash = crypto.hash.sha1(\n      crypto.cfb.mdc(sessionKeyAlgorithm, key, this.encrypted) + decrypted.substring(0, decrypted.length - 20));\n\n\n    var mdc = decrypted.substr(decrypted.length - 20, 20);\n\n    if (this.hash != mdc) {\n      throw new Error('Modification detected.');\n    } else\n      this.packets.read(decrypted.substr(0, decrypted.length - 22));\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_s2k = require('../type/s2k.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_sym_encrypted_session_key() {\n  this.tag = 3;\n  this.sessionKeyEncryptionAlgorithm = null;\n  this.sessionKeyAlgorithm = 'aes256';\n  this.encrypted = null;\n  this.s2k = new type_s2k();\n\n  /**\n   * Parsing function for a symmetric encrypted session key packet (tag 3).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    // A one-octet version number. The only currently defined version is 4.\n    this.version = bytes[0].charCodeAt();\n\n    // A one-octet number describing the symmetric algorithm used.\n    var algo = enums.read(enums.symmetric, bytes[1].charCodeAt());\n\n    // A string-to-key (S2K) specifier, length as defined above.\n    var s2klength = this.s2k.read(bytes.substr(2));\n\n    // Optionally, the encrypted session key itself, which is decrypted\n    // with the string-to-key object.\n    var done = s2klength + 2;\n\n    if (done < bytes.length) {\n      this.encrypted = bytes.substr(done);\n      this.sessionKeyEncryptionAlgorithm = algo\n    } else\n      this.sessionKeyAlgorithm = algo;\n  }\n\n  this.write = function() {\n    var algo = this.encrypted == null ?\n      this.sessionKeyAlgorithm :\n      this.sessionKeyEncryptionAlgorithm;\n\n    var bytes = String.fromCharCode(this.version) +\n      String.fromCharCode(enums.write(enums.symmetric, algo)) +\n      this.s2k.write();\n\n    if (this.encrypted != null)\n      bytes += this.encrypted;\n    return bytes;\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(passphrase) {\n    var algo = this.sessionKeyEncryptionAlgorithm != null ?\n      this.sessionKeyEncryptionAlgorithm :\n      this.sessionKeyAlgorithm;\n\n\n    var length = crypto.cipher[algo].keySize;\n    var key = this.s2k.produce_key(passphrase, length);\n\n    if (this.encrypted == null) {\n      this.sessionKey = key;\n\n    } else {\n      var decrypted = crypto.cfb.decrypt(\n        this.sessionKeyEncryptionAlgorithm, key, this.encrypted, true);\n\n      this.sessionKeyAlgorithm = enums.read(enums.symmetric,\n        decrypted[0].keyCodeAt());\n\n      this.sessionKey = decrypted.substr(1);\n    }\n  }\n\n  this.encrypt = function(passphrase) {\n    var length = crypto.getKeyLength(this.sessionKeyEncryptionAlgorithm);\n    var key = this.s2k.produce_key(passphrase, length);\n\n    var private_key = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm)) +\n\n    crypto.getRandomBytes(\n      crypto.getKeyLength(this.sessionKeyAlgorithm));\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(this.sessionKeyEncryptionAlgorithm),\n      this.sessionKeyEncryptionAlgorithm, key, private_key, true);\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Symmetrically Encrypted Data Packet (Tag 9)\n * \n * RFC4880 5.7: The Symmetrically Encrypted Data packet contains data encrypted\n * with a symmetric-key algorithm. When it has been decrypted, it contains other\n * packets (usually a literal data packet or compressed data packet, but in\n * theory other Symmetrically Encrypted Data packets or sequences of packets\n * that form whole OpenPGP messages).\n */\n\nmodule.exports = function packet_symmetrically_encrypted() {\n  this.encrypted = null;\n  /** Decrypted packets contained within. \n   * @type {openpgp_packetlist} */\n  this.packets;\n\n  this.read = function(bytes) {\n    this.encrypted = bytes;\n  }\n\n  this.write = function() {\n    return this.encrypted;\n  }\n\n  /**\n   * Symmetrically decrypt the packet data\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *             Symmetric key algorithm to use // See RFC4880 9.2\n   * @param {String} key\n   *             Key as string with the corresponding length to the\n   *            algorithm\n   * @return The decrypted data;\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, true);\n\n    this.packets.read(decrypted);\n  }\n\n  this.encrypt = function(algo, key) {\n    var data = this.packets.write();\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(algo), algo, data, key, true);\n  }\n};\n","module.exports = function packet_trust() {\n\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/** \n * @class\n * @classdesc Implementation of the User Attribute Packet (Tag 17)\n *  The User Attribute packet is a variation of the User ID packet.  It\n *  is capable of storing more types of data than the User ID packet,\n *  which is limited to text.  Like the User ID packet, a User Attribute\n *  packet may be certified by the key owner (\"self-signed\") or any other\n *  key owner who cares to certify it.  Except as noted, a User Attribute\n *  packet may be used anywhere that a User ID packet may be used.\n *\n *  While User Attribute packets are not a required part of the OpenPGP\n *  standard, implementations SHOULD provide at least enough\n *  compatibility to properly handle a certification signature on the\n *  User Attribute packet.  A simple way to do this is by treating the\n *  User Attribute packet as a User ID packet with opaque contents, but\n *  an implementation may use any method desired.\n */\nmodule.exports = function packet_user_attribute() {\n  this.tag = 17;\n  this.attributes = [];\n\n  /**\n   * parsing function for a user attribute packet (tag 17).\n   * @param {String} input payload of a tag 17 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    while (i < bytes.length) {\n      var len = openpgp_packet.read_simple_length(bytes);\n\n      i += len.offset;\n      this.attributes.push(bytes.substr(i, len.len));\n      i += len.len;\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of the User ID Packet (Tag 13)\n * A User ID packet consists of UTF-8 text that is intended to represent\n * the name and email address of the key holder.  By convention, it\n * includes an RFC 2822 [RFC2822] mail name-addr, but there are no\n * restrictions on its content.  The packet length in the header\n * specifies the length of the User ID. \n */\nmodule.exports = function packet_userid() {\n  /** @type {String} A string containing the user id. Usually in the form\n   * John Doe <john@example.com> \n   */\n  this.userid = '';\n\n\n  /**\n   * Parsing function for a user id packet (tag 13).\n   * @param {String} input payload of a tag 13 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input \n   * at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    this.userid = util.decode_utf8(bytes);\n  }\n\n  /**\n   * Creates a string representation of the user id packet\n   * @param {String} user_id the user id as string (\"John Doe <john.doe@mail.us\")\n   * @return {String} string representation\n   */\n  this.write = function() {\n    return util.encode_utf8(this.userid);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of type key id (RFC4880 3.3)\n *  A Key ID is an eight-octet scalar that identifies a key.\n   Implementations SHOULD NOT assume that Key IDs are unique.  The\n   section \"Enhanced Key Formats\" below describes how Key IDs are\n   formed.\n */\nmodule.exports = function keyid() {\n\n  this.bytes = '';\n\n  for (var i = 0; i < 8; i++) {\n    this.bytes += String.fromCharCode(0);\n  }\n  /**\n   * Parsing method for a key id\n   * @param {String} input Input to read the key id from \n   * @param {integer} position Position where to start reading the key \n   * id from input\n   * @return {openpgp_type_keyid} This object\n   */\n  this.read = function(bytes) {\n    this.bytes = bytes.substr(0, 8);\n  }\n\n  this.write = function() {\n    return this.bytes;\n  }\n\n  this.toHex = function() {\n    return util.hexstrdump(this.bytes);\n  }\n\n  this.equals = function(keyid) {\n    return this.bytes == keyid.bytes;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n// Hint: We hold our MPIs as an array of octets in big endian format preceeding a two\n// octet scalar: MPI: [a,b,c,d,e,f]\n// - MPI size: (a << 8) | b \n// - MPI = c | d << 8 | e << ((MPI.length -2)*8) | f ((MPI.length -2)*8)\n\nvar BigInteger = require('../crypto/public_key/jsbn.js'),\n  util = require('../util');\n\n/**\n * @class\n * @classdescImplementation of type MPI (RFC4880 3.2)\n * Multiprecision integers (also called MPIs) are unsigned integers used\n * to hold large integers such as the ones used in cryptographic\n * calculations.\n * An MPI consists of two pieces: a two-octet scalar that is the length\n * of the MPI in bits followed by a string of octets that contain the\n * actual integer.\n */\nmodule.exports = function mpi() {\n  /** An implementation dependent integer */\n  this.data = null;\n\n  /**\n   * Parsing function for a mpi (RFC 4880 3.2).\n   * @param {String} input Payload of mpi data\n   * @param {Integer} position Position to start reading from the input \n   * string\n   * @param {Integer} len Length of the packet or the remaining length of \n   * input at position\n   * @return {openpgp_type_mpi} Object representation\n   */\n  this.read = function(bytes) {\n    var bits = (bytes[0].charCodeAt() << 8) | bytes[1].charCodeAt();\n\n    // Additional rules:\n    //\n    //    The size of an MPI is ((MPI.length + 7) / 8) + 2 octets.\n    //\n    //    The length field of an MPI describes the length starting from its\n    //\t  most significant non-zero bit.  Thus, the MPI [00 02 01] is not\n    //    formed correctly.  It should be [00 01 01].\n\n    // TODO: Verification of this size method! This size calculation as\n    // \t\t specified above is not applicable in JavaScript\n    var bytelen = Math.ceil(bits / 8);\n\n    var raw = bytes.substr(2, bytelen);\n    this.fromBytes(raw);\n\n    return 2 + bytelen;\n  }\n\n  this.fromBytes = function(bytes) {\n    this.data = new BigInteger(util.hexstrdump(bytes), 16);\n  }\n\n  this.toBytes = function() {\n    return this.write().substr(2);\n  }\n\n  this.byteLength = function() {\n    return this.toBytes().length;\n  }\n\n  /**\n   * Converts the mpi object to a string as specified in RFC4880 3.2\n   * @return {String} mpi Byte representation\n   */\n  this.write = function() {\n    return this.data.toMPI();\n  }\n\n  this.toBigInteger = function() {\n    return this.data.clone();\n  }\n\n  this.fromBigInteger = function(bn) {\n    this.data = bn.clone();\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the String-to-key specifier (RFC4880 3.7)\n * String-to-key (S2K) specifiers are used to convert passphrase strings\n   into symmetric-key encryption/decryption keys.  They are used in two\n   places, currently: to encrypt the secret part of private keys in the\n   private keyring, and to convert passphrases to encryption keys for\n   symmetrically encrypted messages.\n */\nmodule.exports = function s2k() {\n  /** @type {openpgp.hash} */\n  this.algorithm = 'sha256';\n  /** @type {openpgp_type_s2k.type} */\n  this.type = 'iterated';\n  this.c = 96;\n  /** @type {openpgp_bytearray} \n   * Eight bytes of salt. */\n  this.salt = crypto.random.getRandomBytes(8);\n\n\n  // Exponen bias, defined in RFC4880\n  var expbias = 6;\n\n  this.get_count = function() {\n    return (16 + (this.c & 15)) << ((this.c >> 4) + expbias);\n  }\n\n  /**\n   * Parsing function for a string-to-key specifier (RFC 4880 3.7).\n   * @param {String} input Payload of string-to-key specifier\n   * @return {Integer} Actual length of the object\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    this.type = enums.read(enums.s2k, bytes[i++].charCodeAt());\n    this.algorithm = enums.read(enums.hash, bytes[i++].charCodeAt());\n\n    switch (this.type) {\n      case 'simple':\n        break;\n\n      case 'salted':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n        break;\n\n      case 'iterated':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n\n        // Octet 10: count, a one-octet, coded value\n        this.c = bytes[i++].charCodeAt();\n        break;\n\n      case 'gnu':\n        if (bytes.substr(i, 3) == \"GNU\") {\n          i += 3; // GNU\n          var gnuExtType = 1000 + bytes[i++].charCodeAt();\n          if (gnuExtType == 1001) {\n            this.type = gnuExtType;\n            // GnuPG extension mode 1001 -- don't write secret key at all\n          } else {\n            throw new Error(\"Unknown s2k gnu protection mode.\");\n          }\n        } else {\n          throw new Error(\"Unknown s2k type.\");\n        }\n        break;\n\n      default:\n        throw new Error(\"Unknown s2k type.\");\n        break;\n    }\n\n    return i;\n  }\n\n\n  /**\n   * writes an s2k hash based on the inputs.\n   * @return {String} Produced key of hashAlgorithm hash length\n   */\n  this.write = function() {\n    var bytes = String.fromCharCode(enums.write(enums.s2k, this.type));\n    bytes += String.fromCharCode(enums.write(enums.hash, this.algorithm));\n\n    switch (this.type) {\n      case 'simple':\n        break;\n      case 'salted':\n        bytes += this.salt;\n        break;\n      case 'iterated':\n        bytes += this.salt;\n        bytes += String.fromCharCode(this.c);\n        break;\n    };\n\n    return bytes;\n  }\n\n  /**\n   * Produces a key using the specified passphrase and the defined \n   * hashAlgorithm \n   * @param {String} passphrase Passphrase containing user input\n   * @return {String} Produced key with a length corresponding to \n   * hashAlgorithm hash length\n   */\n  this.produce_key = function(passphrase, numBytes) {\n    passphrase = util.encode_utf8(passphrase);\n\n    function round(prefix, s2k) {\n      var algorithm = enums.write(enums.hash, s2k.algorithm);\n\n      switch (s2k.type) {\n        case 'simple':\n          return crypto.hash.digest(algorithm, prefix + passphrase);\n\n        case 'salted':\n          return crypto.hash.digest(algorithm,\n            prefix + s2k.salt + passphrase);\n\n        case 'iterated':\n          var isp = [],\n            count = s2k.get_count();\n          data = s2k.salt + passphrase;\n\n          while (isp.length * data.length < count)\n            isp.push(data);\n\n          isp = isp.join('');\n\n          if (isp.length > count)\n            isp = isp.substr(0, count);\n\n          return crypto.hash.digest(algorithm, prefix + isp);\n      };\n    }\n\n    var result = '',\n      prefix = '';\n\n    while (result.length <= numBytes) {\n      result += round(prefix, this);\n      prefix += String.fromCharCode(0);\n    }\n\n    return result.substr(0, numBytes);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar Util = function() {\n\n  this.readNumber = function(bytes) {\n    var n = 0;\n\n    for (var i = 0; i < bytes.length; i++) {\n      n <<= 8;\n      n += bytes[i].charCodeAt();\n    }\n\n    return n;\n  };\n\n  this.writeNumber = function(n, bytes) {\n    var b = '';\n    for (var i = 0; i < bytes; i++) {\n      b += String.fromCharCode((n >> (8 * (bytes - i - 1))) & 0xFF);\n    }\n\n    return b;\n  };\n\n\n\n  this.readDate = function(bytes) {\n    var n = this.readNumber(bytes);\n    var d = new Date();\n    d.setTime(n * 1000);\n    return d;\n  };\n\n  this.writeDate = function(time) {\n    var numeric = Math.round(time.getTime() / 1000);\n\n    return this.writeNumber(numeric, 4);\n  };\n\n  this.emailRegEx =\n    /[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?/;\n\n  this.debug = false;\n\n  this.hexdump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    var i = 0;\n    while (c < e) {\n      h = str.charCodeAt(c++).toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\" \" + h);\n      i++;\n      if (i % 32 == 0)\n        r.push(\"\\n           \");\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create hexstring from a binary\n   * @param {String} str String to convert\n   * @return {String} String containing the hexadecimal values\n   */\n  this.hexstrdump = function(str) {\n    if (str == null)\n      return \"\";\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].charCodeAt().toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create binary string from a hex encoded string\n   * @param {String} str Hex string to convert\n   * @return {String} String containing the binary values\n   */\n  this.hex2bin = function(hex) {\n    var str = '';\n    for (var i = 0; i < hex.length; i += 2)\n      str += String.fromCharCode(parseInt(hex.substr(i, 2), 16));\n    return str;\n  };\n\n  /**\n   * Creating a hex string from an binary array of integers (0..255)\n   * @param {String} str Array of bytes to convert\n   * @return {String} Hexadecimal representation of the array\n   */\n  this.hexidump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n\n  /**\n   * Convert a native javascript string to a string of utf8 bytes\n   * @param {String} str The string to convert\n   * @return {String} A valid squence of utf8 bytes\n   */\n  this.encode_utf8 = function(str) {\n    return unescape(encodeURIComponent(str));\n  };\n\n  /**\n   * Convert a string of utf8 bytes to a native javascript string\n   * @param {String} utf8 A valid squence of utf8 bytes\n   * @return {String} A native javascript string\n   */\n  this.decode_utf8 = function(utf8) {\n    return decodeURIComponent(escape(utf8));\n  };\n\n  var str2bin = function(str, result) {\n    for (var i = 0; i < str.length; i++) {\n      result[i] = str.charCodeAt(i);\n    }\n\n    return result;\n  };\n\n  var bin2str = function(bin) {\n    var result = [];\n\n    for (var i = 0; i < bin.length; i++) {\n      result.push(String.fromCharCode(bin[i]));\n    }\n\n    return result.join('');\n  };\n\n  /**\n   * Convert a string to an array of integers(0.255)\n   * @param {String} str String to convert\n   * @return {Integer[]} An array of (binary) integers\n   */\n  this.str2bin = function(str) {\n    return str2bin(str, new Array(str.length));\n  };\n\n\n  /**\n   * Convert an array of integers(0.255) to a string \n   * @param {Integer[]} bin An array of (binary) integers to convert\n   * @return {String} The string representation of the array\n   */\n  this.bin2str = bin2str;\n\n  /**\n   * Convert a string to a Uint8Array\n   * @param {String} str String to convert\n   * @return {Uint8Array} The array of (binary) integers\n   */\n  this.str2Uint8Array = function(str) {\n    return str2bin(str, new Uint8Array(new ArrayBuffer(str.length)));\n  };\n\n  /**\n   * Convert a Uint8Array to a string. This currently functions \n   * the same as bin2str. \n   * @param {Uint8Array} bin An array of (binary) integers to convert\n   * @return {String} String representation of the array\n   */\n  this.Uint8Array2str = bin2str;\n\n  /**\n   * Calculates a 16bit sum of a string by adding each character \n   * codes modulus 65535\n   * @param {String} text String to create a sum of\n   * @return {Integer} An integer containing the sum of all character \n   * codes % 65535\n   */\n  this.calc_checksum = function(text) {\n    var checksum = {\n      s: 0,\n      add: function(sadd) {\n        this.s = (this.s + sadd) % 65536;\n      }\n    };\n    for (var i = 0; i < text.length; i++) {\n      checksum.add(text.charCodeAt(i));\n    }\n    return checksum.s;\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug = function(str) {\n    if (this.debug) {\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * Different than print_debug because will call hexstrdump iff necessary.\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug_hexstr_dump = function(str, strToHex) {\n    if (this.debug) {\n      str = str + this.hexstrdump(strToHex);\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print an error message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the error message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded error message\n   */\n  this.print_error = function(str) {\n    if (this.debug)\n      throw str;\n    console.log(str);\n  };\n\n  /**\n   * Helper function to print an info message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'.\n   * @param {String} str String of the info message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded info message\n   */\n  this.print_info = function(str) {\n    if (this.debug)\n      console.log(str);\n  };\n\n  this.print_warning = function(str) {\n    console.log(str);\n  };\n\n  this.getLeftNBits = function(string, bitcount) {\n    var rest = bitcount % 8;\n    if (rest == 0)\n      return string.substring(0, bitcount / 8);\n    var bytes = (bitcount - rest) / 8 + 1;\n    var result = string.substring(0, bytes);\n    return this.shiftRight(result, 8 - rest); // +String.fromCharCode(string.charCodeAt(bytes -1) << (8-rest) & 0xFF);\n  };\n\n  /**\n   * Shifting a string to n bits right\n   * @param {String} value The string to shift\n   * @param {Integer} bitcount Amount of bits to shift (MUST be smaller \n   * than 9)\n   * @return {String} Resulting string. \n   */\n  this.shiftRight = function(value, bitcount) {\n    var temp = util.str2bin(value);\n    if (bitcount % 8 != 0) {\n      for (var i = temp.length - 1; i >= 0; i--) {\n        temp[i] >>= bitcount % 8;\n        if (i > 0)\n          temp[i] |= (temp[i - 1] << (8 - (bitcount % 8))) & 0xFF;\n      }\n    } else {\n      return value;\n    }\n    return util.bin2str(temp);\n  };\n\n  /**\n   * Return the algorithm type as string\n   * @return {String} String representing the message type\n   */\n  this.get_hashAlgorithmString = function(algo) {\n    switch (algo) {\n      case 1:\n        return \"MD5\";\n      case 2:\n        return \"SHA1\";\n      case 3:\n        return \"RIPEMD160\";\n      case 8:\n        return \"SHA256\";\n      case 9:\n        return \"SHA384\";\n      case 10:\n        return \"SHA512\";\n      case 11:\n        return \"SHA224\";\n    }\n    return \"unknown\";\n  };\n\n};\n\n/**\n * an instance that should be used. \n */\nmodule.exports = new Util();\n","var unit = require('../../unit.js');\n\nunit.register(\"AES Rijndael cipher test with test vectors from ecb_tbl.txt\", function() {\n  var openpgp = require('../../../');\n  var util = openpgp.util;\n\n  var result = new Array();\n\n  function test_aes(input, key, output) {\n    var aes = new openpgp.cipher.aes128(util.bin2str(key));\n\n    var result = util.bin2str(aes.encrypt(input));\n\n    return util.hexstrdump(result) == util.hexstrdump(util.bin2str(output));\n  };\n\n  var testvectors128 = [[[0x00,0x01,0x02,0x03,0x05,0x06,0x07,0x08,0x0A,0x0B,0x0C,0x0D,0x0F,0x10,0x11,0x12],[0x50,0x68,0x12,0xA4,0x5F,0x08,0xC8,0x89,0xB9,0x7F,0x59,0x80,0x03,0x8B,0x83,0x59],[0xD8,0xF5,0x32,0x53,0x82,0x89,0xEF,0x7D,0x06,0xB5,0x06,0xA4,0xFD,0x5B,0xE9,0xC9]],\n                        [[0x14,0x15,0x16,0x17,0x19,0x1A,0x1B,0x1C,0x1E,0x1F,0x20,0x21,0x23,0x24,0x25,0x26],[0x5C,0x6D,0x71,0xCA,0x30,0xDE,0x8B,0x8B,0x00,0x54,0x99,0x84,0xD2,0xEC,0x7D,0x4B],[0x59,0xAB,0x30,0xF4,0xD4,0xEE,0x6E,0x4F,0xF9,0x90,0x7E,0xF6,0x5B,0x1F,0xB6,0x8C]],\n                        [[0x28,0x29,0x2A,0x2B,0x2D,0x2E,0x2F,0x30,0x32,0x33,0x34,0x35,0x37,0x38,0x39,0x3A],[0x53,0xF3,0xF4,0xC6,0x4F,0x86,0x16,0xE4,0xE7,0xC5,0x61,0x99,0xF4,0x8F,0x21,0xF6],[0xBF,0x1E,0xD2,0xFC,0xB2,0xAF,0x3F,0xD4,0x14,0x43,0xB5,0x6D,0x85,0x02,0x5C,0xB1]],\n                        [[0x3C,0x3D,0x3E,0x3F,0x41,0x42,0x43,0x44,0x46,0x47,0x48,0x49,0x4B,0x4C,0x4D,0x4E],[0xA1,0xEB,0x65,0xA3,0x48,0x71,0x65,0xFB,0x0F,0x1C,0x27,0xFF,0x99,0x59,0xF7,0x03],[0x73,0x16,0x63,0x2D,0x5C,0x32,0x23,0x3E,0xDC,0xB0,0x78,0x05,0x60,0xEA,0xE8,0xB2]],\n                        [[0x50,0x51,0x52,0x53,0x55,0x56,0x57,0x58,0x5A,0x5B,0x5C,0x5D,0x5F,0x60,0x61,0x62],[0x35,0x53,0xEC,0xF0,0xB1,0x73,0x95,0x58,0xB0,0x8E,0x35,0x0A,0x98,0xA3,0x9B,0xFA],[0x40,0x8C,0x07,0x3E,0x3E,0x25,0x38,0x07,0x2B,0x72,0x62,0x5E,0x68,0xB8,0x36,0x4B]],\n                        [[0x64,0x65,0x66,0x67,0x69,0x6A,0x6B,0x6C,0x6E,0x6F,0x70,0x71,0x73,0x74,0x75,0x76],[0x67,0x42,0x99,0x69,0x49,0x0B,0x97,0x11,0xAE,0x2B,0x01,0xDC,0x49,0x7A,0xFD,0xE8],[0xE1,0xF9,0x4D,0xFA,0x77,0x65,0x97,0xBE,0xAC,0xA2,0x62,0xF2,0xF6,0x36,0x6F,0xEA]],\n                        [[0x78,0x79,0x7A,0x7B,0x7D,0x7E,0x7F,0x80,0x82,0x83,0x84,0x85,0x87,0x88,0x89,0x8A],[0x93,0x38,0x5C,0x1F,0x2A,0xEC,0x8B,0xED,0x19,0x2F,0x5A,0x8E,0x16,0x1D,0xD5,0x08],[0xF2,0x9E,0x98,0x6C,0x6A,0x1C,0x27,0xD7,0xB2,0x9F,0xFD,0x7E,0xE9,0x2B,0x75,0xF1]],\n                        [[0x8C,0x8D,0x8E,0x8F,0x91,0x92,0x93,0x94,0x96,0x97,0x98,0x99,0x9B,0x9C,0x9D,0x9E],[0xB5,0xBF,0x94,0x6B,0xE1,0x9B,0xEB,0x8D,0xB3,0x98,0x3B,0x5F,0x4C,0x6E,0x8D,0xDB],[0x13,0x1C,0x88,0x6A,0x57,0xF8,0xC2,0xE7,0x13,0xAB,0xA6,0x95,0x5E,0x2B,0x55,0xB5]],\n                        [[0xA0,0xA1,0xA2,0xA3,0xA5,0xA6,0xA7,0xA8,0xAA,0xAB,0xAC,0xAD,0xAF,0xB0,0xB1,0xB2],[0x41,0x32,0x1E,0xE1,0x0E,0x21,0xBD,0x90,0x72,0x27,0xC4,0x45,0x0F,0xF4,0x23,0x24],[0xD2,0xAB,0x76,0x62,0xDF,0x9B,0x8C,0x74,0x02,0x10,0xE5,0xEE,0xB6,0x1C,0x19,0x9D]],\n                        [[0xB4,0xB5,0xB6,0xB7,0xB9,0xBA,0xBB,0xBC,0xBE,0xBF,0xC0,0xC1,0xC3,0xC4,0xC5,0xC6],[0x00,0xA8,0x2F,0x59,0xC9,0x1C,0x84,0x86,0xD1,0x2C,0x0A,0x80,0x12,0x4F,0x60,0x89],[0x14,0xC1,0x05,0x54,0xB2,0x85,0x9C,0x48,0x4C,0xAB,0x58,0x69,0xBB,0xE7,0xC4,0x70]],\n                        [[0xC8,0xC9,0xCA,0xCB,0xCD,0xCE,0xCF,0xD0,0xD2,0xD3,0xD4,0xD5,0xD7,0xD8,0xD9,0xDA],[0x7C,0xE0,0xFD,0x07,0x67,0x54,0x69,0x1B,0x4B,0xBD,0x9F,0xAF,0x8A,0x13,0x72,0xFE],[0xDB,0x4D,0x49,0x8F,0x0A,0x49,0xCF,0x55,0x44,0x5D,0x50,0x2C,0x1F,0x9A,0xB3,0xB5]],\n                        [[0xDC,0xDD,0xDE,0xDF,0xE1,0xE2,0xE3,0xE4,0xE6,0xE7,0xE8,0xE9,0xEB,0xEC,0xED,0xEE],[0x23,0x60,0x5A,0x82,0x43,0xD0,0x77,0x64,0x54,0x1B,0xC5,0xAD,0x35,0x5B,0x31,0x29],[0x6D,0x96,0xFE,0xF7,0xD6,0x65,0x90,0xA7,0x7A,0x77,0xBB,0x20,0x56,0x66,0x7F,0x7F]],\n                        [[0xF0,0xF1,0xF2,0xF3,0xF5,0xF6,0xF7,0xF8,0xFA,0xFB,0xFC,0xFD,0xFE,0x01,0x00,0x02],[0x12,0xA8,0xCF,0xA2,0x3E,0xA7,0x64,0xFD,0x87,0x62,0x32,0xB4,0xE8,0x42,0xBC,0x44],[0x31,0x6F,0xB6,0x8E,0xDB,0xA7,0x36,0xC5,0x3E,0x78,0x47,0x7B,0xF9,0x13,0x72,0x5C]],\n                        [[0x04,0x05,0x06,0x07,0x09,0x0A,0x0B,0x0C,0x0E,0x0F,0x10,0x11,0x13,0x14,0x15,0x16],[0xBC,0xAF,0x32,0x41,0x5E,0x83,0x08,0xB3,0x72,0x3E,0x5F,0xDD,0x85,0x3C,0xCC,0x80],[0x69,0x36,0xF2,0xB9,0x3A,0xF8,0x39,0x7F,0xD3,0xA7,0x71,0xFC,0x01,0x1C,0x8C,0x37]],\n                        [[0x2C,0x2D,0x2E,0x2F,0x31,0x32,0x33,0x34,0x36,0x37,0x38,0x39,0x3B,0x3C,0x3D,0x3E],[0x89,0xAF,0xAE,0x68,0x5D,0x80,0x1A,0xD7,0x47,0xAC,0xE9,0x1F,0xC4,0x9A,0xDD,0xE0],[0xF3,0xF9,0x2F,0x7A,0x9C,0x59,0x17,0x9C,0x1F,0xCC,0x2C,0x2B,0xA0,0xB0,0x82,0xCD]]];\n  \n  var testvectors192 = [[[0x00,0x01,0x02,0x03,0x05,0x06,0x07,0x08,0x0A,0x0B,0x0C,0x0D,0x0F,0x10,0x11,0x12,0x14,0x15,0x16,0x17,0x19,0x1A,0x1B,0x1C],[0x2D,0x33,0xEE,0xF2,0xC0,0x43,0x0A,0x8A,0x9E,0xBF,0x45,0xE8,0x09,0xC4,0x0B,0xB6],[0xDF,0xF4,0x94,0x5E,0x03,0x36,0xDF,0x4C,0x1C,0x56,0xBC,0x70,0x0E,0xFF,0x83,0x7F]],\n                        [[0x1E,0x1F,0x20,0x21,0x23,0x24,0x25,0x26,0x28,0x29,0x2A,0x2B,0x2D,0x2E,0x2F,0x30,0x32,0x33,0x34,0x35,0x37,0x38,0x39,0x3A],[0x6A,0xA3,0x75,0xD1,0xFA,0x15,0x5A,0x61,0xFB,0x72,0x35,0x3E,0x0A,0x5A,0x87,0x56],[0xB6,0xFD,0xDE,0xF4,0x75,0x27,0x65,0xE3,0x47,0xD5,0xD2,0xDC,0x19,0x6D,0x12,0x52]],\n                        [[0x3C,0x3D,0x3E,0x3F,0x41,0x42,0x43,0x44,0x46,0x47,0x48,0x49,0x4B,0x4C,0x4D,0x4E,0x50,0x51,0x52,0x53,0x55,0x56,0x57,0x58],[0xBC,0x37,0x36,0x51,0x8B,0x94,0x90,0xDC,0xB8,0xED,0x60,0xEB,0x26,0x75,0x8E,0xD4],[0xD2,0x36,0x84,0xE3,0xD9,0x63,0xB3,0xAF,0xCF,0x1A,0x11,0x4A,0xCA,0x90,0xCB,0xD6]],\n                        [[0x5A,0x5B,0x5C,0x5D,0x5F,0x60,0x61,0x62,0x64,0x65,0x66,0x67,0x69,0x6A,0x6B,0x6C,0x6E,0x6F,0x70,0x71,0x73,0x74,0x75,0x76],[0xAA,0x21,0x44,0x02,0xB4,0x6C,0xFF,0xB9,0xF7,0x61,0xEC,0x11,0x26,0x3A,0x31,0x1E],[0x3A,0x7A,0xC0,0x27,0x75,0x3E,0x2A,0x18,0xC2,0xCE,0xAB,0x9E,0x17,0xC1,0x1F,0xD0]],\n                        [[0x78,0x79,0x7A,0x7B,0x7D,0x7E,0x7F,0x80,0x82,0x83,0x84,0x85,0x87,0x88,0x89,0x8A,0x8C,0x8D,0x8E,0x8F,0x91,0x92,0x93,0x94],[0x02,0xAE,0xA8,0x6E,0x57,0x2E,0xEA,0xB6,0x6B,0x2C,0x3A,0xF5,0xE9,0xA4,0x6F,0xD6],[0x8F,0x67,0x86,0xBD,0x00,0x75,0x28,0xBA,0x26,0x60,0x3C,0x16,0x01,0xCD,0xD0,0xD8]],\n                        [[0x96,0x97,0x98,0x99,0x9B,0x9C,0x9D,0x9E,0xA0,0xA1,0xA2,0xA3,0xA5,0xA6,0xA7,0xA8,0xAA,0xAB,0xAC,0xAD,0xAF,0xB0,0xB1,0xB2],[0xE2,0xAE,0xF6,0xAC,0xC3,0x3B,0x96,0x5C,0x4F,0xA1,0xF9,0x1C,0x75,0xFF,0x6F,0x36],[0xD1,0x7D,0x07,0x3B,0x01,0xE7,0x15,0x02,0xE2,0x8B,0x47,0xAB,0x55,0x11,0x68,0xB3]],\n                        [[0xB4,0xB5,0xB6,0xB7,0xB9,0xBA,0xBB,0xBC,0xBE,0xBF,0xC0,0xC1,0xC3,0xC4,0xC5,0xC6,0xC8,0xC9,0xCA,0xCB,0xCD,0xCE,0xCF,0xD0],[0x06,0x59,0xDF,0x46,0x42,0x71,0x62,0xB9,0x43,0x48,0x65,0xDD,0x94,0x99,0xF9,0x1D],[0xA4,0x69,0xDA,0x51,0x71,0x19,0xFA,0xB9,0x58,0x76,0xF4,0x1D,0x06,0xD4,0x0F,0xFA]],\n                        [[0xD2,0xD3,0xD4,0xD5,0xD7,0xD8,0xD9,0xDA,0xDC,0xDD,0xDE,0xDF,0xE1,0xE2,0xE3,0xE4,0xE6,0xE7,0xE8,0xE9,0xEB,0xEC,0xED,0xEE],[0x49,0xA4,0x42,0x39,0xC7,0x48,0xFE,0xB4,0x56,0xF5,0x9C,0x27,0x6A,0x56,0x58,0xDF],[0x60,0x91,0xAA,0x3B,0x69,0x5C,0x11,0xF5,0xC0,0xB6,0xAD,0x26,0xD3,0xD8,0x62,0xFF]],\n                        [[0xF0,0xF1,0xF2,0xF3,0xF5,0xF6,0xF7,0xF8,0xFA,0xFB,0xFC,0xFD,0xFE,0x01,0x00,0x02,0x04,0x05,0x06,0x07,0x09,0x0A,0x0B,0x0C],[0x66,0x20,0x8F,0x6E,0x9D,0x04,0x52,0x5B,0xDE,0xDB,0x27,0x33,0xB6,0xA6,0xBE,0x37],[0x70,0xF9,0xE6,0x7F,0x9F,0x8D,0xF1,0x29,0x41,0x31,0x66,0x2D,0xC6,0xE6,0x93,0x64]],\n                        [[0x0E,0x0F,0x10,0x11,0x13,0x14,0x15,0x16,0x18,0x19,0x1A,0x1B,0x1D,0x1E,0x1F,0x20,0x22,0x23,0x24,0x25,0x27,0x28,0x29,0x2A],[0x33,0x93,0xF8,0xDF,0xC7,0x29,0xC9,0x7F,0x54,0x80,0xB9,0x50,0xBC,0x96,0x66,0xB0],[0xD1,0x54,0xDC,0xAF,0xAD,0x8B,0x20,0x7F,0xA5,0xCB,0xC9,0x5E,0x99,0x96,0xB5,0x59]],\n                        [[0x2C,0x2D,0x2E,0x2F,0x31,0x32,0x33,0x34,0x36,0x37,0x38,0x39,0x3B,0x3C,0x3D,0x3E,0x40,0x41,0x42,0x43,0x45,0x46,0x47,0x48],[0x60,0x68,0x34,0xC8,0xCE,0x06,0x3F,0x32,0x34,0xCF,0x11,0x45,0x32,0x5D,0xBD,0x71],[0x49,0x34,0xD5,0x41,0xE8,0xB4,0x6F,0xA3,0x39,0xC8,0x05,0xA7,0xAE,0xB9,0xE5,0xDA]],\n                        [[0x4A,0x4B,0x4C,0x4D,0x4F,0x50,0x51,0x52,0x54,0x55,0x56,0x57,0x59,0x5A,0x5B,0x5C,0x5E,0x5F,0x60,0x61,0x63,0x64,0x65,0x66],[0xFE,0xC1,0xC0,0x4F,0x52,0x9B,0xBD,0x17,0xD8,0xCE,0xCF,0xCC,0x47,0x18,0xB1,0x7F],[0x62,0x56,0x4C,0x73,0x8F,0x3E,0xFE,0x18,0x6E,0x1A,0x12,0x7A,0x0C,0x4D,0x3C,0x61]],\n                        [[0x68,0x69,0x6A,0x6B,0x6D,0x6E,0x6F,0x70,0x72,0x73,0x74,0x75,0x77,0x78,0x79,0x7A,0x7C,0x7D,0x7E,0x7F,0x81,0x82,0x83,0x84],[0x32,0xDF,0x99,0xB4,0x31,0xED,0x5D,0xC5,0xAC,0xF8,0xCA,0xF6,0xDC,0x6C,0xE4,0x75],[0x07,0x80,0x5A,0xA0,0x43,0x98,0x6E,0xB2,0x36,0x93,0xE2,0x3B,0xEF,0x8F,0x34,0x38]],\n                        [[0x86,0x87,0x88,0x89,0x8B,0x8C,0x8D,0x8E,0x90,0x91,0x92,0x93,0x95,0x96,0x97,0x98,0x9A,0x9B,0x9C,0x9D,0x9F,0xA0,0xA1,0xA2],[0x7F,0xDC,0x2B,0x74,0x6F,0x3F,0x66,0x52,0x96,0x94,0x3B,0x83,0x71,0x0D,0x1F,0x82],[0xDF,0x0B,0x49,0x31,0x03,0x8B,0xAD,0xE8,0x48,0xDE,0xE3,0xB4,0xB8,0x5A,0xA4,0x4B]],\n                        [[0xA4,0xA5,0xA6,0xA7,0xA9,0xAA,0xAB,0xAC,0xAE,0xAF,0xB0,0xB1,0xB3,0xB4,0xB5,0xB6,0xB8,0xB9,0xBA,0xBB,0xBD,0xBE,0xBF,0xC0],[0x8F,0xBA,0x15,0x10,0xA3,0xC5,0xB8,0x7E,0x2E,0xAA,0x3F,0x7A,0x91,0x45,0x5C,0xA2],[0x59,0x2D,0x5F,0xDE,0xD7,0x65,0x82,0xE4,0x14,0x3C,0x65,0x09,0x93,0x09,0x47,0x7C]]];\n  \n  var testvectors256 = [[[0x00,0x01,0x02,0x03,0x05,0x06,0x07,0x08,0x0A,0x0B,0x0C,0x0D,0x0F,0x10,0x11,0x12,0x14,0x15,0x16,0x17,0x19,0x1A,0x1B,0x1C,0x1E,0x1F,0x20,0x21,0x23,0x24,0x25,0x26],[0x83,0x4E,0xAD,0xFC,0xCA,0xC7,0xE1,0xB3,0x06,0x64,0xB1,0xAB,0xA4,0x48,0x15,0xAB],[0x19,0x46,0xDA,0xBF,0x6A,0x03,0xA2,0xA2,0xC3,0xD0,0xB0,0x50,0x80,0xAE,0xD6,0xFC]],\n              [[0x28,0x29,0x2A,0x2B,0x2D,0x2E,0x2F,0x30,0x32,0x33,0x34,0x35,0x37,0x38,0x39,0x3A,0x3C,0x3D,0x3E,0x3F,0x41,0x42,0x43,0x44,0x46,0x47,0x48,0x49,0x4B,0x4C,0x4D,0x4E],[0xD9,0xDC,0x4D,0xBA,0x30,0x21,0xB0,0x5D,0x67,0xC0,0x51,0x8F,0x72,0xB6,0x2B,0xF1],[0x5E,0xD3,0x01,0xD7,0x47,0xD3,0xCC,0x71,0x54,0x45,0xEB,0xDE,0xC6,0x2F,0x2F,0xB4]],\n              [[0x50,0x51,0x52,0x53,0x55,0x56,0x57,0x58,0x5A,0x5B,0x5C,0x5D,0x5F,0x60,0x61,0x62,0x64,0x65,0x66,0x67,0x69,0x6A,0x6B,0x6C,0x6E,0x6F,0x70,0x71,0x73,0x74,0x75,0x76],[0xA2,0x91,0xD8,0x63,0x01,0xA4,0xA7,0x39,0xF7,0x39,0x21,0x73,0xAA,0x3C,0x60,0x4C],[0x65,0x85,0xC8,0xF4,0x3D,0x13,0xA6,0xBE,0xAB,0x64,0x19,0xFC,0x59,0x35,0xB9,0xD0]],\n              [[0x78,0x79,0x7A,0x7B,0x7D,0x7E,0x7F,0x80,0x82,0x83,0x84,0x85,0x87,0x88,0x89,0x8A,0x8C,0x8D,0x8E,0x8F,0x91,0x92,0x93,0x94,0x96,0x97,0x98,0x99,0x9B,0x9C,0x9D,0x9E],[0x42,0x64,0xB2,0x69,0x64,0x98,0xDE,0x4D,0xF7,0x97,0x88,0xA9,0xF8,0x3E,0x93,0x90],[0x2A,0x5B,0x56,0xA5,0x96,0x68,0x0F,0xCC,0x0E,0x05,0xF5,0xE0,0xF1,0x51,0xEC,0xAE]],\n              [[0xA0,0xA1,0xA2,0xA3,0xA5,0xA6,0xA7,0xA8,0xAA,0xAB,0xAC,0xAD,0xAF,0xB0,0xB1,0xB2,0xB4,0xB5,0xB6,0xB7,0xB9,0xBA,0xBB,0xBC,0xBE,0xBF,0xC0,0xC1,0xC3,0xC4,0xC5,0xC6],[0xEE,0x99,0x32,0xB3,0x72,0x18,0x04,0xD5,0xA8,0x3E,0xF5,0x94,0x92,0x45,0xB6,0xF6],[0xF5,0xD6,0xFF,0x41,0x4F,0xD2,0xC6,0x18,0x14,0x94,0xD2,0x0C,0x37,0xF2,0xB8,0xC4]],\n              [[0xC8,0xC9,0xCA,0xCB,0xCD,0xCE,0xCF,0xD0,0xD2,0xD3,0xD4,0xD5,0xD7,0xD8,0xD9,0xDA,0xDC,0xDD,0xDE,0xDF,0xE1,0xE2,0xE3,0xE4,0xE6,0xE7,0xE8,0xE9,0xEB,0xEC,0xED,0xEE],[0xE6,0x24,0x8F,0x55,0xC5,0xFD,0xCB,0xCA,0x9C,0xBB,0xB0,0x1C,0x88,0xA2,0xEA,0x77],[0x85,0x39,0x9C,0x01,0xF5,0x9F,0xFF,0xB5,0x20,0x4F,0x19,0xF8,0x48,0x2F,0x00,0xB8]],\n              [[0xF0,0xF1,0xF2,0xF3,0xF5,0xF6,0xF7,0xF8,0xFA,0xFB,0xFC,0xFD,0xFE,0x01,0x00,0x02,0x04,0x05,0x06,0x07,0x09,0x0A,0x0B,0x0C,0x0E,0x0F,0x10,0x11,0x13,0x14,0x15,0x16],[0xB8,0x35,0x8E,0x41,0xB9,0xDF,0xF6,0x5F,0xD4,0x61,0xD5,0x5A,0x99,0x26,0x62,0x47],[0x92,0x09,0x7B,0x4C,0x88,0xA0,0x41,0xDD,0xF9,0x81,0x44,0xBC,0x8D,0x22,0xE8,0xE7]],\n              [[0x18,0x19,0x1A,0x1B,0x1D,0x1E,0x1F,0x20,0x22,0x23,0x24,0x25,0x27,0x28,0x29,0x2A,0x2C,0x2D,0x2E,0x2F,0x31,0x32,0x33,0x34,0x36,0x37,0x38,0x39,0x3B,0x3C,0x3D,0x3E],[0xF0,0xE2,0xD7,0x22,0x60,0xAF,0x58,0xE2,0x1E,0x01,0x5A,0xB3,0xA4,0xC0,0xD9,0x06],[0x89,0xBD,0x5B,0x73,0xB3,0x56,0xAB,0x41,0x2A,0xEF,0x9F,0x76,0xCE,0xA2,0xD6,0x5C]],\n              [[0x40,0x41,0x42,0x43,0x45,0x46,0x47,0x48,0x4A,0x4B,0x4C,0x4D,0x4F,0x50,0x51,0x52,0x54,0x55,0x56,0x57,0x59,0x5A,0x5B,0x5C,0x5E,0x5F,0x60,0x61,0x63,0x64,0x65,0x66],[0x47,0x5B,0x8B,0x82,0x3C,0xE8,0x89,0x3D,0xB3,0xC4,0x4A,0x9F,0x2A,0x37,0x9F,0xF7],[0x25,0x36,0x96,0x90,0x93,0xC5,0x5F,0xF9,0x45,0x46,0x92,0xF2,0xFA,0xC2,0xF5,0x30]],\n              [[0x68,0x69,0x6A,0x6B,0x6D,0x6E,0x6F,0x70,0x72,0x73,0x74,0x75,0x77,0x78,0x79,0x7A,0x7C,0x7D,0x7E,0x7F,0x81,0x82,0x83,0x84,0x86,0x87,0x88,0x89,0x8B,0x8C,0x8D,0x8E],[0x68,0x8F,0x52,0x81,0x94,0x58,0x12,0x86,0x2F,0x5F,0x30,0x76,0xCF,0x80,0x41,0x2F],[0x07,0xFC,0x76,0xA8,0x72,0x84,0x3F,0x3F,0x6E,0x00,0x81,0xEE,0x93,0x96,0xD6,0x37]],\n              [[0x90,0x91,0x92,0x93,0x95,0x96,0x97,0x98,0x9A,0x9B,0x9C,0x9D,0x9F,0xA0,0xA1,0xA2,0xA4,0xA5,0xA6,0xA7,0xA9,0xAA,0xAB,0xAC,0xAE,0xAF,0xB0,0xB1,0xB3,0xB4,0xB5,0xB6],[0x08,0xD1,0xD2,0xBC,0x75,0x0A,0xF5,0x53,0x36,0x5D,0x35,0xE7,0x5A,0xFA,0xCE,0xAA],[0xE3,0x8B,0xA8,0xEC,0x2A,0xA7,0x41,0x35,0x8D,0xCC,0x93,0xE8,0xF1,0x41,0xC4,0x91]],\n              [[0xB8,0xB9,0xBA,0xBB,0xBD,0xBE,0xBF,0xC0,0xC2,0xC3,0xC4,0xC5,0xC7,0xC8,0xC9,0xCA,0xCC,0xCD,0xCE,0xCF,0xD1,0xD2,0xD3,0xD4,0xD6,0xD7,0xD8,0xD9,0xDB,0xDC,0xDD,0xDE],[0x87,0x07,0x12,0x1F,0x47,0xCC,0x3E,0xFC,0xEC,0xA5,0xF9,0xA8,0x47,0x49,0x50,0xA1],[0xD0,0x28,0xEE,0x23,0xE4,0xA8,0x90,0x75,0xD0,0xB0,0x3E,0x86,0x8D,0x7D,0x3A,0x42]],\n              [[0xE0,0xE1,0xE2,0xE3,0xE5,0xE6,0xE7,0xE8,0xEA,0xEB,0xEC,0xED,0xEF,0xF0,0xF1,0xF2,0xF4,0xF5,0xF6,0xF7,0xF9,0xFA,0xFB,0xFC,0xFE,0xFE,0x01,0x01,0x03,0x04,0x05,0x06],[0xE5,0x1A,0xA0,0xB1,0x35,0xDB,0xA5,0x66,0x93,0x9C,0x3B,0x63,0x59,0xA9,0x80,0xC5],[0x8C,0xD9,0x42,0x3D,0xFC,0x45,0x9E,0x54,0x71,0x55,0xC5,0xD1,0xD5,0x22,0xE5,0x40]],\n              [[0x08,0x09,0x0A,0x0B,0x0D,0x0E,0x0F,0x10,0x12,0x13,0x14,0x15,0x17,0x18,0x19,0x1A,0x1C,0x1D,0x1E,0x1F,0x21,0x22,0x23,0x24,0x26,0x27,0x28,0x29,0x2B,0x2C,0x2D,0x2E],[0x06,0x9A,0x00,0x7F,0xC7,0x6A,0x45,0x9F,0x98,0xBA,0xF9,0x17,0xFE,0xDF,0x95,0x21],[0x08,0x0E,0x95,0x17,0xEB,0x16,0x77,0x71,0x9A,0xCF,0x72,0x80,0x86,0x04,0x0A,0xE3]],\n              [[0x30,0x31,0x32,0x33,0x35,0x36,0x37,0x38,0x3A,0x3B,0x3C,0x3D,0x3F,0x40,0x41,0x42,0x44,0x45,0x46,0x47,0x49,0x4A,0x4B,0x4C,0x4E,0x4F,0x50,0x51,0x53,0x54,0x55,0x56],[0x72,0x61,0x65,0xC1,0x72,0x3F,0xBC,0xF6,0xC0,0x26,0xD7,0xD0,0x0B,0x09,0x10,0x27],[0x7C,0x17,0x00,0x21,0x1A,0x39,0x91,0xFC,0x0E,0xCD,0xED,0x0A,0xB3,0xE5,0x76,0xB0]]];\n\n  var res = true;\n  var j = 0;\n  for (var i = 0; i < testvectors128.length; i++) {\n    var res2 = test_aes(testvectors128[i][1],testvectors128[i][0],testvectors128[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing 128 bit key vector with block \"+\n        util.hexidump(testvectors128[i][1])+\n        \" and key \"+util.hexidump(testvectors128[i][0])+\n        \" should be \"+util.hexidump(testvectors128[i][2]),\n        false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"128 bit key test vectors completed.\",true)\n    j++;\n  }\n  \n  res = true;\n  for (var i = 0; i < testvectors192.length; i++) {\n    var res2 = test_aes(testvectors192[i][1],testvectors192[i][0],testvectors192[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing 192 bit key vector with block \"+\n        util.hexidump(testvectors192[i][1])+\n        \" and key \"+util.hexidump(testvectors192[i][0])+\n        \" should be \"+util.hexidump(testvectors192[i][2]),\n        false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"192 bit key test vectors completed.\",true)\n    j++;\n  }\n\n  res = true;\n  for (var i = 0; i < testvectors256.length; i++) {\n    var res2 = test_aes(testvectors256[i][1],testvectors256[i][0],testvectors256[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing 256 bit key vector with block \"+\n        util.hexidump(testvectors256[i][1])+\n        \" and key \"+util.hexidump(testvectors256[i][0])+\n        \" should be \"+util.hexidump(testvectors256[i][2]),\n        false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"256 bit key test vectors completed.\", true)\n    j++;\n  }\n\n  return result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"Blowfish cipher test with test vectors from http://www.schneier.com/code/vectors.txt\", function() {\n  var openpgp = require('../../../'),\n    util = openpgp.util,\n    BFencrypt = openpgp.cipher.blowfish;\n\n  var result = [];\n  function test_bf(input, key, output) {\n    var blowfish = new openpgp.cipher.blowfish(util.bin2str(key));\n    var result = util.bin2str(blowfish.encrypt(input));\n\n    return (util.hexstrdump(result) == util.hexstrdump(util.bin2str(output)));\n  }\n  var testvectors = [[[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4E,0xF9,0x97,0x45,0x61,0x98,0xDD,0x78]],\n                     [[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x51,0x86,0x6F,0xD5,0xB8,0x5E,0xCB,0x8A]],\n                     [[0x30,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x01],[0x7D,0x85,0x6F,0x9A,0x61,0x30,0x63,0xF2]],\n                     [[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x24,0x66,0xDD,0x87,0x8B,0x96,0x3C,0x9D]],\n                     [[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x61,0xF9,0xC3,0x80,0x22,0x81,0xB0,0x96]],\n                     [[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x7D,0x0C,0xC6,0x30,0xAF,0xDA,0x1E,0xC7]],\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4E,0xF9,0x97,0x45,0x61,0x98,0xDD,0x78]],\n                     [[0xFE,0xDC,0xBA,0x98,0x76,0x54,0x32,0x10],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x0A,0xCE,0xAB,0x0F,0xC6,0xA0,0xA2,0x8D]],\n                     [[0x7C,0xA1,0x10,0x45,0x4A,0x1A,0x6E,0x57],[0x01,0xA1,0xD6,0xD0,0x39,0x77,0x67,0x42],[0x59,0xC6,0x82,0x45,0xEB,0x05,0x28,0x2B]],\n                     [[0x01,0x31,0xD9,0x61,0x9D,0xC1,0x37,0x6E],[0x5C,0xD5,0x4C,0xA8,0x3D,0xEF,0x57,0xDA],[0xB1,0xB8,0xCC,0x0B,0x25,0x0F,0x09,0xA0]],\n                     [[0x07,0xA1,0x13,0x3E,0x4A,0x0B,0x26,0x86],[0x02,0x48,0xD4,0x38,0x06,0xF6,0x71,0x72],[0x17,0x30,0xE5,0x77,0x8B,0xEA,0x1D,0xA4]],\n                     [[0x38,0x49,0x67,0x4C,0x26,0x02,0x31,0x9E],[0x51,0x45,0x4B,0x58,0x2D,0xDF,0x44,0x0A],[0xA2,0x5E,0x78,0x56,0xCF,0x26,0x51,0xEB]],\n                     [[0x04,0xB9,0x15,0xBA,0x43,0xFE,0xB5,0xB6],[0x42,0xFD,0x44,0x30,0x59,0x57,0x7F,0xA2],[0x35,0x38,0x82,0xB1,0x09,0xCE,0x8F,0x1A]],\n                     [[0x01,0x13,0xB9,0x70,0xFD,0x34,0xF2,0xCE],[0x05,0x9B,0x5E,0x08,0x51,0xCF,0x14,0x3A],[0x48,0xF4,0xD0,0x88,0x4C,0x37,0x99,0x18]],\n                     [[0x01,0x70,0xF1,0x75,0x46,0x8F,0xB5,0xE6],[0x07,0x56,0xD8,0xE0,0x77,0x47,0x61,0xD2],[0x43,0x21,0x93,0xB7,0x89,0x51,0xFC,0x98]],\n                     [[0x43,0x29,0x7F,0xAD,0x38,0xE3,0x73,0xFE],[0x76,0x25,0x14,0xB8,0x29,0xBF,0x48,0x6A],[0x13,0xF0,0x41,0x54,0xD6,0x9D,0x1A,0xE5]],\n                     [[0x07,0xA7,0x13,0x70,0x45,0xDA,0x2A,0x16],[0x3B,0xDD,0x11,0x90,0x49,0x37,0x28,0x02],[0x2E,0xED,0xDA,0x93,0xFF,0xD3,0x9C,0x79]],\n                     [[0x04,0x68,0x91,0x04,0xC2,0xFD,0x3B,0x2F],[0x26,0x95,0x5F,0x68,0x35,0xAF,0x60,0x9A],[0xD8,0x87,0xE0,0x39,0x3C,0x2D,0xA6,0xE3]],\n                     [[0x37,0xD0,0x6B,0xB5,0x16,0xCB,0x75,0x46],[0x16,0x4D,0x5E,0x40,0x4F,0x27,0x52,0x32],[0x5F,0x99,0xD0,0x4F,0x5B,0x16,0x39,0x69]],\n                     [[0x1F,0x08,0x26,0x0D,0x1A,0xC2,0x46,0x5E],[0x6B,0x05,0x6E,0x18,0x75,0x9F,0x5C,0xCA],[0x4A,0x05,0x7A,0x3B,0x24,0xD3,0x97,0x7B]],\n                     [[0x58,0x40,0x23,0x64,0x1A,0xBA,0x61,0x76],[0x00,0x4B,0xD6,0xEF,0x09,0x17,0x60,0x62],[0x45,0x20,0x31,0xC1,0xE4,0xFA,0xDA,0x8E]],\n                     [[0x02,0x58,0x16,0x16,0x46,0x29,0xB0,0x07],[0x48,0x0D,0x39,0x00,0x6E,0xE7,0x62,0xF2],[0x75,0x55,0xAE,0x39,0xF5,0x9B,0x87,0xBD]],\n                     [[0x49,0x79,0x3E,0xBC,0x79,0xB3,0x25,0x8F],[0x43,0x75,0x40,0xC8,0x69,0x8F,0x3C,0xFA],[0x53,0xC5,0x5F,0x9C,0xB4,0x9F,0xC0,0x19]],\n                     [[0x4F,0xB0,0x5E,0x15,0x15,0xAB,0x73,0xA7],[0x07,0x2D,0x43,0xA0,0x77,0x07,0x52,0x92],[0x7A,0x8E,0x7B,0xFA,0x93,0x7E,0x89,0xA3]],\n                     [[0x49,0xE9,0x5D,0x6D,0x4C,0xA2,0x29,0xBF],[0x02,0xFE,0x55,0x77,0x81,0x17,0xF1,0x2A],[0xCF,0x9C,0x5D,0x7A,0x49,0x86,0xAD,0xB5]],\n                     [[0x01,0x83,0x10,0xDC,0x40,0x9B,0x26,0xD6],[0x1D,0x9D,0x5C,0x50,0x18,0xF7,0x28,0xC2],[0xD1,0xAB,0xB2,0x90,0x65,0x8B,0xC7,0x78]],\n                     [[0x1C,0x58,0x7F,0x1C,0x13,0x92,0x4F,0xEF],[0x30,0x55,0x32,0x28,0x6D,0x6F,0x29,0x5A],[0x55,0xCB,0x37,0x74,0xD1,0x3E,0xF2,0x01]],\n                     [[0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0xFA,0x34,0xEC,0x48,0x47,0xB2,0x68,0xB2]],\n                     [[0x1F,0x1F,0x1F,0x1F,0x0E,0x0E,0x0E,0x0E],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0xA7,0x90,0x79,0x51,0x08,0xEA,0x3C,0xAE]],\n                     [[0xE0,0xFE,0xE0,0xFE,0xF1,0xFE,0xF1,0xFE],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0xC3,0x9E,0x07,0x2D,0x9F,0xAC,0x63,0x1D]],\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x01,0x49,0x33,0xE0,0xCD,0xAF,0xF6,0xE4]],\n                     [[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xF2,0x1E,0x9A,0x77,0xB7,0x1C,0x49,0xBC]],\n                     [[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x24,0x59,0x46,0x88,0x57,0x54,0x36,0x9A]],\n                     [[0xFE,0xDC,0xBA,0x98,0x76,0x54,0x32,0x10],[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x6B,0x5C,0x5A,0x9C,0x5D,0x9E,0x0A,0x5A]]];\n\n  var res = true;\n  var j = 0;\n  for (var i = 0; i < testvectors.length; i++) {\n    var res2 = test_bf(testvectors[i][1],testvectors[i][0],testvectors[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing vector \"+i+\" with block \"+\n        util.hexidump(testvectors[i][0])+\n        \" and key \"+util.hexidump(testvectors[i][1])+\n        \" should be \"+util.hexidump(testvectors[i][2]), false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"34 test vectors completed \", true);\n  }\n  return result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"CAST-128 cipher test with test vectors from RFC2144\", function() {\n\tvar openpgp = require('../../../'),\n\t\tutil = openpgp.util;\n\n\tvar result = [];\n\tfunction test_cast(input, key, output) {\n\t\tvar cast5 = new openpgp.cipher.cast5(util.bin2str(key));\n\t\tvar result = util.bin2str(cast5.encrypt(input));\n\n\t\treturn util.hexstrdump(result) == util.hexstrdump(util.bin2str(output));\n\t}\n\t\n\tvar testvectors = [[[0x01,0x23,0x45,0x67,0x12,0x34,0x56,0x78,0x23,0x45,0x67,0x89,0x34,0x56,0x78,0x9A],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x23,0x8B,0x4F,0xE5,0x84,0x7E,0x44,0xB2]]];\n\n\tfor (var i = 0; i < testvectors.length; i++) {\n\t\tresult[i] = new unit.result(\"Testing vector with block \"+\n\t\t\t\tutil.hexidump(testvectors[i][0])+\n\t\t\t\t\" and key \"+util.hexidump(testvectors[i][1])+\n\t\t\t\t\" should be \"+util.hexidump(testvectors[i][2]),\n\t\t\ttest_cast(testvectors[i][1],testvectors[i][0],testvectors[i][2]));\n\t}\n\treturn result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"TripleDES (EDE) cipher test with test vectors from http://csrc.nist.gov/publications/nistpubs/800-20/800-20.pdf\", function() {\n  var openpgp = require('../../../'),\n    util = openpgp.util;\n\n  var result = [];\n  var key = util.bin2str([1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1]);\n  var testvectors = [[[0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x95,0xF8,0xA5,0xE5,0xDD,0x31,0xD9,0x00]],\n                     [[0x40,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xDD,0x7F,0x12,0x1C,0xA5,0x01,0x56,0x19]],\n                     [[0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x2E,0x86,0x53,0x10,0x4F,0x38,0x34,0xEA]],\n                     [[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4B,0xD3,0x88,0xFF,0x6C,0xD8,0x1D,0x4F]],\n                     [[0x08,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x20,0xB9,0xE7,0x67,0xB2,0xFB,0x14,0x56]],\n                     [[0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x55,0x57,0x93,0x80,0xD7,0x71,0x38,0xEF]],\n                     [[0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x6C,0xC5,0xDE,0xFA,0xAF,0x04,0x51,0x2F]],\n                     [[0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x0D,0x9F,0x27,0x9B,0xA5,0xD8,0x72,0x60]],\n                     [[0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x00],[0xD9,0x03,0x1B,0x02,0x71,0xBD,0x5A,0x0A]],\n                     [[0x00,0x40,0x00,0x00,0x00,0x00,0x00,0x00],[0x42,0x42,0x50,0xB3,0x7C,0x3D,0xD9,0x51]],\n                     [[0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00],[0xB8,0x06,0x1B,0x7E,0xCD,0x9A,0x21,0xE5]],\n                     [[0x00,0x10,0x00,0x00,0x00,0x00,0x00,0x00],[0xF1,0x5D,0x0F,0x28,0x6B,0x65,0xBD,0x28]],\n                     [[0x00,0x08,0x00,0x00,0x00,0x00,0x00,0x00],[0xAD,0xD0,0xCC,0x8D,0x6E,0x5D,0xEB,0xA1]],\n                     [[0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00],[0xE6,0xD5,0xF8,0x27,0x52,0xAD,0x63,0xD1]],\n                     [[0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00],[0xEC,0xBF,0xE3,0xBD,0x3F,0x59,0x1A,0x5E]],\n                     [[0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x00],[0xF3,0x56,0x83,0x43,0x79,0xD1,0x65,0xCD]],\n                     [[0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00],[0x2B,0x9F,0x98,0x2F,0x20,0x03,0x7F,0xA9]],\n                     [[0x00,0x00,0x40,0x00,0x00,0x00,0x00,0x00],[0x88,0x9D,0xE0,0x68,0xA1,0x6F,0x0B,0xE6]],\n                     [[0x00,0x00,0x20,0x00,0x00,0x00,0x00,0x00],[0xE1,0x9E,0x27,0x5D,0x84,0x6A,0x12,0x98]],\n                     [[0x00,0x00,0x10,0x00,0x00,0x00,0x00,0x00],[0x32,0x9A,0x8E,0xD5,0x23,0xD7,0x1A,0xEC]],\n                     [[0x00,0x00,0x08,0x00,0x00,0x00,0x00,0x00],[0xE7,0xFC,0xE2,0x25,0x57,0xD2,0x3C,0x97]],\n                     [[0x00,0x00,0x04,0x00,0x00,0x00,0x00,0x00],[0x12,0xA9,0xF5,0x81,0x7F,0xF2,0xD6,0x5D]],\n                     [[0x00,0x00,0x02,0x00,0x00,0x00,0x00,0x00],[0xA4,0x84,0xC3,0xAD,0x38,0xDC,0x9C,0x19]],\r\n                     [[0x00,0x00,0x01,0x00,0x00,0x00,0x00,0x00],[0xFB,0xE0,0x0A,0x8A,0x1E,0xF8,0xAD,0x72]],\r\n                     [[0x00,0x00,0x00,0x80,0x00,0x00,0x00,0x00],[0x75,0x0D,0x07,0x94,0x07,0x52,0x13,0x63]],\r\n                     [[0x00,0x00,0x00,0x40,0x00,0x00,0x00,0x00],[0x64,0xFE,0xED,0x9C,0x72,0x4C,0x2F,0xAF]],\r\n                     [[0x00,0x00,0x00,0x20,0x00,0x00,0x00,0x00],[0xF0,0x2B,0x26,0x3B,0x32,0x8E,0x2B,0x60]],\r\n                     [[0x00,0x00,0x00,0x10,0x00,0x00,0x00,0x00],[0x9D,0x64,0x55,0x5A,0x9A,0x10,0xB8,0x52]],\r\n                     [[0x00,0x00,0x00,0x08,0x00,0x00,0x00,0x00],[0xD1,0x06,0xFF,0x0B,0xED,0x52,0x55,0xD7]],\r\n                     [[0x00,0x00,0x00,0x04,0x00,0x00,0x00,0x00],[0xE1,0x65,0x2C,0x6B,0x13,0x8C,0x64,0xA5]],\r\n                     [[0x00,0x00,0x00,0x02,0x00,0x00,0x00,0x00],[0xE4,0x28,0x58,0x11,0x86,0xEC,0x8F,0x46]],\r\n                     [[0x00,0x00,0x00,0x01,0x00,0x00,0x00,0x00],[0xAE,0xB5,0xF5,0xED,0xE2,0x2D,0x1A,0x36]],\r\n                     [[0x00,0x00,0x00,0x00,0x80,0x00,0x00,0x00],[0xE9,0x43,0xD7,0x56,0x8A,0xEC,0x0C,0x5C]],\r\n                     [[0x00,0x00,0x00,0x00,0x40,0x00,0x00,0x00],[0xDF,0x98,0xC8,0x27,0x6F,0x54,0xB0,0x4B]],\r\n                     [[0x00,0x00,0x00,0x00,0x20,0x00,0x00,0x00],[0xB1,0x60,0xE4,0x68,0x0F,0x6C,0x69,0x6F]],\r\n                     [[0x00,0x00,0x00,0x00,0x10,0x00,0x00,0x00],[0xFA,0x07,0x52,0xB0,0x7D,0x9C,0x4A,0xB8]],\r\n                     [[0x00,0x00,0x00,0x00,0x08,0x00,0x00,0x00],[0xCA,0x3A,0x2B,0x03,0x6D,0xBC,0x85,0x02]],\r\n                     [[0x00,0x00,0x00,0x00,0x04,0x00,0x00,0x00],[0x5E,0x09,0x05,0x51,0x7B,0xB5,0x9B,0xCF]],\r\n                     [[0x00,0x00,0x00,0x00,0x02,0x00,0x00,0x00],[0x81,0x4E,0xEB,0x3B,0x91,0xD9,0x07,0x26]],\r\n                     [[0x00,0x00,0x00,0x00,0x01,0x00,0x00,0x00],[0x4D,0x49,0xDB,0x15,0x32,0x91,0x9C,0x9F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x80,0x00,0x00],[0x25,0xEB,0x5F,0xC3,0xF8,0xCF,0x06,0x21]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x40,0x00,0x00],[0xAB,0x6A,0x20,0xC0,0x62,0x0D,0x1C,0x6F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x20,0x00,0x00],[0x79,0xE9,0x0D,0xBC,0x98,0xF9,0x2C,0xCA]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x10,0x00,0x00],[0x86,0x6E,0xCE,0xDD,0x80,0x72,0xBB,0x0E]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x08,0x00,0x00],[0x8B,0x54,0x53,0x6F,0x2F,0x3E,0x64,0xA8]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x04,0x00,0x00],[0xEA,0x51,0xD3,0x97,0x55,0x95,0xB8,0x6B]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x02,0x00,0x00],[0xCA,0xFF,0xC6,0xAC,0x45,0x42,0xDE,0x31]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x01,0x00,0x00],[0x8D,0xD4,0x5A,0x2D,0xDF,0x90,0x79,0x6C]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x80,0x00],[0x10,0x29,0xD5,0x5E,0x88,0x0E,0xC2,0xD0]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x40,0x00],[0x5D,0x86,0xCB,0x23,0x63,0x9D,0xBE,0xA9]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x00],[0x1D,0x1C,0xA8,0x53,0xAE,0x7C,0x0C,0x5F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x10,0x00],[0xCE,0x33,0x23,0x29,0x24,0x8F,0x32,0x28]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x08,0x00],[0x84,0x05,0xD1,0xAB,0xE2,0x4F,0xB9,0x42]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x04,0x00],[0xE6,0x43,0xD7,0x80,0x90,0xCA,0x42,0x07]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x02,0x00],[0x48,0x22,0x1B,0x99,0x37,0x74,0x8A,0x23]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x00],[0xDD,0x7C,0x0B,0xBD,0x61,0xFA,0xFD,0x54]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80],[0x2F,0xBC,0x29,0x1A,0x57,0x0D,0xB5,0xC4]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x40],[0xE0,0x7C,0x30,0xD7,0xE4,0xE2,0x6E,0x12]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x20],[0x09,0x53,0xE2,0x25,0x8E,0x8E,0x90,0xA1]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x10],[0x5B,0x71,0x1B,0xC4,0xCE,0xEB,0xF2,0xEE]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x08],[0xCC,0x08,0x3F,0x1E,0x6D,0x9E,0x85,0xF6]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x04],[0xD2,0xFD,0x88,0x67,0xD5,0x0D,0x2D,0xFE]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x02],[0x06,0xE7,0xEA,0x22,0xCE,0x92,0x70,0x8F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01],[0x16,0x6B,0x40,0xB4,0x4A,0xBA,0x4B,0xD6]]];\r\n\n  var res = true;\n  var j = 0;\n  for (var i = 0; i < testvectors.length; i++) {\n    var des = new openpgp.cipher.des(key);\n\n    var encr = util.bin2str(des.encrypt(testvectors[i][0], key));\n    var res2 = encr == util.bin2str(testvectors[i][1]);\n\n    res &= res2;\n\n    if (!res2) {\n    result[j] = new unit.result(\"Testing vector with block \" +\n        util.hexidump(testvectors[i][0]) +\n        \" and key \" + util.hexstrdump(key) +\n        \" should be \" + util.hexidump(testvectors[i][1]) + \" != \" +\n        util.hexidump(encr),\n      false);\n    j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"All 3DES EDE test vectors completed\", true);\n  }\n  return result;\n});\n\n\nunit.register(\"DES encrypt/decrypt padding tests\", function () {\r\n    var openpgp = require('../../../'),\n      util = openpgp.util;\n\n    var result = [];\r\n    var key = util.bin2str([0x01, 0x23, 0x45, 0x67, 0x89, 0xAB, 0xCD, 0xEF]);\r\n    var testvectors = new Array();\r\n    testvectors[0] = [[[0x01], [0x24, 0xC7, 0x4A, 0x9A, 0x79, 0x75, 0x4B, 0xC7]],\r\n\t                  [[0x02, 0x03], [0xA7, 0x7A, 0x9A, 0x59, 0x8A, 0x86, 0x85, 0xC5]],\r\n\t                  [[0x03, 0x04, 0x05], [0x01, 0xCF, 0xEB, 0x6A, 0x74, 0x60, 0xF5, 0x02]],\r\n\t                  [[0x04, 0x05, 0x06, 0x07], [0xA8, 0xF0, 0x3D, 0x59, 0xBA, 0x6B, 0x0E, 0x76]],\r\n\t                  [[0x05, 0x06, 0x07, 0x08, 0x09], [0x86, 0x40, 0x33, 0x61, 0x3F, 0x55, 0x73, 0x49]],\r\n\t                  [[0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B], [0x13, 0x21, 0x3E, 0x0E, 0xCE, 0x2C, 0x94, 0x01]],\r\n\t                  [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0x30, 0x49, 0x97, 0xC1, 0xDA, 0xD5, 0x59, 0xA5]],\r\n\t                  [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9]]];\r\n    testvectors[1] = [[[0x01], [0xF2, 0xAB, 0x1C, 0x9E, 0x70, 0x7D, 0xCC, 0x92]],\r\n\t                  [[0x02, 0x03], [0x6B, 0x4C, 0x67, 0x24, 0x9F, 0xB7, 0x4D, 0xAC]],\r\n\t                  [[0x03, 0x04, 0x05], [0x68, 0x95, 0xAB, 0xA8, 0xEA, 0x53, 0x13, 0x23]],\r\n\t                  [[0x04, 0x05, 0x06, 0x07], [0xC8, 0xDE, 0x60, 0x8F, 0xF6, 0x09, 0x90, 0xB5]],\r\n\t                  [[0x05, 0x06, 0x07, 0x08, 0x09], [0x19, 0x13, 0x50, 0x20, 0x70, 0x40, 0x2E, 0x09]],\r\n\t                  [[0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B], [0xA8, 0x23, 0x40, 0xC6, 0x17, 0xA6, 0x31, 0x4A]],\r\n\t                  [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0x36, 0x62, 0xF2, 0x99, 0x68, 0xD4, 0xBF, 0x7C]],\r\n\t                  [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9, 0x08, 0x6F, 0x9A, 0x1D, 0x74, 0xC9, 0x4D, 0x4E]]];\r\n    testvectors[2] = [[[0x01], [0x83, 0x68, 0xE4, 0x9C, 0x84, 0xCC, 0xCB, 0xF0]],\r\n\t                  [[0x02, 0x03], [0xBB, 0xA8, 0x0B, 0x66, 0x1B, 0x62, 0xC4, 0xC8]],\r\n\t                  [[0x03, 0x04, 0x05], [0x9A, 0xD7, 0x5A, 0x24, 0xFD, 0x3F, 0xBF, 0x22]],\r\n\t                  [[0x04, 0x05, 0x06, 0x07], [0x14, 0x4E, 0x68, 0x6D, 0x2E, 0xC1, 0xB7, 0x52]],\r\n\t                  [[0x05, 0x06, 0x07, 0x08, 0x09], [0x12, 0x0A, 0x51, 0x08, 0xF9, 0xA3, 0x03, 0x74]],\r\n\t                  [[0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B], [0xB2, 0x07, 0xD1, 0x05, 0xF6, 0x67, 0xAF, 0xBA]],\r\n\t                  [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0xCA, 0x59, 0x61, 0x3A, 0x83, 0x23, 0x26, 0xDD]],\r\n\t                  [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9]]];\r\n\r\n    var des = new openpgp.cipher.originalDes(key);\r\n\r\n    var res = true;\r\n    var j = 0;\r\n\r\n    for (var padding = 0; padding < 3; padding++) {\r\n        var thisVectorSet = testvectors[padding];\r\n\r\n        for (var i = 0; i < thisVectorSet.length; i++) {\r\n            var encrypted = des.encrypt(thisVectorSet[i][0], padding);\r\n            var decrypted = des.decrypt(encrypted, padding);\r\n\r\n            var res2 = (util.bin2str(encrypted) == util.bin2str(thisVectorSet[i][1]));\r\n            var res3 = (util.bin2str(decrypted) == util.bin2str(thisVectorSet[i][0]));\r\n            res &= res2;\r\n            res &= res3;\r\n            if (!res2 || !res3) {\r\n                result[j] = new unit.result(\r\n                    \"Testing vector with block [\" +\r\n                        util.hexidump(thisVectorSet[i][0]) +\r\n                        \"] and key [\" + util.hexstrdump(key) +\r\n                        \"] and padding [\" + padding +\r\n                        \"] should be \" + util.hexidump(thisVectorSet[i][1]) + \" - Actually [ENC:\" + util.hexidump(encrypted) + \", DEC:\" + util.hexidump(decrypted) + \"]\",\r\n                    false);\r\n                j++;\r\n            }\r\n        }\r\n    }\r\n    if (res) {\r\n        result[j] = new unit.result(\"All DES test vectors completed\", true);\r\n    }\r\n    return result;\r\n});\r\n","var unit = require('../../unit.js');\n\nunit.register(\"Twofish test with test vectors from http://www.schneier.com/code/ecb_ival.txt\", function() {\n  var openpgp = require('../../../'),\n    util = openpgp.util;\n\n  function TFencrypt(block, key) {\n    var tf = new openpgp.cipher.twofish(key);\n\n    return tf.encrypt(block);\n  }\n\n\n  var result = [];\n  var start = [0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];\n  var start_short = [0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];\n  var testvectors = [[0x57,0xFF,0x73,0x9D,0x4D,0xC9,0x2C,0x1B,0xD7,0xFC,0x01,0x70,0x0C,0xC8,0x21,0x6F],\n            [0xD4,0x3B,0xB7,0x55,0x6E,0xA3,0x2E,0x46,0xF2,0xA2,0x82,0xB7,0xD4,0x5B,0x4E,0x0D],\n            [0x90,0xAF,0xE9,0x1B,0xB2,0x88,0x54,0x4F,0x2C,0x32,0xDC,0x23,0x9B,0x26,0x35,0xE6],\n            [0x6C,0xB4,0x56,0x1C,0x40,0xBF,0x0A,0x97,0x05,0x93,0x1C,0xB6,0xD4,0x08,0xE7,0xFA],\n            [0x30,0x59,0xD6,0xD6,0x17,0x53,0xB9,0x58,0xD9,0x2F,0x47,0x81,0xC8,0x64,0x0E,0x58],\n            [0xE6,0x94,0x65,0x77,0x05,0x05,0xD7,0xF8,0x0E,0xF6,0x8C,0xA3,0x8A,0xB3,0xA3,0xD6],\n            [0x5A,0xB6,0x7A,0x5F,0x85,0x39,0xA4,0xA5,0xFD,0x9F,0x03,0x73,0xBA,0x46,0x34,0x66],\n            [0xDC,0x09,0x6B,0xCD,0x99,0xFC,0x72,0xF7,0x99,0x36,0xD4,0xC7,0x48,0xE7,0x5A,0xF7],\n            [0xC5,0xA3,0xE7,0xCE,0xE0,0xF1,0xB7,0x26,0x05,0x28,0xA6,0x8F,0xB4,0xEA,0x05,0xF2],\n            [0x43,0xD5,0xCE,0xC3,0x27,0xB2,0x4A,0xB9,0x0A,0xD3,0x4A,0x79,0xD0,0x46,0x91,0x51]];\n  testvectors[47] =  [0x43,0x10,0x58,0xF4,0xDB,0xC7,0xF7,0x34,0xDA,0x4F,0x02,0xF0,0x4C,0xC4,0xF4,0x59];\n  testvectors[48] =  [0x37,0xFE,0x26,0xFF,0x1C,0xF6,0x61,0x75,0xF5,0xDD,0xF4,0xC3,0x3B,0x97,0xA2,0x05];\n  var res = true;\n  var j = 0;\n  for (var i = 0; i < 49; i++) {\n    var res2 = false;\n    var blk, key, ct;\n    if (i === 0) {\n      blk = start_short;\n      key = util.bin2str(start);\n      ct = testvectors[0];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else if (i === 1) {\n      blk = testvectors[0];\n      key = util.bin2str(start);\n      ct = testvectors[1];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else if (i === 2) {\n      blk = testvectors[i-1];\n      key = util.bin2str(testvectors[i-2].concat(start_short));\n      ct = testvectors[i];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else if (i < 10 || i > 46) {\n      blk = testvectors[i-1];\n      key = util.bin2str(testvectors[i-2].concat(testvectors[i-3]));\n      ct = testvectors[i];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else {\n      testvectors[i] = TFencrypt(testvectors[i-1],util.bin2str(testvectors[i-2].concat(testvectors[i-3])));\n      res2 = true;\n    }\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing vector with block \"+util.hexidump(blk)+\" with key \"+ util.hexstrdump(key) +\" should be \"+util.hexidump(ct)+\" but is \"+util.hexidump(TFencrypt(blk,key)), false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"49 test vectors completed\", true);\n  }\n  return result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"MD5 test with test vectors from RFC 1321\", function() {\n\tvar openpgp = require('../../../'),\n\t\tutil = openpgp.util,\n\t\tMD5 = openpgp.hash.md5;\n\n\tvar result = new Array();\n\tresult[0] = new unit.result(\"MD5 (\\\"\\\") = d41d8cd98f00b204e9800998ecf8427e\",\n\t\t\tutil.hexstrdump(MD5(\"\")) == \"d41d8cd98f00b204e9800998ecf8427e\");\n\tresult[1] = new unit.result(\"MD5 (\\\"a\\\") = 0cc175b9c0f1b6a831c399e269772661\",\n\t\t\tutil.hexstrdump(MD5 (\"abc\")) == \"900150983cd24fb0d6963f7d28e17f72\");\n\tresult[2] = new unit.result(\"MD5 (\\\"message digest\\\") = f96b697d7cb7938d525a2f31aaf161d0\",\n\t\t\tutil.hexstrdump(MD5 (\"message digest\")) == \"f96b697d7cb7938d525a2f31aaf161d0\");\n\tresult[3] = new unit.result(\"MD5 (\\\"abcdefghijklmnopqrstuvwxyz\\\") = c3fcd3d76192e4007dfb496cca67e13b\",\n\t\t\tutil.hexstrdump(MD5 (\"abcdefghijklmnopqrstuvwxyz\")) == \"c3fcd3d76192e4007dfb496cca67e13b\");\n\tresult[4] = new unit.result(\"MD5 (\\\"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\\\") = d174ab98d277d9f5a5611c2c9f419d9f\",\n\t\t\tutil.hexstrdump(MD5 (\"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\")) == \"d174ab98d277d9f5a5611c2c9f419d9f\");\n\tresult[5] = new unit.result(\"MD5 (\\\"12345678901234567890123456789012345678901234567890123456789012345678901234567890\\\") = 57edf4a22be3c955ac49da2e2107b67a\",\n\t\t\tutil.hexstrdump(MD5 (\"12345678901234567890123456789012345678901234567890123456789012345678901234567890\")) == \"57edf4a22be3c955ac49da2e2107b67a\");\n\treturn result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"RIPE-MD 160 bits test with test vectors from http://homes.esat.kuleuven.be/~bosselae/ripemd160.html\", function() {\n\n\tvar openpgp = require('../../../'),\n\t\tutil = openpgp.util,\n\t\tRMDstring = openpgp.hash.ripemd;\n\n\tvar result = new Array();\n\tresult[0] = new unit.result(\"RMDstring (\\\"\\\") = 9c1185a5c5e9fc54612808977ee8f548b2258d31\",\n\t\t\tutil.hexstrdump(RMDstring(\"\")) == \"9c1185a5c5e9fc54612808977ee8f548b2258d31\");\n\tresult[1] = new unit.result(\"RMDstring (\\\"a\\\") = 0bdc9d2d256b3ee9daae347be6f4dc835a467ffe\",\n\t\t\tutil.hexstrdump(RMDstring(\"a\")) == \"0bdc9d2d256b3ee9daae347be6f4dc835a467ffe\");\n\tresult[2] = new unit.result(\"RMDstring (\\\"abc\\\") = 8eb208f7e05d987a9b044a8e98c6b087f15a0bfc\",\n\t\t\tutil.hexstrdump(RMDstring(\"abc\")) == \"8eb208f7e05d987a9b044a8e98c6b087f15a0bfc\");\n\tresult[3] = new unit.result(\"RMDstring (\\\"message digest\\\") = 5d0689ef49d2fae572b881b123a85ffa21595f36\",\n\t\t\tutil.hexstrdump(RMDstring(\"message digest\")) == \"5d0689ef49d2fae572b881b123a85ffa21595f36\");\n\treturn result;\n});\n","var unit = require('../../unit.js');\n\n\nunit.register(\"SHA* test with test vectors from NIST FIPS 180-2\", function() {\n\tvar openpgp = require('../../../src'),\n\t\tutil = openpgp.util,\n\t\thash = openpgp.hash;\n\n\tvar result = new Array();\n\t\n\tresult[0] = new unit.result(\"SHA1 - a9993e364706816aba3e25717850c26c9cd0d89d = hash.sha1(\\\"abc\\\") \",\n\t\t\t\"a9993e364706816aba3e25717850c26c9cd0d89d\" == util.hexstrdump(hash.sha1(\"abc\")));\n\tresult[1] = new unit.result(\"SHA1 - 84983e441c3bd26ebaae4aa1f95129e5e54670f1 = hash.sha1(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"84983e441c3bd26ebaae4aa1f95129e5e54670f1\" == util.hexstrdump(hash.sha1(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\n\tresult[2] = new unit.result(\"SHA224 - 23097d223405d8228642a477bda255b32aadbce4bda0b3f7e36c9da7 = hash.sha224(\\\"abc\\\") \",\n\t\t\t\"23097d223405d8228642a477bda255b32aadbce4bda0b3f7e36c9da7\" == util.hexstrdump(hash.sha224(\"abc\")));\n\tresult[3] = new unit.result(\"SHA224 - 75388b16512776cc5dba5da1fd890150b0c6455cb4f58b1952522525 = hash.sha224(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"75388b16512776cc5dba5da1fd890150b0c6455cb4f58b1952522525\" == util.hexstrdump(hash.sha224(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\n\tresult[4] = new unit.result(\"SHA256 - ba7816bf8f01cfea414140de5dae2223b00361a396177a9cb410ff61f20015ad = hash.sha256(\\\"abc\\\") \",\n\t\t\t\"ba7816bf8f01cfea414140de5dae2223b00361a396177a9cb410ff61f20015ad\" == util.hexstrdump(hash.sha256(\"abc\")));\n\tresult[5] = new unit.result(\"SHA256 - 248d6a61d20638b8e5c026930c3e6039a33ce45964ff2167f6ecedd419db06c1 = hash.sha256(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"248d6a61d20638b8e5c026930c3e6039a33ce45964ff2167f6ecedd419db06c1\" == util.hexstrdump(hash.sha256(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\n\tresult[6] = new unit.result(\"SHA384 - cb00753f45a35e8bb5a03d699ac65007272c32ab0eded1631a8b605a43ff5bed8086072ba1e7cc2358baeca134c825a7 = hash.sha384(\\\"abc\\\") \",\n\t\t\t\"cb00753f45a35e8bb5a03d699ac65007272c32ab0eded1631a8b605a43ff5bed8086072ba1e7cc2358baeca134c825a7\" == util.hexstrdump(hash.sha384(\"abc\")));\n\tresult[7] = new unit.result(\"SHA384 - 3391fdddfc8dc7393707a65b1b4709397cf8b1d162af05abfe8f450de5f36bc6b0455a8520bc4e6f5fe95b1fe3c8452b = str384(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"3391fdddfc8dc7393707a65b1b4709397cf8b1d162af05abfe8f450de5f36bc6b0455a8520bc4e6f5fe95b1fe3c8452b\" == util.hexstrdump(hash.sha384(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\t\t\t\t\t\n\tresult[8] = new unit.result(\"SHA512 - ddaf35a193617abacc417349ae20413112e6fa4e89a97ea20a9eeee64b55d39a2192992a274fc1a836ba3c23a3feebbd454d4423643ce80e2a9ac94fa54ca49f = hash.sha512(\\\"abc\\\") \",\n\t\t\t\"ddaf35a193617abacc417349ae20413112e6fa4e89a97ea20a9eeee64b55d39a2192992a274fc1a836ba3c23a3feebbd454d4423643ce80e2a9ac94fa54ca49f\" == util.hexstrdump(hash.sha512(\"abc\")));\n\tresult[9] = new unit.result(\"SHA512 - 204a8fc6dda82f0a0ced7beb8e08a41657c16ef468b228a8279be331a703c33596fd15c13b1b07f9aa1d3bea57789ca031ad85c7a71dd70354ec631238ca3445 = hash.sha512(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"204a8fc6dda82f0a0ced7beb8e08a41657c16ef468b228a8279be331a703c33596fd15c13b1b07f9aa1d3bea57789ca031ad85c7a71dd70354ec631238ca3445\" == util.hexstrdump(hash.sha512(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\t\t\t\t\t\n\treturn result;\n});\n","var unit = require('../unit.js');\n\nunit.register(\"Functional testing of openpgp.crypto.* methods\", function() {\n\tvar openpgp = require('../../');\n  var util = openpgp.util;\n  var result = [];\n  var RSApubMPIstrs = [\n              util.bin2str([0x08,0x00,0xac,0x15,0xb3,0xd6,0xd2,0x0f,0xf0,0x7a,0xdd,0x21,0xb7,\n                            0xbf,0x61,0xfa,0xca,0x93,0x86,0xc8,0x55,0x5a,0x4b,0xa6,0xa4,0x1a,\n                            0x60,0xa2,0x3a,0x37,0x06,0x08,0xd8,0x15,0x8e,0x85,0x45,0xaa,0xb7,\n                            0x1d,0x7b,0x0b,0x73,0x94,0x55,0x0c,0x5c,0xec,0xc0,0x22,0x4b,0xa1,\n                            0x64,0x20,0x7d,0x4d,0xa8,0x96,0x1a,0x64,0x38,0x93,0xcd,0xec,0x73,\n                            0x5d,0xf9,0x89,0x88,0x24,0x3d,0x48,0xff,0x3b,0x87,0x62,0xd0,0x84,\n                            0xea,0xff,0x39,0xb5,0x27,0x70,0xea,0x4a,0xb2,0x2e,0x9d,0xf1,0x7c,\n                            0x23,0xec,0xf4,0x5e,0xea,0x61,0x3d,0xe5,0xd8,0x0d,0xf9,0x59,0x6d,\n                            0x28,0x00,0xeb,0xcb,0xc9,0x55,0x00,0x72,0x30,0x1f,0x65,0x9d,0xd6,\n                            0x17,0x58,0x5f,0xa6,0x4a,0xa0,0xdd,0xe1,0x76,0xf2,0xef,0x21,0x9f,\n                            0x84,0xfc,0xaa,0x5b,0x52,0x6e,0xc1,0xa2,0xb9,0xbd,0xb9,0xf4,0x9e,\n                            0x49,0x92,0xf2,0xaf,0x57,0x86,0xf2,0xef,0x70,0xbf,0x51,0x40,0xfd,\n                            0xbf,0x56,0x51,0xe8,0x2c,0xa2,0x4f,0xf8,0xa4,0xd7,0x36,0x18,0x85,\n                            0xce,0x09,0x0d,0xbc,0x8d,0x65,0x5e,0x8a,0x1d,0x98,0xb0,0x4d,0x9d,\n                            0xc1,0xcf,0x82,0xe1,0xb7,0x43,0x5d,0x5a,0x72,0xcd,0x55,0xd2,0xff,\n                            0xb1,0xb4,0x78,0xbf,0xa1,0x7d,0xac,0xd9,0x1b,0xc4,0xfa,0x39,0x34,\n                            0x92,0x09,0xf9,0x08,0x2a,0x6b,0x9d,0x14,0x56,0x12,0x4c,0xe9,0xa6,\n                            0x29,0xc1,0xf3,0xa9,0x0b,0xfc,0x31,0x75,0x58,0x74,0x2a,0x88,0xaf,\n                            0xee,0xc9,0xa4,0xcd,0x15,0xdc,0x1b,0x8d,0x64,0xc1,0x36,0x17,0xc4,\n                            0x8d,0x5e,0x99,0x7a,0x5b,0x9f,0x39,0xd0,0x00,0x6e,0xf9]),\n              util.bin2str([0x00,0x11,0x01,0x00,0x01])];\n  var RSAsecMPIstrs = [\n              util.bin2str([0x07,0xfe,0x23,0xff,0xce,0x45,0x6c,0x60,0x65,0x40,0x6e,0xae,0x35,\n                            0x10,0x56,0x60,0xee,0xab,0xfa,0x10,0x42,0xba,0xc7,0x04,0xaf,0x63,\n                            0xcd,0x3f,0x62,0xca,0x4b,0xfa,0xe1,0xa9,0x70,0xcd,0x34,0x8b,0xc8,\n                            0x0e,0xe4,0xc4,0xba,0x83,0x17,0x5f,0xa4,0xb8,0xea,0x60,0xc2,0x4d,\n                            0x9a,0xf2,0xa9,0x03,0xeb,0xf6,0xaa,0xc2,0xb8,0x8b,0x43,0x12,0xe9,\n                            0xf7,0x88,0xd2,0x5a,0xa6,0xaa,0x23,0x71,0x31,0x74,0xdb,0x19,0x20,\n                            0x15,0x41,0x1b,0x43,0x68,0x62,0xd8,0xc0,0x93,0x91,0xe8,0xfc,0xb1,\n                            0xa9,0x9a,0x52,0x6c,0xe0,0xbf,0x43,0x01,0xa8,0x37,0x14,0x28,0xbf,\n                            0x0b,0x15,0x56,0x3e,0xa5,0x79,0xa0,0xc4,0x42,0x88,0xee,0xeb,0x1b,\n                            0xf4,0x7a,0x4a,0x58,0x31,0x58,0x81,0xd2,0x3e,0xde,0x07,0x64,0x92,\n                            0xf0,0x60,0xd3,0x9a,0x29,0xca,0xc6,0x67,0x75,0x07,0xca,0x92,0x39,\n                            0x56,0xf6,0x11,0x84,0xba,0x6d,0x4b,0xe6,0x6f,0x66,0xc2,0x17,0xeb,\n                            0x46,0x69,0x1c,0xbb,0xdf,0xc0,0x38,0x00,0xd6,0x01,0xe6,0x70,0x9d,\n                            0x4b,0x9b,0x70,0xed,0x5c,0xb8,0xcf,0xe8,0x68,0x71,0xbe,0x24,0x6d,\n                            0xb1,0xa3,0x13,0xcc,0xf1,0xbc,0x67,0xdc,0xe0,0x69,0x09,0x82,0x3c,\n                            0x3b,0x0f,0x14,0x98,0x48,0x30,0xb2,0x70,0xc6,0x9e,0xfa,0x46,0x8f,\n                            0xf1,0xc0,0x65,0x8e,0xc6,0xae,0xdc,0x47,0x91,0x13,0x1e,0xd6,0x4a,\n                            0xf2,0xad,0xda,0xc2,0xc7,0x39,0x78,0x99,0xde,0x57,0x14,0x45,0x7f,\n                            0x32,0x38,0xa3,0x44,0x0f,0xe7,0x39,0x4c,0x6f,0x0f,0x32,0x7e,0xf1,\n                            0x5c,0x84,0x97,0xdd,0xa0,0x0c,0x87,0x66,0x7d,0x75,0x79]),\n              util.bin2str([0x04,0x00,0xc2,0xbc,0x71,0xf7,0x41,0x4a,0x09,0x66,0x70,0x02,0x68,\n                            0x8b,0xeb,0xe2,0x34,0xd1,0x12,0x83,0x93,0x75,0xe9,0x71,0x32,0xe2,\n                            0xed,0x18,0x6f,0x8e,0x3a,0xff,0x22,0x70,0x28,0x01,0xbf,0x4a,0x39,\n                            0x41,0xbb,0x3c,0x4a,0xbc,0xb8,0x13,0xfc,0x14,0xf6,0x71,0xa1,0x44,\n                            0x1c,0x02,0xa1,0x73,0x81,0xcc,0xa0,0x35,0x02,0x3e,0x97,0xb5,0xc4,\n                            0x94,0x33,0xf1,0xd1,0xdf,0x14,0x3f,0xfb,0x8f,0xb9,0x75,0x70,0xdc,\n                            0x74,0x3f,0x07,0x35,0x8f,0x53,0xaa,0xb2,0xd6,0x88,0x51,0x71,0x4e,\n                            0x01,0x24,0xec,0x7d,0xca,0xf6,0xa2,0xb3,0xbb,0xad,0x2e,0x60,0xfb,\n                            0x1c,0xee,0x49,0xd0,0x4e,0x5c,0xe3,0x1f,0x88,0x48,0xe4,0x68,0x14,\n                            0x3d,0x71,0xba,0xd7,0x4d,0x35,0x10,0x86,0x37,0x62,0xe0,0xa5,0x0b]),\n              util.bin2str([0x04,0x00,0xe2,0x38,0xf9,0xc8,0x3c,0xd1,0xcf,0x62,0x93,0xc3,0x77,\n                            0x76,0x97,0x44,0xe8,0xc8,0xca,0x93,0x9a,0xef,0xf0,0x63,0x76,0x25,\n                            0x3b,0x1c,0x46,0xff,0x90,0x13,0x91,0x15,0x97,0x7e,0x88,0x95,0xd4,\n                            0x7f,0x2f,0x52,0x6e,0x0d,0x55,0x55,0x2e,0xf1,0x58,0x5c,0x7e,0x56,\n                            0xd4,0x48,0xaa,0xdb,0x8c,0x44,0x4d,0x84,0x69,0x33,0x87,0x07,0xb2,\n                            0x7e,0xf5,0xa0,0x60,0xfb,0x73,0x59,0x46,0x29,0xcb,0x1e,0x3f,0x7c,\n                            0x2f,0xa6,0x53,0xe3,0x8c,0xef,0xd5,0xeb,0xbb,0xc8,0x9a,0x8e,0x66,\n                            0x4a,0x47,0x2f,0xe1,0xba,0x5e,0x32,0xd4,0x52,0x04,0x88,0x9d,0x63,\n                            0x3e,0xba,0x71,0x2d,0xf7,0x61,0xd5,0xfc,0x26,0xbf,0xd8,0x60,0x92,\n                            0x7b,0x94,0xf8,0x6f,0x3d,0x97,0x0b,0x0c,0x52,0x8c,0xb3,0xb6,0x8b]),\n              util.bin2str([0x04,0x00,0xb7,0xc5,0x4d,0x6e,0x2f,0xdd,0xef,0xec,0x07,0x70,0xa2,\n                            0x7c,0x1c,0x9d,0x8e,0x66,0x60,0x7c,0x61,0x1e,0x45,0xe9,0xdc,0x82,\n                            0x2f,0xc5,0x7e,0x1a,0xc6,0xd0,0x92,0xc5,0x22,0x9b,0x9a,0xfb,0x73,\n                            0x95,0x99,0xf2,0x7c,0xdb,0x2a,0x93,0x7b,0x5a,0x29,0x73,0x24,0x16,\n                            0x41,0x49,0xb5,0xf2,0x5f,0xbe,0xe7,0x64,0x4d,0xda,0x52,0x9e,0xc1,\n                            0x41,0x40,0x5e,0x03,0x92,0x8d,0x39,0x95,0x1f,0x68,0x9f,0x00,0x2e,\n                            0x0c,0x6f,0xcf,0xd9,0x6d,0x68,0xf7,0x00,0x4f,0x0e,0xc8,0x0b,0xfa,\n                            0x51,0xe0,0x22,0xf0,0xff,0xa7,0x42,0xd4,0xde,0x0b,0x47,0x8f,0x2b,\n                            0xf5,0x4d,0x04,0x32,0x91,0x89,0x4b,0x0e,0x05,0x8d,0x70,0xf9,0xbb,\n                            0xe7,0xd6,0x76,0xea,0x0e,0x1a,0x90,0x30,0xf5,0x98,0x01,0xc5,0x73])];\n  \n  var DSApubMPIstrs = [\n          util.bin2str([0x08,0x00,0xa8,0x85,0x5c,0x28,0x05,0x94,0x03,0xbe,0x07,0x6c,0x13,0x3e,0x65,\n                        0xfb,0xb5,0xe1,0x99,0x7c,0xfa,0x84,0xe3,0xac,0x47,0xa5,0xc4,0x46,0xd8,0x5f,\n                        0x44,0xe9,0xc1,0x6b,0x69,0xf7,0x10,0x76,0x49,0xa7,0x25,0x85,0xf4,0x1b,0xed,\n                        0xc6,0x60,0xc4,0x5b,0xaa,0xd4,0x87,0xd6,0x8f,0x92,0x56,0x7d,0x55,0x3f,0x45,\n                        0xae,0x12,0x73,0xda,0x29,0x8c,0xba,0x32,0xcc,0xd7,0xa4,0xd0,0x24,0xb0,0x7c,\n                        0xd8,0x0c,0x3a,0x91,0x6f,0x98,0x40,0x9c,0x9a,0xa8,0xcc,0x28,0x27,0x95,0x0b,\n                        0xe1,0x5b,0xb9,0x3b,0x1c,0x1c,0xd2,0xec,0xab,0x07,0x25,0x8d,0x7a,0x2a,0x2b,\n                        0x16,0x14,0xe8,0xda,0x71,0xd2,0xab,0xba,0x85,0x14,0x0d,0xc5,0xe0,0x88,0xeb,\n                        0xa5,0xe2,0xd5,0x48,0x3d,0x74,0x0c,0x41,0xeb,0xfd,0xb6,0x4e,0xf9,0x2c,0x82,\n                        0x17,0xdd,0x64,0x1e,0x19,0x39,0xa3,0x7f,0xf9,0x00,0xcd,0x9b,0xda,0x2e,0xbd,\n                        0x71,0x12,0xdf,0x0d,0x7c,0x0a,0x6b,0x2d,0x21,0x3b,0x9c,0x66,0x93,0x4a,0x1e,\n                        0x90,0x79,0xd3,0x5a,0x5b,0xe5,0xb9,0x94,0x1b,0xe6,0x47,0x99,0x06,0x98,0xd8,\n                        0x2a,0xe5,0xe2,0xa6,0x95,0x6a,0x07,0xc8,0xac,0x7c,0xe9,0xfc,0xa2,0x6a,0x16,\n                        0x2c,0x94,0x98,0xbd,0x91,0x0a,0x7c,0x7c,0x2c,0xb9,0x7e,0xa2,0x51,0x8b,0x45,\n                        0x1d,0x46,0x34,0xa8,0x52,0x2b,0xdd,0xd9,0xa8,0xbc,0x46,0x78,0x66,0xe1,0x72,\n                        0x11,0xf1,0xcb,0x1a,0xb6,0x4e,0x05,0x54,0xf7,0xe9,0xbe,0x4c,0x25,0x59,0x08,\n                        0x9f,0xf8,0xea,0x25,0x97,0x33,0xd6,0xc9,0x0f,0x59,0x0e,0xfd,0x9f,0xdc,0xe2,\n                        0xc0,0xcf,0x2f]),\n      util.bin2str([0x01,0x00,0xe1,0x72,0x2c,0xd0,0xbb,0x1a,0x4f,0xb6,0xb6,0x95,0x77,0x71,0x2e,\n                    0x01,0x48,0x3e,0x35,0x54,0x64,0x2b,0xed,0x40,0x5f,0x65,0x0c,0x57,0x28,0x5f,\n                    0xfd,0xfd,0xff,0xd7]),\n      util.bin2str([0x07,0xff,0x5d,0x9f,0xc4,0xb5,0x63,0x25,0x9d,0x72,0x88,0xe5,0x53,0x46,0x98,\n                    0xe3,0xe9,0x62,0xcb,0x0c,0xa1,0xb7,0x75,0x9f,0x18,0x41,0x94,0x32,0x28,0x29,\n                    0x6d,0x69,0xe0,0x3f,0x7d,0x7b,0x2b,0x06,0x5a,0x33,0x5c,0xd4,0x36,0x31,0x09,\n                    0x54,0x85,0x9d,0xb8,0x20,0xfe,0xda,0xfc,0xcd,0x1f,0xb1,0x2c,0x15,0x08,0x9d,\n                    0x32,0x53,0x2f,0xc1,0x42,0x22,0x69,0xff,0x67,0x2e,0x39,0x97,0x50,0x66,0x39,\n                    0xda,0xcf,0xfd,0x64,0x6f,0x91,0x05,0x64,0x37,0xc5,0x07,0x24,0xaa,0x40,0xa0,\n                    0x75,0x82,0x1d,0x97,0x96,0x12,0xf1,0xbd,0x9e,0x09,0x26,0x3c,0x97,0x5d,0x57,\n                    0xb8,0x5c,0x7d,0x89,0x03,0x82,0xcd,0x40,0xe5,0x03,0xe6,0x4a,0xfb,0xbc,0xd2,\n                    0xef,0x7a,0x89,0x02,0x08,0xc8,0x52,0xfa,0x97,0x74,0x66,0x32,0xae,0xa6,0x52,\n                    0x4b,0xef,0x5f,0xce,0x91,0x23,0x3f,0xab,0x9d,0x62,0x21,0xef,0x48,0x6d,0x07,\n                    0x5a,0xba,0xdf,0x00,0x91,0x54,0xea,0x5c,0xfa,0x4b,0x16,0x28,0x1a,0xce,0x48,\n                    0xb7,0x5c,0x50,0xa5,0x59,0xa4,0xb4,0xaf,0x1f,0xeb,0x8d,0x58,0x3f,0x0a,0xa5,\n                    0x97,0x2b,0x51,0x56,0xe8,0x88,0xf6,0x07,0xbc,0xdf,0xfa,0x2b,0x7b,0x88,0xe0,\n                    0x46,0xc8,0x7a,0x3e,0xd8,0x80,0xdb,0x4d,0x87,0x61,0x4f,0x64,0xcd,0xeb,0xe8,\n                    0x0d,0x86,0x16,0xcc,0xdd,0x6c,0x76,0x66,0xc1,0x73,0xb7,0x08,0x98,0x89,0x2f,\n                    0x67,0x69,0xd1,0xfc,0x97,0x4d,0xa2,0xce,0xad,0xbb,0x6f,0xab,0xa5,0xd6,0x18,\n                    0xb3,0x1a,0x96,0x02,0xbc,0x31,0x42,0xa2,0xad,0x77,0xe8,0xe2,0x4c,0x99,0xf9,\n                    0xdd,0xbe,0xcd]),\n      util.bin2str([0x07,0xff,0x5d,0xfe,0x9c,0x98,0xef,0x3a,0xa6,0x49,0xf0,0x10,0x67,0x79,0x2a,\n                    0x9d,0x79,0x43,0x06,0xa4,0xa8,0x6b,0x1a,0x6d,0x1f,0x77,0x6e,0x00,0x31,0xb9,\n                    0xed,0xc9,0x66,0xff,0xf1,0x21,0x32,0xfa,0x62,0x43,0xcd,0x97,0xd3,0x3d,0xaf,\n                    0xb4,0x29,0x29,0x26,0x4e,0x1c,0xa0,0xad,0x1c,0x07,0x28,0x3f,0xe5,0x43,0x10,\n                    0xba,0xb4,0x08,0xe0,0xdc,0xa2,0xc3,0x5b,0x1f,0xbd,0x94,0xc7,0x43,0xe5,0xf2,\n                    0x17,0x30,0x54,0x7f,0x14,0xbe,0xf4,0xbd,0x91,0x3b,0xe4,0x36,0xa4,0x50,0x5b,\n                    0x37,0x89,0x5e,0xcc,0xc7,0x74,0x54,0x32,0x20,0x09,0x63,0x98,0xb7,0xd9,0xaf,\n                    0x7f,0xb0,0x6c,0x27,0x43,0xfe,0x52,0xe6,0x1a,0x88,0x59,0x25,0xfc,0xeb,0x43,\n                    0x50,0xc7,0x65,0x43,0xc1,0x86,0x73,0x58,0x53,0x3a,0xcf,0x7a,0xa3,0x1d,0x56,\n                    0xc8,0x4a,0x80,0x70,0xb7,0xbf,0xf2,0xa3,0xec,0xe8,0x77,0x05,0x33,0x09,0x9d,\n                    0xaa,0xca,0xa0,0xe1,0x64,0x64,0x6f,0x76,0x99,0x41,0x75,0x78,0x90,0xf6,0xe7,\n                    0x23,0xe6,0xec,0x50,0xe5,0x99,0xa8,0x3e,0x1a,0x4b,0xc9,0x88,0x58,0x66,0xae,\n                    0x1a,0x53,0x5e,0xe4,0xb7,0x86,0xcf,0xa6,0xe5,0xad,0xb4,0x80,0xa0,0xf1,0x0d,\n                    0x96,0xb8,0x41,0xd0,0x07,0x9a,0x21,0x8d,0x50,0x7f,0x4f,0x73,0x13,0xa2,0xe2,\n                    0x02,0x07,0xc3,0xa3,0x0f,0x09,0x18,0x7f,0xf7,0x6b,0x90,0x70,0xc0,0xf9,0x0c,\n                    0x67,0x8d,0x9d,0x14,0xb6,0x9d,0x32,0x82,0xd0,0xb5,0xc6,0x57,0xf0,0x91,0xd9,\n                    0xc3,0x26,0xae,0x9f,0xa9,0x67,0x49,0x96,0x5c,0x07,0x3e,0x47,0x5c,0xed,0x60,\n                    0x07,0xac,0x6a])];\n  var DSAsecMPIstrs = [util.bin2str([0x01,0x00,0x9b,0x58,0xa8,0xf4,0x04,0xb1,0xd5,0x14,0x09,0xe1,\n                           0xe1,0xa1,0x8a,0x0b,0xa3,0xc3,0xa3,0x66,0xaa,0x27,0x99,0x50,\n                           0x1c,0x4d,0xba,0x24,0xee,0xdf,0xdf,0xb8,0x8e,0x8e])];\n            \n  var ElgamalpubMPIstrs = \n          [util.bin2str([0x08,0x00,0xea,0xcc,0xbe,0xe2,0xe4,0x5a,0x51,0x18,0x93,0xa1,0x12,0x2f,\n                     0x00,0x99,0x42,0xd8,0x5c,0x1c,0x2f,0xb6,0x3c,0xd9,0x94,0x61,0xb4,0x55,\n                     0x8d,0x4e,0x73,0xe6,0x69,0xbc,0x1d,0x33,0xe3,0x2d,0x91,0x23,0x69,0x95,\n                     0x98,0xd7,0x18,0x5a,0xaf,0xa7,0x93,0xc6,0x05,0x93,0x3a,0xc7,0xea,0xd0,\n                     0xb1,0xa9,0xc7,0xab,0x41,0x89,0xc8,0x38,0x99,0xdc,0x1a,0x57,0x35,0x1a,\n                     0x27,0x62,0x40,0x71,0x9f,0x36,0x1c,0x6d,0x18,0x1c,0x93,0xf7,0xba,0x35,\n                     0x06,0xed,0x30,0xb8,0xd9,0x8a,0x7c,0x03,0xaf,0xba,0x40,0x1f,0x62,0xf1,\n                     0x6d,0x87,0x2c,0xa6,0x2e,0x46,0xb0,0xaa,0xbc,0xbc,0x93,0xfa,0x9b,0x47,\n                     0x3f,0x70,0x1f,0x2a,0xc2,0x66,0x9c,0x7c,0x69,0xe0,0x2b,0x05,0xee,0xb7,\n                     0xa7,0x7f,0xf3,0x21,0x48,0x85,0xc2,0x95,0x5f,0x6f,0x1e,0xb3,0x9b,0x97,\n                     0xf8,0x14,0xc3,0xff,0x4d,0x97,0x25,0x29,0x94,0x41,0x4b,0x90,0xd8,0xba,\n                     0x71,0x45,0x4b,0x1e,0x2f,0xca,0x82,0x5f,0x56,0x77,0xe9,0xd3,0x88,0x5d,\n                     0x8b,0xec,0x92,0x8b,0x8a,0x23,0x88,0x05,0xf8,0x2c,0xa8,0xf1,0x70,0x76,\n                     0xe7,0xbf,0x75,0xa8,0x31,0x14,0x8e,0x76,0xc8,0x01,0xa6,0x25,0x27,0x49,\n                     0xaf,0xdc,0xf4,0xf6,0xf4,0xce,0x90,0x84,0x15,0x2b,0x4d,0xb3,0xcc,0x77,\n                     0xdb,0x65,0x71,0x75,0xd3,0x00,0x1d,0x22,0xc5,0x42,0x2f,0x51,0xfa,0x7b,\n                     0xeb,0x6e,0x03,0xd9,0x41,0xdd,0x2d,0x1a,0xdd,0x07,0x74,0x8b,0xb7,0xa2,\n                     0xfa,0xb2,0x59,0x0e,0x0e,0x94,0x7c,0x00,0xad,0x95,0x23,0x42,0x91,0x18,\n                     0x4c,0x97,0xf1,0x27,0x62,0x77]),\n       util.bin2str([0x00,0x03,0x05]),\n       util.bin2str([0x07,0xff,0x57,0x19,0x76,0xfc,0x09,0x6a,0x7a,0xf7,0xba,0xb2,0x42,0xbf,\n                     0xcd,0x2b,0xc1,0x1a,0x79,0x25,0x8c,0xad,0xf4,0x3a,0x0a,0x7a,0x9b,0x4c,\n                     0x46,0x3c,0xe0,0x4f,0xcc,0x6e,0xe5,0x7a,0x33,0x3a,0x4e,0x80,0xcb,0xd3,\n                     0x62,0xd7,0x8f,0xe2,0xc8,0xb0,0xd0,0xcb,0x49,0xc9,0x9e,0x2d,0x97,0x16,\n                     0x3a,0x7d,0xb1,0xe1,0xd3,0xd9,0xd7,0x3f,0x20,0x60,0xe3,0x3e,0x77,0xea,\n                     0x0c,0xe4,0x7b,0xf0,0x39,0x1a,0x0d,0xd9,0x8f,0x73,0xd2,0x51,0xb8,0x0c,\n                     0x0e,0x15,0x1e,0xad,0x7c,0xd8,0x9d,0x74,0x6e,0xa2,0x17,0x6b,0x58,0x14,\n                     0x2b,0xb7,0xad,0x8a,0xd7,0x66,0xc0,0xdf,0xea,0x2d,0xfc,0xc4,0x6e,0x68,\n                     0xb6,0x4c,0x9a,0x16,0xa4,0x3d,0xc2,0x26,0x0c,0xb7,0xd4,0x13,0x7b,0x22,\n                     0xfd,0x84,0xd7,0x0f,0xdc,0x42,0x75,0x05,0x85,0x29,0x00,0x31,0x1d,0xec,\n                     0x4e,0x22,0x8b,0xf6,0x37,0x83,0x45,0xe5,0xb3,0x31,0x61,0x2c,0x02,0xa1,\n                     0xc6,0x9d,0xea,0xba,0x3d,0x8a,0xab,0x0f,0x61,0x5e,0x14,0x64,0x69,0x1e,\n                     0xa0,0x15,0x48,0x86,0xe5,0x11,0x06,0xe8,0xde,0x34,0xc7,0xa7,0x3d,0x35,\n                     0xd1,0x76,0xc2,0xbe,0x01,0x82,0x61,0x8d,0xe7,0x7e,0x28,0x1d,0x4e,0x8c,\n                     0xb9,0xe8,0x7e,0xa4,0x5f,0xa6,0x3a,0x9e,0x5d,0xac,0xf3,0x60,0x22,0x14,\n                     0xd5,0xd5,0xbe,0x1f,0xf0,0x19,0xe6,0x81,0xfd,0x5d,0xe1,0xf8,0x76,0x5f,\n                     0xe3,0xda,0xba,0x19,0xf3,0xcb,0x10,0xa0,0x6b,0xd0,0x2d,0xbe,0x40,0x42,\n                     0x7b,0x9b,0x15,0xa4,0x2d,0xec,0xcf,0x09,0xd6,0xe3,0x92,0xc3,0x8d,0x65,\n                     0x6b,0x60,0x97,0xda,0x6b,0xca])];\n\n  var ElgamalsecMPIstrs = [\n             util.bin2str([0x01,0x52,0x02,0x80,0x87,0xf6,0xe4,0x49,0xd7,0x2e,0x3e,0xfe,0x60,0xb9,\n                           0xa3,0x2a,0xf0,0x67,0x58,0xe9,0xf6,0x47,0x83,0xde,0x7e,0xfb,0xbb,0xbd,\n                           0xdf,0x48,0x12,0x1b,0x06,0x7d,0x13,0xbc,0x3b,0x49,0xf9,0x86,0xd4,0x53,\n                           0xed,0x2d,0x68])];\n\n  var RSApubMPIs = [];\n  var i;\n  for (i = 0; i < 2; i++) {\n    RSApubMPIs[i] = new openpgp.mpi();\n    RSApubMPIs[i].read(RSApubMPIstrs[i]);\n  }\n\n  var RSAsecMPIs = [];\n  for (i = 0; i < 4; i++) {\n    RSAsecMPIs[i] = new openpgp.mpi();\n    RSAsecMPIs[i].read(RSAsecMPIstrs[i]);\n  }\n    \n  var DSAsecMPIs = [];\n  for (i = 0; i < 1; i++) {\n    DSAsecMPIs[i] = new openpgp.mpi();\n    DSAsecMPIs[i].read(DSAsecMPIstrs[i]);\n  }\n    \n  var DSApubMPIs = [];\n  for (i = 0; i < 4; i++) {\n    DSApubMPIs[i] = new openpgp.mpi();\n    DSApubMPIs[i].read(DSApubMPIstrs[i]);\n  }\n  var ElgamalsecMPIs = [];\n  for (i = 0; i < 1; i++) {\n    ElgamalsecMPIs[i] = new openpgp.mpi();\n    ElgamalsecMPIs[i].read(ElgamalsecMPIstrs[i]);\n  }\n    \n  var ElgamalpubMPIs = [];\n  for (i = 0; i < 3; i++) {\n    ElgamalpubMPIs[i] = new openpgp.mpi();\n    ElgamalpubMPIs[i].read(ElgamalpubMPIstrs[i]);\n  }\n\n  //Originally we passed public and secret MPI separately, now they are joined. Is this what we want to do long term?\n  // RSA\n  var RSAsignedData = openpgp.signature.sign(2, 1, RSApubMPIs.concat(RSAsecMPIs), \"foobar\");\n  var RSAsignedDataMPI = new openpgp.mpi();\n  RSAsignedDataMPI.read(RSAsignedData);\n  result[0] = new unit.result(\"Testing RSA Sign and Verify\",\n      openpgp.signature.verify(1, 2, [RSAsignedDataMPI], RSApubMPIs, \"foobar\"));\n\n  // DSA \n  var DSAsignedData = openpgp.signature.sign(2, 17, DSApubMPIs.concat(DSAsecMPIs), \"foobar\");\n  \n  var DSAmsgMPIs = [];\n    DSAmsgMPIs[0] = new openpgp.mpi();\n    DSAmsgMPIs[1] = new openpgp.mpi();\n    DSAmsgMPIs[0].read(DSAsignedData.substring(0,34));\n    DSAmsgMPIs[1].read(DSAsignedData.substring(34,68));\n  result[1] = new unit.result(\"Testing DSA Sign and Verify\",\n      openpgp.signature.verify(17, 2, DSAmsgMPIs, DSApubMPIs, \"foobar\"));\n  \n  var symmAlgo = \"aes256\"; // AES256\n  var symmKey = openpgp.generateSessionKey(symmAlgo);\n  var symmencDataOCFB = openpgp.cfb.encrypt(openpgp.getPrefixRandom(symmAlgo), symmAlgo, \"foobarfoobar1234567890\", symmKey, true);\n  var symmencDataCFB  = openpgp.cfb.encrypt(openpgp.getPrefixRandom(symmAlgo), symmAlgo, \"foobarfoobar1234567890\", symmKey, false);\n  \n  result[2] = new unit.result(\"Testing symmetric encrypt and decrypt with OpenPGP CFB resync\",\n      openpgp.cfb.decrypt(symmAlgo,symmKey,symmencDataOCFB,true) == \"foobarfoobar1234567890\");\n  result[3] = new unit.result(\"Testing symmetric encrypt and decrypt without OpenPGP CFB resync (used in modification detection code \\\"MDC\\\" packets)\",\n      openpgp.cfb.decrypt(symmAlgo,symmKey,symmencDataCFB,false) == \"foobarfoobar1234567890\");\n  \n  var RSAUnencryptedData = new openpgp.mpi();\n  RSAUnencryptedData.fromBytes(openpgp.pkcs1.eme.encode(symmKey, RSApubMPIs[0].byteLength()));\n  var RSAEncryptedData = openpgp.publicKeyEncrypt(\"rsa_encrypt_sign\", RSApubMPIs, RSAUnencryptedData);\n\n  result[4] = new unit.result(\"Testing asymmetric encrypt and decrypt using RSA with eme_pkcs1 padding\",\n      openpgp.pkcs1.eme.decode(openpgp.publicKeyDecrypt(\"rsa_encrypt_sign\", RSApubMPIs.concat(RSAsecMPIs), RSAEncryptedData).write().substring(2), RSApubMPIs[0].byteLength()) == symmKey);\n\n  var ElgamalUnencryptedData = new openpgp.mpi();\n  ElgamalUnencryptedData.fromBytes(openpgp.pkcs1.eme.encode(symmKey, ElgamalpubMPIs[0].byteLength()));\n  var ElgamalEncryptedData = openpgp.publicKeyEncrypt(\"elgamal\", ElgamalpubMPIs, ElgamalUnencryptedData);\n\n  result[5] = new unit.result(\"Testing asymmetric encrypt and decrypt using Elgamal with eme_pkcs1 padding\",\n      openpgp.pkcs1.eme.decode(openpgp.publicKeyDecrypt(\"elgamal\", ElgamalpubMPIs.concat(ElgamalsecMPIs), ElgamalEncryptedData).write().substring(2), ElgamalpubMPIs[0].byteLength()) == symmKey);\n\n  return result;\n});\n","var unit = require('../unit.js');\n\nunit.register(\"Testing of binary signature checking\", function() {\n  var openpgp = require('../../');\n  var keyring = require('../../src/openpgp.keyring.js');\n  var result = [];\n  var priv_key = openpgp.readArmoredPackets([\n        '-----BEGIN PGP PRIVATE KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'lQHhBFERnrMRBADmM0hIfkI3yosjgbWo9v0Lnr3CCE+8KsMszgVS+hBu0XfGraKm',\n        'ivcA2aaJimHqVYOP7gEnwFAxHBBpeTJcu5wzCFyJwEYqVeS3nnaIhBPplSF14Duf',\n        'i6bB9RV7KxVAg6aunmM2tAutqC+a0y2rDaf7jkJoZ9gWJe2zI+vraD6fiwCgxvHo',\n        '3IgULB9RqIqpLoMgXfcjC+cD/1jeJlKRm+n71ryYwT/ECKsspFz7S36z6q3XyS8Q',\n        'QfrsUz2p1fbFicvJwIOJ8B20J/N2/nit4P0gBUTUxv3QEa7XCM/56/xrGkyBzscW',\n        'AzBoy/AK9K7GN6z13RozuAS60F1xO7MQc6Yi2VU3eASDQEKiyL/Ubf/s/rkZ+sGj',\n        'yJizBACtwCbQzA+z9XBZNUat5NPgcZz5Qeh1nwF9Nxnr6pyBv7tkrLh/3gxRGHqG',\n        '063dMbUk8pmUcJzBUyRsNiIPDoEUsLjY5zmZZmp/waAhpREsnK29WLCbqLdpUors',\n        'c1JJBsObkA1IM8TZY8YUmvsMEvBLCCanuKpclZZXqeRAeOHJ0v4DAwK8WfuTe5B+',\n        'M2BOOeZbN8BpfiA1l//fMMHLRS3UvbLBv4P1+4SyvhyYTR7M76Q0xPc03MFOWHL+',\n        'S9VumbQWVGVzdDIgPHRlc3QyQHRlc3QuY29tPohiBBMRAgAiBQJREZ6zAhsDBgsJ',\n        'CAcDAgYVCAIJCgsEFgIDAQIeAQIXgAAKCRARJ5QDyxae+MXNAKCzWSDR3tMrTrDb',\n        'TAri73N1Xb3j1ACfSl9y+SAah2q7GvmiR1+6+/ekqJGdAVgEURGesxAEANlpMZjW',\n        '33jMxlKHDdyRFXtKOq8RreXhq00plorHbgz9zFEWm4VF53+E/KGnmHGyY5Cy8TKy',\n        'ZjaueZZ9XuG0huZg5If68irFfNZtxdA26jv8//PdZ0Uj+X6J3RVa2peMLDDswTYL',\n        'OL1ZO1fxdtDD40fdAiIZ1QyjwEG0APtz41EfAAMFBAC5/dtgBBPtHe8UjDBaUe4n',\n        'NzHuUBBp6XE+H7eqHNFCuZAJ7yqJLGVHNIaQR419cNy08/OO/+YUQ7rg78LxjFiv',\n        'CH7IzhfU+6yvELSbgRMicY6EnAP2GT+b1+MtFNa3lBGtBHcJla52c2rTAHthYZWk',\n        'fT5R5DnJuQ2cJHBMS9HWyP4DAwK8WfuTe5B+M2C7a/YJSUv6SexdGCaiaTcAm6g/',\n        'PvA6hw/FLzIEP67QcQSSTmhftQIwnddt4S4MyJJH3U4fJaFfYQ1zCniYJohJBBgR',\n        'AgAJBQJREZ6zAhsMAAoJEBEnlAPLFp74QbMAn3V4857xwnO9/+vzIVnL93W3k0/8',\n        'AKC8omYPPomN1E/UJFfXdLDIMi5LoA==',\n        '=LSrW',\n        '-----END PGP PRIVATE KEY BLOCK-----'\n      ].join(\"\\n\"));\n  var pub_key = openpgp.readArmoredPackets(\n      [ '-----BEGIN PGP PUBLIC KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'mQGiBFERlw4RBAD6Bmcf2w1dtUmtCLkdxeqZLArk3vYoQAjdibxA3gXVyur7fsWb',\n        'ro0jVbBHqOCtC6jDxE2l52NP9+tTlWeVMaqqNvUE47LSaPq2DGI8Wx1Rj6bF3mTs',\n        'obYEwhGbGh/MhJnME9AHODarvk8AZbzo0+k1EwrBWF6dTUBPfqO7rGU2ewCg80WV',\n        'x5pt3evj8rRK3jQ8SMKTNRsD/1PhTdxdZTdXARAFzcW1VaaruWW0Rr1+XHKKwDCz',\n        'i7HE76SO9qjnQfZCZG75CdQxI0h8GFeN3zsDqmhob2iSz2aJ1krtjM+iZ1FBFd57',\n        'OqCV6wmk5IT0RBN12ZzMS19YvzN/ONXHrmTZlKExd9Mh9RKLeVNw+bf6JsKQEzcY',\n        'JzFkBACX9X+hDYchO/2hiTwx4iOO9Fhsuh7eIWumB3gt+aUpm1jrSbas/QLTymmk',\n        'uZuQVXI4NtnlvzlNgWv4L5s5RU5WqNGG7WSaKNdcrvJZRC2dgbUJt04J5CKrWp6R',\n        'aIYal/81Ut1778lU01PEt563TcQnUBlnjU5OR25KhfSeN5CZY7QUVGVzdCA8dGVz',\n        'dEB0ZXN0LmNvbT6IYgQTEQIAIgUCURGXDgIbAwYLCQgHAwIGFQgCCQoLBBYCAwEC',\n        'HgECF4AACgkQikDlZK/UvLSspgCfcNaOpTg1W2ucR1JwBbBGvaERfuMAnRgt3/rs',\n        'EplqEakMckCtikEnpxYe',\n        '=b2Ln',\n        '-----END PGP PUBLIC KEY BLOCK-----'\n      ].join(\"\\n\"));\n  var msg = openpgp.readArmoredPackets([\n        '-----BEGIN PGP MESSAGE-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'hQEOA1N4OCSSjECBEAP/diDJCQn4e88193PgqhbfAkohk9RQ0v0MPnXpJbCRTHKO',\n        '8r9nxiAr/TQv4ZOingXdAp2JZEoE9pXxZ3r1UWew04czxmgJ8FP1ztZYWVFAWFVi',\n        'Tj930TBD7L1fY/MD4fK6xjEG7z5GT8k4tn4mLm/PpWMbarIglfMopTy1M/py2cID',\n        '/2Sj7Ikh3UFiG+zm4sViYc5roNbMy8ixeoKixxi99Mx8INa2cxNfqbabjblFyc0Z',\n        'BwmbIc+ZiY2meRNI5y/tk0gRD7hT84IXGGl6/mH00bsX/kkWdKGeTvz8s5G8RDHa',\n        'Za4HgLbXItkX/QarvRS9kvkD01ujHfj+1ZvgmOBttNfP0p8BQLIICqvg1eYD9aPB',\n        '+GtOZ2F3+k5VyBL5yIn/s65SBjNO8Fqs3aL0x+p7s1cfUzx8J8a8nWpqq/qIQIqg',\n        'ZJH6MZRKuQwscwH6NWgsSVwcnVCAXnYOpbHxFQ+j7RbF/+uiuqU+DFH/Rd5pik8b',\n        '0Dqnp0yfefrkjQ0nuvubgB6Rv89mHpnvuJfFJRInpg4lrHwLvRwdpN2HDozFHcKK',\n        'aOU=',\n        '=4iGt',\n        '-----END PGP MESSAGE-----'\n      ].join(\"\\n\"));\n  //TODO need both?\n  priv_key[0].decrypt(\"abcd\");\n  priv_key[3].decrypt(\"abcd\");\n  msg[0].decrypt(priv_key[3]);\n  msg[1].decrypt(msg[0].sessionKeyAlgorithm, msg[0].sessionKey);\n  msg[1].packets[2].verify(pub_key[0], msg[1].packets[1]);\n  result[0] = new unit.result(\"Testing signature checking on CAST5-enciphered message\",\n          msg[1].packets[2].verified === true);\n\n  // exercises the GnuPG s2k type 1001 extension:\n  // the secrets on the primary key have been stripped.\n  var priv_key_gnupg_ext = openpgp.readArmoredPackets([\n        '-----BEGIN PGP PRIVATE KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'lQGqBFERnrMRBADmM0hIfkI3yosjgbWo9v0Lnr3CCE+8KsMszgVS+hBu0XfGraKm',\n        'ivcA2aaJimHqVYOP7gEnwFAxHBBpeTJcu5wzCFyJwEYqVeS3nnaIhBPplSF14Duf',\n        'i6bB9RV7KxVAg6aunmM2tAutqC+a0y2rDaf7jkJoZ9gWJe2zI+vraD6fiwCgxvHo',\n        '3IgULB9RqIqpLoMgXfcjC+cD/1jeJlKRm+n71ryYwT/ECKsspFz7S36z6q3XyS8Q',\n        'QfrsUz2p1fbFicvJwIOJ8B20J/N2/nit4P0gBUTUxv3QEa7XCM/56/xrGkyBzscW',\n        'AzBoy/AK9K7GN6z13RozuAS60F1xO7MQc6Yi2VU3eASDQEKiyL/Ubf/s/rkZ+sGj',\n        'yJizBACtwCbQzA+z9XBZNUat5NPgcZz5Qeh1nwF9Nxnr6pyBv7tkrLh/3gxRGHqG',\n        '063dMbUk8pmUcJzBUyRsNiIPDoEUsLjY5zmZZmp/waAhpREsnK29WLCbqLdpUors',\n        'c1JJBsObkA1IM8TZY8YUmvsMEvBLCCanuKpclZZXqeRAeOHJ0v4DZQJHTlUBtBZU',\n        'ZXN0MiA8dGVzdDJAdGVzdC5jb20+iGIEExECACIFAlERnrMCGwMGCwkIBwMCBhUI',\n        'AgkKCwQWAgMBAh4BAheAAAoJEBEnlAPLFp74xc0AoLNZINHe0ytOsNtMCuLvc3Vd',\n        'vePUAJ9KX3L5IBqHarsa+aJHX7r796SokZ0BWARREZ6zEAQA2WkxmNbfeMzGUocN',\n        '3JEVe0o6rxGt5eGrTSmWisduDP3MURabhUXnf4T8oaeYcbJjkLLxMrJmNq55ln1e',\n        '4bSG5mDkh/ryKsV81m3F0DbqO/z/891nRSP5fondFVral4wsMOzBNgs4vVk7V/F2',\n        '0MPjR90CIhnVDKPAQbQA+3PjUR8AAwUEALn922AEE+0d7xSMMFpR7ic3Me5QEGnp',\n        'cT4ft6oc0UK5kAnvKoksZUc0hpBHjX1w3LTz847/5hRDuuDvwvGMWK8IfsjOF9T7',\n        'rK8QtJuBEyJxjoScA/YZP5vX4y0U1reUEa0EdwmVrnZzatMAe2FhlaR9PlHkOcm5',\n        'DZwkcExL0dbI/gMDArxZ+5N7kH4zYLtr9glJS/pJ7F0YJqJpNwCbqD8+8DqHD8Uv',\n        'MgQ/rtBxBJJOaF+1AjCd123hLgzIkkfdTh8loV9hDXMKeJgmiEkEGBECAAkFAlER',\n        'nrMCGwwACgkQESeUA8sWnvhBswCfdXjznvHCc73/6/MhWcv3dbeTT/wAoLyiZg8+',\n        'iY3UT9QkV9d0sMgyLkug',\n        '=GQsY',\n        '-----END PGP PRIVATE KEY BLOCK-----',\n      ].join(\"\\n\"));\n  priv_key_gnupg_ext[3].decrypt(\"abcd\");\n  msg[0].decrypt(priv_key_gnupg_ext[3]);\n  msg[1].decrypt(msg[0].sessionKeyAlgorithm, msg[0].sessionKey);\n  msg[1].packets[2].verify(pub_key[0], msg[1].packets[1]);\n  result[1] = new unit.result(\"Testing GnuPG stripped-key extensions\",\n          msg[1].packets[2].verified === true);\n\n  // Exercises the ability of openpgp_keyring.getPublicKeysForKeyId to return subkeys\n  keyring.init();\n  keyring.importPacketlist([\n        '-----BEGIN PGP PUBLIC KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'mQGiBFERvI4RBAD0M/HGglCtVNXPF72ehT8riAXrl0rSec4RJC61Bh+UAOhxn5+U',\n        'fDgos5p1SpIzYmn+M87JoSSVLAjfakFk0gHgR9I3bu7SIwq3Bikk1Vw3gO+yDSO6',\n        'TKpLUFGYDiBSSE1MGdxBadWLE1hlRf5B2x62gnGmjSpSVbly33PFkoDmrwCg9rAp',\n        'RmncnF9GhWjOLFkEoQw9Yx8EAOsxvq8Ig5Z1gk+ZKfDZeftpHRe3FdrRtbnhxvYY',\n        '7z+w9uz1EpoZUwDR5G4X3hTwJQ7lXmIOskg/+eRMLEAqEY7b/7tW6RaUJ2d6Ehsi',\n        'dOS89fIxElwjAOnVOM5S24f0FDQTTto7QBOoxcNTfkEJCHXSlpoOUmGAP85fXh3l',\n        'yPTGBACJfKc76Un3UWC1sWIRDxYiyh3ZpZyNEskoV6ESW8jEI1RnMnv5TrfGJH5K',\n        'E8jWX7TTnoFyPJtBTjlucAtkQaS4Bb7dg1LLja17zAqKNGOJK2b9fb2Z+lnTjPiY',\n        'i7DPH1XHnfaEexjlh/U7mYa5RrwIphRxNi8gCuxv874ZMmhEn7QWVGVzdDMgPHRl',\n        'c3QzQHRlc3QuY29tPohiBBMRAgAiBQJREbyOAhsDBgsJCAcDAgYVCAIJCgsEFgID',\n        'AQIeAQIXgAAKCRC0u8O0Moa2JYxyAJ9Oi2UlcUT0VJNgwjyl/VF9Xcjf9gCeJPvy',\n        'g/fp4EAU8MJIaN2yMI8pLFS5AaIEURG8nhEEAKVgeNDuYDPufLuJ0GrJV/CbXEjj',\n        'aEPA0iTUqV0nTCPdAfQ/nmE3gh5UlNMr/zSHJ+c4FQhYdLrzRGDOSzV+mfPHH3t+',\n        'YVx+wat0BYwABpHAtsIuLIVo2RQqYZYH85tatwBkm71HHT3jmlEAvr6NFH38+v3s',\n        '3w4Wl0/sdHyaeiSXAKCxJ4X1eOdN7L1rrbJozQ/gDCFuVQP/dcV6Ksss8Aw443jG',\n        'AYBLHWh6o4GhAY6/h1kijF0xD+uc+tNmTQnQi1tEOoTeIZMXnSRwtk8XEuJkkbAP',\n        '+uyvMgyV3wrk9zkaTAin7nrjAERxezFOdBEOtnB1CovJxtMn+RRxaMEGpC4GnETy',\n        'N5+6FkLuLcNXiCQP75ajzOAN1aID/juNjUNpBbNpfqBV7j1K+Kn0n9HYTyQl9ghy',\n        '026+/4c8ag2HV+bg3BD7c2VTVu9xBODHsfu0q8Ql/QB9W8tmYugU6DeXMHaeWPUH',\n        'ph98guM9kF2yHIiRBvAd5i7wOjwn+I/Ir6nBR2yxJ3p31CDUnUlbjTPYg7mtQvHW',\n        'EY2Cp4SWiJEEGBECAAkFAlERvJ4CGwIAUgkQtLvDtDKGtiVHIAQZEQIABgUCURG8',\n        'ngAKCRAMiMeR296Y2SjyAJ9V3wRJJ2Szazqal4khWGfLu5R6/wCfQQIRD24yVdz8',\n        '2a+2eCrwyALT2GAihACfS0nWM3a0gtITqngpJsRws+Ep+eIAn15qD2itutxNb8NI',\n        'bR2gBB5QmVJ3',\n        '=pGA6',\n        '-----END PGP PUBLIC KEY BLOCK-----'\n        ].join(\"\\n\"));\n\n  var msg2 = openpgp.readArmoredPackets([\n        '-----BEGIN PGP MESSAGE-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'kA0DAAIRDIjHkdvemNkBrB1iB2Zvby50eHRREbz3VEVTVCBEQVRBIDEyMzQ1NohG',\n        'BAARAgAGBQJREbz3AAoJEAyIx5Hb3pjZ2TcAn32LpDEuHe9QrSRlyvSuREKNOFwz',\n        'AJ9zh4zsK4GIPuEu81YPNmHsju7DYg==',\n        '=WaSx',\n        '-----END PGP MESSAGE-----'\n        ].join(\"\\n\"));\n  var packetlists = keyring.getPacketlistsForKeyId(msg2[0].signingKeyId.write());\n  var pubKey = packetlists[0];\n  msg2[2].verify(pubKey[3], msg2[1]);\n  result[2] = new unit.result(\"Testing keyring public subkey support\",\n          packetlists !== null && \n          packetlists.length == 1 && \n          msg2[2].verified);\n  return result;\n});\n\n","var unit = require('../unit.js');\n\nunit.register(\"Key generation/encryption/decryption\", function() {\n  var openpgp = require('../../');\n  var keyring = require('../../src/openpgp.keyring.js');\n  var result = [];\n  var testHelper = function(passphrase, userid, message) {\n    var key = openpgp.generateKeyPair(openpgp.enums.publicKey.rsa_encrypt_sign, 512, \n                                      userid, message, passphrase);\n\n    var info = '\\npassphrase: ' + passphrase + '\\n'\n        + 'userid: ' + userid + '\\n'\n        + 'message: ' + message;\n\n    var keyPacketlist = openpgp.readArmoredPackets(key);\n    debugger;\n    var privKey = openpgp.getKeyFromPacketlist(keyPacketlist, passphrase);\n\n    var encrypted = openpgp.encryptMessage(keyPacketlist, message);\n\n    openpgp.keyring.importPublicKey(key.publicKeyArmored);\n\n    var msg = openpgp.read_message(encrypted);\n    var keymat = null;\n    var sesskey = null;\n\n    // Find the private (sub)key for the session key of the message\n    for (var i = 0; i< msg[0].sessionKeys.length; i++) {\n      if (priv_key.privateKeyPacket.publicKey.getKeyId().write() == msg[0].sessionKeys[i].keyId.bytes) {\n        keymat = { key: priv_key, keymaterial: priv_key.privateKeyPacket};\n        sesskey = msg[0].sessionKeys[i];\n        break;\n      }\n      for (var j = 0; j < priv_key.subKeys.length; j++) {\n        if (priv_key.subKeys[j].publicKey.getKeyId().write() == msg[0].sessionKeys[i].keyId.bytes) {\n          keymat = { key: priv_key, keymaterial: priv_key.subKeys[j]};\n          sesskey = msg[0].sessionKeys[i];\n          break;\n        }\n      }\n    }\n\n    var decrypted = '';\n    if (keymat !== null) {\n      if (!keymat.keymaterial.decryptSecretMPIs(passphrase)) {\n        return new test_result(\"Password for secrect key was incorrect!\", \n          + info, false);\n      }\n\n      decrypted = msg[0].decrypt(keymat, sesskey);\n    } else {\n      return new test_result(\"No private key found!\" + info, false);\n    }\n    return new test_result(message + ' == ' + decrypted + info, message == decrypted);\n  };\n\n  result.push(testHelper('password', 'Test McTestington <test@example.com>', 'hello world'));\n  result.push(testHelper('●●●●', '♔♔♔♔ <test@example.com>', 'łäóć'));\n\n  return result;\n});\n\nunit.register(\"Encryption/decryption\", function() {\n  var openpgp = require('../../src');\n\n  var pub_key = \n     ['-----BEGIN PGP PUBLIC KEY BLOCK-----',\n      'Version: GnuPG v2.0.19 (GNU/Linux)',\n      'Type: RSA/RSA',\n      '',\n      'mI0EUmEvTgEEANyWtQQMOybQ9JltDqmaX0WnNPJeLILIM36sw6zL0nfTQ5zXSS3+',\n      'fIF6P29lJFxpblWk02PSID5zX/DYU9/zjM2xPO8Oa4xo0cVTOTLj++Ri5mtr//f5',\n      'GLsIXxFrBJhD/ghFsL3Op0GXOeLJ9A5bsOn8th7x6JucNKuaRB6bQbSPABEBAAG0',\n      'JFRlc3QgTWNUZXN0aW5ndG9uIDx0ZXN0QGV4YW1wbGUuY29tPoi5BBMBAgAjBQJS',\n      'YS9OAhsvBwsJCAcDAgEGFQgCCQoLBBYCAwECHgECF4AACgkQSmNhOk1uQJQwDAP6',\n      'AgrTyqkRlJVqz2pb46TfbDM2TDF7o9CBnBzIGoxBhlRwpqALz7z2kxBDmwpQa+ki',\n      'Bq3jZN/UosY9y8bhwMAlnrDY9jP1gdCo+H0sD48CdXybblNwaYpwqC8VSpDdTndf',\n      '9j2wE/weihGp/DAdy/2kyBCaiOY1sjhUfJ1GogF49rC4jQRSYS9OAQQA6R/PtBFa',\n      'JaT4jq10yqASk4sqwVMsc6HcifM5lSdxzExFP74naUMMyEsKHP53QxTF0Grqusag',\n      'Qg/ZtgT0CN1HUM152y7ACOdp1giKjpMzOTQClqCoclyvWOFB+L/SwGEIJf7LSCEr',\n      'woBuJifJc8xAVr0XX0JthoW+uP91eTQ3XpsAEQEAAYkBPQQYAQIACQUCUmEvTgIb',\n      'LgCoCRBKY2E6TW5AlJ0gBBkBAgAGBQJSYS9OAAoJEOCE90RsICyXuqIEANmmiRCA',\n      'SF7YK7PvFkieJNwzeK0V3F2lGX+uu6Y3Q/Zxdtwc4xR+me/CSBmsURyXTO29OWhP',\n      'GLszPH9zSJU9BdDi6v0yNprmFPX/1Ng0Abn/sCkwetvjxC1YIvTLFwtUL/7v6NS2',\n      'bZpsUxRTg9+cSrMWWSNjiY9qUKajm1tuzPDZXAUEAMNmAN3xXN/Kjyvj2OK2ck0X',\n      'W748sl/tc3qiKPMJ+0AkMF7Pjhmh9nxqE9+QCEl7qinFqqBLjuzgUhBU4QlwX1GD',\n      'AtNTq6ihLMD5v1d82ZC7tNatdlDMGWnIdvEMCv2GZcuIqDQ9rXWs49e7tq1NncLY',\n      'hz3tYjKhoFTKEIq3y3Pp',\n      '=h/aX',\n      '-----END PGP PUBLIC KEY BLOCK-----'].join('\\n');\n\n  var priv_key =\n      ['-----BEGIN PGP PRIVATE KEY BLOCK-----',\n      'Version: GnuPG v2.0.19 (GNU/Linux)',\n      'Type: RSA/RSA',\n      'Pwd: hello world',\n      '',\n      'lQH+BFJhL04BBADclrUEDDsm0PSZbQ6pml9FpzTyXiyCyDN+rMOsy9J300Oc10kt',\n      '/nyBej9vZSRcaW5VpNNj0iA+c1/w2FPf84zNsTzvDmuMaNHFUzky4/vkYuZra//3',\n      '+Ri7CF8RawSYQ/4IRbC9zqdBlzniyfQOW7Dp/LYe8eibnDSrmkQem0G0jwARAQAB',\n      '/gMDAu7L//czBpE40p1ZqO8K3k7UejemjsQqc7kOqnlDYd1Z6/3NEA/UM30Siipr',\n      'KjdIFY5+hp0hcs6EiiNq0PDfm/W2j+7HfrZ5kpeQVxDek4irezYZrl7JS2xezaLv',\n      'k0Fv/6fxasnFtjOM6Qbstu67s5Gpl9y06ZxbP3VpT62+Xeibn/swWrfiJjuGEEhM',\n      'bgnsMpHtzAz/L8y6KSzViG/05hBaqrvk3/GeEA6nE+o0+0a6r0LYLTemmq6FbaA1',\n      'PHo+x7k7oFcBFUUeSzgx78GckuPwqr2mNfeF+IuSRnrlpZl3kcbHASPAOfEkyMXS',\n      'sWGE7grCAjbyQyM3OEXTSyqnehvGS/1RdB6kDDxGwgE/QFbwNyEh6K4eaaAThW2j',\n      'IEEI0WEnRkPi9fXyxhFsCLSI1XhqTaq7iDNqJTxE+AX2b9ZuZXAxI3Tc/7++vEyL',\n      '3p18N/MB2kt1Wb1azmXWL2EKlT1BZ5yDaJuBQ8BhphM3tCRUZXN0IE1jVGVzdGlu',\n      'Z3RvbiA8dGVzdEBleGFtcGxlLmNvbT6IuQQTAQIAIwUCUmEvTgIbLwcLCQgHAwIB',\n      'BhUIAgkKCwQWAgMBAh4BAheAAAoJEEpjYTpNbkCUMAwD+gIK08qpEZSVas9qW+Ok',\n      '32wzNkwxe6PQgZwcyBqMQYZUcKagC8+89pMQQ5sKUGvpIgat42Tf1KLGPcvG4cDA',\n      'JZ6w2PYz9YHQqPh9LA+PAnV8m25TcGmKcKgvFUqQ3U53X/Y9sBP8HooRqfwwHcv9',\n      'pMgQmojmNbI4VHydRqIBePawnQH+BFJhL04BBADpH8+0EVolpPiOrXTKoBKTiyrB',\n      'UyxzodyJ8zmVJ3HMTEU/vidpQwzISwoc/ndDFMXQauq6xqBCD9m2BPQI3UdQzXnb',\n      'LsAI52nWCIqOkzM5NAKWoKhyXK9Y4UH4v9LAYQgl/stIISvCgG4mJ8lzzEBWvRdf',\n      'Qm2Ghb64/3V5NDdemwARAQAB/gMDAu7L//czBpE40iPcpLzL7GwBbWFhSWgSLy53',\n      'Md99Kxw3cApWCok2E8R9/4VS0490xKZIa5y2I/K8thVhqk96Z8Kbt7MRMC1WLHgC',\n      'qJvkeQCI6PrFM0PUIPLHAQtDJYKtaLXxYuexcAdKzZj3FHdtLNWCooK6n3vJlL1c',\n      'WjZcHJ1PH7USlj1jup4XfxsbziuysRUSyXkjn92GZLm+64vCIiwhqAYoizF2NHHG',\n      'hRTN4gQzxrxgkeVchl+ag7DkQUDANIIVI+A63JeLJgWJiH1fbYlwESByHW+zBFNt',\n      'qStjfIOhjrfNIc3RvsggbDdWQLcbxmLZj4sB0ydPSgRKoaUdRHJY0S4vp9ouKOtl',\n      '2au/P1BP3bhD0fDXl91oeheYth+MSmsJFDg/vZJzCJhFaQ9dp+2EnjN5auNCNbaI',\n      'beFJRHFf9cha8p3hh+AK54NRCT++B2MXYf+TPwqX88jYMBv8kk8vYUgo8128r1zQ',\n      'EzjviQE9BBgBAgAJBQJSYS9OAhsuAKgJEEpjYTpNbkCUnSAEGQECAAYFAlJhL04A',\n      'CgkQ4IT3RGwgLJe6ogQA2aaJEIBIXtgrs+8WSJ4k3DN4rRXcXaUZf667pjdD9nF2',\n      '3BzjFH6Z78JIGaxRHJdM7b05aE8YuzM8f3NIlT0F0OLq/TI2muYU9f/U2DQBuf+w',\n      'KTB62+PELVgi9MsXC1Qv/u/o1LZtmmxTFFOD35xKsxZZI2OJj2pQpqObW27M8Nlc',\n      'BQQAw2YA3fFc38qPK+PY4rZyTRdbvjyyX+1zeqIo8wn7QCQwXs+OGaH2fGoT35AI',\n      'SXuqKcWqoEuO7OBSEFThCXBfUYMC01OrqKEswPm/V3zZkLu01q12UMwZach28QwK',\n      '/YZly4ioND2tdazj17u2rU2dwtiHPe1iMqGgVMoQirfLc+k=',\n      '=lw5e',\n      '-----END PGP PRIVATE KEY BLOCK-----'].join('\\n');\n\n  var armored = \n     ['-----BEGIN PGP MESSAGE-----',\n      'Version: OpenPGP.js VERSION',\n      'Comment: http://openpgpjs.org',\n      '',\n      'wYwD4IT3RGwgLJcBA/wN8H9qUvsJwsarDOcsczk1L9Iif47jy+3vF6LcpSgA',\n      'DfKSARWvatyakvEJmuCNfcpzS8NUPnFA51p0VWmI7FnYG5LkPVUIHb4sN07G',\n      '9PeqhaayZIeNVvS6DoYuxiTG2sbDmyrv6MMSLivs7VcTAN6L7XXJRI2IumOS',\n      'x2WFgYNKANJFAQmen4R4IGf9nZDI7NJ4QHUlK1GENgLix9RzMK+Cri6p9Gx5',\n      '4MDV23jgCBWrUHfFYximgcXiUk0NfpVD3x6chcnKUKJv',\n      '=Eaix',\n      '-----END PGP MESSAGE-----'].join('\\n');\n\n\n  var plaintext = 'short message';\n\n  var key = openpgp.key.readArmored(pub_key);\n\n  var encrypted = openpgp.encryptMessage([key], plaintext);\n\n  var message = openpgp.message.readArmored(encrypted);\n\n  var privKey = openpgp.key.readArmored(priv_key);\n\n  privKey.decrypt('hello world');\n\n  var decrypted = openpgp.decryptMessage(privKey, message);\n\n  return new test_result('Encrypt plain text and afterwards decrypt leads to same result', plaintext == decrypted);\n\n});\n","module.exports=require('QjPZ1q');","var process=require(\"__browserify_process\");\nmodule.exports = {\n\ttests: [],\n\tregister: function(str_title, func_runtest) {\n\t\tthis.tests.push({ title: str_title, run: func_runtest });\n\t},\n\t\n\trun: function() {\n\t\tvar test = this.tests.shift();\n\n\t\tvar result = {\n\t\t\ttitle: test.title\n\t\t};\n\n\n\t\tresult.tests = test.run();\n\n\t\treturn result;\n\t},\n\n\trun_all: function() {\n\t\tvar passed = true;\n\n\t\twhile(this.tests.length > 0) {\n\t\t\tvar result = this.run();\n\n\t\t\tconsole.log('Test: ' + result.title);\n\n\t\t\tfor(var i in result.tests) {\n\n\t\t\t\tvar res = result.tests[i].result ?\n\t\t\t\t\t'SUCCESS' : 'FAILED';\n\n\t\t\t\tconsole.log(result.tests[i].description + ' ' + res);\n\n\t\t\t\tpassed = passed && result.tests[i].result;\n\t\t\t}\t\t\t\t\n\t\t}\n\n\t\tif(!passed) process.exit(1);\n\t},\n\t\t\n\tresult: function(str_description, boolean_result) {\n\t\tthis.description = str_description;\n\t\tthis.result = boolean_result;\n\t}\n}\n\n"]} +//@ sourceMappingURL=data:application/json;base64,{"version":3,"file":"generated.js","sources":["/home/toberndo/dev/node/lib/node_modules/browserify/node_modules/insert-module-globals/node_modules/process/browser.js","/home/toberndo/dev/openpgpjs-devel/src/compression/jxg.js","/home/toberndo/dev/openpgpjs-devel/src/config/config.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cfb.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/aes.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/blowfish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/cast5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/des.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/cipher/twofish.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/crypto.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/md5.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/ripe-md.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/hash/sha.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/pkcs1.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/dsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/elgamal.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/index.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/jsbn.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/public_key/rsa.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/random.js","/home/toberndo/dev/openpgpjs-devel/src/crypto/signature.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/armor.js","/home/toberndo/dev/openpgpjs-devel/src/encoding/base64.js","/home/toberndo/dev/openpgpjs-devel/src/enums.js","/home/toberndo/dev/openpgpjs-devel/src/index.js","/home/toberndo/dev/openpgpjs-devel/src/key.js","/home/toberndo/dev/openpgpjs-devel/src/keyring.js","/home/toberndo/dev/openpgpjs-devel/src/message.js","/home/toberndo/dev/openpgpjs-devel/src/openpgp.js","/home/toberndo/dev/openpgpjs-devel/src/packet/all_packets.js","/home/toberndo/dev/openpgpjs-devel/src/packet/compressed.js","/home/toberndo/dev/openpgpjs-devel/src/packet/index.js","/home/toberndo/dev/openpgpjs-devel/src/packet/literal.js","/home/toberndo/dev/openpgpjs-devel/src/packet/marker.js","/home/toberndo/dev/openpgpjs-devel/src/packet/one_pass_signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packet.js","/home/toberndo/dev/openpgpjs-devel/src/packet/packetlist.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_key_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/public_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/secret_subkey.js","/home/toberndo/dev/openpgpjs-devel/src/packet/signature.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_integrity_protected.js","/home/toberndo/dev/openpgpjs-devel/src/packet/sym_encrypted_session_key.js","/home/toberndo/dev/openpgpjs-devel/src/packet/symmetrically_encrypted.js","/home/toberndo/dev/openpgpjs-devel/src/packet/trust.js","/home/toberndo/dev/openpgpjs-devel/src/packet/user_attribute.js","/home/toberndo/dev/openpgpjs-devel/src/packet/userid.js","/home/toberndo/dev/openpgpjs-devel/src/type/keyid.js","/home/toberndo/dev/openpgpjs-devel/src/type/mpi.js","/home/toberndo/dev/openpgpjs-devel/src/type/s2k.js","/home/toberndo/dev/openpgpjs-devel/src/util/util.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/aes.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/blowfish.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/cast5.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/des.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/cipher/twofish.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/hash/md5.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/hash/ripemd.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/hash/sha.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/openpgp.crypto.js","/home/toberndo/dev/openpgpjs-devel/test/crypto/openpgp.sigcheck.js","/home/toberndo/dev/openpgpjs-devel/test/general/openpgp.basic.js","/home/toberndo/dev/openpgpjs-devel/test/test-all.js","/home/toberndo/dev/openpgpjs-devel/test/unit.js"],"names":[],"mappings":";AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3fA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5ZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5lCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;;ACLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzqDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACVA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClgBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpEA;AACA;AACA;AACA;;ACHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9VA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3JA;;;;;;;;;;;;;;;;;;;;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","sourcesContent":["// shim for using process in browser\n\nvar process = module.exports = {};\n\nprocess.nextTick = (function () {\n    var canSetImmediate = typeof window !== 'undefined'\n    && window.setImmediate;\n    var canPost = typeof window !== 'undefined'\n    && window.postMessage && window.addEventListener\n    ;\n\n    if (canSetImmediate) {\n        return function (f) { return window.setImmediate(f) };\n    }\n\n    if (canPost) {\n        var queue = [];\n        window.addEventListener('message', function (ev) {\n            if (ev.source === window && ev.data === 'process-tick') {\n                ev.stopPropagation();\n                if (queue.length > 0) {\n                    var fn = queue.shift();\n                    fn();\n                }\n            }\n        }, true);\n\n        return function nextTick(fn) {\n            queue.push(fn);\n            window.postMessage('process-tick', '*');\n        };\n    }\n\n    return function nextTick(fn) {\n        setTimeout(fn, 0);\n    };\n})();\n\nprocess.title = 'browser';\nprocess.browser = true;\nprocess.env = {};\nprocess.argv = [];\n\nprocess.binding = function (name) {\n    throw new Error('process.binding is not supported');\n}\n\n// TODO(shtylman)\nprocess.cwd = function () { return '/' };\nprocess.chdir = function (dir) {\n    throw new Error('process.chdir is not supported');\n};\n","JXG = {\n  exists: (function(undefined) {\n    return function(v) {\n      return !(v === undefined || v === null);\n    }\n  })()\n};\nJXG.decompress = function(str) {\n  return unescape((new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(str))).unzip()[0][0]);\n};\n/*\n    Copyright 2008-2012\n        Matthias Ehmann,\n        Michael Gerhaeuser,\n        Carsten Miller,\n        Bianca Valentin,\n        Alfred Wassermann,\n        Peter Wilfahrt\n\n    This file is part of JSXGraph.\n    \n    Dual licensed under the Apache License Version 2.0, or LGPL Version 3 licenses.\n\n    You should have received a copy of the GNU Lesser General Public License\n    along with JSXCompressor.  If not, see <http://www.gnu.org/licenses/>.\n    \n    You should have received a copy of the Apache License along with JSXCompressor.  \n    If not, see <http://www.apache.org/licenses/>.\n\n*/\n\n/**\n * @class Util class\n * @classdesc Utilities for uncompressing and base64 decoding\n * Class for gunzipping, unzipping and base64 decoding of files.\n * It is used for reading GEONExT, Geogebra and Intergeo files.\n *\n * Only Huffman codes are decoded in gunzip.\n * The code is based on the source code for gunzip.c by Pasi Ojala \n * {@link http://www.cs.tut.fi/~albert/Dev/gunzip/gunzip.c}\n * {@link http://www.cs.tut.fi/~albert}\n */\nJXG.Util = {};\n\n/**\n * Unzip zip files\n */\nJXG.Util.Unzip = function(barray) {\n  var outputArr = [],\n    output = \"\",\n    debug = false,\n    gpflags,\n    files = 0,\n    unzipped = [],\n    crc,\n    buf32k = new Array(32768),\n    bIdx = 0,\n    modeZIP = false,\n\n    CRC, SIZE,\n\n    bitReverse = [\n        0x00, 0x80, 0x40, 0xc0, 0x20, 0xa0, 0x60, 0xe0,\n        0x10, 0x90, 0x50, 0xd0, 0x30, 0xb0, 0x70, 0xf0,\n        0x08, 0x88, 0x48, 0xc8, 0x28, 0xa8, 0x68, 0xe8,\n        0x18, 0x98, 0x58, 0xd8, 0x38, 0xb8, 0x78, 0xf8,\n        0x04, 0x84, 0x44, 0xc4, 0x24, 0xa4, 0x64, 0xe4,\n        0x14, 0x94, 0x54, 0xd4, 0x34, 0xb4, 0x74, 0xf4,\n        0x0c, 0x8c, 0x4c, 0xcc, 0x2c, 0xac, 0x6c, 0xec,\n        0x1c, 0x9c, 0x5c, 0xdc, 0x3c, 0xbc, 0x7c, 0xfc,\n        0x02, 0x82, 0x42, 0xc2, 0x22, 0xa2, 0x62, 0xe2,\n        0x12, 0x92, 0x52, 0xd2, 0x32, 0xb2, 0x72, 0xf2,\n        0x0a, 0x8a, 0x4a, 0xca, 0x2a, 0xaa, 0x6a, 0xea,\n        0x1a, 0x9a, 0x5a, 0xda, 0x3a, 0xba, 0x7a, 0xfa,\n        0x06, 0x86, 0x46, 0xc6, 0x26, 0xa6, 0x66, 0xe6,\n        0x16, 0x96, 0x56, 0xd6, 0x36, 0xb6, 0x76, 0xf6,\n        0x0e, 0x8e, 0x4e, 0xce, 0x2e, 0xae, 0x6e, 0xee,\n        0x1e, 0x9e, 0x5e, 0xde, 0x3e, 0xbe, 0x7e, 0xfe,\n        0x01, 0x81, 0x41, 0xc1, 0x21, 0xa1, 0x61, 0xe1,\n        0x11, 0x91, 0x51, 0xd1, 0x31, 0xb1, 0x71, 0xf1,\n        0x09, 0x89, 0x49, 0xc9, 0x29, 0xa9, 0x69, 0xe9,\n        0x19, 0x99, 0x59, 0xd9, 0x39, 0xb9, 0x79, 0xf9,\n        0x05, 0x85, 0x45, 0xc5, 0x25, 0xa5, 0x65, 0xe5,\n        0x15, 0x95, 0x55, 0xd5, 0x35, 0xb5, 0x75, 0xf5,\n        0x0d, 0x8d, 0x4d, 0xcd, 0x2d, 0xad, 0x6d, 0xed,\n        0x1d, 0x9d, 0x5d, 0xdd, 0x3d, 0xbd, 0x7d, 0xfd,\n        0x03, 0x83, 0x43, 0xc3, 0x23, 0xa3, 0x63, 0xe3,\n        0x13, 0x93, 0x53, 0xd3, 0x33, 0xb3, 0x73, 0xf3,\n        0x0b, 0x8b, 0x4b, 0xcb, 0x2b, 0xab, 0x6b, 0xeb,\n        0x1b, 0x9b, 0x5b, 0xdb, 0x3b, 0xbb, 0x7b, 0xfb,\n        0x07, 0x87, 0x47, 0xc7, 0x27, 0xa7, 0x67, 0xe7,\n        0x17, 0x97, 0x57, 0xd7, 0x37, 0xb7, 0x77, 0xf7,\n        0x0f, 0x8f, 0x4f, 0xcf, 0x2f, 0xaf, 0x6f, 0xef,\n        0x1f, 0x9f, 0x5f, 0xdf, 0x3f, 0xbf, 0x7f, 0xff\n    ],\n\n    cplens = [\n        3, 4, 5, 6, 7, 8, 9, 10, 11, 13, 15, 17, 19, 23, 27, 31,\n        35, 43, 51, 59, 67, 83, 99, 115, 131, 163, 195, 227, 258, 0, 0\n    ],\n\n    cplext = [\n        0, 0, 0, 0, 0, 0, 0, 0, 1, 1, 1, 1, 2, 2, 2, 2,\n        3, 3, 3, 3, 4, 4, 4, 4, 5, 5, 5, 5, 0, 99, 99\n    ],\n    /* 99==invalid */\n\n    cpdist = [\n        0x0001, 0x0002, 0x0003, 0x0004, 0x0005, 0x0007, 0x0009, 0x000d,\n        0x0011, 0x0019, 0x0021, 0x0031, 0x0041, 0x0061, 0x0081, 0x00c1,\n        0x0101, 0x0181, 0x0201, 0x0301, 0x0401, 0x0601, 0x0801, 0x0c01,\n        0x1001, 0x1801, 0x2001, 0x3001, 0x4001, 0x6001\n    ],\n\n    cpdext = [\n        0, 0, 0, 0, 1, 1, 2, 2,\n        3, 3, 4, 4, 5, 5, 6, 6,\n        7, 7, 8, 8, 9, 9, 10, 10,\n        11, 11, 12, 12, 13, 13\n    ],\n\n    border = [16, 17, 18, 0, 8, 7, 9, 6, 10, 5, 11, 4, 12, 3, 13, 2, 14, 1, 15],\n\n    bA = barray,\n\n    bytepos = 0,\n    bitpos = 0,\n    bb = 1,\n    bits = 0,\n\n    NAMEMAX = 256,\n\n    nameBuf = [],\n\n    fileout;\n\n  function readByte() {\n    bits += 8;\n    if (bytepos < bA.length) {\n      //if (debug)\n      //    document.write(bytepos+\": \"+bA[bytepos]+\"<br>\");\n      return bA[bytepos++];\n    } else\n      return -1;\n  };\n\n  function byteAlign() {\n    bb = 1;\n  };\n\n  function readBit() {\n    var carry;\n    bits++;\n    carry = (bb & 1);\n    bb >>= 1;\n    if (bb == 0) {\n      bb = readByte();\n      carry = (bb & 1);\n      bb = (bb >> 1) | 0x80;\n    }\n    return carry;\n  };\n\n  function readBits(a) {\n    var res = 0,\n      i = a;\n\n    while (i--) {\n      res = (res << 1) | readBit();\n    }\n    if (a) {\n      res = bitReverse[res] >> (8 - a);\n    }\n    return res;\n  };\n\n  function flushBuffer() {\n    //document.write('FLUSHBUFFER:'+buf32k);\n    bIdx = 0;\n  };\n\n  function addBuffer(a) {\n    SIZE++;\n    //CRC=updcrc(a,crc);\n    buf32k[bIdx++] = a;\n    outputArr.push(String.fromCharCode(a));\n    //output+=String.fromCharCode(a);\n    if (bIdx == 0x8000) {\n      //document.write('ADDBUFFER:'+buf32k);\n      bIdx = 0;\n    }\n  };\n\n  function HufNode() {\n    this.b0 = 0;\n    this.b1 = 0;\n    this.jump = null;\n    this.jumppos = -1;\n  };\n\n  var LITERALS = 288;\n\n  var literalTree = new Array(LITERALS);\n  var distanceTree = new Array(32);\n  var treepos = 0;\n  var Places = null;\n  var Places2 = null;\n\n  var impDistanceTree = new Array(64);\n  var impLengthTree = new Array(64);\n\n  var len = 0;\n  var fpos = new Array(17);\n  fpos[0] = 0;\n  var flens;\n  var fmax;\n\n  function IsPat() {\n    while (1) {\n      if (fpos[len] >= fmax)\n        return -1;\n      if (flens[fpos[len]] == len)\n        return fpos[len]++;\n      fpos[len]++;\n    }\n  };\n\n  function Rec() {\n    var curplace = Places[treepos];\n    var tmp;\n    if (debug)\n      document.write(\"<br>len:\" + len + \" treepos:\" + treepos);\n    if (len == 17) { //war 17\n      return -1;\n    }\n    treepos++;\n    len++;\n\n    tmp = IsPat();\n    if (debug)\n      document.write(\"<br>IsPat \" + tmp);\n    if (tmp >= 0) {\n      curplace.b0 = tmp; /* leaf cell for 0-bit */\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n    } else {\n      /* Not a Leaf cell */\n      curplace.b0 = 0x8000;\n      if (debug)\n        document.write(\"<br>b0 \" + curplace.b0);\n      if (Rec())\n        return -1;\n    }\n    tmp = IsPat();\n    if (tmp >= 0) {\n      curplace.b1 = tmp; /* leaf cell for 1-bit */\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = null; /* Just for the display routine */\n    } else {\n      /* Not a Leaf cell */\n      curplace.b1 = 0x8000;\n      if (debug)\n        document.write(\"<br>b1 \" + curplace.b1);\n      curplace.jump = Places[treepos];\n      curplace.jumppos = treepos;\n      if (Rec())\n        return -1;\n    }\n    len--;\n    return 0;\n  };\n\n  function CreateTree(currentTree, numval, lengths, show) {\n    var i;\n    /* Create the Huffman decode tree/table */\n    //document.write(\"<br>createtree<br>\");\n    if (debug)\n      document.write(\"currentTree \" + currentTree + \" numval \" + numval + \" lengths \" + lengths + \" show \" + show);\n    Places = currentTree;\n    treepos = 0;\n    flens = lengths;\n    fmax = numval;\n    for (i = 0; i < 17; i++)\n      fpos[i] = 0;\n    len = 0;\n    if (Rec()) {\n      //fprintf(stderr, \"invalid huffman tree\\n\");\n      if (debug)\n        alert(\"invalid huffman tree\\n\");\n      return -1;\n    }\n    if (debug) {\n      document.write('<br>Tree: ' + Places.length);\n      for (var a = 0; a < 32; a++) {\n        document.write(\"Places[\" + a + \"].b0=\" + Places[a].b0 + \"<br>\");\n        document.write(\"Places[\" + a + \"].b1=\" + Places[a].b1 + \"<br>\");\n      }\n    }\n\n    /*if(show) {\n            var tmp;\n            for(tmp=currentTree;tmp<Places;tmp++) {\n                fprintf(stdout, \"0x%03x  0x%03x (0x%04x)\",tmp-currentTree, tmp->jump?tmp->jump-currentTree:0,(tmp->jump?tmp->jump-currentTree:0)*6+0xcf0);\n                if(!(tmp.b0 & 0x8000)) {\n                    //fprintf(stdout, \"  0x%03x (%c)\", tmp->b0,(tmp->b0<256 && isprint(tmp->b0))?tmp->b0:'�');\n                }\n                if(!(tmp.b1 & 0x8000)) {\n                    if((tmp.b0 & 0x8000))\n                        fprintf(stdout, \"           \");\n                    fprintf(stdout, \"  0x%03x (%c)\", tmp->b1,(tmp->b1<256 && isprint(tmp->b1))?tmp->b1:'�');\n                }\n                fprintf(stdout, \"\\n\");\n            }\n        }*/\n    return 0;\n  };\n\n  function DecodeValue(currentTree) {\n    var len, i,\n      xtreepos = 0,\n      X = currentTree[xtreepos],\n      b;\n\n    /* decode one symbol of the data */\n    while (1) {\n      b = readBit();\n      if (debug)\n        document.write(\"b=\" + b);\n      if (b) {\n        if (!(X.b1 & 0x8000)) {\n          if (debug)\n            document.write(\"ret1\");\n          return X.b1; /* If leaf node, return data */\n        }\n        X = X.jump;\n        len = currentTree.length;\n        for (i = 0; i < len; i++) {\n          if (currentTree[i] === X) {\n            xtreepos = i;\n            break;\n          }\n        }\n        //xtreepos++;\n      } else {\n        if (!(X.b0 & 0x8000)) {\n          if (debug)\n            document.write(\"ret2\");\n          return X.b0; /* If leaf node, return data */\n        }\n        //X++; //??????????????????\n        xtreepos++;\n        X = currentTree[xtreepos];\n      }\n    }\n    if (debug)\n      document.write(\"ret3\");\n    return -1;\n  };\n\n  function DeflateLoop() {\n    var last, c, type, i, len;\n\n    do {\n      /*if((last = readBit())){\n            fprintf(errfp, \"Last Block: \");\n        } else {\n            fprintf(errfp, \"Not Last Block: \");\n        }*/\n      last = readBit();\n      type = readBits(2);\n      switch (type) {\n        case 0:\n          if (debug)\n            alert(\"Stored\\n\");\n          break;\n        case 1:\n          if (debug)\n            alert(\"Fixed Huffman codes\\n\");\n          break;\n        case 2:\n          if (debug)\n            alert(\"Dynamic Huffman codes\\n\");\n          break;\n        case 3:\n          if (debug)\n            alert(\"Reserved block type!!\\n\");\n          break;\n        default:\n          if (debug)\n            alert(\"Unexpected value %d!\\n\", type);\n          break;\n      }\n\n      if (type == 0) {\n        var blockLen, cSum;\n\n        // Stored \n        byteAlign();\n        blockLen = readByte();\n        blockLen |= (readByte() << 8);\n\n        cSum = readByte();\n        cSum |= (readByte() << 8);\n\n        if (((blockLen ^ ~cSum) & 0xffff)) {\n          document.write(\"BlockLen checksum mismatch\\n\");\n        }\n        while (blockLen--) {\n          c = readByte();\n          addBuffer(c);\n        }\n      } else if (type == 1) {\n        var j;\n\n        /* Fixed Huffman tables -- fixed decode routine */\n        while (1) {\n          /*\n                256    0000000        0\n                :   :     :\n                279    0010111        23\n                0   00110000    48\n                :    :      :\n                143    10111111    191\n                280 11000000    192\n                :    :      :\n                287 11000111    199\n                144    110010000    400\n                :    :       :\n                255    111111111    511\n    \n                Note the bit order!\n                */\n\n          j = (bitReverse[readBits(7)] >> 1);\n          if (j > 23) {\n            j = (j << 1) | readBit(); /* 48..255 */\n\n            if (j > 199) { /* 200..255 */\n              j -= 128; /*  72..127 */\n              j = (j << 1) | readBit(); /* 144..255 << */\n            } else { /*  48..199 */\n              j -= 48; /*   0..151 */\n              if (j > 143) {\n                j = j + 136; /* 280..287 << */\n                /*   0..143 << */\n              }\n            }\n          } else { /*   0..23 */\n            j += 256; /* 256..279 << */\n          }\n          if (j < 256) {\n            addBuffer(j);\n            //document.write(\"out:\"+String.fromCharCode(j));\n            /*fprintf(errfp, \"@%d %02x\\n\", SIZE, j);*/\n          } else if (j == 256) {\n            /* EOF */\n            break;\n          } else {\n            var len, dist;\n\n            j -= 256 + 1; /* bytes + EOF */\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = bitReverse[readBits(5)] >> 3;\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n\n            /*fprintf(errfp, \"@%d (l%02x,d%04x)\\n\", SIZE, len, dist);*/\n            for (j = 0; j < len; j++) {\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          }\n        } // while\n      } else if (type == 2) {\n        var j, n, literalCodes, distCodes, lenCodes;\n        var ll = new Array(288 + 32); // \"static\" just to preserve stack\n\n        // Dynamic Huffman tables \n\n        literalCodes = 257 + readBits(5);\n        distCodes = 1 + readBits(5);\n        lenCodes = 4 + readBits(4);\n        //document.write(\"<br>param: \"+literalCodes+\" \"+distCodes+\" \"+lenCodes+\"<br>\");\n        for (j = 0; j < 19; j++) {\n          ll[j] = 0;\n        }\n\n        // Get the decode tree code lengths\n\n        //document.write(\"<br>\");\n        for (j = 0; j < lenCodes; j++) {\n          ll[border[j]] = readBits(3);\n          //document.write(ll[border[j]]+\" \");\n        }\n        //fprintf(errfp, \"\\n\");\n        //document.write('<br>ll:'+ll);\n        len = distanceTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        if (CreateTree(distanceTree, 19, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug) {\n          document.write(\"<br>distanceTree\");\n          for (var a = 0; a < distanceTree.length; a++) {\n            document.write(\"<br>\" + distanceTree[a].b0 + \" \" + distanceTree[a].b1 + \" \" + distanceTree[a].jump + \" \" +\n              distanceTree[a].jumppos);\n            /*if (distanceTree[a].jumppos!=-1)\n                    \tdocument.write(\" \"+distanceTree[a].jump.b0+\" \"+distanceTree[a].jump.b1);\n                \t*/\n          }\n        }\n        //document.write('<BR>tree created');\n\n        //read in literal and distance code lengths\n        n = literalCodes + distCodes;\n        i = 0;\n        var z = -1;\n        if (debug)\n          document.write(\"<br>n=\" + n + \" bits: \" + bits + \"<br>\");\n        while (i < n) {\n          z++;\n          j = DecodeValue(distanceTree);\n          if (debug)\n            document.write(\"<br>\" + z + \" i:\" + i + \" decode: \" + j + \"    bits \" + bits + \"<br>\");\n          if (j < 16) { // length of code in bits (0..15)\n            ll[i++] = j;\n          } else if (j == 16) { // repeat last length 3 to 6 times \n            var l;\n            j = 3 + readBits(2);\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            l = i ? ll[i - 1] : 0;\n            while (j--) {\n              ll[i++] = l;\n            }\n          } else {\n            if (j == 17) { // 3 to 10 zero length codes\n              j = 3 + readBits(3);\n            } else { // j == 18: 11 to 138 zero length codes \n              j = 11 + readBits(7);\n            }\n            if (i + j > n) {\n              flushBuffer();\n              return 1;\n            }\n            while (j--) {\n              ll[i++] = 0;\n            }\n          }\n        }\n        /*for(j=0; j<literalCodes+distCodes; j++) {\n                //fprintf(errfp, \"%d \", ll[j]);\n                if ((j&7)==7)\n                    fprintf(errfp, \"\\n\");\n            }\n            fprintf(errfp, \"\\n\");*/\n        // Can overwrite tree decode tree as it is not used anymore\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          literalTree[i] = new HufNode();\n        if (CreateTree(literalTree, literalCodes, ll, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        len = literalTree.length;\n        for (i = 0; i < len; i++)\n          distanceTree[i] = new HufNode();\n        var ll2 = new Array();\n        for (i = literalCodes; i < ll.length; i++) {\n          ll2[i - literalCodes] = ll[i];\n        }\n        if (CreateTree(distanceTree, distCodes, ll2, 0)) {\n          flushBuffer();\n          return 1;\n        }\n        if (debug)\n          document.write(\"<br>literalTree\");\n        outer: while (1) {\n          j = DecodeValue(literalTree);\n          if (j >= 256) { // In C64: if carry set\n            var len, dist;\n            j -= 256;\n            if (j == 0) {\n              // EOF\n              break;\n            }\n            j--;\n            len = readBits(cplext[j]) + cplens[j];\n\n            j = DecodeValue(distanceTree);\n            if (cpdext[j] > 8) {\n              dist = readBits(8);\n              dist |= (readBits(cpdext[j] - 8) << 8);\n            } else {\n              dist = readBits(cpdext[j]);\n            }\n            dist += cpdist[j];\n            while (len--) {\n              if (bIdx - dist < 0) {\n                break outer;\n              }\n              var c = buf32k[(bIdx - dist) & 0x7fff];\n              addBuffer(c);\n            }\n          } else {\n            addBuffer(j);\n          }\n        }\n      }\n    } while (!last);\n    flushBuffer();\n\n    byteAlign();\n    return 0;\n  };\n\n  JXG.Util.Unzip.prototype.unzipFile = function(name) {\n    var i;\n    this.unzip();\n    //alert(unzipped[0][1]);\n    for (i = 0; i < unzipped.length; i++) {\n      if (unzipped[i][1] == name) {\n        return unzipped[i][0];\n      }\n    }\n\n  };\n\n  JXG.Util.Unzip.prototype.deflate = function() {\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    DeflateLoop();\n    if (debug)\n      alert(outputArr.join(''));\n    unzipped[files] = new Array(2);\n    unzipped[files][0] = outputArr.join('');\n    unzipped[files][1] = \"DEFLATE\";\n    files++;\n    return unzipped;\n  }\n\n  JXG.Util.Unzip.prototype.unzip = function() {\n    //convertToByteArray(input);\n    if (debug)\n      alert(bA);\n    /*for (i=0;i<bA.length*8;i++){\n\t\tdocument.write(readBit());\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}*/\n    /*for (i=0;i<bA.length;i++){\n\t\tdocument.write(readByte()+\" \");\n\t\tif ((i+1)%8==0)\n\t\t\tdocument.write(\" \");\n\t}\n\tfor (i=0;i<bA.length;i++){\n\t\tdocument.write(bA[i]+\" \");\n\t\tif ((i+1)%16==0)\n\t\t\tdocument.write(\"<br>\");\n\t}\t\n\t*/\n    //alert(bA);\n    nextFile();\n    return unzipped;\n  };\n\n  function nextFile() {\n    if (debug)\n      alert(\"NEXTFILE\");\n    outputArr = [];\n    var tmp = [];\n    modeZIP = false;\n    tmp[0] = readByte();\n    tmp[1] = readByte();\n    if (debug)\n      alert(\"type: \" + tmp[0] + \" \" + tmp[1]);\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"da\", 16)) { //GZIP\n      if (debug)\n        alert(\"GEONExT-GZIP\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"geonext.gxt\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"78\", 16) && tmp[1] == parseInt(\"9c\", 16)) { //ZLIB\n      if (debug)\n        alert(\"ZLIB\");\n      DeflateLoop();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"ZLIB\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"1f\", 16) && tmp[1] == parseInt(\"8b\", 16)) { //GZIP\n      if (debug)\n        alert(\"GZIP\");\n      //DeflateLoop();\n      skipdir();\n      if (debug)\n        alert(outputArr.join(''));\n      unzipped[files] = new Array(2);\n      unzipped[files][0] = outputArr.join('');\n      unzipped[files][1] = \"file\";\n      files++;\n    }\n    if (tmp[0] == parseInt(\"50\", 16) && tmp[1] == parseInt(\"4b\", 16)) { //ZIP\n      modeZIP = true;\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n      if (tmp[2] == parseInt(\"3\", 16) && tmp[3] == parseInt(\"4\", 16)) {\n        //MODE_ZIP\n        tmp[0] = readByte();\n        tmp[1] = readByte();\n        if (debug)\n          alert(\"ZIP-Version: \" + tmp[1] + \" \" + tmp[0] / 10 + \".\" + tmp[0] % 10);\n\n        gpflags = readByte();\n        gpflags |= (readByte() << 8);\n        if (debug)\n          alert(\"gpflags: \" + gpflags);\n\n        var method = readByte();\n        method |= (readByte() << 8);\n        if (debug)\n          alert(\"method: \" + method);\n\n        readByte();\n        readByte();\n        readByte();\n        readByte();\n\n        var crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n\n        var compSize = readByte();\n        compSize |= (readByte() << 8);\n        compSize |= (readByte() << 16);\n        compSize |= (readByte() << 24);\n\n        var size = readByte();\n        size |= (readByte() << 8);\n        size |= (readByte() << 16);\n        size |= (readByte() << 24);\n\n        if (debug)\n          alert(\"local CRC: \" + crc + \"\\nlocal Size: \" + size + \"\\nlocal CompSize: \" + compSize);\n\n        var filelen = readByte();\n        filelen |= (readByte() << 8);\n\n        var extralen = readByte();\n        extralen |= (readByte() << 8);\n\n        if (debug)\n          alert(\"filelen \" + filelen);\n        i = 0;\n        nameBuf = [];\n        while (filelen--) {\n          var c = readByte();\n          if (c == \"/\" | c == \":\") {\n            i = 0;\n          } else if (i < NAMEMAX - 1)\n            nameBuf[i++] = String.fromCharCode(c);\n        }\n        if (debug)\n          alert(\"nameBuf: \" + nameBuf);\n\n        //nameBuf[i] = \"\\0\";\n        if (!fileout)\n          fileout = nameBuf;\n\n        var i = 0;\n        while (i < extralen) {\n          c = readByte();\n          i++;\n        }\n\n        CRC = 0xffffffff;\n        SIZE = 0;\n\n        if (size = 0 && fileOut.charAt(fileout.length - 1) == \"/\") {\n          //skipdir\n          if (debug)\n            alert(\"skipdir\");\n        }\n        if (method == 8) {\n          DeflateLoop();\n          if (debug)\n            alert(outputArr.join(''));\n          unzipped[files] = new Array(2);\n          unzipped[files][0] = outputArr.join('');\n          unzipped[files][1] = nameBuf.join('');\n          files++;\n          //return outputArr.join('');\n        }\n        skipdir();\n      }\n    }\n  };\n\n  function skipdir() {\n    var crc,\n      tmp = [],\n      compSize, size, os, i, c;\n\n    if ((gpflags & 8)) {\n      tmp[0] = readByte();\n      tmp[1] = readByte();\n      tmp[2] = readByte();\n      tmp[3] = readByte();\n\n      if (tmp[0] == parseInt(\"50\", 16) &&\n        tmp[1] == parseInt(\"4b\", 16) &&\n        tmp[2] == parseInt(\"07\", 16) &&\n        tmp[3] == parseInt(\"08\", 16)) {\n        crc = readByte();\n        crc |= (readByte() << 8);\n        crc |= (readByte() << 16);\n        crc |= (readByte() << 24);\n      } else {\n        crc = tmp[0] | (tmp[1] << 8) | (tmp[2] << 16) | (tmp[3] << 24);\n      }\n\n      compSize = readByte();\n      compSize |= (readByte() << 8);\n      compSize |= (readByte() << 16);\n      compSize |= (readByte() << 24);\n\n      size = readByte();\n      size |= (readByte() << 8);\n      size |= (readByte() << 16);\n      size |= (readByte() << 24);\n\n      if (debug)\n        alert(\"CRC:\");\n    }\n\n    if (modeZIP)\n      nextFile();\n\n    tmp[0] = readByte();\n    if (tmp[0] != 8) {\n      if (debug)\n        alert(\"Unknown compression method!\");\n      return 0;\n    }\n\n    gpflags = readByte();\n    if (debug) {\n      if ((gpflags & ~(parseInt(\"1f\", 16))))\n        alert(\"Unknown flags set!\");\n    }\n\n    readByte();\n    readByte();\n    readByte();\n    readByte();\n\n    readByte();\n    os = readByte();\n\n    if ((gpflags & 4)) {\n      tmp[0] = readByte();\n      tmp[2] = readByte();\n      len = tmp[0] + 256 * tmp[1];\n      if (debug)\n        alert(\"Extra field size: \" + len);\n      for (i = 0; i < len; i++)\n        readByte();\n    }\n\n    if ((gpflags & 8)) {\n      i = 0;\n      nameBuf = [];\n      while (c = readByte()) {\n        if (c == \"7\" || c == \":\")\n          i = 0;\n        if (i < NAMEMAX - 1)\n          nameBuf[i++] = c;\n      }\n      //nameBuf[i] = \"\\0\";\n      if (debug)\n        alert(\"original file name: \" + nameBuf);\n    }\n\n    if ((gpflags & 16)) {\n      while (c = readByte()) {\n        //FILE COMMENT\n      }\n    }\n\n    if ((gpflags & 2)) {\n      readByte();\n      readByte();\n    }\n\n    DeflateLoop();\n\n    crc = readByte();\n    crc |= (readByte() << 8);\n    crc |= (readByte() << 16);\n    crc |= (readByte() << 24);\n\n    size = readByte();\n    size |= (readByte() << 8);\n    size |= (readByte() << 16);\n    size |= (readByte() << 24);\n\n    if (modeZIP)\n      nextFile();\n\n  };\n\n};\n\n/**\n *  Base64 encoding / decoding\n *  {@link http://www.webtoolkit.info/}\n */\nJXG.Util.Base64 = {\n\n  // private property\n  _keyStr: \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=\",\n\n  // public method for encoding\n  encode: function(input) {\n    var output = [],\n      chr1, chr2, chr3, enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = JXG.Util.Base64._utf8_encode(input);\n\n    while (i < input.length) {\n\n      chr1 = input.charCodeAt(i++);\n      chr2 = input.charCodeAt(i++);\n      chr3 = input.charCodeAt(i++);\n\n      enc1 = chr1 >> 2;\n      enc2 = ((chr1 & 3) << 4) | (chr2 >> 4);\n      enc3 = ((chr2 & 15) << 2) | (chr3 >> 6);\n      enc4 = chr3 & 63;\n\n      if (isNaN(chr2)) {\n        enc3 = enc4 = 64;\n      } else if (isNaN(chr3)) {\n        enc4 = 64;\n      }\n\n      output.push([this._keyStr.charAt(enc1),\n          this._keyStr.charAt(enc2),\n          this._keyStr.charAt(enc3),\n          this._keyStr.charAt(enc4)\n      ].join(''));\n    }\n\n    return output.join('');\n  },\n\n  // public method for decoding\n  decode: function(input, utf8) {\n    var output = [],\n      chr1, chr2, chr3,\n      enc1, enc2, enc3, enc4,\n      i = 0;\n\n    input = input.replace(/[^A-Za-z0-9\\+\\/\\=]/g, \"\");\n\n    while (i < input.length) {\n\n      enc1 = this._keyStr.indexOf(input.charAt(i++));\n      enc2 = this._keyStr.indexOf(input.charAt(i++));\n      enc3 = this._keyStr.indexOf(input.charAt(i++));\n      enc4 = this._keyStr.indexOf(input.charAt(i++));\n\n      chr1 = (enc1 << 2) | (enc2 >> 4);\n      chr2 = ((enc2 & 15) << 4) | (enc3 >> 2);\n      chr3 = ((enc3 & 3) << 6) | enc4;\n\n      output.push(String.fromCharCode(chr1));\n\n      if (enc3 != 64) {\n        output.push(String.fromCharCode(chr2));\n      }\n      if (enc4 != 64) {\n        output.push(String.fromCharCode(chr3));\n      }\n    }\n\n    output = output.join('');\n\n    if (utf8) {\n      output = JXG.Util.Base64._utf8_decode(output);\n    }\n    return output;\n\n  },\n\n  // private method for UTF-8 encoding\n  _utf8_encode: function(string) {\n    string = string.replace(/\\r\\n/g, \"\\n\");\n    var utftext = \"\";\n\n    for (var n = 0; n < string.length; n++) {\n\n      var c = string.charCodeAt(n);\n\n      if (c < 128) {\n        utftext += String.fromCharCode(c);\n      } else if ((c > 127) && (c < 2048)) {\n        utftext += String.fromCharCode((c >> 6) | 192);\n        utftext += String.fromCharCode((c & 63) | 128);\n      } else {\n        utftext += String.fromCharCode((c >> 12) | 224);\n        utftext += String.fromCharCode(((c >> 6) & 63) | 128);\n        utftext += String.fromCharCode((c & 63) | 128);\n      }\n\n    }\n\n    return utftext;\n  },\n\n  // private method for UTF-8 decoding\n  _utf8_decode: function(utftext) {\n    var string = [],\n      i = 0,\n      c = 0,\n      c2 = 0,\n      c3 = 0;\n\n    while (i < utftext.length) {\n      c = utftext.charCodeAt(i);\n      if (c < 128) {\n        string.push(String.fromCharCode(c));\n        i++;\n      } else if ((c > 191) && (c < 224)) {\n        c2 = utftext.charCodeAt(i + 1);\n        string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n        i += 2;\n      } else {\n        c2 = utftext.charCodeAt(i + 1);\n        c3 = utftext.charCodeAt(i + 2);\n        string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n        i += 3;\n      }\n    }\n    return string.join('');\n  },\n\n  _destrip: function(stripped, wrap) {\n    var lines = [],\n      lineno, i,\n      destripped = [];\n\n    if (wrap == null)\n      wrap = 76;\n\n    stripped.replace(/ /g, \"\");\n    lineno = stripped.length / wrap;\n    for (i = 0; i < lineno; i++)\n      lines[i] = stripped.substr(i * wrap, wrap);\n    if (lineno != stripped.length / wrap)\n      lines[lines.length] = stripped.substr(lineno * wrap, stripped.length - (lineno * wrap));\n\n    for (i = 0; i < lines.length; i++)\n      destripped.push(lines[i]);\n    return destripped.join('\\n');\n  },\n\n  decodeAsArray: function(input) {\n    var dec = this.decode(input),\n      ar = [],\n      i;\n    for (i = 0; i < dec.length; i++) {\n      ar[i] = dec.charCodeAt(i);\n    }\n    return ar;\n  },\n\n  decodeGEONExT: function(input) {\n    return decodeAsArray(destrip(input), false);\n  }\n};\n\n/**\n * @private\n */\nJXG.Util.asciiCharCodeAt = function(str, i) {\n  var c = str.charCodeAt(i);\n  if (c > 255) {\n    switch (c) {\n      case 8364:\n        c = 128;\n        break;\n      case 8218:\n        c = 130;\n        break;\n      case 402:\n        c = 131;\n        break;\n      case 8222:\n        c = 132;\n        break;\n      case 8230:\n        c = 133;\n        break;\n      case 8224:\n        c = 134;\n        break;\n      case 8225:\n        c = 135;\n        break;\n      case 710:\n        c = 136;\n        break;\n      case 8240:\n        c = 137;\n        break;\n      case 352:\n        c = 138;\n        break;\n      case 8249:\n        c = 139;\n        break;\n      case 338:\n        c = 140;\n        break;\n      case 381:\n        c = 142;\n        break;\n      case 8216:\n        c = 145;\n        break;\n      case 8217:\n        c = 146;\n        break;\n      case 8220:\n        c = 147;\n        break;\n      case 8221:\n        c = 148;\n        break;\n      case 8226:\n        c = 149;\n        break;\n      case 8211:\n        c = 150;\n        break;\n      case 8212:\n        c = 151;\n        break;\n      case 732:\n        c = 152;\n        break;\n      case 8482:\n        c = 153;\n        break;\n      case 353:\n        c = 154;\n        break;\n      case 8250:\n        c = 155;\n        break;\n      case 339:\n        c = 156;\n        break;\n      case 382:\n        c = 158;\n        break;\n      case 376:\n        c = 159;\n        break;\n      default:\n        break;\n    }\n  }\n  return c;\n};\n\n/**\n * Decoding string into utf-8\n * @param {String} string to decode\n * @return {String} utf8 decoded string\n */\nJXG.Util.utf8Decode = function(utftext) {\n  var string = [];\n  var i = 0;\n  var c = 0,\n    c1 = 0,\n    c2 = 0,\n    c3;\n  if (!JXG.exists(utftext)) return '';\n\n  while (i < utftext.length) {\n    c = utftext.charCodeAt(i);\n\n    if (c < 128) {\n      string.push(String.fromCharCode(c));\n      i++;\n    } else if ((c > 191) && (c < 224)) {\n      c2 = utftext.charCodeAt(i + 1);\n      string.push(String.fromCharCode(((c & 31) << 6) | (c2 & 63)));\n      i += 2;\n    } else {\n      c2 = utftext.charCodeAt(i + 1);\n      c3 = utftext.charCodeAt(i + 2);\n      string.push(String.fromCharCode(((c & 15) << 12) | ((c2 & 63) << 6) | (c3 & 63)));\n      i += 3;\n    }\n  };\n  return string.join('');\n};\n\n/**\n * Generate a random uuid.\n * http://www.broofa.com\n * mailto:robert@broofa.com\n *\n * Copyright (c) 2010 Robert Kieffer\n * Dual licensed under the MIT and GPL licenses.\n *\n * EXAMPLES:\n *   >>> Math.uuid()\n *   \"92329D39-6F5C-4520-ABFC-AAB64544E172\"\n */\nJXG.Util.genUUID = function() {\n  // Private array of chars to use\n  var chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split(''),\n    uuid = new Array(36),\n    rnd = 0,\n    r;\n\n  for (var i = 0; i < 36; i++) {\n    if (i == 8 || i == 13 || i == 18 || i == 23) {\n      uuid[i] = '-';\n    } else if (i == 14) {\n      uuid[i] = '4';\n    } else {\n      if (rnd <= 0x02) rnd = 0x2000000 + (Math.random() * 0x1000000) | 0;\n      r = rnd & 0xf;\n      rnd = rnd >> 4;\n      uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r];\n    }\n  }\n\n  return uuid.join('');\n};\n\n\nmodule.exports = JXG;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js');\n\n/**\n *\n * This object contains configuration values and implements\n * storing and retrieving configuration them from HTML5 local storage.\n *\n * This object can be accessed after calling openpgp.init()\n * using openpgp.config\n * Stored config parameters can be accessed using\n * openpgp.config.config\n * @class\n * @classdesc Implementation of the GPG4Browsers config object\n */\nvar config = function() {\n  /**\n   * @property {Integer} prefer_hash_algorithm\n   * @property {Integer} encryption_cipher\n   * @property {Integer} compression\n   * @property {Boolean} show_version\n   * @property {Boolean} show_comment\n   * @property {Boolean} integrity_protect\n   * @property {String} keyserver\n   */\n  this.prefer_hash_algorithm = enums.hash.sha256;\n  this.encryption_cipher = enums.symmetric.aes256;\n  this.compression = enums.compression.zip;\n  this.show_version = true;\n  this.show_comment = true;\n  this.integrity_protect = true;\n  this.keyserver = \"keyserver.linux.it\"; // \"pgp.mit.edu:11371\"\n\n  this.versionstring = \"OpenPGP.js VERSION\";\n  this.commentstring = \"http://openpgpjs.org\";\n\n  /**\n   * If enabled, debug messages will be printed\n   */\n  this.debug = false;\n\n};\n\nmodule.exports = new config();\n","// Modified by Recurity Labs GmbH \n\n// modified version of http://www.hanewin.net/encrypt/PGdecode.js:\n\n/* OpenPGP encryption using RSA/AES\n * Copyright 2005-2006 Herbert Hanewinkel, www.haneWIN.de\n * version 2.0, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../util'),\n  cipher = require('./cipher');\n\nmodule.exports = {\n\n  /**\n   * An array of bytes, that is integers with values from 0 to 255\n   * @typedef {(Array|Uint8Array)} openpgp_byte_array\n   */\n\n  /**\n   * Block cipher function\n   * @callback openpgp_cipher_block_fn\n   * @param {openpgp_byte_array} block A block to perform operations on\n   * @param {openpgp_byte_array} key to use in encryption/decryption\n   * @return {openpgp_byte_array} Encrypted/decrypted block\n   */\n\n\n  // --------------------------------------\n  /**\n   * This function encrypts a given with the specified prefixrandom \n   * using the specified blockcipher to encrypt a message\n   * @param {String} prefixrandom random bytes of block_size length provided \n   *  as a string to be used in prefixing the data\n   * @param {openpgp_cipher_block_fn} blockcipherfn the algorithm encrypt function to encrypt\n   *  data in one block_size encryption. \n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext data to be encrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to encrypt the data. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Encryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the encrypted data\n   */\n  encrypt: function(prefixrandom, cipherfn, plaintext, key, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var FR = new Array(block_size);\n    var FRE = new Array(block_size);\n\n    prefixrandom = prefixrandom + prefixrandom.charAt(block_size - 2) + prefixrandom.charAt(block_size - 1);\n    util.print_debug(\"prefixrandom:\" + util.hexstrdump(prefixrandom));\n    var ciphertext = \"\";\n    // 1.  The feedback register (FR) is set to the IV, which is all zeros.\n    for (var i = 0; i < block_size; i++) FR[i] = 0;\n\n    // 2.  FR is encrypted to produce FRE (FR Encrypted).  This is the\n    //     encryption of an all-zero value.\n    FRE = cipherfn.encrypt(FR);\n    // 3.  FRE is xored with the first BS octets of random data prefixed to\n    //     the plaintext to produce C[1] through C[BS], the first BS octets\n    //     of ciphertext.\n    for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ prefixrandom.charCodeAt(i));\n\n    // 4.  FR is loaded with C[1] through C[BS].\n    for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n\n    // 5.  FR is encrypted to produce FRE, the encryption of the first BS\n    // \t   octets of ciphertext.\n    FRE = cipherfn.encrypt(FR);\n\n    // 6.  The left two octets of FRE get xored with the next two octets of\n    //     data that were prefixed to the plaintext.  This produces C[BS+1]\n    //     and C[BS+2], the next two octets of ciphertext.\n    ciphertext += String.fromCharCode(FRE[0] ^ prefixrandom.charCodeAt(block_size));\n    ciphertext += String.fromCharCode(FRE[1] ^ prefixrandom.charCodeAt(block_size + 1));\n\n    if (resync) {\n      // 7.  (The resync step) FR is loaded with C3-C10.\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i + 2);\n    } else {\n      for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(i);\n    }\n    // 8.  FR is encrypted to produce FRE.\n    FRE = cipherfn.encrypt(FR, key);\n\n    if (resync) {\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 0; i < block_size; i++)\n        ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      for (n = block_size + 2; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = ciphertext.charCodeAt(n + i);\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        // next 8 octets of ciphertext.  These are loaded into FR and the\n        // process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt((n - 2) +\n            i));\n      }\n    } else {\n      plaintext = \"  \" + plaintext;\n      // 9.  FRE is xored with the first 8 octets of the given plaintext, now\n      //\t   that we have finished encrypting the 10 octets of prefixed data.\n      // \t   This produces C11-C18, the next 8 octets of ciphertext.\n      for (var i = 2; i < block_size; i++) ciphertext += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(i));\n      var tempCiphertext = ciphertext.substring(0, 2 * block_size).split('');\n      var tempCiphertextString = ciphertext.substring(block_size);\n      for (n = block_size; n < plaintext.length; n += block_size) {\n        // 10. FR is loaded with C11-C18\n        for (var i = 0; i < block_size; i++) FR[i] = tempCiphertextString.charCodeAt(i);\n        tempCiphertextString = '';\n\n        // 11. FR is encrypted to produce FRE.\n        FRE = cipherfn.encrypt(FR);\n\n        // 12. FRE is xored with the next 8 octets of plaintext, to produce the\n        //     next 8 octets of ciphertext.  These are loaded into FR and the\n        //     process is repeated until the plaintext is used up.\n        for (var i = 0; i < block_size; i++) {\n          tempCiphertext.push(String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i)));\n          tempCiphertextString += String.fromCharCode(FRE[i] ^ plaintext.charCodeAt(n + i));\n        }\n      }\n      ciphertext = tempCiphertext.join('');\n\n    }\n\n    ciphertext = ciphertext.substring(0, plaintext.length + 2 + block_size);\n\n    return ciphertext;\n  },\n\n  /**\n   * Decrypts the prefixed data for the Modification Detection Code (MDC) computation\n   * @param {openpgp_block_cipher_fn} cipherfn.encrypt Cipher function to use\n   * @param {Integer} block_size Blocksize of the algorithm\n   * @param {openpgp_byte_array} key The key for encryption\n   * @param {String} ciphertext The encrypted data\n   * @return {String} plaintext Data of D(ciphertext) with blocksize length +2\n   */\n  mdc: function(cipherfn, key, ciphertext) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i;\n\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock);\n\n    return util.bin2str(iblock) +\n      String.fromCharCode(ablock[0] ^ ciphertext.charCodeAt(block_size)) +\n      String.fromCharCode(ablock[1] ^ ciphertext.charCodeAt(block_size + 1));\n  },\n  /**\n   * This function decrypts a given plaintext using the specified\n   * blockcipher to decrypt a message\n   * @param {openpgp_cipher_block_fn} blockcipherfn The algorithm _encrypt_ function to encrypt\n   *  data in one block_size encryption.\n   * @param {Integer} block_size the block size in bytes of the algorithm used\n   * @param {String} plaintext ciphertext to be decrypted provided as a string\n   * @param {openpgp_byte_array} key key to be used to decrypt the ciphertext. This will be passed to the \n   *  blockcipherfn\n   * @param {Boolean} resync a boolean value specifying if a resync of the \n   *  IV should be used or not. The encrypteddatapacket uses the \n   *  \"old\" style with a resync. Decryption within an \n   *  encryptedintegrityprotecteddata packet is not resyncing the IV.\n   * @return {String} a string with the plaintext data\n   */\n\n  decrypt: function(cipherfn, key, ciphertext, resync) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var iblock = new Array(block_size);\n    var ablock = new Array(block_size);\n    var i, n = '';\n    var text = [];\n\n    // initialisation vector\n    for (i = 0; i < block_size; i++) iblock[i] = 0;\n\n    iblock = cipherfn.encrypt(iblock, key);\n    for (i = 0; i < block_size; i++) {\n      ablock[i] = ciphertext.charCodeAt(i);\n      iblock[i] ^= ablock[i];\n    }\n\n    ablock = cipherfn.encrypt(ablock, key);\n\n    // test check octets\n    if (iblock[block_size - 2] != (ablock[0] ^ ciphertext.charCodeAt(block_size)) || iblock[block_size - 1] != (ablock[\n      1] ^ ciphertext.charCodeAt(block_size + 1))) {\n      throw new Error('Invalid data.');\n    }\n\n    /*  RFC4880: Tag 18 and Resync:\n\t\t *  [...] Unlike the Symmetrically Encrypted Data Packet, no\n\t\t *  special CFB resynchronization is done after encrypting this prefix\n\t\t *  data.  See \"OpenPGP CFB Mode\" below for more details.\n\n\t\t */\n\n    if (resync) {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i + 2);\n      for (n = block_size + 2; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    } else {\n      for (i = 0; i < block_size; i++) iblock[i] = ciphertext.charCodeAt(i);\n      for (n = block_size; n < ciphertext.length; n += block_size) {\n        ablock = cipherfn.encrypt(iblock);\n        for (i = 0; i < block_size && i + n < ciphertext.length; i++) {\n          iblock[i] = ciphertext.charCodeAt(n + i);\n          text.push(String.fromCharCode(ablock[i] ^ iblock[i]));\n        }\n      }\n    }\n\n    var n = resync ? 0 : 2;\n\n    text = text.join('');\n\n    text = text.substring(n, ciphertext.length - block_size - 2 + n);\n\n\n    return text;\n  },\n\n\n  normalEncrypt: function(cipherfn, key, plaintext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blocki = \"\";\n    var blockc = \"\";\n    var pos = 0;\n    var cyphertext = [];\n    var tempBlock = [];\n    blockc = iv.substring(0, block_size);\n    while (plaintext.length > block_size * pos) {\n      var encblock = cipherfn.encrypt(util.str2bin(blockc));\n      blocki = plaintext.substring((pos * block_size), (pos * block_size) + block_size);\n      for (var i = 0; i < blocki.length; i++)\n        tempBlock.push(String.fromCharCode(blocki.charCodeAt(i) ^ encblock[i]));\n      blockc = tempBlock.join('');\n      tempBlock = [];\n      cyphertext.push(blockc);\n      pos++;\n    }\n    return cyphertext.join('');\n  },\n\n  normalDecrypt: function(cipherfn, key, ciphertext, iv) {\n    cipherfn = new cipher[cipherfn](key);\n    var block_size = cipherfn.blockSize;\n\n    var blockp = \"\";\n    var pos = 0;\n    var plaintext = [];\n    var offset = 0;\n    if (iv == null)\n      for (var i = 0; i < block_size; i++) blockp += String.fromCharCode(0);\n    else\n      blockp = iv.substring(0, block_size);\n    while (ciphertext.length > (block_size * pos)) {\n      var decblock = cipherfn.encrypt(util.str2bin(blockp));\n      blockp = ciphertext.substring((pos * (block_size)) + offset, (pos * (block_size)) + (block_size) + offset);\n      for (var i = 0; i < blockp.length; i++) {\n        plaintext.push(String.fromCharCode(blockp.charCodeAt(i) ^ decblock[i]));\n      }\n      pos++;\n    }\n\n    return plaintext.join('');\n  }\n}\n","/* Rijndael (AES) Encryption\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.1, check www.haneWIN.de for the latest version\n\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other\n * materials provided with the application or distribution.\n */\n\nvar util = require('../../util');\n\n// The round constants used in subkey expansion\nvar Rcon = [\n    0x01, 0x02, 0x04, 0x08, 0x10, 0x20, 0x40, 0x80, 0x1b, 0x36, 0x6c, 0xd8,\n    0xab, 0x4d, 0x9a, 0x2f, 0x5e, 0xbc, 0x63, 0xc6, 0x97, 0x35, 0x6a, 0xd4,\n    0xb3, 0x7d, 0xfa, 0xef, 0xc5, 0x91\n];\n\n// Precomputed lookup table for the SBox\nvar S = [\n    99, 124, 119, 123, 242, 107, 111, 197, 48, 1, 103, 43, 254, 215, 171,\n    118, 202, 130, 201, 125, 250, 89, 71, 240, 173, 212, 162, 175, 156, 164,\n    114, 192, 183, 253, 147, 38, 54, 63, 247, 204, 52, 165, 229, 241, 113,\n    216, 49, 21, 4, 199, 35, 195, 24, 150, 5, 154, 7, 18, 128, 226,\n    235, 39, 178, 117, 9, 131, 44, 26, 27, 110, 90, 160, 82, 59, 214,\n    179, 41, 227, 47, 132, 83, 209, 0, 237, 32, 252, 177, 91, 106, 203,\n    190, 57, 74, 76, 88, 207, 208, 239, 170, 251, 67, 77, 51, 133, 69,\n    249, 2, 127, 80, 60, 159, 168, 81, 163, 64, 143, 146, 157, 56, 245,\n    188, 182, 218, 33, 16, 255, 243, 210, 205, 12, 19, 236, 95, 151, 68,\n    23, 196, 167, 126, 61, 100, 93, 25, 115, 96, 129, 79, 220, 34, 42,\n    144, 136, 70, 238, 184, 20, 222, 94, 11, 219, 224, 50, 58, 10, 73,\n    6, 36, 92, 194, 211, 172, 98, 145, 149, 228, 121, 231, 200, 55, 109,\n    141, 213, 78, 169, 108, 86, 244, 234, 101, 122, 174, 8, 186, 120, 37,\n    46, 28, 166, 180, 198, 232, 221, 116, 31, 75, 189, 139, 138, 112, 62,\n    181, 102, 72, 3, 246, 14, 97, 53, 87, 185, 134, 193, 29, 158, 225,\n    248, 152, 17, 105, 217, 142, 148, 155, 30, 135, 233, 206, 85, 40, 223,\n    140, 161, 137, 13, 191, 230, 66, 104, 65, 153, 45, 15, 176, 84, 187,\n    22\n];\n\nvar T1 = [\n    0xa56363c6, 0x847c7cf8, 0x997777ee, 0x8d7b7bf6,\n    0x0df2f2ff, 0xbd6b6bd6, 0xb16f6fde, 0x54c5c591,\n    0x50303060, 0x03010102, 0xa96767ce, 0x7d2b2b56,\n    0x19fefee7, 0x62d7d7b5, 0xe6abab4d, 0x9a7676ec,\n    0x45caca8f, 0x9d82821f, 0x40c9c989, 0x877d7dfa,\n    0x15fafaef, 0xeb5959b2, 0xc947478e, 0x0bf0f0fb,\n    0xecadad41, 0x67d4d4b3, 0xfda2a25f, 0xeaafaf45,\n    0xbf9c9c23, 0xf7a4a453, 0x967272e4, 0x5bc0c09b,\n    0xc2b7b775, 0x1cfdfde1, 0xae93933d, 0x6a26264c,\n    0x5a36366c, 0x413f3f7e, 0x02f7f7f5, 0x4fcccc83,\n    0x5c343468, 0xf4a5a551, 0x34e5e5d1, 0x08f1f1f9,\n    0x937171e2, 0x73d8d8ab, 0x53313162, 0x3f15152a,\n    0x0c040408, 0x52c7c795, 0x65232346, 0x5ec3c39d,\n    0x28181830, 0xa1969637, 0x0f05050a, 0xb59a9a2f,\n    0x0907070e, 0x36121224, 0x9b80801b, 0x3de2e2df,\n    0x26ebebcd, 0x6927274e, 0xcdb2b27f, 0x9f7575ea,\n    0x1b090912, 0x9e83831d, 0x742c2c58, 0x2e1a1a34,\n    0x2d1b1b36, 0xb26e6edc, 0xee5a5ab4, 0xfba0a05b,\n    0xf65252a4, 0x4d3b3b76, 0x61d6d6b7, 0xceb3b37d,\n    0x7b292952, 0x3ee3e3dd, 0x712f2f5e, 0x97848413,\n    0xf55353a6, 0x68d1d1b9, 0x00000000, 0x2cededc1,\n    0x60202040, 0x1ffcfce3, 0xc8b1b179, 0xed5b5bb6,\n    0xbe6a6ad4, 0x46cbcb8d, 0xd9bebe67, 0x4b393972,\n    0xde4a4a94, 0xd44c4c98, 0xe85858b0, 0x4acfcf85,\n    0x6bd0d0bb, 0x2aefefc5, 0xe5aaaa4f, 0x16fbfbed,\n    0xc5434386, 0xd74d4d9a, 0x55333366, 0x94858511,\n    0xcf45458a, 0x10f9f9e9, 0x06020204, 0x817f7ffe,\n    0xf05050a0, 0x443c3c78, 0xba9f9f25, 0xe3a8a84b,\n    0xf35151a2, 0xfea3a35d, 0xc0404080, 0x8a8f8f05,\n    0xad92923f, 0xbc9d9d21, 0x48383870, 0x04f5f5f1,\n    0xdfbcbc63, 0xc1b6b677, 0x75dadaaf, 0x63212142,\n    0x30101020, 0x1affffe5, 0x0ef3f3fd, 0x6dd2d2bf,\n    0x4ccdcd81, 0x140c0c18, 0x35131326, 0x2fececc3,\n    0xe15f5fbe, 0xa2979735, 0xcc444488, 0x3917172e,\n    0x57c4c493, 0xf2a7a755, 0x827e7efc, 0x473d3d7a,\n    0xac6464c8, 0xe75d5dba, 0x2b191932, 0x957373e6,\n    0xa06060c0, 0x98818119, 0xd14f4f9e, 0x7fdcdca3,\n    0x66222244, 0x7e2a2a54, 0xab90903b, 0x8388880b,\n    0xca46468c, 0x29eeeec7, 0xd3b8b86b, 0x3c141428,\n    0x79dedea7, 0xe25e5ebc, 0x1d0b0b16, 0x76dbdbad,\n    0x3be0e0db, 0x56323264, 0x4e3a3a74, 0x1e0a0a14,\n    0xdb494992, 0x0a06060c, 0x6c242448, 0xe45c5cb8,\n    0x5dc2c29f, 0x6ed3d3bd, 0xefacac43, 0xa66262c4,\n    0xa8919139, 0xa4959531, 0x37e4e4d3, 0x8b7979f2,\n    0x32e7e7d5, 0x43c8c88b, 0x5937376e, 0xb76d6dda,\n    0x8c8d8d01, 0x64d5d5b1, 0xd24e4e9c, 0xe0a9a949,\n    0xb46c6cd8, 0xfa5656ac, 0x07f4f4f3, 0x25eaeacf,\n    0xaf6565ca, 0x8e7a7af4, 0xe9aeae47, 0x18080810,\n    0xd5baba6f, 0x887878f0, 0x6f25254a, 0x722e2e5c,\n    0x241c1c38, 0xf1a6a657, 0xc7b4b473, 0x51c6c697,\n    0x23e8e8cb, 0x7cdddda1, 0x9c7474e8, 0x211f1f3e,\n    0xdd4b4b96, 0xdcbdbd61, 0x868b8b0d, 0x858a8a0f,\n    0x907070e0, 0x423e3e7c, 0xc4b5b571, 0xaa6666cc,\n    0xd8484890, 0x05030306, 0x01f6f6f7, 0x120e0e1c,\n    0xa36161c2, 0x5f35356a, 0xf95757ae, 0xd0b9b969,\n    0x91868617, 0x58c1c199, 0x271d1d3a, 0xb99e9e27,\n    0x38e1e1d9, 0x13f8f8eb, 0xb398982b, 0x33111122,\n    0xbb6969d2, 0x70d9d9a9, 0x898e8e07, 0xa7949433,\n    0xb69b9b2d, 0x221e1e3c, 0x92878715, 0x20e9e9c9,\n    0x49cece87, 0xff5555aa, 0x78282850, 0x7adfdfa5,\n    0x8f8c8c03, 0xf8a1a159, 0x80898909, 0x170d0d1a,\n    0xdabfbf65, 0x31e6e6d7, 0xc6424284, 0xb86868d0,\n    0xc3414182, 0xb0999929, 0x772d2d5a, 0x110f0f1e,\n    0xcbb0b07b, 0xfc5454a8, 0xd6bbbb6d, 0x3a16162c\n];\n\nvar T2 = [\n    0x6363c6a5, 0x7c7cf884, 0x7777ee99, 0x7b7bf68d,\n    0xf2f2ff0d, 0x6b6bd6bd, 0x6f6fdeb1, 0xc5c59154,\n    0x30306050, 0x01010203, 0x6767cea9, 0x2b2b567d,\n    0xfefee719, 0xd7d7b562, 0xabab4de6, 0x7676ec9a,\n    0xcaca8f45, 0x82821f9d, 0xc9c98940, 0x7d7dfa87,\n    0xfafaef15, 0x5959b2eb, 0x47478ec9, 0xf0f0fb0b,\n    0xadad41ec, 0xd4d4b367, 0xa2a25ffd, 0xafaf45ea,\n    0x9c9c23bf, 0xa4a453f7, 0x7272e496, 0xc0c09b5b,\n    0xb7b775c2, 0xfdfde11c, 0x93933dae, 0x26264c6a,\n    0x36366c5a, 0x3f3f7e41, 0xf7f7f502, 0xcccc834f,\n    0x3434685c, 0xa5a551f4, 0xe5e5d134, 0xf1f1f908,\n    0x7171e293, 0xd8d8ab73, 0x31316253, 0x15152a3f,\n    0x0404080c, 0xc7c79552, 0x23234665, 0xc3c39d5e,\n    0x18183028, 0x969637a1, 0x05050a0f, 0x9a9a2fb5,\n    0x07070e09, 0x12122436, 0x80801b9b, 0xe2e2df3d,\n    0xebebcd26, 0x27274e69, 0xb2b27fcd, 0x7575ea9f,\n    0x0909121b, 0x83831d9e, 0x2c2c5874, 0x1a1a342e,\n    0x1b1b362d, 0x6e6edcb2, 0x5a5ab4ee, 0xa0a05bfb,\n    0x5252a4f6, 0x3b3b764d, 0xd6d6b761, 0xb3b37dce,\n    0x2929527b, 0xe3e3dd3e, 0x2f2f5e71, 0x84841397,\n    0x5353a6f5, 0xd1d1b968, 0x00000000, 0xededc12c,\n    0x20204060, 0xfcfce31f, 0xb1b179c8, 0x5b5bb6ed,\n    0x6a6ad4be, 0xcbcb8d46, 0xbebe67d9, 0x3939724b,\n    0x4a4a94de, 0x4c4c98d4, 0x5858b0e8, 0xcfcf854a,\n    0xd0d0bb6b, 0xefefc52a, 0xaaaa4fe5, 0xfbfbed16,\n    0x434386c5, 0x4d4d9ad7, 0x33336655, 0x85851194,\n    0x45458acf, 0xf9f9e910, 0x02020406, 0x7f7ffe81,\n    0x5050a0f0, 0x3c3c7844, 0x9f9f25ba, 0xa8a84be3,\n    0x5151a2f3, 0xa3a35dfe, 0x404080c0, 0x8f8f058a,\n    0x92923fad, 0x9d9d21bc, 0x38387048, 0xf5f5f104,\n    0xbcbc63df, 0xb6b677c1, 0xdadaaf75, 0x21214263,\n    0x10102030, 0xffffe51a, 0xf3f3fd0e, 0xd2d2bf6d,\n    0xcdcd814c, 0x0c0c1814, 0x13132635, 0xececc32f,\n    0x5f5fbee1, 0x979735a2, 0x444488cc, 0x17172e39,\n    0xc4c49357, 0xa7a755f2, 0x7e7efc82, 0x3d3d7a47,\n    0x6464c8ac, 0x5d5dbae7, 0x1919322b, 0x7373e695,\n    0x6060c0a0, 0x81811998, 0x4f4f9ed1, 0xdcdca37f,\n    0x22224466, 0x2a2a547e, 0x90903bab, 0x88880b83,\n    0x46468cca, 0xeeeec729, 0xb8b86bd3, 0x1414283c,\n    0xdedea779, 0x5e5ebce2, 0x0b0b161d, 0xdbdbad76,\n    0xe0e0db3b, 0x32326456, 0x3a3a744e, 0x0a0a141e,\n    0x494992db, 0x06060c0a, 0x2424486c, 0x5c5cb8e4,\n    0xc2c29f5d, 0xd3d3bd6e, 0xacac43ef, 0x6262c4a6,\n    0x919139a8, 0x959531a4, 0xe4e4d337, 0x7979f28b,\n    0xe7e7d532, 0xc8c88b43, 0x37376e59, 0x6d6ddab7,\n    0x8d8d018c, 0xd5d5b164, 0x4e4e9cd2, 0xa9a949e0,\n    0x6c6cd8b4, 0x5656acfa, 0xf4f4f307, 0xeaeacf25,\n    0x6565caaf, 0x7a7af48e, 0xaeae47e9, 0x08081018,\n    0xbaba6fd5, 0x7878f088, 0x25254a6f, 0x2e2e5c72,\n    0x1c1c3824, 0xa6a657f1, 0xb4b473c7, 0xc6c69751,\n    0xe8e8cb23, 0xdddda17c, 0x7474e89c, 0x1f1f3e21,\n    0x4b4b96dd, 0xbdbd61dc, 0x8b8b0d86, 0x8a8a0f85,\n    0x7070e090, 0x3e3e7c42, 0xb5b571c4, 0x6666ccaa,\n    0x484890d8, 0x03030605, 0xf6f6f701, 0x0e0e1c12,\n    0x6161c2a3, 0x35356a5f, 0x5757aef9, 0xb9b969d0,\n    0x86861791, 0xc1c19958, 0x1d1d3a27, 0x9e9e27b9,\n    0xe1e1d938, 0xf8f8eb13, 0x98982bb3, 0x11112233,\n    0x6969d2bb, 0xd9d9a970, 0x8e8e0789, 0x949433a7,\n    0x9b9b2db6, 0x1e1e3c22, 0x87871592, 0xe9e9c920,\n    0xcece8749, 0x5555aaff, 0x28285078, 0xdfdfa57a,\n    0x8c8c038f, 0xa1a159f8, 0x89890980, 0x0d0d1a17,\n    0xbfbf65da, 0xe6e6d731, 0x424284c6, 0x6868d0b8,\n    0x414182c3, 0x999929b0, 0x2d2d5a77, 0x0f0f1e11,\n    0xb0b07bcb, 0x5454a8fc, 0xbbbb6dd6, 0x16162c3a\n];\n\nvar T3 = [\n    0x63c6a563, 0x7cf8847c, 0x77ee9977, 0x7bf68d7b,\n    0xf2ff0df2, 0x6bd6bd6b, 0x6fdeb16f, 0xc59154c5,\n    0x30605030, 0x01020301, 0x67cea967, 0x2b567d2b,\n    0xfee719fe, 0xd7b562d7, 0xab4de6ab, 0x76ec9a76,\n    0xca8f45ca, 0x821f9d82, 0xc98940c9, 0x7dfa877d,\n    0xfaef15fa, 0x59b2eb59, 0x478ec947, 0xf0fb0bf0,\n    0xad41ecad, 0xd4b367d4, 0xa25ffda2, 0xaf45eaaf,\n    0x9c23bf9c, 0xa453f7a4, 0x72e49672, 0xc09b5bc0,\n    0xb775c2b7, 0xfde11cfd, 0x933dae93, 0x264c6a26,\n    0x366c5a36, 0x3f7e413f, 0xf7f502f7, 0xcc834fcc,\n    0x34685c34, 0xa551f4a5, 0xe5d134e5, 0xf1f908f1,\n    0x71e29371, 0xd8ab73d8, 0x31625331, 0x152a3f15,\n    0x04080c04, 0xc79552c7, 0x23466523, 0xc39d5ec3,\n    0x18302818, 0x9637a196, 0x050a0f05, 0x9a2fb59a,\n    0x070e0907, 0x12243612, 0x801b9b80, 0xe2df3de2,\n    0xebcd26eb, 0x274e6927, 0xb27fcdb2, 0x75ea9f75,\n    0x09121b09, 0x831d9e83, 0x2c58742c, 0x1a342e1a,\n    0x1b362d1b, 0x6edcb26e, 0x5ab4ee5a, 0xa05bfba0,\n    0x52a4f652, 0x3b764d3b, 0xd6b761d6, 0xb37dceb3,\n    0x29527b29, 0xe3dd3ee3, 0x2f5e712f, 0x84139784,\n    0x53a6f553, 0xd1b968d1, 0x00000000, 0xedc12ced,\n    0x20406020, 0xfce31ffc, 0xb179c8b1, 0x5bb6ed5b,\n    0x6ad4be6a, 0xcb8d46cb, 0xbe67d9be, 0x39724b39,\n    0x4a94de4a, 0x4c98d44c, 0x58b0e858, 0xcf854acf,\n    0xd0bb6bd0, 0xefc52aef, 0xaa4fe5aa, 0xfbed16fb,\n    0x4386c543, 0x4d9ad74d, 0x33665533, 0x85119485,\n    0x458acf45, 0xf9e910f9, 0x02040602, 0x7ffe817f,\n    0x50a0f050, 0x3c78443c, 0x9f25ba9f, 0xa84be3a8,\n    0x51a2f351, 0xa35dfea3, 0x4080c040, 0x8f058a8f,\n    0x923fad92, 0x9d21bc9d, 0x38704838, 0xf5f104f5,\n    0xbc63dfbc, 0xb677c1b6, 0xdaaf75da, 0x21426321,\n    0x10203010, 0xffe51aff, 0xf3fd0ef3, 0xd2bf6dd2,\n    0xcd814ccd, 0x0c18140c, 0x13263513, 0xecc32fec,\n    0x5fbee15f, 0x9735a297, 0x4488cc44, 0x172e3917,\n    0xc49357c4, 0xa755f2a7, 0x7efc827e, 0x3d7a473d,\n    0x64c8ac64, 0x5dbae75d, 0x19322b19, 0x73e69573,\n    0x60c0a060, 0x81199881, 0x4f9ed14f, 0xdca37fdc,\n    0x22446622, 0x2a547e2a, 0x903bab90, 0x880b8388,\n    0x468cca46, 0xeec729ee, 0xb86bd3b8, 0x14283c14,\n    0xdea779de, 0x5ebce25e, 0x0b161d0b, 0xdbad76db,\n    0xe0db3be0, 0x32645632, 0x3a744e3a, 0x0a141e0a,\n    0x4992db49, 0x060c0a06, 0x24486c24, 0x5cb8e45c,\n    0xc29f5dc2, 0xd3bd6ed3, 0xac43efac, 0x62c4a662,\n    0x9139a891, 0x9531a495, 0xe4d337e4, 0x79f28b79,\n    0xe7d532e7, 0xc88b43c8, 0x376e5937, 0x6ddab76d,\n    0x8d018c8d, 0xd5b164d5, 0x4e9cd24e, 0xa949e0a9,\n    0x6cd8b46c, 0x56acfa56, 0xf4f307f4, 0xeacf25ea,\n    0x65caaf65, 0x7af48e7a, 0xae47e9ae, 0x08101808,\n    0xba6fd5ba, 0x78f08878, 0x254a6f25, 0x2e5c722e,\n    0x1c38241c, 0xa657f1a6, 0xb473c7b4, 0xc69751c6,\n    0xe8cb23e8, 0xdda17cdd, 0x74e89c74, 0x1f3e211f,\n    0x4b96dd4b, 0xbd61dcbd, 0x8b0d868b, 0x8a0f858a,\n    0x70e09070, 0x3e7c423e, 0xb571c4b5, 0x66ccaa66,\n    0x4890d848, 0x03060503, 0xf6f701f6, 0x0e1c120e,\n    0x61c2a361, 0x356a5f35, 0x57aef957, 0xb969d0b9,\n    0x86179186, 0xc19958c1, 0x1d3a271d, 0x9e27b99e,\n    0xe1d938e1, 0xf8eb13f8, 0x982bb398, 0x11223311,\n    0x69d2bb69, 0xd9a970d9, 0x8e07898e, 0x9433a794,\n    0x9b2db69b, 0x1e3c221e, 0x87159287, 0xe9c920e9,\n    0xce8749ce, 0x55aaff55, 0x28507828, 0xdfa57adf,\n    0x8c038f8c, 0xa159f8a1, 0x89098089, 0x0d1a170d,\n    0xbf65dabf, 0xe6d731e6, 0x4284c642, 0x68d0b868,\n    0x4182c341, 0x9929b099, 0x2d5a772d, 0x0f1e110f,\n    0xb07bcbb0, 0x54a8fc54, 0xbb6dd6bb, 0x162c3a16\n];\n\nvar T4 = [\n    0xc6a56363, 0xf8847c7c, 0xee997777, 0xf68d7b7b,\n    0xff0df2f2, 0xd6bd6b6b, 0xdeb16f6f, 0x9154c5c5,\n    0x60503030, 0x02030101, 0xcea96767, 0x567d2b2b,\n    0xe719fefe, 0xb562d7d7, 0x4de6abab, 0xec9a7676,\n    0x8f45caca, 0x1f9d8282, 0x8940c9c9, 0xfa877d7d,\n    0xef15fafa, 0xb2eb5959, 0x8ec94747, 0xfb0bf0f0,\n    0x41ecadad, 0xb367d4d4, 0x5ffda2a2, 0x45eaafaf,\n    0x23bf9c9c, 0x53f7a4a4, 0xe4967272, 0x9b5bc0c0,\n    0x75c2b7b7, 0xe11cfdfd, 0x3dae9393, 0x4c6a2626,\n    0x6c5a3636, 0x7e413f3f, 0xf502f7f7, 0x834fcccc,\n    0x685c3434, 0x51f4a5a5, 0xd134e5e5, 0xf908f1f1,\n    0xe2937171, 0xab73d8d8, 0x62533131, 0x2a3f1515,\n    0x080c0404, 0x9552c7c7, 0x46652323, 0x9d5ec3c3,\n    0x30281818, 0x37a19696, 0x0a0f0505, 0x2fb59a9a,\n    0x0e090707, 0x24361212, 0x1b9b8080, 0xdf3de2e2,\n    0xcd26ebeb, 0x4e692727, 0x7fcdb2b2, 0xea9f7575,\n    0x121b0909, 0x1d9e8383, 0x58742c2c, 0x342e1a1a,\n    0x362d1b1b, 0xdcb26e6e, 0xb4ee5a5a, 0x5bfba0a0,\n    0xa4f65252, 0x764d3b3b, 0xb761d6d6, 0x7dceb3b3,\n    0x527b2929, 0xdd3ee3e3, 0x5e712f2f, 0x13978484,\n    0xa6f55353, 0xb968d1d1, 0x00000000, 0xc12ceded,\n    0x40602020, 0xe31ffcfc, 0x79c8b1b1, 0xb6ed5b5b,\n    0xd4be6a6a, 0x8d46cbcb, 0x67d9bebe, 0x724b3939,\n    0x94de4a4a, 0x98d44c4c, 0xb0e85858, 0x854acfcf,\n    0xbb6bd0d0, 0xc52aefef, 0x4fe5aaaa, 0xed16fbfb,\n    0x86c54343, 0x9ad74d4d, 0x66553333, 0x11948585,\n    0x8acf4545, 0xe910f9f9, 0x04060202, 0xfe817f7f,\n    0xa0f05050, 0x78443c3c, 0x25ba9f9f, 0x4be3a8a8,\n    0xa2f35151, 0x5dfea3a3, 0x80c04040, 0x058a8f8f,\n    0x3fad9292, 0x21bc9d9d, 0x70483838, 0xf104f5f5,\n    0x63dfbcbc, 0x77c1b6b6, 0xaf75dada, 0x42632121,\n    0x20301010, 0xe51affff, 0xfd0ef3f3, 0xbf6dd2d2,\n    0x814ccdcd, 0x18140c0c, 0x26351313, 0xc32fecec,\n    0xbee15f5f, 0x35a29797, 0x88cc4444, 0x2e391717,\n    0x9357c4c4, 0x55f2a7a7, 0xfc827e7e, 0x7a473d3d,\n    0xc8ac6464, 0xbae75d5d, 0x322b1919, 0xe6957373,\n    0xc0a06060, 0x19988181, 0x9ed14f4f, 0xa37fdcdc,\n    0x44662222, 0x547e2a2a, 0x3bab9090, 0x0b838888,\n    0x8cca4646, 0xc729eeee, 0x6bd3b8b8, 0x283c1414,\n    0xa779dede, 0xbce25e5e, 0x161d0b0b, 0xad76dbdb,\n    0xdb3be0e0, 0x64563232, 0x744e3a3a, 0x141e0a0a,\n    0x92db4949, 0x0c0a0606, 0x486c2424, 0xb8e45c5c,\n    0x9f5dc2c2, 0xbd6ed3d3, 0x43efacac, 0xc4a66262,\n    0x39a89191, 0x31a49595, 0xd337e4e4, 0xf28b7979,\n    0xd532e7e7, 0x8b43c8c8, 0x6e593737, 0xdab76d6d,\n    0x018c8d8d, 0xb164d5d5, 0x9cd24e4e, 0x49e0a9a9,\n    0xd8b46c6c, 0xacfa5656, 0xf307f4f4, 0xcf25eaea,\n    0xcaaf6565, 0xf48e7a7a, 0x47e9aeae, 0x10180808,\n    0x6fd5baba, 0xf0887878, 0x4a6f2525, 0x5c722e2e,\n    0x38241c1c, 0x57f1a6a6, 0x73c7b4b4, 0x9751c6c6,\n    0xcb23e8e8, 0xa17cdddd, 0xe89c7474, 0x3e211f1f,\n    0x96dd4b4b, 0x61dcbdbd, 0x0d868b8b, 0x0f858a8a,\n    0xe0907070, 0x7c423e3e, 0x71c4b5b5, 0xccaa6666,\n    0x90d84848, 0x06050303, 0xf701f6f6, 0x1c120e0e,\n    0xc2a36161, 0x6a5f3535, 0xaef95757, 0x69d0b9b9,\n    0x17918686, 0x9958c1c1, 0x3a271d1d, 0x27b99e9e,\n    0xd938e1e1, 0xeb13f8f8, 0x2bb39898, 0x22331111,\n    0xd2bb6969, 0xa970d9d9, 0x07898e8e, 0x33a79494,\n    0x2db69b9b, 0x3c221e1e, 0x15928787, 0xc920e9e9,\n    0x8749cece, 0xaaff5555, 0x50782828, 0xa57adfdf,\n    0x038f8c8c, 0x59f8a1a1, 0x09808989, 0x1a170d0d,\n    0x65dabfbf, 0xd731e6e6, 0x84c64242, 0xd0b86868,\n    0x82c34141, 0x29b09999, 0x5a772d2d, 0x1e110f0f,\n    0x7bcbb0b0, 0xa8fc5454, 0x6dd6bbbb, 0x2c3a1616\n];\n\nfunction B0(x) {\n  return (x & 255);\n}\n\nfunction B1(x) {\n  return ((x >> 8) & 255);\n}\n\nfunction B2(x) {\n  return ((x >> 16) & 255);\n}\n\nfunction B3(x) {\n  return ((x >> 24) & 255);\n}\n\nfunction F1(x0, x1, x2, x3) {\n  return B1(T1[x0 & 255]) | (B1(T1[(x1 >> 8) & 255]) << 8) | (B1(T1[(x2 >> 16) & 255]) << 16) | (B1(T1[x3 >>> 24]) <<\n    24);\n}\n\nfunction packBytes(octets) {\n  var i, j;\n  var len = octets.length;\n  var b = new Array(len / 4);\n\n  if (!octets || len % 4) return;\n\n  for (i = 0, j = 0; j < len; j += 4)\n    b[i++] = octets[j] | (octets[j + 1] << 8) | (octets[j + 2] << 16) | (octets[j + 3] << 24);\n\n  return b;\n}\n\nfunction unpackBytes(packed) {\n  var j;\n  var i = 0,\n    l = packed.length;\n  var r = new Array(l * 4);\n\n  for (j = 0; j < l; j++) {\n    r[i++] = B0(packed[j]);\n    r[i++] = B1(packed[j]);\n    r[i++] = B2(packed[j]);\n    r[i++] = B3(packed[j]);\n  }\n  return r;\n}\n\n// ------------------------------------------------\n\nvar maxkc = 8;\nvar maxrk = 14;\n\nfunction keyExpansion(key) {\n  var kc, i, j, r, t;\n  var rounds;\n  var keySched = new Array(maxrk + 1);\n  var keylen = key.length;\n  var k = new Array(maxkc);\n  var tk = new Array(maxkc);\n  var rconpointer = 0;\n\n  if (keylen == 16) {\n    rounds = 10;\n    kc = 4;\n  } else if (keylen == 24) {\n    rounds = 12;\n    kc = 6;\n  } else if (keylen == 32) {\n    rounds = 14;\n    kc = 8;\n  } else {\n    util.print_error('aes.js: Invalid key-length for AES key:' + keylen);\n    return;\n  }\n\n  for (i = 0; i < maxrk + 1; i++) keySched[i] = new Array(4);\n\n  for (i = 0, j = 0; j < keylen; j++, i += 4)\n    k[j] = key.charCodeAt(i) | (key.charCodeAt(i + 1) << 8) | (key.charCodeAt(i + 2) << 16) | (key.charCodeAt(i + 3) <<\n      24);\n\n  for (j = kc - 1; j >= 0; j--) tk[j] = k[j];\n\n  r = 0;\n  t = 0;\n  for (j = 0;\n  (j < kc) && (r < rounds + 1);) {\n    for (;\n    (j < kc) && (t < 4); j++, t++) {\n      keySched[r][t] = tk[j];\n    }\n    if (t == 4) {\n      r++;\n      t = 0;\n    }\n  }\n\n  while (r < rounds + 1) {\n    var temp = tk[kc - 1];\n\n    tk[0] ^= S[B1(temp)] | (S[B2(temp)] << 8) | (S[B3(temp)] << 16) | (S[B0(temp)] << 24);\n    tk[0] ^= Rcon[rconpointer++];\n\n    if (kc != 8) {\n      for (j = 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    } else {\n      for (j = 1; j < kc / 2; j++) tk[j] ^= tk[j - 1];\n\n      temp = tk[kc / 2 - 1];\n      tk[kc / 2] ^= S[B0(temp)] | (S[B1(temp)] << 8) | (S[B2(temp)] << 16) | (S[B3(temp)] << 24);\n\n      for (j = kc / 2 + 1; j < kc; j++) tk[j] ^= tk[j - 1];\n    }\n\n    for (j = 0;\n    (j < kc) && (r < rounds + 1);) {\n      for (;\n      (j < kc) && (t < 4); j++, t++) {\n        keySched[r][t] = tk[j];\n      }\n      if (t == 4) {\n        r++;\n        t = 0;\n      }\n    }\n  }\n  this.rounds = rounds;\n  this.rk = keySched;\n  return this;\n}\n\nfunction AESencrypt(block, ctx) {\n  var r;\n  var t0, t1, t2, t3;\n\n  var b = packBytes(block);\n  var rounds = ctx.rounds;\n  var b0 = b[0];\n  var b1 = b[1];\n  var b2 = b[2];\n  var b3 = b[3];\n\n  for (r = 0; r < rounds - 1; r++) {\n    t0 = b0 ^ ctx.rk[r][0];\n    t1 = b1 ^ ctx.rk[r][1];\n    t2 = b2 ^ ctx.rk[r][2];\n    t3 = b3 ^ ctx.rk[r][3];\n\n    b0 = T1[t0 & 255] ^ T2[(t1 >> 8) & 255] ^ T3[(t2 >> 16) & 255] ^ T4[t3 >>> 24];\n    b1 = T1[t1 & 255] ^ T2[(t2 >> 8) & 255] ^ T3[(t3 >> 16) & 255] ^ T4[t0 >>> 24];\n    b2 = T1[t2 & 255] ^ T2[(t3 >> 8) & 255] ^ T3[(t0 >> 16) & 255] ^ T4[t1 >>> 24];\n    b3 = T1[t3 & 255] ^ T2[(t0 >> 8) & 255] ^ T3[(t1 >> 16) & 255] ^ T4[t2 >>> 24];\n  }\n\n  // last round is special\n  r = rounds - 1;\n\n  t0 = b0 ^ ctx.rk[r][0];\n  t1 = b1 ^ ctx.rk[r][1];\n  t2 = b2 ^ ctx.rk[r][2];\n  t3 = b3 ^ ctx.rk[r][3];\n\n  b[0] = F1(t0, t1, t2, t3) ^ ctx.rk[rounds][0];\n  b[1] = F1(t1, t2, t3, t0) ^ ctx.rk[rounds][1];\n  b[2] = F1(t2, t3, t0, t1) ^ ctx.rk[rounds][2];\n  b[3] = F1(t3, t0, t1, t2) ^ ctx.rk[rounds][3];\n\n  return unpackBytes(b);\n}\n\nfunction makeClass(length) {\n\n  var c = function(key) {\n    this.key = keyExpansion(key);\n\n    this.encrypt = function(block) {\n      return AESencrypt(block, this.key);\n    }\n  }\n\n  c.blockSize = c.prototype.blockSize = 16;\n  c.keySize = c.prototype.keySize = length / 8;\n\n  return c;\n}\n\nmodule.exports = {}\n\nvar types = [128, 192, 256];\n\nfor (var i in types) {\n  module.exports[types[i]] = makeClass(types[i]);\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Originally written by nklein software (nklein.com)\n */\n\n/* \n * Javascript implementation based on Bruce Schneier's reference implementation.\n *\n *\n * The constructor doesn't do much of anything.  It's just here\n * so we can start defining properties and methods and such.\n */\nfunction Blowfish() {};\n\n/*\n * Declare the block size so that protocols know what size\n * Initialization Vector (IV) they will need.\n */\nBlowfish.prototype.BLOCKSIZE = 8;\n\n/*\n * These are the default SBOXES.\n */\nBlowfish.prototype.SBOXES = [\n  [\n      0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96,\n      0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16,\n      0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658,\n      0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013,\n      0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e,\n      0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60,\n      0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6,\n      0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a,\n      0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c,\n      0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193,\n      0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1,\n      0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239,\n      0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a,\n      0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3,\n      0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176,\n      0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe,\n      0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706,\n      0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b,\n      0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b,\n      0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463,\n      0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c,\n      0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3,\n      0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a,\n      0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8,\n      0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760,\n      0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db,\n      0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8,\n      0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b,\n      0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33,\n      0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4,\n      0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0,\n      0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c,\n      0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777,\n      0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299,\n      0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705,\n      0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf,\n      0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e,\n      0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa,\n      0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9,\n      0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915,\n      0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f,\n      0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664,\n      0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a\n  ],\n  [\n      0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d,\n      0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1,\n      0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65,\n      0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1,\n      0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9,\n      0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737,\n      0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d,\n      0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd,\n      0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc,\n      0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41,\n      0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908,\n      0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af,\n      0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124,\n      0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c,\n      0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908,\n      0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd,\n      0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b,\n      0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e,\n      0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa,\n      0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a,\n      0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d,\n      0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66,\n      0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5,\n      0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84,\n      0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96,\n      0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14,\n      0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca,\n      0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7,\n      0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77,\n      0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99,\n      0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054,\n      0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73,\n      0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea,\n      0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105,\n      0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646,\n      0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285,\n      0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea,\n      0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb,\n      0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e,\n      0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc,\n      0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd,\n      0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20,\n      0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7\n  ],\n  [\n      0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7,\n      0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af,\n      0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af,\n      0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504,\n      0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4,\n      0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee,\n      0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec,\n      0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b,\n      0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332,\n      0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527,\n      0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58,\n      0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c,\n      0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22,\n      0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17,\n      0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60,\n      0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115,\n      0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99,\n      0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0,\n      0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74,\n      0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d,\n      0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3,\n      0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3,\n      0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979,\n      0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c,\n      0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa,\n      0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a,\n      0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086,\n      0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc,\n      0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24,\n      0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2,\n      0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84,\n      0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c,\n      0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09,\n      0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10,\n      0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe,\n      0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027,\n      0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0,\n      0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634,\n      0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188,\n      0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc,\n      0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8,\n      0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837,\n      0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0\n  ],\n  [\n      0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742,\n      0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b,\n      0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79,\n      0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6,\n      0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a,\n      0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4,\n      0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1,\n      0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59,\n      0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797,\n      0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28,\n      0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6,\n      0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28,\n      0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba,\n      0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a,\n      0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5,\n      0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f,\n      0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce,\n      0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680,\n      0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd,\n      0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb,\n      0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb,\n      0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370,\n      0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc,\n      0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048,\n      0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc,\n      0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9,\n      0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a,\n      0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f,\n      0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a,\n      0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1,\n      0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b,\n      0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e,\n      0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e,\n      0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f,\n      0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623,\n      0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc,\n      0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a,\n      0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6,\n      0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3,\n      0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060,\n      0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c,\n      0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f,\n      0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6\n  ]\n];\n\n//*\n//* This is the default PARRAY\n//*\nBlowfish.prototype.PARRAY = [\n    0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0,\n    0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c,\n    0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b\n];\n\n//*\n//* This is the number of rounds the cipher will go\n//*\nBlowfish.prototype.NN = 16;\n\n//*\n//* This function is needed to get rid of problems\n//* with the high-bit getting set.  If we don't do\n//* this, then sometimes ( aa & 0x00FFFFFFFF ) is not\n//* equal to ( bb & 0x00FFFFFFFF ) even when they\n//* agree bit-for-bit for the first 32 bits.\n//*\nBlowfish.prototype._clean = function(xx) {\n  if (xx < 0) {\n    var yy = xx & 0x7FFFFFFF;\n    xx = yy + 0x80000000;\n  }\n  return xx;\n};\n\n//*\n//* This is the mixing function that uses the sboxes\n//*\nBlowfish.prototype._F = function(xx) {\n  var aa;\n  var bb;\n  var cc;\n  var dd;\n  var yy;\n\n  dd = xx & 0x00FF;\n  xx >>>= 8;\n  cc = xx & 0x00FF;\n  xx >>>= 8;\n  bb = xx & 0x00FF;\n  xx >>>= 8;\n  aa = xx & 0x00FF;\n\n  yy = this.sboxes[0][aa] + this.sboxes[1][bb];\n  yy = yy ^ this.sboxes[2][cc];\n  yy = yy + this.sboxes[3][dd];\n\n  return yy;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and does NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._encrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = 0; ii < this.NN; ++ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[this.NN + 0];\n  dataR = dataR ^ this.parray[this.NN + 1];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a vector of numbers and turns them\n//* into long words so that they can be processed by the\n//* real algorithm.\n//*\n//* Maybe I should make the real algorithm above take a vector\n//* instead.  That will involve more looping, but it won't require\n//* the F() method to deconstruct the vector.\n//*\nBlowfish.prototype.encrypt_block = function(vector) {\n  var ii;\n  var vals = [0, 0];\n  var off = this.BLOCKSIZE / 2;\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    vals[0] = (vals[0] << 8) | (vector[ii + 0] & 0x00FF);\n    vals[1] = (vals[1] << 8) | (vector[ii + off] & 0x00FF);\n  }\n\n  this._encrypt_block(vals);\n\n  var ret = [];\n  for (ii = 0; ii < this.BLOCKSIZE / 2; ++ii) {\n    ret[ii + 0] = (vals[0] >>> (24 - 8 * (ii)) & 0x00FF);\n    ret[ii + off] = (vals[1] >>> (24 - 8 * (ii)) & 0x00FF);\n    // vals[ 0 ] = ( vals[ 0 ] >>> 8 );\n    // vals[ 1 ] = ( vals[ 1 ] >>> 8 );\n  }\n\n  return ret;\n};\n\n//*\n//* This method takes an array with two values, left and right\n//* and undoes NN rounds of Blowfish on them.\n//*\nBlowfish.prototype._decrypt_block = function(vals) {\n  var dataL = vals[0];\n  var dataR = vals[1];\n\n  var ii;\n\n  for (ii = this.NN + 1; ii > 1; --ii) {\n    dataL = dataL ^ this.parray[ii];\n    dataR = this._F(dataL) ^ dataR;\n\n    var tmp = dataL;\n    dataL = dataR;\n    dataR = tmp;\n  }\n\n  dataL = dataL ^ this.parray[1];\n  dataR = dataR ^ this.parray[0];\n\n  vals[0] = this._clean(dataR);\n  vals[1] = this._clean(dataL);\n};\n\n//*\n//* This method takes a key array and initializes the\n//* sboxes and parray for this encryption.\n//*\nBlowfish.prototype.init = function(key) {\n  var ii;\n  var jj = 0;\n\n  this.parray = [];\n  for (ii = 0; ii < this.NN + 2; ++ii) {\n    var data = 0x00000000;\n    var kk;\n    for (kk = 0; kk < 4; ++kk) {\n      data = (data << 8) | (key[jj] & 0x00FF);\n      if (++jj >= key.length) {\n        jj = 0;\n      }\n    }\n    this.parray[ii] = this.PARRAY[ii] ^ data;\n  }\n\n  this.sboxes = [];\n  for (ii = 0; ii < 4; ++ii) {\n    this.sboxes[ii] = [];\n    for (jj = 0; jj < 256; ++jj) {\n      this.sboxes[ii][jj] = this.SBOXES[ii][jj];\n    }\n  }\n\n  var vals = [0x00000000, 0x00000000];\n\n  for (ii = 0; ii < this.NN + 2; ii += 2) {\n    this._encrypt_block(vals);\n    this.parray[ii + 0] = vals[0];\n    this.parray[ii + 1] = vals[1];\n  }\n\n  for (ii = 0; ii < 4; ++ii) {\n    for (jj = 0; jj < 256; jj += 2) {\n      this._encrypt_block(vals);\n      this.sboxes[ii][jj + 0] = vals[0];\n      this.sboxes[ii][jj + 1] = vals[1];\n    }\n  }\n};\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction BFencrypt(block, key) {\n  var bf = new Blowfish();\n  bf.init(util.str2bin(key));\n  return bf.encrypt_block(block);\n}\n\nfunction BF(key) {\n  this.bf = new Blowfish();\n  this.bf.init(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.bf.encrypt_block(block);\n  }\n}\n\n\nmodule.exports = BF;\nmodule.exports.keySize = BF.prototype.keySize = 16;\nmodule.exports.blockSize = BF.prototype.blockSize = 16;\n","// Use of this source code is governed by a BSD-style\n// license that can be found in the LICENSE file.\n\n// Copyright 2010 pjacobs@xeekr.com . All rights reserved.\n\n// Modified by Recurity Labs GmbH\n\n// fixed/modified by Herbert Hanewinkel, www.haneWIN.de\n// check www.haneWIN.de for the latest version\n\n// cast5.js is a Javascript implementation of CAST-128, as defined in RFC 2144.\n// CAST-128 is a common OpenPGP cipher.\n\n\n// CAST5 constructor\n\n\n\nfunction openpgp_symenc_cast5() {\n  this.BlockSize = 8;\n  this.KeySize = 16;\n\n  this.setKey = function(key) {\n    this.masking = new Array(16);\n    this.rotate = new Array(16);\n\n    this.reset();\n\n    if (key.length == this.KeySize) {\n      this.keySchedule(key);\n    } else {\n      util.print_error('cast5.js: CAST-128: keys must be 16 bytes');\n      return false;\n    }\n    return true;\n  };\n\n  this.reset = function() {\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = 0;\n      this.rotate[i] = 0;\n    }\n  };\n\n  this.getBlockSize = function() {\n    return BlockSize;\n  };\n\n  this.encrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >>> 16) & 255;\n      dst[i + 6] = (l >>> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n\n  this.decrypt = function(src) {\n    var dst = new Array(src.length);\n\n    for (var i = 0; i < src.length; i += 8) {\n      var l = src[i] << 24 | src[i + 1] << 16 | src[i + 2] << 8 | src[i + 3];\n      var r = src[i + 4] << 24 | src[i + 5] << 16 | src[i + 6] << 8 | src[i + 7];\n      var t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[15], this.rotate[15]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[14], this.rotate[14]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[13], this.rotate[13]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[12], this.rotate[12]);\n      l = t;\n\n      t = r;\n      r = l ^ f3(r, this.masking[11], this.rotate[11]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[10], this.rotate[10]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[9], this.rotate[9]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[8], this.rotate[8]);\n      l = t;\n\n      t = r;\n      r = l ^ f2(r, this.masking[7], this.rotate[7]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[6], this.rotate[6]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[5], this.rotate[5]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[4], this.rotate[4]);\n      l = t;\n\n      t = r;\n      r = l ^ f1(r, this.masking[3], this.rotate[3]);\n      l = t;\n      t = r;\n      r = l ^ f3(r, this.masking[2], this.rotate[2]);\n      l = t;\n      t = r;\n      r = l ^ f2(r, this.masking[1], this.rotate[1]);\n      l = t;\n      t = r;\n      r = l ^ f1(r, this.masking[0], this.rotate[0]);\n      l = t;\n\n      dst[i] = (r >>> 24) & 255;\n      dst[i + 1] = (r >>> 16) & 255;\n      dst[i + 2] = (r >>> 8) & 255;\n      dst[i + 3] = r & 255;\n      dst[i + 4] = (l >>> 24) & 255;\n      dst[i + 5] = (l >> 16) & 255;\n      dst[i + 6] = (l >> 8) & 255;\n      dst[i + 7] = l & 255;\n    }\n\n    return dst;\n  };\n  var scheduleA = new Array(4);\n\n  scheduleA[0] = new Array(4);\n  scheduleA[0][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 0x8);\n  scheduleA[0][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[0][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[0][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n  scheduleA[1] = new Array(4);\n  scheduleA[1][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[1][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[1][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[1][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  scheduleA[2] = new Array(4);\n  scheduleA[2][0] = new Array(4, 0, 0xd, 0xf, 0xc, 0xe, 8);\n  scheduleA[2][1] = new Array(5, 2, 16 + 0, 16 + 2, 16 + 1, 16 + 3, 0xa);\n  scheduleA[2][2] = new Array(6, 3, 16 + 7, 16 + 6, 16 + 5, 16 + 4, 9);\n  scheduleA[2][3] = new Array(7, 1, 16 + 0xa, 16 + 9, 16 + 0xb, 16 + 8, 0xb);\n\n\n  scheduleA[3] = new Array(4);\n  scheduleA[3][0] = new Array(0, 6, 16 + 5, 16 + 7, 16 + 4, 16 + 6, 16 + 0);\n  scheduleA[3][1] = new Array(1, 4, 0, 2, 1, 3, 16 + 2);\n  scheduleA[3][2] = new Array(2, 5, 7, 6, 5, 4, 16 + 1);\n  scheduleA[3][3] = new Array(3, 7, 0xa, 9, 0xb, 8, 16 + 3);\n\n  var scheduleB = new Array(4);\n\n  scheduleB[0] = new Array(4);\n  scheduleB[0][0] = new Array(16 + 8, 16 + 9, 16 + 7, 16 + 6, 16 + 2);\n  scheduleB[0][1] = new Array(16 + 0xa, 16 + 0xb, 16 + 5, 16 + 4, 16 + 6);\n  scheduleB[0][2] = new Array(16 + 0xc, 16 + 0xd, 16 + 3, 16 + 2, 16 + 9);\n  scheduleB[0][3] = new Array(16 + 0xe, 16 + 0xf, 16 + 1, 16 + 0, 16 + 0xc);\n\n  scheduleB[1] = new Array(4);\n  scheduleB[1][0] = new Array(3, 2, 0xc, 0xd, 8);\n  scheduleB[1][1] = new Array(1, 0, 0xe, 0xf, 0xd);\n  scheduleB[1][2] = new Array(7, 6, 8, 9, 3);\n  scheduleB[1][3] = new Array(5, 4, 0xa, 0xb, 7);\n\n\n  scheduleB[2] = new Array(4);\n  scheduleB[2][0] = new Array(16 + 3, 16 + 2, 16 + 0xc, 16 + 0xd, 16 + 9);\n  scheduleB[2][1] = new Array(16 + 1, 16 + 0, 16 + 0xe, 16 + 0xf, 16 + 0xc);\n  scheduleB[2][2] = new Array(16 + 7, 16 + 6, 16 + 8, 16 + 9, 16 + 2);\n  scheduleB[2][3] = new Array(16 + 5, 16 + 4, 16 + 0xa, 16 + 0xb, 16 + 6);\n\n\n  scheduleB[3] = new Array(4);\n  scheduleB[3][0] = new Array(8, 9, 7, 6, 3);\n  scheduleB[3][1] = new Array(0xa, 0xb, 5, 4, 7);\n  scheduleB[3][2] = new Array(0xc, 0xd, 3, 2, 8);\n  scheduleB[3][3] = new Array(0xe, 0xf, 1, 0, 0xd);\n\n  // changed 'in' to 'inn' (in javascript 'in' is a reserved word)\n  this.keySchedule = function(inn) {\n    var t = new Array(8);\n    var k = new Array(32);\n\n    for (var i = 0; i < 4; i++) {\n      var j = i * 4;\n      t[i] = inn[j] << 24 | inn[j + 1] << 16 | inn[j + 2] << 8 | inn[j + 3];\n    }\n\n    var x = [6, 7, 4, 5];\n    var ki = 0;\n\n    for (var half = 0; half < 2; half++) {\n      for (var round = 0; round < 4; round++) {\n        for (var j = 0; j < 4; j++) {\n          var a = scheduleA[round][j];\n          var w = t[a[1]];\n\n          w ^= sBox[4][(t[a[2] >>> 2] >>> (24 - 8 * (a[2] & 3))) & 0xff];\n          w ^= sBox[5][(t[a[3] >>> 2] >>> (24 - 8 * (a[3] & 3))) & 0xff];\n          w ^= sBox[6][(t[a[4] >>> 2] >>> (24 - 8 * (a[4] & 3))) & 0xff];\n          w ^= sBox[7][(t[a[5] >>> 2] >>> (24 - 8 * (a[5] & 3))) & 0xff];\n          w ^= sBox[x[j]][(t[a[6] >>> 2] >>> (24 - 8 * (a[6] & 3))) & 0xff];\n          t[a[0]] = w;\n        }\n\n        for (var j = 0; j < 4; j++) {\n          var b = scheduleB[round][j];\n          var w = sBox[4][(t[b[0] >>> 2] >>> (24 - 8 * (b[0] & 3))) & 0xff];\n\n          w ^= sBox[5][(t[b[1] >>> 2] >>> (24 - 8 * (b[1] & 3))) & 0xff];\n          w ^= sBox[6][(t[b[2] >>> 2] >>> (24 - 8 * (b[2] & 3))) & 0xff];\n          w ^= sBox[7][(t[b[3] >>> 2] >>> (24 - 8 * (b[3] & 3))) & 0xff];\n          w ^= sBox[4 + j][(t[b[4] >>> 2] >>> (24 - 8 * (b[4] & 3))) & 0xff];\n          k[ki] = w;\n          ki++;\n        }\n      }\n    }\n\n    for (var i = 0; i < 16; i++) {\n      this.masking[i] = k[i];\n      this.rotate[i] = k[16 + i] & 0x1f;\n    }\n  };\n\n  // These are the three 'f' functions. See RFC 2144, section 2.2.\n\n  function f1(d, m, r) {\n    var t = m + d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] ^ sBox[1][(I >>> 16) & 255]) - sBox[2][(I >>> 8) & 255]) + sBox[3][I & 255];\n  }\n\n  function f2(d, m, r) {\n    var t = m ^ d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] - sBox[1][(I >>> 16) & 255]) + sBox[2][(I >>> 8) & 255]) ^ sBox[3][I & 255];\n  }\n\n  function f3(d, m, r) {\n    var t = m - d;\n    var I = (t << r) | (t >>> (32 - r));\n    return ((sBox[0][I >>> 24] + sBox[1][(I >>> 16) & 255]) ^ sBox[2][(I >>> 8) & 255]) - sBox[3][I & 255];\n  }\n\n  var sBox = new Array(8);\n  sBox[0] = new Array(\n    0x30fb40d4, 0x9fa0ff0b, 0x6beccd2f, 0x3f258c7a, 0x1e213f2f, 0x9c004dd3, 0x6003e540, 0xcf9fc949,\n    0xbfd4af27, 0x88bbbdb5, 0xe2034090, 0x98d09675, 0x6e63a0e0, 0x15c361d2, 0xc2e7661d, 0x22d4ff8e,\n    0x28683b6f, 0xc07fd059, 0xff2379c8, 0x775f50e2, 0x43c340d3, 0xdf2f8656, 0x887ca41a, 0xa2d2bd2d,\n    0xa1c9e0d6, 0x346c4819, 0x61b76d87, 0x22540f2f, 0x2abe32e1, 0xaa54166b, 0x22568e3a, 0xa2d341d0,\n    0x66db40c8, 0xa784392f, 0x004dff2f, 0x2db9d2de, 0x97943fac, 0x4a97c1d8, 0x527644b7, 0xb5f437a7,\n    0xb82cbaef, 0xd751d159, 0x6ff7f0ed, 0x5a097a1f, 0x827b68d0, 0x90ecf52e, 0x22b0c054, 0xbc8e5935,\n    0x4b6d2f7f, 0x50bb64a2, 0xd2664910, 0xbee5812d, 0xb7332290, 0xe93b159f, 0xb48ee411, 0x4bff345d,\n    0xfd45c240, 0xad31973f, 0xc4f6d02e, 0x55fc8165, 0xd5b1caad, 0xa1ac2dae, 0xa2d4b76d, 0xc19b0c50,\n    0x882240f2, 0x0c6e4f38, 0xa4e4bfd7, 0x4f5ba272, 0x564c1d2f, 0xc59c5319, 0xb949e354, 0xb04669fe,\n    0xb1b6ab8a, 0xc71358dd, 0x6385c545, 0x110f935d, 0x57538ad5, 0x6a390493, 0xe63d37e0, 0x2a54f6b3,\n    0x3a787d5f, 0x6276a0b5, 0x19a6fcdf, 0x7a42206a, 0x29f9d4d5, 0xf61b1891, 0xbb72275e, 0xaa508167,\n    0x38901091, 0xc6b505eb, 0x84c7cb8c, 0x2ad75a0f, 0x874a1427, 0xa2d1936b, 0x2ad286af, 0xaa56d291,\n    0xd7894360, 0x425c750d, 0x93b39e26, 0x187184c9, 0x6c00b32d, 0x73e2bb14, 0xa0bebc3c, 0x54623779,\n    0x64459eab, 0x3f328b82, 0x7718cf82, 0x59a2cea6, 0x04ee002e, 0x89fe78e6, 0x3fab0950, 0x325ff6c2,\n    0x81383f05, 0x6963c5c8, 0x76cb5ad6, 0xd49974c9, 0xca180dcf, 0x380782d5, 0xc7fa5cf6, 0x8ac31511,\n    0x35e79e13, 0x47da91d0, 0xf40f9086, 0xa7e2419e, 0x31366241, 0x051ef495, 0xaa573b04, 0x4a805d8d,\n    0x548300d0, 0x00322a3c, 0xbf64cddf, 0xba57a68e, 0x75c6372b, 0x50afd341, 0xa7c13275, 0x915a0bf5,\n    0x6b54bfab, 0x2b0b1426, 0xab4cc9d7, 0x449ccd82, 0xf7fbf265, 0xab85c5f3, 0x1b55db94, 0xaad4e324,\n    0xcfa4bd3f, 0x2deaa3e2, 0x9e204d02, 0xc8bd25ac, 0xeadf55b3, 0xd5bd9e98, 0xe31231b2, 0x2ad5ad6c,\n    0x954329de, 0xadbe4528, 0xd8710f69, 0xaa51c90f, 0xaa786bf6, 0x22513f1e, 0xaa51a79b, 0x2ad344cc,\n    0x7b5a41f0, 0xd37cfbad, 0x1b069505, 0x41ece491, 0xb4c332e6, 0x032268d4, 0xc9600acc, 0xce387e6d,\n    0xbf6bb16c, 0x6a70fb78, 0x0d03d9c9, 0xd4df39de, 0xe01063da, 0x4736f464, 0x5ad328d8, 0xb347cc96,\n    0x75bb0fc3, 0x98511bfb, 0x4ffbcc35, 0xb58bcf6a, 0xe11f0abc, 0xbfc5fe4a, 0xa70aec10, 0xac39570a,\n    0x3f04442f, 0x6188b153, 0xe0397a2e, 0x5727cb79, 0x9ceb418f, 0x1cacd68d, 0x2ad37c96, 0x0175cb9d,\n    0xc69dff09, 0xc75b65f0, 0xd9db40d8, 0xec0e7779, 0x4744ead4, 0xb11c3274, 0xdd24cb9e, 0x7e1c54bd,\n    0xf01144f9, 0xd2240eb1, 0x9675b3fd, 0xa3ac3755, 0xd47c27af, 0x51c85f4d, 0x56907596, 0xa5bb15e6,\n    0x580304f0, 0xca042cf1, 0x011a37ea, 0x8dbfaadb, 0x35ba3e4a, 0x3526ffa0, 0xc37b4d09, 0xbc306ed9,\n    0x98a52666, 0x5648f725, 0xff5e569d, 0x0ced63d0, 0x7c63b2cf, 0x700b45e1, 0xd5ea50f1, 0x85a92872,\n    0xaf1fbda7, 0xd4234870, 0xa7870bf3, 0x2d3b4d79, 0x42e04198, 0x0cd0ede7, 0x26470db8, 0xf881814c,\n    0x474d6ad7, 0x7c0c5e5c, 0xd1231959, 0x381b7298, 0xf5d2f4db, 0xab838653, 0x6e2f1e23, 0x83719c9e,\n    0xbd91e046, 0x9a56456e, 0xdc39200c, 0x20c8c571, 0x962bda1c, 0xe1e696ff, 0xb141ab08, 0x7cca89b9,\n    0x1a69e783, 0x02cc4843, 0xa2f7c579, 0x429ef47d, 0x427b169c, 0x5ac9f049, 0xdd8f0f00, 0x5c8165bf);\n\n  sBox[1] = new Array(\n    0x1f201094, 0xef0ba75b, 0x69e3cf7e, 0x393f4380, 0xfe61cf7a, 0xeec5207a, 0x55889c94, 0x72fc0651,\n    0xada7ef79, 0x4e1d7235, 0xd55a63ce, 0xde0436ba, 0x99c430ef, 0x5f0c0794, 0x18dcdb7d, 0xa1d6eff3,\n    0xa0b52f7b, 0x59e83605, 0xee15b094, 0xe9ffd909, 0xdc440086, 0xef944459, 0xba83ccb3, 0xe0c3cdfb,\n    0xd1da4181, 0x3b092ab1, 0xf997f1c1, 0xa5e6cf7b, 0x01420ddb, 0xe4e7ef5b, 0x25a1ff41, 0xe180f806,\n    0x1fc41080, 0x179bee7a, 0xd37ac6a9, 0xfe5830a4, 0x98de8b7f, 0x77e83f4e, 0x79929269, 0x24fa9f7b,\n    0xe113c85b, 0xacc40083, 0xd7503525, 0xf7ea615f, 0x62143154, 0x0d554b63, 0x5d681121, 0xc866c359,\n    0x3d63cf73, 0xcee234c0, 0xd4d87e87, 0x5c672b21, 0x071f6181, 0x39f7627f, 0x361e3084, 0xe4eb573b,\n    0x602f64a4, 0xd63acd9c, 0x1bbc4635, 0x9e81032d, 0x2701f50c, 0x99847ab4, 0xa0e3df79, 0xba6cf38c,\n    0x10843094, 0x2537a95e, 0xf46f6ffe, 0xa1ff3b1f, 0x208cfb6a, 0x8f458c74, 0xd9e0a227, 0x4ec73a34,\n    0xfc884f69, 0x3e4de8df, 0xef0e0088, 0x3559648d, 0x8a45388c, 0x1d804366, 0x721d9bfd, 0xa58684bb,\n    0xe8256333, 0x844e8212, 0x128d8098, 0xfed33fb4, 0xce280ae1, 0x27e19ba5, 0xd5a6c252, 0xe49754bd,\n    0xc5d655dd, 0xeb667064, 0x77840b4d, 0xa1b6a801, 0x84db26a9, 0xe0b56714, 0x21f043b7, 0xe5d05860,\n    0x54f03084, 0x066ff472, 0xa31aa153, 0xdadc4755, 0xb5625dbf, 0x68561be6, 0x83ca6b94, 0x2d6ed23b,\n    0xeccf01db, 0xa6d3d0ba, 0xb6803d5c, 0xaf77a709, 0x33b4a34c, 0x397bc8d6, 0x5ee22b95, 0x5f0e5304,\n    0x81ed6f61, 0x20e74364, 0xb45e1378, 0xde18639b, 0x881ca122, 0xb96726d1, 0x8049a7e8, 0x22b7da7b,\n    0x5e552d25, 0x5272d237, 0x79d2951c, 0xc60d894c, 0x488cb402, 0x1ba4fe5b, 0xa4b09f6b, 0x1ca815cf,\n    0xa20c3005, 0x8871df63, 0xb9de2fcb, 0x0cc6c9e9, 0x0beeff53, 0xe3214517, 0xb4542835, 0x9f63293c,\n    0xee41e729, 0x6e1d2d7c, 0x50045286, 0x1e6685f3, 0xf33401c6, 0x30a22c95, 0x31a70850, 0x60930f13,\n    0x73f98417, 0xa1269859, 0xec645c44, 0x52c877a9, 0xcdff33a6, 0xa02b1741, 0x7cbad9a2, 0x2180036f,\n    0x50d99c08, 0xcb3f4861, 0xc26bd765, 0x64a3f6ab, 0x80342676, 0x25a75e7b, 0xe4e6d1fc, 0x20c710e6,\n    0xcdf0b680, 0x17844d3b, 0x31eef84d, 0x7e0824e4, 0x2ccb49eb, 0x846a3bae, 0x8ff77888, 0xee5d60f6,\n    0x7af75673, 0x2fdd5cdb, 0xa11631c1, 0x30f66f43, 0xb3faec54, 0x157fd7fa, 0xef8579cc, 0xd152de58,\n    0xdb2ffd5e, 0x8f32ce19, 0x306af97a, 0x02f03ef8, 0x99319ad5, 0xc242fa0f, 0xa7e3ebb0, 0xc68e4906,\n    0xb8da230c, 0x80823028, 0xdcdef3c8, 0xd35fb171, 0x088a1bc8, 0xbec0c560, 0x61a3c9e8, 0xbca8f54d,\n    0xc72feffa, 0x22822e99, 0x82c570b4, 0xd8d94e89, 0x8b1c34bc, 0x301e16e6, 0x273be979, 0xb0ffeaa6,\n    0x61d9b8c6, 0x00b24869, 0xb7ffce3f, 0x08dc283b, 0x43daf65a, 0xf7e19798, 0x7619b72f, 0x8f1c9ba4,\n    0xdc8637a0, 0x16a7d3b1, 0x9fc393b7, 0xa7136eeb, 0xc6bcc63e, 0x1a513742, 0xef6828bc, 0x520365d6,\n    0x2d6a77ab, 0x3527ed4b, 0x821fd216, 0x095c6e2e, 0xdb92f2fb, 0x5eea29cb, 0x145892f5, 0x91584f7f,\n    0x5483697b, 0x2667a8cc, 0x85196048, 0x8c4bacea, 0x833860d4, 0x0d23e0f9, 0x6c387e8a, 0x0ae6d249,\n    0xb284600c, 0xd835731d, 0xdcb1c647, 0xac4c56ea, 0x3ebd81b3, 0x230eabb0, 0x6438bc87, 0xf0b5b1fa,\n    0x8f5ea2b3, 0xfc184642, 0x0a036b7a, 0x4fb089bd, 0x649da589, 0xa345415e, 0x5c038323, 0x3e5d3bb9,\n    0x43d79572, 0x7e6dd07c, 0x06dfdf1e, 0x6c6cc4ef, 0x7160a539, 0x73bfbe70, 0x83877605, 0x4523ecf1);\n\n  sBox[2] = new Array(\n    0x8defc240, 0x25fa5d9f, 0xeb903dbf, 0xe810c907, 0x47607fff, 0x369fe44b, 0x8c1fc644, 0xaececa90,\n    0xbeb1f9bf, 0xeefbcaea, 0xe8cf1950, 0x51df07ae, 0x920e8806, 0xf0ad0548, 0xe13c8d83, 0x927010d5,\n    0x11107d9f, 0x07647db9, 0xb2e3e4d4, 0x3d4f285e, 0xb9afa820, 0xfade82e0, 0xa067268b, 0x8272792e,\n    0x553fb2c0, 0x489ae22b, 0xd4ef9794, 0x125e3fbc, 0x21fffcee, 0x825b1bfd, 0x9255c5ed, 0x1257a240,\n    0x4e1a8302, 0xbae07fff, 0x528246e7, 0x8e57140e, 0x3373f7bf, 0x8c9f8188, 0xa6fc4ee8, 0xc982b5a5,\n    0xa8c01db7, 0x579fc264, 0x67094f31, 0xf2bd3f5f, 0x40fff7c1, 0x1fb78dfc, 0x8e6bd2c1, 0x437be59b,\n    0x99b03dbf, 0xb5dbc64b, 0x638dc0e6, 0x55819d99, 0xa197c81c, 0x4a012d6e, 0xc5884a28, 0xccc36f71,\n    0xb843c213, 0x6c0743f1, 0x8309893c, 0x0feddd5f, 0x2f7fe850, 0xd7c07f7e, 0x02507fbf, 0x5afb9a04,\n    0xa747d2d0, 0x1651192e, 0xaf70bf3e, 0x58c31380, 0x5f98302e, 0x727cc3c4, 0x0a0fb402, 0x0f7fef82,\n    0x8c96fdad, 0x5d2c2aae, 0x8ee99a49, 0x50da88b8, 0x8427f4a0, 0x1eac5790, 0x796fb449, 0x8252dc15,\n    0xefbd7d9b, 0xa672597d, 0xada840d8, 0x45f54504, 0xfa5d7403, 0xe83ec305, 0x4f91751a, 0x925669c2,\n    0x23efe941, 0xa903f12e, 0x60270df2, 0x0276e4b6, 0x94fd6574, 0x927985b2, 0x8276dbcb, 0x02778176,\n    0xf8af918d, 0x4e48f79e, 0x8f616ddf, 0xe29d840e, 0x842f7d83, 0x340ce5c8, 0x96bbb682, 0x93b4b148,\n    0xef303cab, 0x984faf28, 0x779faf9b, 0x92dc560d, 0x224d1e20, 0x8437aa88, 0x7d29dc96, 0x2756d3dc,\n    0x8b907cee, 0xb51fd240, 0xe7c07ce3, 0xe566b4a1, 0xc3e9615e, 0x3cf8209d, 0x6094d1e3, 0xcd9ca341,\n    0x5c76460e, 0x00ea983b, 0xd4d67881, 0xfd47572c, 0xf76cedd9, 0xbda8229c, 0x127dadaa, 0x438a074e,\n    0x1f97c090, 0x081bdb8a, 0x93a07ebe, 0xb938ca15, 0x97b03cff, 0x3dc2c0f8, 0x8d1ab2ec, 0x64380e51,\n    0x68cc7bfb, 0xd90f2788, 0x12490181, 0x5de5ffd4, 0xdd7ef86a, 0x76a2e214, 0xb9a40368, 0x925d958f,\n    0x4b39fffa, 0xba39aee9, 0xa4ffd30b, 0xfaf7933b, 0x6d498623, 0x193cbcfa, 0x27627545, 0x825cf47a,\n    0x61bd8ba0, 0xd11e42d1, 0xcead04f4, 0x127ea392, 0x10428db7, 0x8272a972, 0x9270c4a8, 0x127de50b,\n    0x285ba1c8, 0x3c62f44f, 0x35c0eaa5, 0xe805d231, 0x428929fb, 0xb4fcdf82, 0x4fb66a53, 0x0e7dc15b,\n    0x1f081fab, 0x108618ae, 0xfcfd086d, 0xf9ff2889, 0x694bcc11, 0x236a5cae, 0x12deca4d, 0x2c3f8cc5,\n    0xd2d02dfe, 0xf8ef5896, 0xe4cf52da, 0x95155b67, 0x494a488c, 0xb9b6a80c, 0x5c8f82bc, 0x89d36b45,\n    0x3a609437, 0xec00c9a9, 0x44715253, 0x0a874b49, 0xd773bc40, 0x7c34671c, 0x02717ef6, 0x4feb5536,\n    0xa2d02fff, 0xd2bf60c4, 0xd43f03c0, 0x50b4ef6d, 0x07478cd1, 0x006e1888, 0xa2e53f55, 0xb9e6d4bc,\n    0xa2048016, 0x97573833, 0xd7207d67, 0xde0f8f3d, 0x72f87b33, 0xabcc4f33, 0x7688c55d, 0x7b00a6b0,\n    0x947b0001, 0x570075d2, 0xf9bb88f8, 0x8942019e, 0x4264a5ff, 0x856302e0, 0x72dbd92b, 0xee971b69,\n    0x6ea22fde, 0x5f08ae2b, 0xaf7a616d, 0xe5c98767, 0xcf1febd2, 0x61efc8c2, 0xf1ac2571, 0xcc8239c2,\n    0x67214cb8, 0xb1e583d1, 0xb7dc3e62, 0x7f10bdce, 0xf90a5c38, 0x0ff0443d, 0x606e6dc6, 0x60543a49,\n    0x5727c148, 0x2be98a1d, 0x8ab41738, 0x20e1be24, 0xaf96da0f, 0x68458425, 0x99833be5, 0x600d457d,\n    0x282f9350, 0x8334b362, 0xd91d1120, 0x2b6d8da0, 0x642b1e31, 0x9c305a00, 0x52bce688, 0x1b03588a,\n    0xf7baefd5, 0x4142ed9c, 0xa4315c11, 0x83323ec5, 0xdfef4636, 0xa133c501, 0xe9d3531c, 0xee353783);\n\n  sBox[3] = new Array(\n    0x9db30420, 0x1fb6e9de, 0xa7be7bef, 0xd273a298, 0x4a4f7bdb, 0x64ad8c57, 0x85510443, 0xfa020ed1,\n    0x7e287aff, 0xe60fb663, 0x095f35a1, 0x79ebf120, 0xfd059d43, 0x6497b7b1, 0xf3641f63, 0x241e4adf,\n    0x28147f5f, 0x4fa2b8cd, 0xc9430040, 0x0cc32220, 0xfdd30b30, 0xc0a5374f, 0x1d2d00d9, 0x24147b15,\n    0xee4d111a, 0x0fca5167, 0x71ff904c, 0x2d195ffe, 0x1a05645f, 0x0c13fefe, 0x081b08ca, 0x05170121,\n    0x80530100, 0xe83e5efe, 0xac9af4f8, 0x7fe72701, 0xd2b8ee5f, 0x06df4261, 0xbb9e9b8a, 0x7293ea25,\n    0xce84ffdf, 0xf5718801, 0x3dd64b04, 0xa26f263b, 0x7ed48400, 0x547eebe6, 0x446d4ca0, 0x6cf3d6f5,\n    0x2649abdf, 0xaea0c7f5, 0x36338cc1, 0x503f7e93, 0xd3772061, 0x11b638e1, 0x72500e03, 0xf80eb2bb,\n    0xabe0502e, 0xec8d77de, 0x57971e81, 0xe14f6746, 0xc9335400, 0x6920318f, 0x081dbb99, 0xffc304a5,\n    0x4d351805, 0x7f3d5ce3, 0xa6c866c6, 0x5d5bcca9, 0xdaec6fea, 0x9f926f91, 0x9f46222f, 0x3991467d,\n    0xa5bf6d8e, 0x1143c44f, 0x43958302, 0xd0214eeb, 0x022083b8, 0x3fb6180c, 0x18f8931e, 0x281658e6,\n    0x26486e3e, 0x8bd78a70, 0x7477e4c1, 0xb506e07c, 0xf32d0a25, 0x79098b02, 0xe4eabb81, 0x28123b23,\n    0x69dead38, 0x1574ca16, 0xdf871b62, 0x211c40b7, 0xa51a9ef9, 0x0014377b, 0x041e8ac8, 0x09114003,\n    0xbd59e4d2, 0xe3d156d5, 0x4fe876d5, 0x2f91a340, 0x557be8de, 0x00eae4a7, 0x0ce5c2ec, 0x4db4bba6,\n    0xe756bdff, 0xdd3369ac, 0xec17b035, 0x06572327, 0x99afc8b0, 0x56c8c391, 0x6b65811c, 0x5e146119,\n    0x6e85cb75, 0xbe07c002, 0xc2325577, 0x893ff4ec, 0x5bbfc92d, 0xd0ec3b25, 0xb7801ab7, 0x8d6d3b24,\n    0x20c763ef, 0xc366a5fc, 0x9c382880, 0x0ace3205, 0xaac9548a, 0xeca1d7c7, 0x041afa32, 0x1d16625a,\n    0x6701902c, 0x9b757a54, 0x31d477f7, 0x9126b031, 0x36cc6fdb, 0xc70b8b46, 0xd9e66a48, 0x56e55a79,\n    0x026a4ceb, 0x52437eff, 0x2f8f76b4, 0x0df980a5, 0x8674cde3, 0xedda04eb, 0x17a9be04, 0x2c18f4df,\n    0xb7747f9d, 0xab2af7b4, 0xefc34d20, 0x2e096b7c, 0x1741a254, 0xe5b6a035, 0x213d42f6, 0x2c1c7c26,\n    0x61c2f50f, 0x6552daf9, 0xd2c231f8, 0x25130f69, 0xd8167fa2, 0x0418f2c8, 0x001a96a6, 0x0d1526ab,\n    0x63315c21, 0x5e0a72ec, 0x49bafefd, 0x187908d9, 0x8d0dbd86, 0x311170a7, 0x3e9b640c, 0xcc3e10d7,\n    0xd5cad3b6, 0x0caec388, 0xf73001e1, 0x6c728aff, 0x71eae2a1, 0x1f9af36e, 0xcfcbd12f, 0xc1de8417,\n    0xac07be6b, 0xcb44a1d8, 0x8b9b0f56, 0x013988c3, 0xb1c52fca, 0xb4be31cd, 0xd8782806, 0x12a3a4e2,\n    0x6f7de532, 0x58fd7eb6, 0xd01ee900, 0x24adffc2, 0xf4990fc5, 0x9711aac5, 0x001d7b95, 0x82e5e7d2,\n    0x109873f6, 0x00613096, 0xc32d9521, 0xada121ff, 0x29908415, 0x7fbb977f, 0xaf9eb3db, 0x29c9ed2a,\n    0x5ce2a465, 0xa730f32c, 0xd0aa3fe8, 0x8a5cc091, 0xd49e2ce7, 0x0ce454a9, 0xd60acd86, 0x015f1919,\n    0x77079103, 0xdea03af6, 0x78a8565e, 0xdee356df, 0x21f05cbe, 0x8b75e387, 0xb3c50651, 0xb8a5c3ef,\n    0xd8eeb6d2, 0xe523be77, 0xc2154529, 0x2f69efdf, 0xafe67afb, 0xf470c4b2, 0xf3e0eb5b, 0xd6cc9876,\n    0x39e4460c, 0x1fda8538, 0x1987832f, 0xca007367, 0xa99144f8, 0x296b299e, 0x492fc295, 0x9266beab,\n    0xb5676e69, 0x9bd3ddda, 0xdf7e052f, 0xdb25701c, 0x1b5e51ee, 0xf65324e6, 0x6afce36c, 0x0316cc04,\n    0x8644213e, 0xb7dc59d0, 0x7965291f, 0xccd6fd43, 0x41823979, 0x932bcdf6, 0xb657c34d, 0x4edfd282,\n    0x7ae5290c, 0x3cb9536b, 0x851e20fe, 0x9833557e, 0x13ecf0b0, 0xd3ffb372, 0x3f85c5c1, 0x0aef7ed2);\n\n  sBox[4] = new Array(\n    0x7ec90c04, 0x2c6e74b9, 0x9b0e66df, 0xa6337911, 0xb86a7fff, 0x1dd358f5, 0x44dd9d44, 0x1731167f,\n    0x08fbf1fa, 0xe7f511cc, 0xd2051b00, 0x735aba00, 0x2ab722d8, 0x386381cb, 0xacf6243a, 0x69befd7a,\n    0xe6a2e77f, 0xf0c720cd, 0xc4494816, 0xccf5c180, 0x38851640, 0x15b0a848, 0xe68b18cb, 0x4caadeff,\n    0x5f480a01, 0x0412b2aa, 0x259814fc, 0x41d0efe2, 0x4e40b48d, 0x248eb6fb, 0x8dba1cfe, 0x41a99b02,\n    0x1a550a04, 0xba8f65cb, 0x7251f4e7, 0x95a51725, 0xc106ecd7, 0x97a5980a, 0xc539b9aa, 0x4d79fe6a,\n    0xf2f3f763, 0x68af8040, 0xed0c9e56, 0x11b4958b, 0xe1eb5a88, 0x8709e6b0, 0xd7e07156, 0x4e29fea7,\n    0x6366e52d, 0x02d1c000, 0xc4ac8e05, 0x9377f571, 0x0c05372a, 0x578535f2, 0x2261be02, 0xd642a0c9,\n    0xdf13a280, 0x74b55bd2, 0x682199c0, 0xd421e5ec, 0x53fb3ce8, 0xc8adedb3, 0x28a87fc9, 0x3d959981,\n    0x5c1ff900, 0xfe38d399, 0x0c4eff0b, 0x062407ea, 0xaa2f4fb1, 0x4fb96976, 0x90c79505, 0xb0a8a774,\n    0xef55a1ff, 0xe59ca2c2, 0xa6b62d27, 0xe66a4263, 0xdf65001f, 0x0ec50966, 0xdfdd55bc, 0x29de0655,\n    0x911e739a, 0x17af8975, 0x32c7911c, 0x89f89468, 0x0d01e980, 0x524755f4, 0x03b63cc9, 0x0cc844b2,\n    0xbcf3f0aa, 0x87ac36e9, 0xe53a7426, 0x01b3d82b, 0x1a9e7449, 0x64ee2d7e, 0xcddbb1da, 0x01c94910,\n    0xb868bf80, 0x0d26f3fd, 0x9342ede7, 0x04a5c284, 0x636737b6, 0x50f5b616, 0xf24766e3, 0x8eca36c1,\n    0x136e05db, 0xfef18391, 0xfb887a37, 0xd6e7f7d4, 0xc7fb7dc9, 0x3063fcdf, 0xb6f589de, 0xec2941da,\n    0x26e46695, 0xb7566419, 0xf654efc5, 0xd08d58b7, 0x48925401, 0xc1bacb7f, 0xe5ff550f, 0xb6083049,\n    0x5bb5d0e8, 0x87d72e5a, 0xab6a6ee1, 0x223a66ce, 0xc62bf3cd, 0x9e0885f9, 0x68cb3e47, 0x086c010f,\n    0xa21de820, 0xd18b69de, 0xf3f65777, 0xfa02c3f6, 0x407edac3, 0xcbb3d550, 0x1793084d, 0xb0d70eba,\n    0x0ab378d5, 0xd951fb0c, 0xded7da56, 0x4124bbe4, 0x94ca0b56, 0x0f5755d1, 0xe0e1e56e, 0x6184b5be,\n    0x580a249f, 0x94f74bc0, 0xe327888e, 0x9f7b5561, 0xc3dc0280, 0x05687715, 0x646c6bd7, 0x44904db3,\n    0x66b4f0a3, 0xc0f1648a, 0x697ed5af, 0x49e92ff6, 0x309e374f, 0x2cb6356a, 0x85808573, 0x4991f840,\n    0x76f0ae02, 0x083be84d, 0x28421c9a, 0x44489406, 0x736e4cb8, 0xc1092910, 0x8bc95fc6, 0x7d869cf4,\n    0x134f616f, 0x2e77118d, 0xb31b2be1, 0xaa90b472, 0x3ca5d717, 0x7d161bba, 0x9cad9010, 0xaf462ba2,\n    0x9fe459d2, 0x45d34559, 0xd9f2da13, 0xdbc65487, 0xf3e4f94e, 0x176d486f, 0x097c13ea, 0x631da5c7,\n    0x445f7382, 0x175683f4, 0xcdc66a97, 0x70be0288, 0xb3cdcf72, 0x6e5dd2f3, 0x20936079, 0x459b80a5,\n    0xbe60e2db, 0xa9c23101, 0xeba5315c, 0x224e42f2, 0x1c5c1572, 0xf6721b2c, 0x1ad2fff3, 0x8c25404e,\n    0x324ed72f, 0x4067b7fd, 0x0523138e, 0x5ca3bc78, 0xdc0fd66e, 0x75922283, 0x784d6b17, 0x58ebb16e,\n    0x44094f85, 0x3f481d87, 0xfcfeae7b, 0x77b5ff76, 0x8c2302bf, 0xaaf47556, 0x5f46b02a, 0x2b092801,\n    0x3d38f5f7, 0x0ca81f36, 0x52af4a8a, 0x66d5e7c0, 0xdf3b0874, 0x95055110, 0x1b5ad7a8, 0xf61ed5ad,\n    0x6cf6e479, 0x20758184, 0xd0cefa65, 0x88f7be58, 0x4a046826, 0x0ff6f8f3, 0xa09c7f70, 0x5346aba0,\n    0x5ce96c28, 0xe176eda3, 0x6bac307f, 0x376829d2, 0x85360fa9, 0x17e3fe2a, 0x24b79767, 0xf5a96b20,\n    0xd6cd2595, 0x68ff1ebf, 0x7555442c, 0xf19f06be, 0xf9e0659a, 0xeeb9491d, 0x34010718, 0xbb30cab8,\n    0xe822fe15, 0x88570983, 0x750e6249, 0xda627e55, 0x5e76ffa8, 0xb1534546, 0x6d47de08, 0xefe9e7d4);\n\n  sBox[5] = new Array(\n    0xf6fa8f9d, 0x2cac6ce1, 0x4ca34867, 0xe2337f7c, 0x95db08e7, 0x016843b4, 0xeced5cbc, 0x325553ac,\n    0xbf9f0960, 0xdfa1e2ed, 0x83f0579d, 0x63ed86b9, 0x1ab6a6b8, 0xde5ebe39, 0xf38ff732, 0x8989b138,\n    0x33f14961, 0xc01937bd, 0xf506c6da, 0xe4625e7e, 0xa308ea99, 0x4e23e33c, 0x79cbd7cc, 0x48a14367,\n    0xa3149619, 0xfec94bd5, 0xa114174a, 0xeaa01866, 0xa084db2d, 0x09a8486f, 0xa888614a, 0x2900af98,\n    0x01665991, 0xe1992863, 0xc8f30c60, 0x2e78ef3c, 0xd0d51932, 0xcf0fec14, 0xf7ca07d2, 0xd0a82072,\n    0xfd41197e, 0x9305a6b0, 0xe86be3da, 0x74bed3cd, 0x372da53c, 0x4c7f4448, 0xdab5d440, 0x6dba0ec3,\n    0x083919a7, 0x9fbaeed9, 0x49dbcfb0, 0x4e670c53, 0x5c3d9c01, 0x64bdb941, 0x2c0e636a, 0xba7dd9cd,\n    0xea6f7388, 0xe70bc762, 0x35f29adb, 0x5c4cdd8d, 0xf0d48d8c, 0xb88153e2, 0x08a19866, 0x1ae2eac8,\n    0x284caf89, 0xaa928223, 0x9334be53, 0x3b3a21bf, 0x16434be3, 0x9aea3906, 0xefe8c36e, 0xf890cdd9,\n    0x80226dae, 0xc340a4a3, 0xdf7e9c09, 0xa694a807, 0x5b7c5ecc, 0x221db3a6, 0x9a69a02f, 0x68818a54,\n    0xceb2296f, 0x53c0843a, 0xfe893655, 0x25bfe68a, 0xb4628abc, 0xcf222ebf, 0x25ac6f48, 0xa9a99387,\n    0x53bddb65, 0xe76ffbe7, 0xe967fd78, 0x0ba93563, 0x8e342bc1, 0xe8a11be9, 0x4980740d, 0xc8087dfc,\n    0x8de4bf99, 0xa11101a0, 0x7fd37975, 0xda5a26c0, 0xe81f994f, 0x9528cd89, 0xfd339fed, 0xb87834bf,\n    0x5f04456d, 0x22258698, 0xc9c4c83b, 0x2dc156be, 0x4f628daa, 0x57f55ec5, 0xe2220abe, 0xd2916ebf,\n    0x4ec75b95, 0x24f2c3c0, 0x42d15d99, 0xcd0d7fa0, 0x7b6e27ff, 0xa8dc8af0, 0x7345c106, 0xf41e232f,\n    0x35162386, 0xe6ea8926, 0x3333b094, 0x157ec6f2, 0x372b74af, 0x692573e4, 0xe9a9d848, 0xf3160289,\n    0x3a62ef1d, 0xa787e238, 0xf3a5f676, 0x74364853, 0x20951063, 0x4576698d, 0xb6fad407, 0x592af950,\n    0x36f73523, 0x4cfb6e87, 0x7da4cec0, 0x6c152daa, 0xcb0396a8, 0xc50dfe5d, 0xfcd707ab, 0x0921c42f,\n    0x89dff0bb, 0x5fe2be78, 0x448f4f33, 0x754613c9, 0x2b05d08d, 0x48b9d585, 0xdc049441, 0xc8098f9b,\n    0x7dede786, 0xc39a3373, 0x42410005, 0x6a091751, 0x0ef3c8a6, 0x890072d6, 0x28207682, 0xa9a9f7be,\n    0xbf32679d, 0xd45b5b75, 0xb353fd00, 0xcbb0e358, 0x830f220a, 0x1f8fb214, 0xd372cf08, 0xcc3c4a13,\n    0x8cf63166, 0x061c87be, 0x88c98f88, 0x6062e397, 0x47cf8e7a, 0xb6c85283, 0x3cc2acfb, 0x3fc06976,\n    0x4e8f0252, 0x64d8314d, 0xda3870e3, 0x1e665459, 0xc10908f0, 0x513021a5, 0x6c5b68b7, 0x822f8aa0,\n    0x3007cd3e, 0x74719eef, 0xdc872681, 0x073340d4, 0x7e432fd9, 0x0c5ec241, 0x8809286c, 0xf592d891,\n    0x08a930f6, 0x957ef305, 0xb7fbffbd, 0xc266e96f, 0x6fe4ac98, 0xb173ecc0, 0xbc60b42a, 0x953498da,\n    0xfba1ae12, 0x2d4bd736, 0x0f25faab, 0xa4f3fceb, 0xe2969123, 0x257f0c3d, 0x9348af49, 0x361400bc,\n    0xe8816f4a, 0x3814f200, 0xa3f94043, 0x9c7a54c2, 0xbc704f57, 0xda41e7f9, 0xc25ad33a, 0x54f4a084,\n    0xb17f5505, 0x59357cbe, 0xedbd15c8, 0x7f97c5ab, 0xba5ac7b5, 0xb6f6deaf, 0x3a479c3a, 0x5302da25,\n    0x653d7e6a, 0x54268d49, 0x51a477ea, 0x5017d55b, 0xd7d25d88, 0x44136c76, 0x0404a8c8, 0xb8e5a121,\n    0xb81a928a, 0x60ed5869, 0x97c55b96, 0xeaec991b, 0x29935913, 0x01fdb7f1, 0x088e8dfa, 0x9ab6f6f5,\n    0x3b4cbf9f, 0x4a5de3ab, 0xe6051d35, 0xa0e1d855, 0xd36b4cf1, 0xf544edeb, 0xb0e93524, 0xbebb8fbd,\n    0xa2d762cf, 0x49c92f54, 0x38b5f331, 0x7128a454, 0x48392905, 0xa65b1db8, 0x851c97bd, 0xd675cf2f);\n\n  sBox[6] = new Array(\n    0x85e04019, 0x332bf567, 0x662dbfff, 0xcfc65693, 0x2a8d7f6f, 0xab9bc912, 0xde6008a1, 0x2028da1f,\n    0x0227bce7, 0x4d642916, 0x18fac300, 0x50f18b82, 0x2cb2cb11, 0xb232e75c, 0x4b3695f2, 0xb28707de,\n    0xa05fbcf6, 0xcd4181e9, 0xe150210c, 0xe24ef1bd, 0xb168c381, 0xfde4e789, 0x5c79b0d8, 0x1e8bfd43,\n    0x4d495001, 0x38be4341, 0x913cee1d, 0x92a79c3f, 0x089766be, 0xbaeeadf4, 0x1286becf, 0xb6eacb19,\n    0x2660c200, 0x7565bde4, 0x64241f7a, 0x8248dca9, 0xc3b3ad66, 0x28136086, 0x0bd8dfa8, 0x356d1cf2,\n    0x107789be, 0xb3b2e9ce, 0x0502aa8f, 0x0bc0351e, 0x166bf52a, 0xeb12ff82, 0xe3486911, 0xd34d7516,\n    0x4e7b3aff, 0x5f43671b, 0x9cf6e037, 0x4981ac83, 0x334266ce, 0x8c9341b7, 0xd0d854c0, 0xcb3a6c88,\n    0x47bc2829, 0x4725ba37, 0xa66ad22b, 0x7ad61f1e, 0x0c5cbafa, 0x4437f107, 0xb6e79962, 0x42d2d816,\n    0x0a961288, 0xe1a5c06e, 0x13749e67, 0x72fc081a, 0xb1d139f7, 0xf9583745, 0xcf19df58, 0xbec3f756,\n    0xc06eba30, 0x07211b24, 0x45c28829, 0xc95e317f, 0xbc8ec511, 0x38bc46e9, 0xc6e6fa14, 0xbae8584a,\n    0xad4ebc46, 0x468f508b, 0x7829435f, 0xf124183b, 0x821dba9f, 0xaff60ff4, 0xea2c4e6d, 0x16e39264,\n    0x92544a8b, 0x009b4fc3, 0xaba68ced, 0x9ac96f78, 0x06a5b79a, 0xb2856e6e, 0x1aec3ca9, 0xbe838688,\n    0x0e0804e9, 0x55f1be56, 0xe7e5363b, 0xb3a1f25d, 0xf7debb85, 0x61fe033c, 0x16746233, 0x3c034c28,\n    0xda6d0c74, 0x79aac56c, 0x3ce4e1ad, 0x51f0c802, 0x98f8f35a, 0x1626a49f, 0xeed82b29, 0x1d382fe3,\n    0x0c4fb99a, 0xbb325778, 0x3ec6d97b, 0x6e77a6a9, 0xcb658b5c, 0xd45230c7, 0x2bd1408b, 0x60c03eb7,\n    0xb9068d78, 0xa33754f4, 0xf430c87d, 0xc8a71302, 0xb96d8c32, 0xebd4e7be, 0xbe8b9d2d, 0x7979fb06,\n    0xe7225308, 0x8b75cf77, 0x11ef8da4, 0xe083c858, 0x8d6b786f, 0x5a6317a6, 0xfa5cf7a0, 0x5dda0033,\n    0xf28ebfb0, 0xf5b9c310, 0xa0eac280, 0x08b9767a, 0xa3d9d2b0, 0x79d34217, 0x021a718d, 0x9ac6336a,\n    0x2711fd60, 0x438050e3, 0x069908a8, 0x3d7fedc4, 0x826d2bef, 0x4eeb8476, 0x488dcf25, 0x36c9d566,\n    0x28e74e41, 0xc2610aca, 0x3d49a9cf, 0xbae3b9df, 0xb65f8de6, 0x92aeaf64, 0x3ac7d5e6, 0x9ea80509,\n    0xf22b017d, 0xa4173f70, 0xdd1e16c3, 0x15e0d7f9, 0x50b1b887, 0x2b9f4fd5, 0x625aba82, 0x6a017962,\n    0x2ec01b9c, 0x15488aa9, 0xd716e740, 0x40055a2c, 0x93d29a22, 0xe32dbf9a, 0x058745b9, 0x3453dc1e,\n    0xd699296e, 0x496cff6f, 0x1c9f4986, 0xdfe2ed07, 0xb87242d1, 0x19de7eae, 0x053e561a, 0x15ad6f8c,\n    0x66626c1c, 0x7154c24c, 0xea082b2a, 0x93eb2939, 0x17dcb0f0, 0x58d4f2ae, 0x9ea294fb, 0x52cf564c,\n    0x9883fe66, 0x2ec40581, 0x763953c3, 0x01d6692e, 0xd3a0c108, 0xa1e7160e, 0xe4f2dfa6, 0x693ed285,\n    0x74904698, 0x4c2b0edd, 0x4f757656, 0x5d393378, 0xa132234f, 0x3d321c5d, 0xc3f5e194, 0x4b269301,\n    0xc79f022f, 0x3c997e7e, 0x5e4f9504, 0x3ffafbbd, 0x76f7ad0e, 0x296693f4, 0x3d1fce6f, 0xc61e45be,\n    0xd3b5ab34, 0xf72bf9b7, 0x1b0434c0, 0x4e72b567, 0x5592a33d, 0xb5229301, 0xcfd2a87f, 0x60aeb767,\n    0x1814386b, 0x30bcc33d, 0x38a0c07d, 0xfd1606f2, 0xc363519b, 0x589dd390, 0x5479f8e6, 0x1cb8d647,\n    0x97fd61a9, 0xea7759f4, 0x2d57539d, 0x569a58cf, 0xe84e63ad, 0x462e1b78, 0x6580f87e, 0xf3817914,\n    0x91da55f4, 0x40a230f3, 0xd1988f35, 0xb6e318d2, 0x3ffa50bc, 0x3d40f021, 0xc3c0bdae, 0x4958c24c,\n    0x518f36b2, 0x84b1d370, 0x0fedce83, 0x878ddada, 0xf2a279c7, 0x94e01be8, 0x90716f4b, 0x954b8aa3);\n\n  sBox[7] = new Array(\n    0xe216300d, 0xbbddfffc, 0xa7ebdabd, 0x35648095, 0x7789f8b7, 0xe6c1121b, 0x0e241600, 0x052ce8b5,\n    0x11a9cfb0, 0xe5952f11, 0xece7990a, 0x9386d174, 0x2a42931c, 0x76e38111, 0xb12def3a, 0x37ddddfc,\n    0xde9adeb1, 0x0a0cc32c, 0xbe197029, 0x84a00940, 0xbb243a0f, 0xb4d137cf, 0xb44e79f0, 0x049eedfd,\n    0x0b15a15d, 0x480d3168, 0x8bbbde5a, 0x669ded42, 0xc7ece831, 0x3f8f95e7, 0x72df191b, 0x7580330d,\n    0x94074251, 0x5c7dcdfa, 0xabbe6d63, 0xaa402164, 0xb301d40a, 0x02e7d1ca, 0x53571dae, 0x7a3182a2,\n    0x12a8ddec, 0xfdaa335d, 0x176f43e8, 0x71fb46d4, 0x38129022, 0xce949ad4, 0xb84769ad, 0x965bd862,\n    0x82f3d055, 0x66fb9767, 0x15b80b4e, 0x1d5b47a0, 0x4cfde06f, 0xc28ec4b8, 0x57e8726e, 0x647a78fc,\n    0x99865d44, 0x608bd593, 0x6c200e03, 0x39dc5ff6, 0x5d0b00a3, 0xae63aff2, 0x7e8bd632, 0x70108c0c,\n    0xbbd35049, 0x2998df04, 0x980cf42a, 0x9b6df491, 0x9e7edd53, 0x06918548, 0x58cb7e07, 0x3b74ef2e,\n    0x522fffb1, 0xd24708cc, 0x1c7e27cd, 0xa4eb215b, 0x3cf1d2e2, 0x19b47a38, 0x424f7618, 0x35856039,\n    0x9d17dee7, 0x27eb35e6, 0xc9aff67b, 0x36baf5b8, 0x09c467cd, 0xc18910b1, 0xe11dbf7b, 0x06cd1af8,\n    0x7170c608, 0x2d5e3354, 0xd4de495a, 0x64c6d006, 0xbcc0c62c, 0x3dd00db3, 0x708f8f34, 0x77d51b42,\n    0x264f620f, 0x24b8d2bf, 0x15c1b79e, 0x46a52564, 0xf8d7e54e, 0x3e378160, 0x7895cda5, 0x859c15a5,\n    0xe6459788, 0xc37bc75f, 0xdb07ba0c, 0x0676a3ab, 0x7f229b1e, 0x31842e7b, 0x24259fd7, 0xf8bef472,\n    0x835ffcb8, 0x6df4c1f2, 0x96f5b195, 0xfd0af0fc, 0xb0fe134c, 0xe2506d3d, 0x4f9b12ea, 0xf215f225,\n    0xa223736f, 0x9fb4c428, 0x25d04979, 0x34c713f8, 0xc4618187, 0xea7a6e98, 0x7cd16efc, 0x1436876c,\n    0xf1544107, 0xbedeee14, 0x56e9af27, 0xa04aa441, 0x3cf7c899, 0x92ecbae6, 0xdd67016d, 0x151682eb,\n    0xa842eedf, 0xfdba60b4, 0xf1907b75, 0x20e3030f, 0x24d8c29e, 0xe139673b, 0xefa63fb8, 0x71873054,\n    0xb6f2cf3b, 0x9f326442, 0xcb15a4cc, 0xb01a4504, 0xf1e47d8d, 0x844a1be5, 0xbae7dfdc, 0x42cbda70,\n    0xcd7dae0a, 0x57e85b7a, 0xd53f5af6, 0x20cf4d8c, 0xcea4d428, 0x79d130a4, 0x3486ebfb, 0x33d3cddc,\n    0x77853b53, 0x37effcb5, 0xc5068778, 0xe580b3e6, 0x4e68b8f4, 0xc5c8b37e, 0x0d809ea2, 0x398feb7c,\n    0x132a4f94, 0x43b7950e, 0x2fee7d1c, 0x223613bd, 0xdd06caa2, 0x37df932b, 0xc4248289, 0xacf3ebc3,\n    0x5715f6b7, 0xef3478dd, 0xf267616f, 0xc148cbe4, 0x9052815e, 0x5e410fab, 0xb48a2465, 0x2eda7fa4,\n    0xe87b40e4, 0xe98ea084, 0x5889e9e1, 0xefd390fc, 0xdd07d35b, 0xdb485694, 0x38d7e5b2, 0x57720101,\n    0x730edebc, 0x5b643113, 0x94917e4f, 0x503c2fba, 0x646f1282, 0x7523d24a, 0xe0779695, 0xf9c17a8f,\n    0x7a5b2121, 0xd187b896, 0x29263a4d, 0xba510cdf, 0x81f47c9f, 0xad1163ed, 0xea7b5965, 0x1a00726e,\n    0x11403092, 0x00da6d77, 0x4a0cdd61, 0xad1f4603, 0x605bdfb0, 0x9eedc364, 0x22ebe6a8, 0xcee7d28a,\n    0xa0e736a0, 0x5564a6b9, 0x10853209, 0xc7eb8f37, 0x2de705ca, 0x8951570f, 0xdf09822b, 0xbd691a6c,\n    0xaa12e4f2, 0x87451c0f, 0xe0f6a27a, 0x3ada4819, 0x4cf1764f, 0x0d771c2b, 0x67cdb156, 0x350d8384,\n    0x5938fa0f, 0x42399ef3, 0x36997b07, 0x0e84093d, 0x4aa93e61, 0x8360d87b, 0x1fa98b0c, 0x1149382c,\n    0xe97625a5, 0x0614d1b7, 0x0e25244b, 0x0c768347, 0x589e8d82, 0x0d2059d1, 0xa466bb1e, 0xf8da0a82,\n    0x04f19130, 0xba6e4ec0, 0x99265164, 0x1ee7230d, 0x50b2ad80, 0xeaee6801, 0x8db2a283, 0xea8bf59e);\n\n};\n\nvar util = require('../../util');\n\nfunction cast5(key) {\n  this.cast5 = new openpgp_symenc_cast5();\n  this.cast5.setKey(util.str2bin(key));\n\n  this.encrypt = function(block) {\n    return this.cast5.encrypt(block);\n  }\n}\n\nmodule.exports = cast5;\nmodule.exports.blockSize = cast5.prototype.blockSize = 8;\nmodule.exports.keySize = cast5.prototype.keySize = 16;\n","//Paul Tero, July 2001\n//http://www.tero.co.uk/des/\n//\n//Optimised for performance with large blocks by Michael Hayworth, November 2001\n//http://www.netdealing.com\n//\n// Modified by Recurity Labs GmbH\n\n//THIS SOFTWARE IS PROVIDED \"AS IS\" AND\n//ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE\n//IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE\n//ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE\n//FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL\n//DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS\n//OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)\n//HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT\n//LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY\n//OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF\n//SUCH DAMAGE.\n\n//des\n//this takes the key, the message, and whether to encrypt or decrypt\n\n\n\n\nfunction des(keys, message, encrypt, mode, iv, padding) {\n  //declaring this locally speeds things up a bit\n  var spfunction1 = new Array(0x1010400, 0, 0x10000, 0x1010404, 0x1010004, 0x10404, 0x4, 0x10000, 0x400, 0x1010400,\n    0x1010404, 0x400, 0x1000404, 0x1010004, 0x1000000, 0x4, 0x404, 0x1000400, 0x1000400, 0x10400, 0x10400, 0x1010000,\n    0x1010000, 0x1000404, 0x10004, 0x1000004, 0x1000004, 0x10004, 0, 0x404, 0x10404, 0x1000000, 0x10000, 0x1010404, 0x4,\n    0x1010000, 0x1010400, 0x1000000, 0x1000000, 0x400, 0x1010004, 0x10000, 0x10400, 0x1000004, 0x400, 0x4, 0x1000404,\n    0x10404, 0x1010404, 0x10004, 0x1010000, 0x1000404, 0x1000004, 0x404, 0x10404, 0x1010400, 0x404, 0x1000400,\n    0x1000400, 0, 0x10004, 0x10400, 0, 0x1010004);\n  var spfunction2 = new Array(-0x7fef7fe0, -0x7fff8000, 0x8000, 0x108020, 0x100000, 0x20, -0x7fefffe0, -0x7fff7fe0, -\n    0x7fffffe0, -0x7fef7fe0, -0x7fef8000, -0x80000000, -0x7fff8000, 0x100000, 0x20, -0x7fefffe0, 0x108000, 0x100020, -\n    0x7fff7fe0, 0, -0x80000000, 0x8000, 0x108020, -0x7ff00000, 0x100020, -0x7fffffe0, 0, 0x108000, 0x8020, -0x7fef8000, -\n    0x7ff00000, 0x8020, 0, 0x108020, -0x7fefffe0, 0x100000, -0x7fff7fe0, -0x7ff00000, -0x7fef8000, 0x8000, -0x7ff00000, -\n    0x7fff8000, 0x20, -0x7fef7fe0, 0x108020, 0x20, 0x8000, -0x80000000, 0x8020, -0x7fef8000, 0x100000, -0x7fffffe0,\n    0x100020, -0x7fff7fe0, -0x7fffffe0, 0x100020, 0x108000, 0, -0x7fff8000, 0x8020, -0x80000000, -0x7fefffe0, -\n    0x7fef7fe0, 0x108000);\n  var spfunction3 = new Array(0x208, 0x8020200, 0, 0x8020008, 0x8000200, 0, 0x20208, 0x8000200, 0x20008, 0x8000008,\n    0x8000008, 0x20000, 0x8020208, 0x20008, 0x8020000, 0x208, 0x8000000, 0x8, 0x8020200, 0x200, 0x20200, 0x8020000,\n    0x8020008, 0x20208, 0x8000208, 0x20200, 0x20000, 0x8000208, 0x8, 0x8020208, 0x200, 0x8000000, 0x8020200, 0x8000000,\n    0x20008, 0x208, 0x20000, 0x8020200, 0x8000200, 0, 0x200, 0x20008, 0x8020208, 0x8000200, 0x8000008, 0x200, 0,\n    0x8020008, 0x8000208, 0x20000, 0x8000000, 0x8020208, 0x8, 0x20208, 0x20200, 0x8000008, 0x8020000, 0x8000208, 0x208,\n    0x8020000, 0x20208, 0x8, 0x8020008, 0x20200);\n  var spfunction4 = new Array(0x802001, 0x2081, 0x2081, 0x80, 0x802080, 0x800081, 0x800001, 0x2001, 0, 0x802000,\n    0x802000, 0x802081, 0x81, 0, 0x800080, 0x800001, 0x1, 0x2000, 0x800000, 0x802001, 0x80, 0x800000, 0x2001, 0x2080,\n    0x800081, 0x1, 0x2080, 0x800080, 0x2000, 0x802080, 0x802081, 0x81, 0x800080, 0x800001, 0x802000, 0x802081, 0x81, 0,\n    0, 0x802000, 0x2080, 0x800080, 0x800081, 0x1, 0x802001, 0x2081, 0x2081, 0x80, 0x802081, 0x81, 0x1, 0x2000, 0x800001,\n    0x2001, 0x802080, 0x800081, 0x2001, 0x2080, 0x800000, 0x802001, 0x80, 0x800000, 0x2000, 0x802080);\n  var spfunction5 = new Array(0x100, 0x2080100, 0x2080000, 0x42000100, 0x80000, 0x100, 0x40000000, 0x2080000,\n    0x40080100, 0x80000, 0x2000100, 0x40080100, 0x42000100, 0x42080000, 0x80100, 0x40000000, 0x2000000, 0x40080000,\n    0x40080000, 0, 0x40000100, 0x42080100, 0x42080100, 0x2000100, 0x42080000, 0x40000100, 0, 0x42000000, 0x2080100,\n    0x2000000, 0x42000000, 0x80100, 0x80000, 0x42000100, 0x100, 0x2000000, 0x40000000, 0x2080000, 0x42000100,\n    0x40080100, 0x2000100, 0x40000000, 0x42080000, 0x2080100, 0x40080100, 0x100, 0x2000000, 0x42080000, 0x42080100,\n    0x80100, 0x42000000, 0x42080100, 0x2080000, 0, 0x40080000, 0x42000000, 0x80100, 0x2000100, 0x40000100, 0x80000, 0,\n    0x40080000, 0x2080100, 0x40000100);\n  var spfunction6 = new Array(0x20000010, 0x20400000, 0x4000, 0x20404010, 0x20400000, 0x10, 0x20404010, 0x400000,\n    0x20004000, 0x404010, 0x400000, 0x20000010, 0x400010, 0x20004000, 0x20000000, 0x4010, 0, 0x400010, 0x20004010,\n    0x4000, 0x404000, 0x20004010, 0x10, 0x20400010, 0x20400010, 0, 0x404010, 0x20404000, 0x4010, 0x404000, 0x20404000,\n    0x20000000, 0x20004000, 0x10, 0x20400010, 0x404000, 0x20404010, 0x400000, 0x4010, 0x20000010, 0x400000, 0x20004000,\n    0x20000000, 0x4010, 0x20000010, 0x20404010, 0x404000, 0x20400000, 0x404010, 0x20404000, 0, 0x20400010, 0x10, 0x4000,\n    0x20400000, 0x404010, 0x4000, 0x400010, 0x20004010, 0, 0x20404000, 0x20000000, 0x400010, 0x20004010);\n  var spfunction7 = new Array(0x200000, 0x4200002, 0x4000802, 0, 0x800, 0x4000802, 0x200802, 0x4200800, 0x4200802,\n    0x200000, 0, 0x4000002, 0x2, 0x4000000, 0x4200002, 0x802, 0x4000800, 0x200802, 0x200002, 0x4000800, 0x4000002,\n    0x4200000, 0x4200800, 0x200002, 0x4200000, 0x800, 0x802, 0x4200802, 0x200800, 0x2, 0x4000000, 0x200800, 0x4000000,\n    0x200800, 0x200000, 0x4000802, 0x4000802, 0x4200002, 0x4200002, 0x2, 0x200002, 0x4000000, 0x4000800, 0x200000,\n    0x4200800, 0x802, 0x200802, 0x4200800, 0x802, 0x4000002, 0x4200802, 0x4200000, 0x200800, 0, 0x2, 0x4200802, 0,\n    0x200802, 0x4200000, 0x800, 0x4000002, 0x4000800, 0x800, 0x200002);\n  var spfunction8 = new Array(0x10001040, 0x1000, 0x40000, 0x10041040, 0x10000000, 0x10001040, 0x40, 0x10000000,\n    0x40040, 0x10040000, 0x10041040, 0x41000, 0x10041000, 0x41040, 0x1000, 0x40, 0x10040000, 0x10000040, 0x10001000,\n    0x1040, 0x41000, 0x40040, 0x10040040, 0x10041000, 0x1040, 0, 0, 0x10040040, 0x10000040, 0x10001000, 0x41040,\n    0x40000, 0x41040, 0x40000, 0x10041000, 0x1000, 0x40, 0x10040040, 0x1000, 0x41040, 0x10001000, 0x40, 0x10000040,\n    0x10040000, 0x10040040, 0x10000000, 0x40000, 0x10001040, 0, 0x10041040, 0x40040, 0x10000040, 0x10040000, 0x10001000,\n    0x10001040, 0, 0x10041040, 0x41000, 0x41000, 0x1040, 0x1040, 0x40040, 0x10000000, 0x10041000);\n\n  //create the 16 or 48 subkeys we will need\n  var m = 0,\n    i, j, temp, temp2, right1, right2, left, right, looping;\n  var cbcleft, cbcleft2, cbcright, cbcright2\n  var endloop, loopinc;\n  var len = message.length;\n  var chunk = 0;\n  //set up the loops for single and triple des\n  var iterations = keys.length == 32 ? 3 : 9; //single or triple des\n  if (iterations == 3) {\n    looping = encrypt ? new Array(0, 32, 2) : new Array(30, -2, -2);\n  } else {\n    looping = encrypt ? new Array(0, 32, 2, 62, 30, -2, 64, 96, 2) : new Array(94, 62, -2, 32, 64, 2, 30, -2, -2);\n  }\n\n  //pad the message depending on the padding parameter\n  //only add padding if encrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (encrypt) {\n    message = des_addPadding(message, padding);\n    len = message.length;\n  }\n\n  //store the result here\n  result = \"\";\n  tempresult = \"\";\n\n  if (mode == 1) { //CBC mode\n    cbcleft = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    cbcright = (iv.charCodeAt(m++) << 24) | (iv.charCodeAt(m++) << 16) | (iv.charCodeAt(m++) << 8) | iv.charCodeAt(m++);\n    m = 0;\n  }\n\n  //loop through each 64 bit chunk of the message\n  while (m < len) {\n    left = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) | message\n      .charCodeAt(m++);\n    right = (message.charCodeAt(m++) << 24) | (message.charCodeAt(m++) << 16) | (message.charCodeAt(m++) << 8) |\n      message.charCodeAt(m++);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        left ^= cbcleft;\n        right ^= cbcright;\n      } else {\n        cbcleft2 = cbcleft;\n        cbcright2 = cbcright;\n        cbcleft = left;\n        cbcright = right;\n      }\n    }\n\n    //first each 64 but chunk of the message must be permuted according to IP\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    left = ((left << 1) | (left >>> 31));\n    right = ((right << 1) | (right >>> 31));\n\n    //do this either 1 or 3 times for each chunk of the message\n    for (j = 0; j < iterations; j += 3) {\n      endloop = looping[j + 1];\n      loopinc = looping[j + 2];\n      //now go through and perform the encryption or decryption  \n      for (i = looping[j]; i != endloop; i += loopinc) { //for efficiency\n        right1 = right ^ keys[i];\n        right2 = ((right >>> 4) | (right << 28)) ^ keys[i + 1];\n        //the result is attained by passing these bytes through the S selection functions\n        temp = left;\n        left = right;\n        right = temp ^ (spfunction2[(right1 >>> 24) & 0x3f] | spfunction4[(right1 >>> 16) & 0x3f] | spfunction6[(right1 >>>\n          8) & 0x3f] | spfunction8[right1 & 0x3f] | spfunction1[(right2 >>> 24) & 0x3f] | spfunction3[(right2 >>> 16) &\n          0x3f] | spfunction5[(right2 >>> 8) & 0x3f] | spfunction7[right2 & 0x3f]);\n      }\n      temp = left;\n      left = right;\n      right = temp; //unreverse left and right\n    } //for either 1 or 3 iterations\n\n    //move then each one bit to the right\n    left = ((left >>> 1) | (left << 31));\n    right = ((right >>> 1) | (right << 31));\n\n    //now perform IP-1, which is IP in the opposite direction\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((right >>> 2) ^ left) & 0x33333333;\n    left ^= temp;\n    right ^= (temp << 2);\n    temp = ((left >>> 16) ^ right) & 0x0000ffff;\n    right ^= temp;\n    left ^= (temp << 16);\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n\n    //for Cipher Block Chaining mode, xor the message with the previous result\n    if (mode == 1) {\n      if (encrypt) {\n        cbcleft = left;\n        cbcright = right;\n      } else {\n        left ^= cbcleft2;\n        right ^= cbcright2;\n      }\n    }\n    tempresult += String.fromCharCode((left >>> 24), ((left >>> 16) & 0xff), ((left >>> 8) & 0xff), (left & 0xff), (\n      right >>> 24), ((right >>> 16) & 0xff), ((right >>> 8) & 0xff), (right & 0xff));\n\n    chunk += 8;\n    if (chunk == 512) {\n      result += tempresult;\n      tempresult = \"\";\n      chunk = 0;\n    }\n  } //for every 8 characters, or 64 bits in the message\n\n  //return the result as an array\n  result += tempresult;\n\n  //only remove padding if decrypting - note that you need to use the same padding option for both encrypt and decrypt\n  if (!encrypt) {\n    result = des_removePadding(result, padding);\n  }\n\n  return result;\n} //end of des\n\n\n\n//des_createKeys\n//this takes as input a 64 bit key (even though only 56 bits are used)\n//as an array of 2 integers, and returns 16 48 bit keys\n\nfunction des_createKeys(key) {\n  //declaring this locally speeds things up a bit\n  pc2bytes0 = new Array(0, 0x4, 0x20000000, 0x20000004, 0x10000, 0x10004, 0x20010000, 0x20010004, 0x200, 0x204,\n    0x20000200, 0x20000204, 0x10200, 0x10204, 0x20010200, 0x20010204);\n  pc2bytes1 = new Array(0, 0x1, 0x100000, 0x100001, 0x4000000, 0x4000001, 0x4100000, 0x4100001, 0x100, 0x101, 0x100100,\n    0x100101, 0x4000100, 0x4000101, 0x4100100, 0x4100101);\n  pc2bytes2 = new Array(0, 0x8, 0x800, 0x808, 0x1000000, 0x1000008, 0x1000800, 0x1000808, 0, 0x8, 0x800, 0x808,\n    0x1000000, 0x1000008, 0x1000800, 0x1000808);\n  pc2bytes3 = new Array(0, 0x200000, 0x8000000, 0x8200000, 0x2000, 0x202000, 0x8002000, 0x8202000, 0x20000, 0x220000,\n    0x8020000, 0x8220000, 0x22000, 0x222000, 0x8022000, 0x8222000);\n  pc2bytes4 = new Array(0, 0x40000, 0x10, 0x40010, 0, 0x40000, 0x10, 0x40010, 0x1000, 0x41000, 0x1010, 0x41010, 0x1000,\n    0x41000, 0x1010, 0x41010);\n  pc2bytes5 = new Array(0, 0x400, 0x20, 0x420, 0, 0x400, 0x20, 0x420, 0x2000000, 0x2000400, 0x2000020, 0x2000420,\n    0x2000000, 0x2000400, 0x2000020, 0x2000420);\n  pc2bytes6 = new Array(0, 0x10000000, 0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002, 0, 0x10000000,\n    0x80000, 0x10080000, 0x2, 0x10000002, 0x80002, 0x10080002);\n  pc2bytes7 = new Array(0, 0x10000, 0x800, 0x10800, 0x20000000, 0x20010000, 0x20000800, 0x20010800, 0x20000, 0x30000,\n    0x20800, 0x30800, 0x20020000, 0x20030000, 0x20020800, 0x20030800);\n  pc2bytes8 = new Array(0, 0x40000, 0, 0x40000, 0x2, 0x40002, 0x2, 0x40002, 0x2000000, 0x2040000, 0x2000000, 0x2040000,\n    0x2000002, 0x2040002, 0x2000002, 0x2040002);\n  pc2bytes9 = new Array(0, 0x10000000, 0x8, 0x10000008, 0, 0x10000000, 0x8, 0x10000008, 0x400, 0x10000400, 0x408,\n    0x10000408, 0x400, 0x10000400, 0x408, 0x10000408);\n  pc2bytes10 = new Array(0, 0x20, 0, 0x20, 0x100000, 0x100020, 0x100000, 0x100020, 0x2000, 0x2020, 0x2000, 0x2020,\n    0x102000, 0x102020, 0x102000, 0x102020);\n  pc2bytes11 = new Array(0, 0x1000000, 0x200, 0x1000200, 0x200000, 0x1200000, 0x200200, 0x1200200, 0x4000000, 0x5000000,\n    0x4000200, 0x5000200, 0x4200000, 0x5200000, 0x4200200, 0x5200200);\n  pc2bytes12 = new Array(0, 0x1000, 0x8000000, 0x8001000, 0x80000, 0x81000, 0x8080000, 0x8081000, 0x10, 0x1010,\n    0x8000010, 0x8001010, 0x80010, 0x81010, 0x8080010, 0x8081010);\n  pc2bytes13 = new Array(0, 0x4, 0x100, 0x104, 0, 0x4, 0x100, 0x104, 0x1, 0x5, 0x101, 0x105, 0x1, 0x5, 0x101, 0x105);\n\n  //how many iterations (1 for des, 3 for triple des)\n  var iterations = key.length > 8 ? 3 : 1; //changed by Paul 16/6/2007 to use Triple DES for 9+ byte keys\n  //stores the return keys\n  var keys = new Array(32 * iterations);\n  //now define the left shifts which need to be done\n  var shifts = new Array(0, 0, 1, 1, 1, 1, 1, 1, 0, 1, 1, 1, 1, 1, 1, 0);\n  //other variables\n  var lefttemp, righttemp, m = 0,\n    n = 0,\n    temp;\n\n  for (var j = 0; j < iterations; j++) { //either 1 or 3 iterations\n    left = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n    right = (key.charCodeAt(m++) << 24) | (key.charCodeAt(m++) << 16) | (key.charCodeAt(m++) << 8) | key.charCodeAt(m++);\n\n    temp = ((left >>> 4) ^ right) & 0x0f0f0f0f;\n    right ^= temp;\n    left ^= (temp << 4);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 2) ^ right) & 0x33333333;\n    right ^= temp;\n    left ^= (temp << 2);\n    temp = ((right >>> -16) ^ left) & 0x0000ffff;\n    left ^= temp;\n    right ^= (temp << -16);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n    temp = ((right >>> 8) ^ left) & 0x00ff00ff;\n    left ^= temp;\n    right ^= (temp << 8);\n    temp = ((left >>> 1) ^ right) & 0x55555555;\n    right ^= temp;\n    left ^= (temp << 1);\n\n    //the right side needs to be shifted and to get the last four bits of the left side\n    temp = (left << 8) | ((right >>> 20) & 0x000000f0);\n    //left needs to be put upside down\n    left = (right << 24) | ((right << 8) & 0xff0000) | ((right >>> 8) & 0xff00) | ((right >>> 24) & 0xf0);\n    right = temp;\n\n    //now go through and perform these shifts on the left and right keys\n    for (i = 0; i < shifts.length; i++) {\n      //shift the keys either one or two bits to the left\n      if (shifts[i]) {\n        left = (left << 2) | (left >>> 26);\n        right = (right << 2) | (right >>> 26);\n      } else {\n        left = (left << 1) | (left >>> 27);\n        right = (right << 1) | (right >>> 27);\n      }\n      left &= -0xf;\n      right &= -0xf;\n\n      //now apply PC-2, in such a way that E is easier when encrypting or decrypting\n      //this conversion will look like PC-2 except only the last 6 bits of each byte are used\n      //rather than 48 consecutive bits and the order of lines will be according to \n      //how the S selection functions will be applied: S2, S4, S6, S8, S1, S3, S5, S7\n      lefttemp = pc2bytes0[left >>> 28] | pc2bytes1[(left >>> 24) & 0xf] | pc2bytes2[(left >>> 20) & 0xf] | pc2bytes3[(\n        left >>> 16) & 0xf] | pc2bytes4[(left >>> 12) & 0xf] | pc2bytes5[(left >>> 8) & 0xf] | pc2bytes6[(left >>> 4) &\n        0xf];\n      righttemp = pc2bytes7[right >>> 28] | pc2bytes8[(right >>> 24) & 0xf] | pc2bytes9[(right >>> 20) & 0xf] |\n        pc2bytes10[(right >>> 16) & 0xf] | pc2bytes11[(right >>> 12) & 0xf] | pc2bytes12[(right >>> 8) & 0xf] |\n        pc2bytes13[(right >>> 4) & 0xf];\n      temp = ((righttemp >>> 16) ^ lefttemp) & 0x0000ffff;\n      keys[n++] = lefttemp ^ temp;\n      keys[n++] = righttemp ^ (temp << 16);\n    }\n  } //for each iterations\n  //return the keys we've created\n  return keys;\n} //end of des_createKeys\n\n\nfunction des_addPadding(message, padding) {\n  var padLength = 8 - (message.length % 8);\n  if ((padding == 2) && (padLength < 8)) { //pad the message with spaces\n    message += \"        \".substr(0, padLength);\n  } else if (padding == 1) { //PKCS7 padding\n    message += String.fromCharCode(padLength, padLength, padLength, padLength, padLength, padLength, padLength,\n      padLength).substr(0, padLength);\n  } else if (!padding && (padLength < 8)) { //pad the message out with null bytes\n    message += \"\\0\\0\\0\\0\\0\\0\\0\\0\".substr(0, padLength);\n  }\n  return message;\n}\n\nfunction des_removePadding(message, padding) {\n  if (padding == 2) { // space padded\n    message = message.replace(/ *$/g, \"\");\n  } else if (padding == 1) { // PKCS7\n    var padCount = message.charCodeAt(message.length - 1);\n    message = message.substr(0, message.length - padCount);\n  } else if (!padding) { // null padding\n    message = message.replace(/\\0*$/g, \"\");\n  }\n  return message;\n}\n\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction Des(key) {\n  this.key = [];\n\n  for (var i = 0; i < 3; i++) {\n    this.key.push(key.substr(i * 8, 8));\n  }\n\n  this.encrypt = function(block) {\n    return util.str2bin(des(des_createKeys(this.key[2]),\n      des(des_createKeys(this.key[1]),\n      des(des_createKeys(this.key[0]),\n      util.bin2str(block), true, 0, null, null),\n      false, 0, null, null), true, 0, null, null));\n  }\n}\n\nDes.keySize = Des.prototype.keySize = 24;\nDes.blockSize = Des.prototype.blockSize = 8;\n\n// This is \"original\" DES - Des is actually Triple DES.\n// This is only exported so we can unit test.\n\nfunction OriginalDes(key) {\n  this.key = key;\n\n  this.encrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), true, 0, null, padding));\n  }\n\n  this.decrypt = function(block, padding) {\n    var keys = des_createKeys(this.key);\n    return util.str2bin(des(keys, util.bin2str(block), false, 0, null, padding));\n  }\n}\n\nmodule.exports = {\n  des: Des,\n  originalDes: OriginalDes\n}\n","var desModule = require('./des.js');\n\nmodule.exports = {\n  des: desModule['des'],\n  originalDes: desModule['originalDes'],\n  cast5: require('./cast5.js'),\n  twofish: require('./twofish.js'),\n  blowfish: require('./blowfish.js')\n}\n\nvar aes = require('./aes.js');\n\nfor (var i in aes) {\n  module.exports['aes' + i] = aes[i];\n}\n","/* Modified by Recurity Labs GmbH \n * \n * Cipher.js\n * A block-cipher algorithm implementation on JavaScript\n * See Cipher.readme.txt for further information.\n *\n * Copyright(c) 2009 Atsushi Oka [ http://oka.nu/ ]\n * This script file is distributed under the LGPL\n *\n * ACKNOWLEDGMENT\n *\n *     The main subroutines are written by Michiel van Everdingen.\n * \n *     Michiel van Everdingen\n *     http://home.versatel.nl/MAvanEverdingen/index.html\n * \n *     All rights for these routines are reserved to Michiel van Everdingen.\n *\n */\n\n\n\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n//Math\n////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nvar MAXINT = 0xFFFFFFFF;\n\nfunction rotb(b, n) {\n  return (b << n | b >>> (8 - n)) & 0xFF;\n}\n\nfunction rotw(w, n) {\n  return (w << n | w >>> (32 - n)) & MAXINT;\n}\n\nfunction getW(a, i) {\n  return a[i] | a[i + 1] << 8 | a[i + 2] << 16 | a[i + 3] << 24;\n}\n\nfunction setW(a, i, w) {\n  a.splice(i, 4, w & 0xFF, (w >>> 8) & 0xFF, (w >>> 16) & 0xFF, (w >>> 24) & 0xFF);\n}\n\nfunction setWInv(a, i, w) {\n  a.splice(i, 4, (w >>> 24) & 0xFF, (w >>> 16) & 0xFF, (w >>> 8) & 0xFF, w & 0xFF);\n}\n\nfunction getB(x, n) {\n  return (x >>> (n * 8)) & 0xFF;\n}\n\nfunction getNrBits(i) {\n  var n = 0;\n  while (i > 0) {\n    n++;\n    i >>>= 1;\n  }\n  return n;\n}\n\nfunction getMask(n) {\n  return (1 << n) - 1;\n}\n\n//added 2008/11/13 XXX MUST USE ONE-WAY HASH FUNCTION FOR SECURITY REASON\n\nfunction randByte() {\n  return Math.floor(Math.random() * 256);\n}\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n// Twofish\n// //////////////////////////////////////////////////////////////////////////////////////////////////////////////////////\n\nfunction createTwofish() {\n  //\n  var keyBytes = null;\n  var dataBytes = null;\n  var dataOffset = -1;\n  // var dataLength = -1;\n  var algorithmName = null;\n  // var idx2 = -1;\n  //\n\n  algorithmName = \"twofish\";\n\n  var tfsKey = [];\n  var tfsM = [\n    [],\n    [],\n    [],\n    []\n  ];\n\n  function tfsInit(key) {\n    keyBytes = key;\n    var i, a, b, c, d, meKey = [],\n      moKey = [],\n      inKey = [];\n    var kLen;\n    var sKey = [];\n    var f01, f5b, fef;\n\n    var q0 = [\n      [8, 1, 7, 13, 6, 15, 3, 2, 0, 11, 5, 9, 14, 12, 10, 4],\n      [2, 8, 11, 13, 15, 7, 6, 14, 3, 1, 9, 4, 0, 10, 12, 5]\n    ];\n    var q1 = [\n      [14, 12, 11, 8, 1, 2, 3, 5, 15, 4, 10, 6, 7, 0, 9, 13],\n      [1, 14, 2, 11, 4, 12, 3, 7, 6, 13, 10, 5, 15, 9, 0, 8]\n    ];\n    var q2 = [\n      [11, 10, 5, 14, 6, 13, 9, 0, 12, 8, 15, 3, 2, 4, 7, 1],\n      [4, 12, 7, 5, 1, 6, 9, 10, 0, 14, 13, 8, 2, 11, 3, 15]\n    ];\n    var q3 = [\n      [13, 7, 15, 4, 1, 2, 6, 14, 9, 11, 3, 0, 8, 5, 12, 10],\n      [11, 9, 5, 1, 12, 3, 13, 14, 6, 4, 7, 15, 2, 0, 8, 10]\n    ];\n    var ror4 = [0, 8, 1, 9, 2, 10, 3, 11, 4, 12, 5, 13, 6, 14, 7, 15];\n    var ashx = [0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12, 5, 14, 7];\n    var q = [\n      [],\n      []\n    ];\n    var m = [\n      [],\n      [],\n      [],\n      []\n    ];\n\n    function ffm5b(x) {\n      return x ^ (x >> 2) ^ [0, 90, 180, 238][x & 3];\n    }\n\n    function ffmEf(x) {\n      return x ^ (x >> 1) ^ (x >> 2) ^ [0, 238, 180, 90][x & 3];\n    }\n\n    function mdsRem(p, q) {\n      var i, t, u;\n      for (i = 0; i < 8; i++) {\n        t = q >>> 24;\n        q = ((q << 8) & MAXINT) | p >>> 24;\n        p = (p << 8) & MAXINT;\n        u = t << 1;\n        if (t & 128) {\n          u ^= 333;\n        }\n        q ^= t ^ (u << 16);\n        u ^= t >>> 1;\n        if (t & 1) {\n          u ^= 166;\n        }\n        q ^= u << 24 | u << 8;\n      }\n      return q;\n    }\n\n    function qp(n, x) {\n      var a, b, c, d;\n      a = x >> 4;\n      b = x & 15;\n      c = q0[n][a ^ b];\n      d = q1[n][ror4[b] ^ ashx[a]];\n      return q3[n][ror4[d] ^ ashx[c]] << 4 | q2[n][c ^ d];\n    }\n\n    function hFun(x, key) {\n      var a = getB(x, 0),\n        b = getB(x, 1),\n        c = getB(x, 2),\n        d = getB(x, 3);\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(key[3], 0);\n          b = q[0][b] ^ getB(key[3], 1);\n          c = q[0][c] ^ getB(key[3], 2);\n          d = q[1][d] ^ getB(key[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(key[2], 0);\n          b = q[1][b] ^ getB(key[2], 1);\n          c = q[0][c] ^ getB(key[2], 2);\n          d = q[0][d] ^ getB(key[2], 3);\n        case 2:\n          a = q[0][q[0][a] ^ getB(key[1], 0)] ^ getB(key[0], 0);\n          b = q[0][q[1][b] ^ getB(key[1], 1)] ^ getB(key[0], 1);\n          c = q[1][q[0][c] ^ getB(key[1], 2)] ^ getB(key[0], 2);\n          d = q[1][q[1][d] ^ getB(key[1], 3)] ^ getB(key[0], 3);\n      }\n      return m[0][a] ^ m[1][b] ^ m[2][c] ^ m[3][d];\n    }\n\n    keyBytes = keyBytes.slice(0, 32);\n    i = keyBytes.length;\n    while (i != 16 && i != 24 && i != 32)\n      keyBytes[i++] = 0;\n\n    for (i = 0; i < keyBytes.length; i += 4) {\n      inKey[i >> 2] = getW(keyBytes, i);\n    }\n    for (i = 0; i < 256; i++) {\n      q[0][i] = qp(0, i);\n      q[1][i] = qp(1, i);\n    }\n    for (i = 0; i < 256; i++) {\n      f01 = q[1][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[0][i] = f01 + (f5b << 8) + (fef << 16) + (fef << 24);\n      m[2][i] = f5b + (fef << 8) + (f01 << 16) + (fef << 24);\n      f01 = q[0][i];\n      f5b = ffm5b(f01);\n      fef = ffmEf(f01);\n      m[1][i] = fef + (fef << 8) + (f5b << 16) + (f01 << 24);\n      m[3][i] = f5b + (f01 << 8) + (fef << 16) + (f5b << 24);\n    }\n\n    kLen = inKey.length / 2;\n    for (i = 0; i < kLen; i++) {\n      a = inKey[i + i];\n      meKey[i] = a;\n      b = inKey[i + i + 1];\n      moKey[i] = b;\n      sKey[kLen - i - 1] = mdsRem(a, b);\n    }\n    for (i = 0; i < 40; i += 2) {\n      a = 0x1010101 * i;\n      b = a + 0x1010101;\n      a = hFun(a, meKey);\n      b = rotw(hFun(b, moKey), 8);\n      tfsKey[i] = (a + b) & MAXINT;\n      tfsKey[i + 1] = rotw(a + 2 * b, 9);\n    }\n    for (i = 0; i < 256; i++) {\n      a = b = c = d = i;\n      switch (kLen) {\n        case 4:\n          a = q[1][a] ^ getB(sKey[3], 0);\n          b = q[0][b] ^ getB(sKey[3], 1);\n          c = q[0][c] ^ getB(sKey[3], 2);\n          d = q[1][d] ^ getB(sKey[3], 3);\n        case 3:\n          a = q[1][a] ^ getB(sKey[2], 0);\n          b = q[1][b] ^ getB(sKey[2], 1);\n          c = q[0][c] ^ getB(sKey[2], 2);\n          d = q[0][d] ^ getB(sKey[2], 3);\n        case 2:\n          tfsM[0][i] = m[0][q[0][q[0][a] ^ getB(sKey[1], 0)] ^ getB(sKey[0], 0)];\n          tfsM[1][i] = m[1][q[0][q[1][b] ^ getB(sKey[1], 1)] ^ getB(sKey[0], 1)];\n          tfsM[2][i] = m[2][q[1][q[0][c] ^ getB(sKey[1], 2)] ^ getB(sKey[0], 2)];\n          tfsM[3][i] = m[3][q[1][q[1][d] ^ getB(sKey[1], 3)] ^ getB(sKey[0], 3)];\n      }\n    }\n  }\n\n  function tfsG0(x) {\n    return tfsM[0][getB(x, 0)] ^ tfsM[1][getB(x, 1)] ^ tfsM[2][getB(x, 2)] ^ tfsM[3][getB(x, 3)];\n  }\n\n  function tfsG1(x) {\n    return tfsM[0][getB(x, 3)] ^ tfsM[1][getB(x, 0)] ^ tfsM[2][getB(x, 1)] ^ tfsM[3][getB(x, 2)];\n  }\n\n  function tfsFrnd(r, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2] ^ (a + b + tfsKey[4 * r + 8]) & MAXINT, 31);\n    blk[3] = rotw(blk[3], 1) ^ (a + 2 * b + tfsKey[4 * r + 9]) & MAXINT;\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0] ^ (a + b + tfsKey[4 * r + 10]) & MAXINT, 31);\n    blk[1] = rotw(blk[1], 1) ^ (a + 2 * b + tfsKey[4 * r + 11]) & MAXINT;\n  }\n\n  function tfsIrnd(i, blk) {\n    var a = tfsG0(blk[0]);\n    var b = tfsG1(blk[1]);\n    blk[2] = rotw(blk[2], 1) ^ (a + b + tfsKey[4 * i + 10]) & MAXINT;\n    blk[3] = rotw(blk[3] ^ (a + 2 * b + tfsKey[4 * i + 11]) & MAXINT, 31);\n    a = tfsG0(blk[2]);\n    b = tfsG1(blk[3]);\n    blk[0] = rotw(blk[0], 1) ^ (a + b + tfsKey[4 * i + 8]) & MAXINT;\n    blk[1] = rotw(blk[1] ^ (a + 2 * b + tfsKey[4 * i + 9]) & MAXINT, 31);\n  }\n\n  function tfsClose() {\n    tfsKey = [];\n    tfsM = [\n      [],\n      [],\n      [],\n      []\n    ];\n  }\n\n  function tfsEncrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[0],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[1],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[2],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[3]\n    ];\n    for (var j = 0; j < 8; j++) {\n      tfsFrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[4]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[5]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[6]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[7]);\n    dataOffset += 16;\n    return dataBytes;\n  }\n\n  function tfsDecrypt(data, offset) {\n    dataBytes = data;\n    dataOffset = offset;\n    var blk = [getW(dataBytes, dataOffset) ^ tfsKey[4],\n        getW(dataBytes, dataOffset + 4) ^ tfsKey[5],\n        getW(dataBytes, dataOffset + 8) ^ tfsKey[6],\n        getW(dataBytes, dataOffset + 12) ^ tfsKey[7]\n    ];\n    for (var j = 7; j >= 0; j--) {\n      tfsIrnd(j, blk);\n    }\n    setW(dataBytes, dataOffset, blk[2] ^ tfsKey[0]);\n    setW(dataBytes, dataOffset + 4, blk[3] ^ tfsKey[1]);\n    setW(dataBytes, dataOffset + 8, blk[0] ^ tfsKey[2]);\n    setW(dataBytes, dataOffset + 12, blk[1] ^ tfsKey[3]);\n    dataOffset += 16;\n  }\n\n  // added by Recurity Labs\n\n  function tfsFinal() {\n    return dataBytes;\n  }\n\n  return {\n    name: \"twofish\",\n    blocksize: 128 / 8,\n    open: tfsInit,\n    close: tfsClose,\n    encrypt: tfsEncrypt,\n    decrypt: tfsDecrypt,\n    // added by Recurity Labs\n    finalize: tfsFinal\n  };\n}\n\nvar util = require('../../util');\n\n// added by Recurity Labs\n\nfunction TFencrypt(block, key) {\n  var block_copy = [].concat(block);\n  var tf = createTwofish();\n  tf.open(util.str2bin(key), 0);\n  var result = tf.encrypt(block_copy, 0);\n  tf.close();\n  return result;\n}\n\nfunction TF(key) {\n  this.tf = createTwofish();\n  this.tf.open(util.str2bin(key), 0);\n\n  this.encrypt = function(block) {\n    return this.tf.encrypt([].concat(block), 0);\n  }\n}\n\n\nmodule.exports = TF;\nmodule.exports.keySize = TF.prototype.keySize = 32;\nmodule.exports.blockSize = TF.prototype.blockSize = 16;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar random = require('./random.js'),\n  cipher = require('./cipher'),\n  publicKey = require('./public_key'),\n  type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Encrypts data using the specified public key multiprecision integers \n   * and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi[]} if RSA an openpgp_type_mpi; \n   * if elgamal encryption an array of two openpgp_type_mpi is returned; otherwise null\n   */\n  publicKeyEncrypt: function(algo, publicMPIs, data) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          var rsa = new publicKey.rsa();\n          var n = publicMPIs[0].toBigInteger();\n          var e = publicMPIs[1].toBigInteger();\n          var m = data.toBigInteger();\n          return [rsa.encrypt(m, e, n)];\n\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var p = publicMPIs[0].toBigInteger();\n          var g = publicMPIs[1].toBigInteger();\n          var y = publicMPIs[2].toBigInteger();\n          var m = data.toBigInteger();\n          return elgamal.encrypt(m, g, p, y);\n\n        default:\n          return [];\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n  /**\n   * Decrypts data using the specified public key multiprecision integers of the private key,\n   * the specified secretMPIs of the private key and the specified algorithm.\n   * @param {Integer} algo Algorithm to be used (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Algorithm dependent multiprecision integers \n   * of the public key part of the private key\n   * @param {openpgp_type_mpi[]} secretMPIs Algorithm dependent multiprecision integers \n   * of the private key used\n   * @param {openpgp_type_mpi} data Data to be encrypted as MPI\n   * @return {openpgp_type_mpi} returns a big integer containing the decrypted data; otherwise null\n   */\n\n  publicKeyDecrypt: function(algo, keyIntegers, dataIntegers) {\n    var bn = (function() {\n      switch (algo) {\n        case 'rsa_encrypt_sign':\n        case 'rsa_encrypt':\n          var rsa = new publicKey.rsa();\n          // 0 and 1 are the public key.\n          var d = keyIntegers[2].toBigInteger();\n          var p = keyIntegers[3].toBigInteger();\n          var q = keyIntegers[4].toBigInteger();\n          var u = keyIntegers[5].toBigInteger();\n          var m = dataIntegers[0].toBigInteger();\n          return rsa.decrypt(m, d, p, q, u);\n        case 'elgamal':\n          var elgamal = new publicKey.elgamal();\n          var x = keyIntegers[3].toBigInteger();\n          var c1 = dataIntegers[0].toBigInteger();\n          var c2 = dataIntegers[1].toBigInteger();\n          var p = keyIntegers[0].toBigInteger();\n          return elgamal.decrypt(c1, c2, p, x);\n        default:\n          return null;\n      }\n    })();\n\n    var result = new type_mpi();\n    result.fromBigInteger(bn);\n    return result;\n  },\n\n  /** Returns the number of integers comprising the private key of an algorithm\n   * @param {openpgp.publickey} algo The public key algorithm\n   * @return {Integer} The number of integers.\n   */\n  getPrivateMpiCount: function(algo) {\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        //   Algorithm-Specific Fields for RSA secret keys:\n        //   - multiprecision integer (MPI) of RSA secret exponent d.\n        //   - MPI of RSA secret prime value p.\n        //   - MPI of RSA secret prime value q (p < q).\n        //   - MPI of u, the multiplicative inverse of p, mod q.\n        return 4;\n      case 'elgamal':\n        // Algorithm-Specific Fields for Elgamal secret keys:\n        //   - MPI of Elgamal secret exponent x.\n        return 1;\n      case 'dsa':\n        // Algorithm-Specific Fields for DSA secret keys:\n        //   - MPI of DSA secret exponent x.\n        return 1;\n      default:\n        throw new Error('Unknown algorithm');\n    }\n  },\n\n  getPublicMpiCount: function(algo) {\n    // - A series of multiprecision integers comprising the key material:\n    //   Algorithm-Specific Fields for RSA public keys:\n    //       - a multiprecision integer (MPI) of RSA public modulus n;\n    //       - an MPI of RSA public encryption exponent e.\n    switch (algo) {\n      case 'rsa_encrypt':\n      case 'rsa_encrypt_sign':\n      case 'rsa_sign':\n        return 2;\n\n        //   Algorithm-Specific Fields for Elgamal public keys:\n        //     - MPI of Elgamal prime p;\n        //     - MPI of Elgamal group generator g;\n        //     - MPI of Elgamal public key value y (= g**x mod p where x  is secret).\n      case 'elgamal':\n        return 3;\n\n        //   Algorithm-Specific Fields for DSA public keys:\n        //       - MPI of DSA prime p;\n        //       - MPI of DSA group order q (q is a prime divisor of p-1);\n        //       - MPI of DSA group generator g;\n        //       - MPI of DSA public-key value y (= g**x mod p where x  is secret).\n      case 'dsa':\n        return 4;\n\n      default:\n        throw new Error('Unknown algorithm.');\n    }\n  },\n\n  generateMpi: function(algo, bits) {\n    var result = (function() {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n        case 'rsa_sign':\n          //remember \"publicKey\" refers to the crypto/public_key dir\n          var rsa = new publicKey.rsa();\n          var keyObject = rsa.generate(bits, \"10001\");\n          var output = [];\n          output.push(keyObject.n);\n          output.push(keyObject.ee);\n          output.push(keyObject.d);\n          output.push(keyObject.p);\n          output.push(keyObject.q);\n          output.push(keyObject.u);\n          return output;\n        default:\n          throw new Error('Unsupported algorithm for key generation.');\n      }\n    })();\n\n    return result.map(function(bn) {\n      var mpi = new type_mpi();\n      mpi.fromBigInteger(bn);\n      return mpi;\n    });\n  },\n\n\n  /**\n   * generate random byte prefix as string for the specified algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes with length equal to the block\n   * size of the cipher\n   */\n  getPrefixRandom: function(algo) {\n    return random.getRandomBytes(cipher[algo].blockSize);\n  },\n\n  /**\n   * Generating a session key for the specified symmetric algorithm\n   * @param {Integer} algo Algorithm to use (see RFC4880 9.2)\n   * @return {String} Random bytes as a string to be used as a key\n   */\n  generateSessionKey: function(algo) {\n    return random.getRandomBytes(cipher[algo].keySize);\n  },\n\n};\n","var sha = require('./sha.js');\n\nmodule.exports = {\n  md5: require('./md5.js'),\n  sha1: sha.sha1,\n  sha256: sha.sha256,\n  sha224: sha.sha224,\n  sha384: sha.sha384,\n  sha512: sha.sha512,\n  ripemd: require('./ripe-md.js'),\n\n  /**\n   * Create a hash on the specified data using the specified algorithm\n   * @param {Integer} algo Hash algorithm type (see RFC4880 9.4)\n   * @param {String} data Data to be hashed\n   * @return {String} hash value\n   */\n  digest: function(algo, data) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return this.md5(data);\n      case 2:\n        // - SHA-1 [FIPS180]\n        return this.sha1(data);\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return this.ripemd(data);\n      case 8:\n        // - SHA256 [FIPS180]\n        return this.sha256(data);\n      case 9:\n        // - SHA384 [FIPS180]\n        return this.sha384(data);\n      case 10:\n        // - SHA512 [FIPS180]\n        return this.sha512(data);\n      case 11:\n        // - SHA224 [FIPS180]\n        return this.sha224(data);\n      default:\n        throw new Error('Invalid hash function.');\n    }\n  },\n\n  /**\n   * Returns the hash size in bytes of the specified hash algorithm type\n   * @param {Integer} algo Hash algorithm type (See RFC4880 9.4)\n   * @return {Integer} Size in bytes of the resulting hash\n   */\n  getHashByteLength: function(algo) {\n    switch (algo) {\n      case 1:\n        // - MD5 [HAC]\n        return 16;\n      case 2:\n        // - SHA-1 [FIPS180]\n      case 3:\n        // - RIPE-MD/160 [HAC]\n        return 20;\n      case 8:\n        // - SHA256 [FIPS180]\n        return 32;\n      case 9:\n        // - SHA384 [FIPS180]\n        return 48\n      case 10:\n        // - SHA512 [FIPS180]\n        return 64;\n      case 11:\n        // - SHA224 [FIPS180]\n        return 28;\n      default:\n        throw new Error('Invalid hash algorithm.');\n    }\n  }\n\n}\n","/**\n * A fast MD5 JavaScript implementation\n * Copyright (c) 2012 Joseph Myers\n * http://www.myersdaily.org/joseph/javascript/md5-text.html\n *\n * Permission to use, copy, modify, and distribute this software\n * and its documentation for any purposes and without\n * fee is hereby granted provided that this copyright notice\n * appears in all copies.\n *\n * Of course, this soft is provided \"as is\" without express or implied\n * warranty of any kind.\n */\n\nvar util = require('../../util/util.js');\n\nfunction MD5(entree) {\n  var hex = md5(entree);\n  var bin = util.hex2bin(hex);\n  return bin;\n}\n\nfunction md5cycle(x, k) {\n  var a = x[0],\n    b = x[1],\n    c = x[2],\n    d = x[3];\n\n  a = ff(a, b, c, d, k[0], 7, -680876936);\n  d = ff(d, a, b, c, k[1], 12, -389564586);\n  c = ff(c, d, a, b, k[2], 17, 606105819);\n  b = ff(b, c, d, a, k[3], 22, -1044525330);\n  a = ff(a, b, c, d, k[4], 7, -176418897);\n  d = ff(d, a, b, c, k[5], 12, 1200080426);\n  c = ff(c, d, a, b, k[6], 17, -1473231341);\n  b = ff(b, c, d, a, k[7], 22, -45705983);\n  a = ff(a, b, c, d, k[8], 7, 1770035416);\n  d = ff(d, a, b, c, k[9], 12, -1958414417);\n  c = ff(c, d, a, b, k[10], 17, -42063);\n  b = ff(b, c, d, a, k[11], 22, -1990404162);\n  a = ff(a, b, c, d, k[12], 7, 1804603682);\n  d = ff(d, a, b, c, k[13], 12, -40341101);\n  c = ff(c, d, a, b, k[14], 17, -1502002290);\n  b = ff(b, c, d, a, k[15], 22, 1236535329);\n\n  a = gg(a, b, c, d, k[1], 5, -165796510);\n  d = gg(d, a, b, c, k[6], 9, -1069501632);\n  c = gg(c, d, a, b, k[11], 14, 643717713);\n  b = gg(b, c, d, a, k[0], 20, -373897302);\n  a = gg(a, b, c, d, k[5], 5, -701558691);\n  d = gg(d, a, b, c, k[10], 9, 38016083);\n  c = gg(c, d, a, b, k[15], 14, -660478335);\n  b = gg(b, c, d, a, k[4], 20, -405537848);\n  a = gg(a, b, c, d, k[9], 5, 568446438);\n  d = gg(d, a, b, c, k[14], 9, -1019803690);\n  c = gg(c, d, a, b, k[3], 14, -187363961);\n  b = gg(b, c, d, a, k[8], 20, 1163531501);\n  a = gg(a, b, c, d, k[13], 5, -1444681467);\n  d = gg(d, a, b, c, k[2], 9, -51403784);\n  c = gg(c, d, a, b, k[7], 14, 1735328473);\n  b = gg(b, c, d, a, k[12], 20, -1926607734);\n\n  a = hh(a, b, c, d, k[5], 4, -378558);\n  d = hh(d, a, b, c, k[8], 11, -2022574463);\n  c = hh(c, d, a, b, k[11], 16, 1839030562);\n  b = hh(b, c, d, a, k[14], 23, -35309556);\n  a = hh(a, b, c, d, k[1], 4, -1530992060);\n  d = hh(d, a, b, c, k[4], 11, 1272893353);\n  c = hh(c, d, a, b, k[7], 16, -155497632);\n  b = hh(b, c, d, a, k[10], 23, -1094730640);\n  a = hh(a, b, c, d, k[13], 4, 681279174);\n  d = hh(d, a, b, c, k[0], 11, -358537222);\n  c = hh(c, d, a, b, k[3], 16, -722521979);\n  b = hh(b, c, d, a, k[6], 23, 76029189);\n  a = hh(a, b, c, d, k[9], 4, -640364487);\n  d = hh(d, a, b, c, k[12], 11, -421815835);\n  c = hh(c, d, a, b, k[15], 16, 530742520);\n  b = hh(b, c, d, a, k[2], 23, -995338651);\n\n  a = ii(a, b, c, d, k[0], 6, -198630844);\n  d = ii(d, a, b, c, k[7], 10, 1126891415);\n  c = ii(c, d, a, b, k[14], 15, -1416354905);\n  b = ii(b, c, d, a, k[5], 21, -57434055);\n  a = ii(a, b, c, d, k[12], 6, 1700485571);\n  d = ii(d, a, b, c, k[3], 10, -1894986606);\n  c = ii(c, d, a, b, k[10], 15, -1051523);\n  b = ii(b, c, d, a, k[1], 21, -2054922799);\n  a = ii(a, b, c, d, k[8], 6, 1873313359);\n  d = ii(d, a, b, c, k[15], 10, -30611744);\n  c = ii(c, d, a, b, k[6], 15, -1560198380);\n  b = ii(b, c, d, a, k[13], 21, 1309151649);\n  a = ii(a, b, c, d, k[4], 6, -145523070);\n  d = ii(d, a, b, c, k[11], 10, -1120210379);\n  c = ii(c, d, a, b, k[2], 15, 718787259);\n  b = ii(b, c, d, a, k[9], 21, -343485551);\n\n  x[0] = add32(a, x[0]);\n  x[1] = add32(b, x[1]);\n  x[2] = add32(c, x[2]);\n  x[3] = add32(d, x[3]);\n\n}\n\nfunction cmn(q, a, b, x, s, t) {\n  a = add32(add32(a, q), add32(x, t));\n  return add32((a << s) | (a >>> (32 - s)), b);\n}\n\nfunction ff(a, b, c, d, x, s, t) {\n  return cmn((b & c) | ((~b) & d), a, b, x, s, t);\n}\n\nfunction gg(a, b, c, d, x, s, t) {\n  return cmn((b & d) | (c & (~d)), a, b, x, s, t);\n}\n\nfunction hh(a, b, c, d, x, s, t) {\n  return cmn(b ^ c ^ d, a, b, x, s, t);\n}\n\nfunction ii(a, b, c, d, x, s, t) {\n  return cmn(c ^ (b | (~d)), a, b, x, s, t);\n}\n\nfunction md51(s) {\n  txt = '';\n  var n = s.length,\n    state = [1732584193, -271733879, -1732584194, 271733878],\n    i;\n  for (i = 64; i <= s.length; i += 64) {\n    md5cycle(state, md5blk(s.substring(i - 64, i)));\n  }\n  s = s.substring(i - 64);\n  var tail = [0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0];\n  for (i = 0; i < s.length; i++)\n    tail[i >> 2] |= s.charCodeAt(i) << ((i % 4) << 3);\n  tail[i >> 2] |= 0x80 << ((i % 4) << 3);\n  if (i > 55) {\n    md5cycle(state, tail);\n    for (i = 0; i < 16; i++) tail[i] = 0;\n  }\n  tail[14] = n * 8;\n  md5cycle(state, tail);\n  return state;\n}\n\n/* there needs to be support for Unicode here,\n * unless we pretend that we can redefine the MD-5\n * algorithm for multi-byte characters (perhaps\n * by adding every four 16-bit characters and\n * shortening the sum to 32 bits). Otherwise\n * I suggest performing MD-5 as if every character\n * was two bytes--e.g., 0040 0025 = @%--but then\n * how will an ordinary MD-5 sum be matched?\n * There is no way to standardize text to something\n * like UTF-8 before transformation; speed cost is\n * utterly prohibitive. The JavaScript standard\n * itself needs to look at this: it should start\n * providing access to strings as preformed UTF-8\n * 8-bit unsigned value arrays.\n */\nfunction md5blk(s) { /* I figured global was faster.   */\n  var md5blks = [],\n    i; /* Andy King said do it this way. */\n  for (i = 0; i < 64; i += 4) {\n    md5blks[i >> 2] = s.charCodeAt(i) + (s.charCodeAt(i + 1) << 8) + (s.charCodeAt(i + 2) << 16) + (s.charCodeAt(i + 3) <<\n      24);\n  }\n  return md5blks;\n}\n\nvar hex_chr = '0123456789abcdef'.split('');\n\nfunction rhex(n) {\n  var s = '',\n    j = 0;\n  for (; j < 4; j++)\n    s += hex_chr[(n >> (j * 8 + 4)) & 0x0F] + hex_chr[(n >> (j * 8)) & 0x0F];\n  return s;\n}\n\nfunction hex(x) {\n  for (var i = 0; i < x.length; i++)\n    x[i] = rhex(x[i]);\n  return x.join('');\n}\n\nfunction md5(s) {\n  return hex(md51(s));\n}\n\n/* this function is much faster,\nso if possible we use it. Some IEs\nare the only ones I know of that\nneed the idiotic second function,\ngenerated by an if clause.  */\n\nfunction add32(a, b) {\n  return (a + b) & 0xFFFFFFFF;\n}\n\nif (md5('hello') != '5d41402abc4b2a76b9719d911017c592') {\n  function add32(x, y) {\n    var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n      msw = (x >> 16) + (y >> 16) + (lsw >> 16);\n    return (msw << 16) | (lsw & 0xFFFF);\n  }\n}\n\nmodule.exports = MD5\n","/*\n * CryptoMX Tools\n * Copyright (C) 2004 - 2006 Derek Buitenhuis\n *\n * This program is free software; you can redistribute it and/or\n * modify it under the terms of the GNU General Public License\n * as published by the Free Software Foundation; either version 2\n * of the License, or (at your option) any later version.\n *\n * This program is distributed in the hope that it will be useful,\n * but WITHOUT ANY WARRANTY; without even the implied warranty of\n * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\n * GNU General Public License for more details.\n *\n * You should have received a copy of the GNU General Public License\n * along with this program; if not, write to the Free Software\n * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.\n */\n\n/* Modified by Recurity Labs GmbH\n */\n\nvar RMDsize = 160;\nvar X = new Array();\n\nfunction ROL(x, n) {\n  return new Number((x << n) | (x >>> (32 - n)));\n}\n\nfunction F(x, y, z) {\n  return new Number(x ^ y ^ z);\n}\n\nfunction G(x, y, z) {\n  return new Number((x & y) | (~x & z));\n}\n\nfunction H(x, y, z) {\n  return new Number((x | ~y) ^ z);\n}\n\nfunction I(x, y, z) {\n  return new Number((x & z) | (y & ~z));\n}\n\nfunction J(x, y, z) {\n  return new Number(x ^ (y | ~z));\n}\n\nfunction mixOneRound(a, b, c, d, e, x, s, roundNumber) {\n  switch (roundNumber) {\n    case 0:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n    case 1:\n      a += G(b, c, d) + x + 0x5a827999;\n      break;\n    case 2:\n      a += H(b, c, d) + x + 0x6ed9eba1;\n      break;\n    case 3:\n      a += I(b, c, d) + x + 0x8f1bbcdc;\n      break;\n    case 4:\n      a += J(b, c, d) + x + 0xa953fd4e;\n      break;\n    case 5:\n      a += J(b, c, d) + x + 0x50a28be6;\n      break;\n    case 6:\n      a += I(b, c, d) + x + 0x5c4dd124;\n      break;\n    case 7:\n      a += H(b, c, d) + x + 0x6d703ef3;\n      break;\n    case 8:\n      a += G(b, c, d) + x + 0x7a6d76e9;\n      break;\n    case 9:\n      a += F(b, c, d) + x + 0x00000000;\n      break;\n\n    default:\n      document.write(\"Bogus round number\");\n      break;\n  }\n\n  a = ROL(a, s) + e;\n  c = ROL(c, 10);\n\n  a &= 0xffffffff;\n  b &= 0xffffffff;\n  c &= 0xffffffff;\n  d &= 0xffffffff;\n  e &= 0xffffffff;\n\n  var retBlock = new Array();\n  retBlock[0] = a;\n  retBlock[1] = b;\n  retBlock[2] = c;\n  retBlock[3] = d;\n  retBlock[4] = e;\n  retBlock[5] = x;\n  retBlock[6] = s;\n\n  return retBlock;\n}\n\nfunction MDinit(MDbuf) {\n  MDbuf[0] = 0x67452301;\n  MDbuf[1] = 0xefcdab89;\n  MDbuf[2] = 0x98badcfe;\n  MDbuf[3] = 0x10325476;\n  MDbuf[4] = 0xc3d2e1f0;\n}\n\nvar ROLs = [\n  [11, 14, 15, 12, 5, 8, 7, 9, 11, 13, 14, 15, 6, 7, 9, 8],\n  [7, 6, 8, 13, 11, 9, 7, 15, 7, 12, 15, 9, 11, 7, 13, 12],\n  [11, 13, 6, 7, 14, 9, 13, 15, 14, 8, 13, 6, 5, 12, 7, 5],\n  [11, 12, 14, 15, 14, 15, 9, 8, 9, 14, 5, 6, 8, 6, 5, 12],\n  [9, 15, 5, 11, 6, 8, 13, 12, 5, 12, 13, 14, 11, 8, 5, 6],\n  [8, 9, 9, 11, 13, 15, 15, 5, 7, 7, 8, 11, 14, 14, 12, 6],\n  [9, 13, 15, 7, 12, 8, 9, 11, 7, 7, 12, 7, 6, 15, 13, 11],\n  [9, 7, 15, 11, 8, 6, 6, 14, 12, 13, 5, 14, 13, 13, 7, 5],\n  [15, 5, 8, 11, 14, 14, 6, 14, 6, 9, 12, 9, 12, 5, 15, 8],\n  [8, 5, 12, 9, 12, 5, 14, 6, 8, 13, 6, 5, 15, 13, 11, 11]\n];\n\nvar indexes = [\n  [0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15],\n  [7, 4, 13, 1, 10, 6, 15, 3, 12, 0, 9, 5, 2, 14, 11, 8],\n  [3, 10, 14, 4, 9, 15, 8, 1, 2, 7, 0, 6, 13, 11, 5, 12],\n  [1, 9, 11, 10, 0, 8, 12, 4, 13, 3, 7, 15, 14, 5, 6, 2],\n  [4, 0, 5, 9, 7, 12, 2, 10, 14, 1, 3, 8, 11, 6, 15, 13],\n  [5, 14, 7, 0, 9, 2, 11, 4, 13, 6, 15, 8, 1, 10, 3, 12],\n  [6, 11, 3, 7, 0, 13, 5, 10, 14, 15, 8, 12, 4, 9, 1, 2],\n  [15, 5, 1, 3, 7, 14, 6, 9, 11, 8, 12, 2, 10, 0, 4, 13],\n  [8, 6, 4, 1, 3, 11, 15, 0, 5, 12, 2, 13, 9, 7, 10, 14],\n  [12, 15, 10, 4, 1, 5, 8, 7, 6, 2, 13, 14, 0, 3, 9, 11]\n];\n\nfunction compress(MDbuf, X) {\n  blockA = new Array();\n  blockB = new Array();\n\n  var retBlock;\n\n  for (var i = 0; i < 5; i++) {\n    blockA[i] = new Number(MDbuf[i]);\n    blockB[i] = new Number(MDbuf[i]);\n  }\n\n  var step = 0;\n  for (var j = 0; j < 5; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockA[(step + 0) % 5],\n        blockA[(step + 1) % 5],\n        blockA[(step + 2) % 5],\n        blockA[(step + 3) % 5],\n        blockA[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockA[(step + 0) % 5] = retBlock[0];\n      blockA[(step + 1) % 5] = retBlock[1];\n      blockA[(step + 2) % 5] = retBlock[2];\n      blockA[(step + 3) % 5] = retBlock[3];\n      blockA[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  step = 0;\n  for (var j = 5; j < 10; j++) {\n    for (var i = 0; i < 16; i++) {\n      retBlock = mixOneRound(\n        blockB[(step + 0) % 5],\n        blockB[(step + 1) % 5],\n        blockB[(step + 2) % 5],\n        blockB[(step + 3) % 5],\n        blockB[(step + 4) % 5],\n        X[indexes[j][i]],\n        ROLs[j][i],\n        j);\n\n      blockB[(step + 0) % 5] = retBlock[0];\n      blockB[(step + 1) % 5] = retBlock[1];\n      blockB[(step + 2) % 5] = retBlock[2];\n      blockB[(step + 3) % 5] = retBlock[3];\n      blockB[(step + 4) % 5] = retBlock[4];\n\n      step += 4;\n    }\n  }\n\n  blockB[3] += blockA[2] + MDbuf[1];\n  MDbuf[1] = MDbuf[2] + blockA[3] + blockB[4];\n  MDbuf[2] = MDbuf[3] + blockA[4] + blockB[0];\n  MDbuf[3] = MDbuf[4] + blockA[0] + blockB[1];\n  MDbuf[4] = MDbuf[0] + blockA[1] + blockB[2];\n  MDbuf[0] = blockB[3];\n}\n\nfunction zeroX(X) {\n  for (var i = 0; i < 16; i++) {\n    X[i] = 0;\n  }\n}\n\nfunction MDfinish(MDbuf, strptr, lswlen, mswlen) {\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var i = 0; i < (lswlen & 63); i++) {\n    X[i >>> 2] ^= (strptr.charCodeAt(j++) & 255) << (8 * (i & 3));\n  }\n\n  X[(lswlen >>> 2) & 15] ^= 1 << (8 * (lswlen & 3) + 7);\n\n  if ((lswlen & 63) > 55) {\n    compress(MDbuf, X);\n    var X = new Array(16);\n    zeroX(X);\n  }\n\n  X[14] = lswlen << 3;\n  X[15] = (lswlen >>> 29) | (mswlen << 3);\n\n  compress(MDbuf, X);\n}\n\nfunction BYTES_TO_DWORD(fourChars) {\n  var tmp = (fourChars.charCodeAt(3) & 255) << 24;\n  tmp |= (fourChars.charCodeAt(2) & 255) << 16;\n  tmp |= (fourChars.charCodeAt(1) & 255) << 8;\n  tmp |= (fourChars.charCodeAt(0) & 255);\n\n  return tmp;\n}\n\nfunction RMD(message) {\n  var MDbuf = new Array(RMDsize / 32);\n  var hashcode = new Array(RMDsize / 8);\n  var length;\n  var nbytes;\n\n  MDinit(MDbuf);\n  length = message.length;\n\n  var X = new Array(16);\n  zeroX(X);\n\n  var j = 0;\n  for (var nbytes = length; nbytes > 63; nbytes -= 64) {\n    for (var i = 0; i < 16; i++) {\n      X[i] = BYTES_TO_DWORD(message.substr(j, 4));\n      j += 4;\n    }\n    compress(MDbuf, X);\n  }\n\n  MDfinish(MDbuf, message.substr(j), length, 0);\n\n  for (var i = 0; i < RMDsize / 8; i += 4) {\n    hashcode[i] = MDbuf[i >>> 2] & 255;\n    hashcode[i + 1] = (MDbuf[i >>> 2] >>> 8) & 255;\n    hashcode[i + 2] = (MDbuf[i >>> 2] >>> 16) & 255;\n    hashcode[i + 3] = (MDbuf[i >>> 2] >>> 24) & 255;\n  }\n\n  return hashcode;\n}\n\n\nfunction RMDstring(message) {\n  var hashcode = RMD(message);\n  var retString = \"\";\n\n  for (var i = 0; i < RMDsize / 8; i++) {\n    retString += String.fromCharCode(hashcode[i]);\n  }\n\n  return retString;\n}\n\nmodule.exports = RMDstring;\n","/* A JavaScript implementation of the SHA family of hashes, as defined in FIPS \n * PUB 180-2 as well as the corresponding HMAC implementation as defined in\n * FIPS PUB 198a\n *\n * Version 1.3 Copyright Brian Turek 2008-2010\n * Distributed under the BSD License\n * See http://jssha.sourceforge.net/ for more information\n *\n * Several functions taken from Paul Johnson\n */\n\n/* Modified by Recurity Labs GmbH\n * \n * This code has been slightly modified direct string output:\n * - bin2bstr has been added\n * - following wrappers of this library have been added:\n *   - str_sha1\n *   - str_sha256\n *   - str_sha224\n *   - str_sha384\n *   - str_sha512\n */\n\nvar jsSHA = (function() {\n\n  /*\n   * Configurable variables. Defaults typically work\n   */\n  /* Number of Bits Per character (8 for ASCII, 16 for Unicode) */\n  var charSize = 8,\n    /* base-64 pad character. \"=\" for strict RFC compliance */\n    b64pad = \"\",\n    /* hex output format. 0 - lowercase; 1 - uppercase */\n    hexCase = 0,\n\n    /*\n     * Int_64 is a object for 2 32-bit numbers emulating a 64-bit number\n     *\n     * @constructor\n     * @param {Number} msint_32 The most significant 32-bits of a 64-bit number\n     * @param {Number} lsint_32 The least significant 32-bits of a 64-bit number\n     */\n    Int_64 = function(msint_32, lsint_32) {\n      this.highOrder = msint_32;\n      this.lowOrder = lsint_32;\n    },\n\n    /*\n     * Convert a string to an array of big-endian words\n     * If charSize is ASCII, characters >255 have their hi-byte silently\n     * ignored.\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    str2binb = function(str) {\n      var bin = [],\n        mask = (1 << charSize) - 1,\n        length = str.length * charSize,\n        i;\n\n      for (i = 0; i < length; i += charSize) {\n        bin[i >> 5] |= (str.charCodeAt(i / charSize) & mask) <<\n          (32 - charSize - (i % 32));\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert a hex string to an array of big-endian words\n     *\n     * @param {String} str String to be converted to binary representation\n     * @return Integer array representation of the parameter\n     */\n    hex2binb = function(str) {\n      var bin = [],\n        length = str.length,\n        i, num;\n\n      for (i = 0; i < length; i += 2) {\n        num = parseInt(str.substr(i, 2), 16);\n        if (!isNaN(num)) {\n          bin[i >> 3] |= num << (24 - (4 * (i % 8)));\n        } else {\n          return \"INVALID HEX STRING\";\n        }\n      }\n\n      return bin;\n    },\n\n    /*\n     * Convert an array of big-endian words to a hex string.\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to hexidecimal\n     *\t representation\n     * @return Hexidecimal representation of the parameter in String form\n     */\n    binb2hex = function(binarray) {\n      var hex_tab = (hexCase) ? \"0123456789ABCDEF\" : \"0123456789abcdef\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, srcByte;\n\n      for (i = 0; i < length; i += 1) {\n        srcByte = binarray[i >> 2] >> ((3 - (i % 4)) * 8);\n        str += hex_tab.charAt((srcByte >> 4) & 0xF) +\n          hex_tab.charAt(srcByte & 0xF);\n      }\n\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a base-64 string\n     *\n     * @private\n     * @param {Array} binarray Array of integers to be converted to base-64\n     *\t representation\n     * @return Base-64 encoded representation of the parameter in String form\n     */\n    binb2b64 = function(binarray) {\n      var tab = \"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz\" +\n        \"0123456789+/\",\n        str = \"\",\n        length = binarray.length * 4,\n        i, j,\n        triplet;\n\n      for (i = 0; i < length; i += 3) {\n        triplet = (((binarray[i >> 2] >> 8 * (3 - i % 4)) & 0xFF) << 16) |\n          (((binarray[i + 1 >> 2] >> 8 * (3 - (i + 1) % 4)) & 0xFF) << 8) |\n          ((binarray[i + 2 >> 2] >> 8 * (3 - (i + 2) % 4)) & 0xFF);\n        for (j = 0; j < 4; j += 1) {\n          if (i * 8 + j * 6 <= binarray.length * 32) {\n            str += tab.charAt((triplet >> 6 * (3 - j)) & 0x3F);\n          } else {\n            str += b64pad;\n          }\n        }\n      }\n      return str;\n    },\n\n    /*\n     * Convert an array of big-endian words to a string\n     */\n    binb2str = function(bin) {\n      var str = \"\";\n      var mask = (1 << 8) - 1;\n      for (var i = 0; i < bin.length * 32; i += 8)\n        str += String.fromCharCode((bin[i >> 5] >>> (24 - i % 32)) & mask);\n      return str;\n    },\n    /*\n     * The 32-bit implementation of circular rotate left\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotl_32 = function(x, n) {\n      return (x << n) | (x >>> (32 - n));\n    },\n\n    /*\n     * The 32-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_32 = function(x, n) {\n      return (x >>> n) | (x << (32 - n));\n    },\n\n    /*\n     * The 64-bit implementation of circular rotate right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted circularly by n bits\n     */\n    rotr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n        (x.highOrder >>> n) | (x.lowOrder << (32 - n)), (x.lowOrder >>> n) | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n        (x.lowOrder >>> n) | (x.highOrder << (32 - n)), (x.highOrder >>> n) | (x.lowOrder << (32 - n)));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of shift right\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_32 = function(x, n) {\n      return x >>> n;\n    },\n\n    /*\n     * The 64-bit implementation of shift right\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @param {Number} n The number of bits to shift\n     * @return The x shifted by n bits\n     */\n    shr_64 = function(x, n) {\n      if (n <= 32) {\n        return new Int_64(\n          x.highOrder >>> n,\n          x.lowOrder >>> n | (x.highOrder << (32 - n)));\n      } else {\n        return new Int_64(\n          0,\n          x.highOrder << (32 - n));\n      }\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Parity function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    parity_32 = function(x, y, z) {\n      return x ^ y ^ z;\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_32 = function(x, y, z) {\n      return (x & y) ^ (~x & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Ch function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    ch_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^ (~x.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^ (~x.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument\n     * @param {Number} y The second 32-bit integer argument\n     * @param {Number} z The third 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_32 = function(x, y, z) {\n      return (x & y) ^ (x & z) ^ (y & z);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Maj function\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument\n     * @param {Int_64} y The second 64-bit integer argument\n     * @param {Int_64} z The third 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    maj_64 = function(x, y, z) {\n      return new Int_64(\n      (x.highOrder & y.highOrder) ^\n        (x.highOrder & z.highOrder) ^\n        (y.highOrder & z.highOrder), (x.lowOrder & y.lowOrder) ^\n        (x.lowOrder & z.lowOrder) ^\n        (y.lowOrder & z.lowOrder));\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_32 = function(x) {\n      return rotr_32(x, 2) ^ rotr_32(x, 13) ^ rotr_32(x, 22);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma0_64 = function(x) {\n      var rotr28 = rotr_64(x, 28),\n        rotr34 = rotr_64(x, 34),\n        rotr39 = rotr_64(x, 39);\n\n      return new Int_64(\n        rotr28.highOrder ^ rotr34.highOrder ^ rotr39.highOrder,\n        rotr28.lowOrder ^ rotr34.lowOrder ^ rotr39.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_32 = function(x) {\n      return rotr_32(x, 6) ^ rotr_32(x, 11) ^ rotr_32(x, 25);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Sigma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    sigma1_64 = function(x) {\n      var rotr14 = rotr_64(x, 14),\n        rotr18 = rotr_64(x, 18),\n        rotr41 = rotr_64(x, 41);\n\n      return new Int_64(\n        rotr14.highOrder ^ rotr18.highOrder ^ rotr41.highOrder,\n        rotr14.lowOrder ^ rotr18.lowOrder ^ rotr41.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_32 = function(x) {\n      return rotr_32(x, 7) ^ rotr_32(x, 18) ^ shr_32(x, 3);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma0 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma0_64 = function(x) {\n      var rotr1 = rotr_64(x, 1),\n        rotr8 = rotr_64(x, 8),\n        shr7 = shr_64(x, 7);\n\n      return new Int_64(\n        rotr1.highOrder ^ rotr8.highOrder ^ shr7.highOrder,\n        rotr1.lowOrder ^ rotr8.lowOrder ^ shr7.lowOrder);\n    },\n\n    /*\n     * The 32-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Number} x The 32-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_32 = function(x) {\n      return rotr_32(x, 17) ^ rotr_32(x, 19) ^ shr_32(x, 10);\n    },\n\n    /*\n     * The 64-bit implementation of the NIST specified Gamma1 function\n     *\n     * @private\n     * @param {Int_64} x The 64-bit integer argument\n     * @return The NIST specified output of the function\n     */\n    gamma1_64 = function(x) {\n      var rotr19 = rotr_64(x, 19),\n        rotr61 = rotr_64(x, 61),\n        shr6 = shr_64(x, 6);\n\n      return new Int_64(\n        rotr19.highOrder ^ rotr61.highOrder ^ shr6.highOrder,\n        rotr19.lowOrder ^ rotr61.lowOrder ^ shr6.lowOrder);\n    },\n\n    /*\n     * Add two 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} x The first 32-bit integer argument to be added\n     * @param {Number} y The second 32-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_32_2 = function(x, y) {\n      var lsw = (x & 0xFFFF) + (y & 0xFFFF),\n        msw = (x >>> 16) + (y >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add four 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_32_4 = function(a, b, c, d) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add five 32-bit integers, wrapping at 2^32. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Number} a The first 32-bit integer argument to be added\n     * @param {Number} b The second 32-bit integer argument to be added\n     * @param {Number} c The third 32-bit integer argument to be added\n     * @param {Number} d The fourth 32-bit integer argument to be added\n     * @param {Number} e The fifth 32-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_32_5 = function(a, b, c, d, e) {\n      var lsw = (a & 0xFFFF) + (b & 0xFFFF) + (c & 0xFFFF) + (d & 0xFFFF) +\n        (e & 0xFFFF),\n        msw = (a >>> 16) + (b >>> 16) + (c >>> 16) + (d >>> 16) +\n          (e >>> 16) + (lsw >>> 16);\n\n      return ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n    },\n\n    /*\n     * Add two 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} x The first 64-bit integer argument to be added\n     * @param {Int_64} y The second 64-bit integer argument to be added\n     * @return The sum of x + y\n     */\n    safeAdd_64_2 = function(x, y) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (x.lowOrder & 0xFFFF) + (y.lowOrder & 0xFFFF);\n      msw = (x.lowOrder >>> 16) + (y.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (x.highOrder & 0xFFFF) + (y.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (x.highOrder >>> 16) + (y.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add four 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d\n     */\n    safeAdd_64_4 = function(a, b, c, d) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Add five 64-bit integers, wrapping at 2^64. This uses 16-bit operations\n     * internally to work around bugs in some JS interpreters.\n     *\n     * @private\n     * @param {Int_64} a The first 64-bit integer argument to be added\n     * @param {Int_64} b The second 64-bit integer argument to be added\n     * @param {Int_64} c The third 64-bit integer argument to be added\n     * @param {Int_64} d The fouth 64-bit integer argument to be added\n     * @param {Int_64} e The fouth 64-bit integer argument to be added\n     * @return The sum of a + b + c + d + e\n     */\n    safeAdd_64_5 = function(a, b, c, d, e) {\n      var lsw, msw, lowOrder, highOrder;\n\n      lsw = (a.lowOrder & 0xFFFF) + (b.lowOrder & 0xFFFF) +\n        (c.lowOrder & 0xFFFF) + (d.lowOrder & 0xFFFF) +\n        (e.lowOrder & 0xFFFF);\n      msw = (a.lowOrder >>> 16) + (b.lowOrder >>> 16) +\n        (c.lowOrder >>> 16) + (d.lowOrder >>> 16) + (e.lowOrder >>> 16) +\n        (lsw >>> 16);\n      lowOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      lsw = (a.highOrder & 0xFFFF) + (b.highOrder & 0xFFFF) +\n        (c.highOrder & 0xFFFF) + (d.highOrder & 0xFFFF) +\n        (e.highOrder & 0xFFFF) + (msw >>> 16);\n      msw = (a.highOrder >>> 16) + (b.highOrder >>> 16) +\n        (c.highOrder >>> 16) + (d.highOrder >>> 16) +\n        (e.highOrder >>> 16) + (lsw >>> 16);\n      highOrder = ((msw & 0xFFFF) << 16) | (lsw & 0xFFFF);\n\n      return new Int_64(highOrder, lowOrder);\n    },\n\n    /*\n     * Calculates the SHA-1 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} message The binary array representation of the string to\n     *\t hash\n     * @param {Number} messageLen The number of bits in the message\n     * @return The array of integers representing the SHA-1 hash of message\n     */\n    coreSHA1 = function(message, messageLen) {\n      var W = [],\n        a, b, c, d, e, T, ch = ch_32,\n        parity = parity_32,\n        maj = maj_32,\n        rotl = rotl_32,\n        safeAdd_2 = safeAdd_32_2,\n        i, t,\n        safeAdd_5 = safeAdd_32_5,\n        appendedMessageLength,\n        H = [\n            0x67452301, 0xefcdab89, 0x98badcfe, 0x10325476, 0xc3d2e1f0\n        ],\n        K = [\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x5a827999, 0x5a827999, 0x5a827999, 0x5a827999,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1, 0x6ed9eba1,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc, 0x8f1bbcdc,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6,\n            0xca62c1d6, 0xca62c1d6, 0xca62c1d6, 0xca62c1d6\n        ];\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - (messageLen % 32));\n      /* Append length of binary string in the position such that the new\n\t\tlength is a multiple of 512.  Logic does not work for even multiples\n\t\tof 512 but there can never be even multiples of 512 */\n      message[(((messageLen + 65) >> 9) << 4) + 15] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += 16) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n\n        for (t = 0; t < 80; t += 1) {\n          if (t < 16) {\n            W[t] = message[t + i];\n          } else {\n            W[t] = rotl(W[t - 3] ^ W[t - 8] ^ W[t - 14] ^ W[t - 16], 1);\n          }\n\n          if (t < 20) {\n            T = safeAdd_5(rotl(a, 5), ch(b, c, d), e, K[t], W[t]);\n          } else if (t < 40) {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          } else if (t < 60) {\n            T = safeAdd_5(rotl(a, 5), maj(b, c, d), e, K[t], W[t]);\n          } else {\n            T = safeAdd_5(rotl(a, 5), parity(b, c, d), e, K[t], W[t]);\n          }\n\n          e = d;\n          d = c;\n          c = rotl(b, 30);\n          b = a;\n          a = T;\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n      }\n\n      return H;\n    },\n\n    /*\n     * Calculates the desired SHA-2 hash of the string set at instantiation\n     *\n     * @private\n     * @param {Array} The binary array representation of the string to hash\n     * @param {Number} The number of bits in message\n     * @param {String} variant The desired SHA-2 variant\n     * @return The array of integers representing the SHA-2 hash of message\n     */\n    coreSHA2 = function(message, messageLen, variant) {\n      var a, b, c, d, e, f, g, h, T1, T2, H, numRounds, lengthPosition, i, t,\n        binaryStringInc, binaryStringMult, safeAdd_2, safeAdd_4, safeAdd_5,\n        gamma0, gamma1, sigma0, sigma1, ch, maj, Int, K, W = [],\n        appendedMessageLength;\n\n      /* Set up the various function handles and variable for the specific \n       * variant */\n      if (variant === \"SHA-224\" || variant === \"SHA-256\") {\n        /* 32-bit variant */\n        numRounds = 64;\n        lengthPosition = (((messageLen + 65) >> 9) << 4) + 15;\n        binaryStringInc = 16;\n        binaryStringMult = 1;\n        Int = Number;\n        safeAdd_2 = safeAdd_32_2;\n        safeAdd_4 = safeAdd_32_4;\n        safeAdd_5 = safeAdd_32_5;\n        gamma0 = gamma0_32;\n        gamma1 = gamma1_32;\n        sigma0 = sigma0_32;\n        sigma1 = sigma1_32;\n        maj = maj_32;\n        ch = ch_32;\n        K = [\n            0x428A2F98, 0x71374491, 0xB5C0FBCF, 0xE9B5DBA5,\n            0x3956C25B, 0x59F111F1, 0x923F82A4, 0xAB1C5ED5,\n            0xD807AA98, 0x12835B01, 0x243185BE, 0x550C7DC3,\n            0x72BE5D74, 0x80DEB1FE, 0x9BDC06A7, 0xC19BF174,\n            0xE49B69C1, 0xEFBE4786, 0x0FC19DC6, 0x240CA1CC,\n            0x2DE92C6F, 0x4A7484AA, 0x5CB0A9DC, 0x76F988DA,\n            0x983E5152, 0xA831C66D, 0xB00327C8, 0xBF597FC7,\n            0xC6E00BF3, 0xD5A79147, 0x06CA6351, 0x14292967,\n            0x27B70A85, 0x2E1B2138, 0x4D2C6DFC, 0x53380D13,\n            0x650A7354, 0x766A0ABB, 0x81C2C92E, 0x92722C85,\n            0xA2BFE8A1, 0xA81A664B, 0xC24B8B70, 0xC76C51A3,\n            0xD192E819, 0xD6990624, 0xF40E3585, 0x106AA070,\n            0x19A4C116, 0x1E376C08, 0x2748774C, 0x34B0BCB5,\n            0x391C0CB3, 0x4ED8AA4A, 0x5B9CCA4F, 0x682E6FF3,\n            0x748F82EE, 0x78A5636F, 0x84C87814, 0x8CC70208,\n            0x90BEFFFA, 0xA4506CEB, 0xBEF9A3F7, 0xC67178F2\n        ];\n\n        if (variant === \"SHA-224\") {\n          H = [\n              0xc1059ed8, 0x367cd507, 0x3070dd17, 0xf70e5939,\n              0xffc00b31, 0x68581511, 0x64f98fa7, 0xbefa4fa4\n          ];\n        } else {\n          H = [\n              0x6A09E667, 0xBB67AE85, 0x3C6EF372, 0xA54FF53A,\n              0x510E527F, 0x9B05688C, 0x1F83D9AB, 0x5BE0CD19\n          ];\n        }\n      } else if (variant === \"SHA-384\" || variant === \"SHA-512\") {\n        /* 64-bit variant */\n        numRounds = 80;\n        lengthPosition = (((messageLen + 128) >> 10) << 5) + 31;\n        binaryStringInc = 32;\n        binaryStringMult = 2;\n        Int = Int_64;\n        safeAdd_2 = safeAdd_64_2;\n        safeAdd_4 = safeAdd_64_4;\n        safeAdd_5 = safeAdd_64_5;\n        gamma0 = gamma0_64;\n        gamma1 = gamma1_64;\n        sigma0 = sigma0_64;\n        sigma1 = sigma1_64;\n        maj = maj_64;\n        ch = ch_64;\n\n        K = [\n            new Int(0x428a2f98, 0xd728ae22), new Int(0x71374491, 0x23ef65cd),\n            new Int(0xb5c0fbcf, 0xec4d3b2f), new Int(0xe9b5dba5, 0x8189dbbc),\n            new Int(0x3956c25b, 0xf348b538), new Int(0x59f111f1, 0xb605d019),\n            new Int(0x923f82a4, 0xaf194f9b), new Int(0xab1c5ed5, 0xda6d8118),\n            new Int(0xd807aa98, 0xa3030242), new Int(0x12835b01, 0x45706fbe),\n            new Int(0x243185be, 0x4ee4b28c), new Int(0x550c7dc3, 0xd5ffb4e2),\n            new Int(0x72be5d74, 0xf27b896f), new Int(0x80deb1fe, 0x3b1696b1),\n            new Int(0x9bdc06a7, 0x25c71235), new Int(0xc19bf174, 0xcf692694),\n            new Int(0xe49b69c1, 0x9ef14ad2), new Int(0xefbe4786, 0x384f25e3),\n            new Int(0x0fc19dc6, 0x8b8cd5b5), new Int(0x240ca1cc, 0x77ac9c65),\n            new Int(0x2de92c6f, 0x592b0275), new Int(0x4a7484aa, 0x6ea6e483),\n            new Int(0x5cb0a9dc, 0xbd41fbd4), new Int(0x76f988da, 0x831153b5),\n            new Int(0x983e5152, 0xee66dfab), new Int(0xa831c66d, 0x2db43210),\n            new Int(0xb00327c8, 0x98fb213f), new Int(0xbf597fc7, 0xbeef0ee4),\n            new Int(0xc6e00bf3, 0x3da88fc2), new Int(0xd5a79147, 0x930aa725),\n            new Int(0x06ca6351, 0xe003826f), new Int(0x14292967, 0x0a0e6e70),\n            new Int(0x27b70a85, 0x46d22ffc), new Int(0x2e1b2138, 0x5c26c926),\n            new Int(0x4d2c6dfc, 0x5ac42aed), new Int(0x53380d13, 0x9d95b3df),\n            new Int(0x650a7354, 0x8baf63de), new Int(0x766a0abb, 0x3c77b2a8),\n            new Int(0x81c2c92e, 0x47edaee6), new Int(0x92722c85, 0x1482353b),\n            new Int(0xa2bfe8a1, 0x4cf10364), new Int(0xa81a664b, 0xbc423001),\n            new Int(0xc24b8b70, 0xd0f89791), new Int(0xc76c51a3, 0x0654be30),\n            new Int(0xd192e819, 0xd6ef5218), new Int(0xd6990624, 0x5565a910),\n            new Int(0xf40e3585, 0x5771202a), new Int(0x106aa070, 0x32bbd1b8),\n            new Int(0x19a4c116, 0xb8d2d0c8), new Int(0x1e376c08, 0x5141ab53),\n            new Int(0x2748774c, 0xdf8eeb99), new Int(0x34b0bcb5, 0xe19b48a8),\n            new Int(0x391c0cb3, 0xc5c95a63), new Int(0x4ed8aa4a, 0xe3418acb),\n            new Int(0x5b9cca4f, 0x7763e373), new Int(0x682e6ff3, 0xd6b2b8a3),\n            new Int(0x748f82ee, 0x5defb2fc), new Int(0x78a5636f, 0x43172f60),\n            new Int(0x84c87814, 0xa1f0ab72), new Int(0x8cc70208, 0x1a6439ec),\n            new Int(0x90befffa, 0x23631e28), new Int(0xa4506ceb, 0xde82bde9),\n            new Int(0xbef9a3f7, 0xb2c67915), new Int(0xc67178f2, 0xe372532b),\n            new Int(0xca273ece, 0xea26619c), new Int(0xd186b8c7, 0x21c0c207),\n            new Int(0xeada7dd6, 0xcde0eb1e), new Int(0xf57d4f7f, 0xee6ed178),\n            new Int(0x06f067aa, 0x72176fba), new Int(0x0a637dc5, 0xa2c898a6),\n            new Int(0x113f9804, 0xbef90dae), new Int(0x1b710b35, 0x131c471b),\n            new Int(0x28db77f5, 0x23047d84), new Int(0x32caab7b, 0x40c72493),\n            new Int(0x3c9ebe0a, 0x15c9bebc), new Int(0x431d67c4, 0x9c100d4c),\n            new Int(0x4cc5d4be, 0xcb3e42b6), new Int(0x597f299c, 0xfc657e2a),\n            new Int(0x5fcb6fab, 0x3ad6faec), new Int(0x6c44198c, 0x4a475817)\n        ];\n\n        if (variant === \"SHA-384\") {\n          H = [\n              new Int(0xcbbb9d5d, 0xc1059ed8), new Int(0x0629a292a, 0x367cd507),\n              new Int(0x9159015a, 0x3070dd17), new Int(0x0152fecd8, 0xf70e5939),\n              new Int(0x67332667, 0xffc00b31), new Int(0x98eb44a87, 0x68581511),\n              new Int(0xdb0c2e0d, 0x64f98fa7), new Int(0x047b5481d, 0xbefa4fa4)\n          ];\n        } else {\n          H = [\n              new Int(0x6a09e667, 0xf3bcc908), new Int(0xbb67ae85, 0x84caa73b),\n              new Int(0x3c6ef372, 0xfe94f82b), new Int(0xa54ff53a, 0x5f1d36f1),\n              new Int(0x510e527f, 0xade682d1), new Int(0x9b05688c, 0x2b3e6c1f),\n              new Int(0x1f83d9ab, 0xfb41bd6b), new Int(0x5be0cd19, 0x137e2179)\n          ];\n        }\n      }\n\n      /* Append '1' at the end of the binary string */\n      message[messageLen >> 5] |= 0x80 << (24 - messageLen % 32);\n      /* Append length of binary string in the position such that the new\n       * length is correct */\n      message[lengthPosition] = messageLen;\n\n      appendedMessageLength = message.length;\n\n      for (i = 0; i < appendedMessageLength; i += binaryStringInc) {\n        a = H[0];\n        b = H[1];\n        c = H[2];\n        d = H[3];\n        e = H[4];\n        f = H[5];\n        g = H[6];\n        h = H[7];\n\n        for (t = 0; t < numRounds; t += 1) {\n          if (t < 16) {\n            /* Bit of a hack - for 32-bit, the second term is ignored */\n            W[t] = new Int(message[t * binaryStringMult + i],\n              message[t * binaryStringMult + i + 1]);\n          } else {\n            W[t] = safeAdd_4(\n              gamma1(W[t - 2]), W[t - 7],\n              gamma0(W[t - 15]), W[t - 16]);\n          }\n\n          T1 = safeAdd_5(h, sigma1(e), ch(e, f, g), K[t], W[t]);\n          T2 = safeAdd_2(sigma0(a), maj(a, b, c));\n          h = g;\n          g = f;\n          f = e;\n          e = safeAdd_2(d, T1);\n          d = c;\n          c = b;\n          b = a;\n          a = safeAdd_2(T1, T2);\n        }\n\n        H[0] = safeAdd_2(a, H[0]);\n        H[1] = safeAdd_2(b, H[1]);\n        H[2] = safeAdd_2(c, H[2]);\n        H[3] = safeAdd_2(d, H[3]);\n        H[4] = safeAdd_2(e, H[4]);\n        H[5] = safeAdd_2(f, H[5]);\n        H[6] = safeAdd_2(g, H[6]);\n        H[7] = safeAdd_2(h, H[7]);\n      }\n\n      switch (variant) {\n        case \"SHA-224\":\n          return [\n            H[0], H[1], H[2], H[3],\n            H[4], H[5], H[6]];\n        case \"SHA-256\":\n          return H;\n        case \"SHA-384\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder];\n        case \"SHA-512\":\n          return [\n            H[0].highOrder, H[0].lowOrder,\n            H[1].highOrder, H[1].lowOrder,\n            H[2].highOrder, H[2].lowOrder,\n            H[3].highOrder, H[3].lowOrder,\n            H[4].highOrder, H[4].lowOrder,\n            H[5].highOrder, H[5].lowOrder,\n            H[6].highOrder, H[6].lowOrder,\n            H[7].highOrder, H[7].lowOrder];\n        default:\n          /* This should never be reached */\n          return [];\n      }\n    },\n\n    /*\n     * jsSHA is the workhorse of the library.  Instantiate it with the string to\n     * be hashed as the parameter\n     *\n     * @constructor\n     * @param {String} srcString The string to be hashed\n     * @param {String} inputFormat The format of srcString, ASCII or HEX\n     */\n    jsSHA = function(srcString, inputFormat) {\n\n      this.sha1 = null;\n      this.sha224 = null;\n      this.sha256 = null;\n      this.sha384 = null;\n      this.sha512 = null;\n\n      this.strBinLen = null;\n      this.strToHash = null;\n\n      /* Convert the input string into the correct type */\n      if (\"HEX\" === inputFormat) {\n        if (0 !== (srcString.length % 2)) {\n          return \"TEXT MUST BE IN BYTE INCREMENTS\";\n        }\n        this.strBinLen = srcString.length * 4;\n        this.strToHash = hex2binb(srcString);\n      } else if ((\"ASCII\" === inputFormat) ||\n        ('undefined' === typeof(inputFormat))) {\n        this.strBinLen = srcString.length * charSize;\n        this.strToHash = str2binb(srcString);\n      } else {\n        return \"UNKNOWN TEXT INPUT TYPE\";\n      }\n    };\n\n  jsSHA.prototype = {\n    /*\n     * Returns the desired SHA hash of the string specified at instantiation\n     * using the specified parameters\n     *\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} format The desired output formatting (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHash: function(variant, format) {\n      var formatFunc = null,\n        message = this.strToHash.slice();\n\n      switch (format) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      switch (variant) {\n        case \"SHA-1\":\n          if (null === this.sha1) {\n            this.sha1 = coreSHA1(message, this.strBinLen);\n          }\n          return formatFunc(this.sha1);\n        case \"SHA-224\":\n          if (null === this.sha224) {\n            this.sha224 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha224);\n        case \"SHA-256\":\n          if (null === this.sha256) {\n            this.sha256 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha256);\n        case \"SHA-384\":\n          if (null === this.sha384) {\n            this.sha384 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha384);\n        case \"SHA-512\":\n          if (null === this.sha512) {\n            this.sha512 = coreSHA2(message, this.strBinLen, variant);\n          }\n          return formatFunc(this.sha512);\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n    },\n\n    /*\n     * Returns the desired HMAC of the string specified at instantiation\n     * using the key and variant param.\n     *\n     * @param {String} key The key used to calculate the HMAC\n     * @param {String} inputFormat The format of key, ASCII or HEX\n     * @param {String} variant The desired SHA variant (SHA-1, SHA-224,\n     *\t SHA-256, SHA-384, or SHA-512)\n     * @param {String} outputFormat The desired output formatting\n     *\t (B64 or HEX)\n     * @return The string representation of the hash in the format specified\n     */\n    getHMAC: function(key, inputFormat, variant, outputFormat) {\n      var formatFunc, keyToUse, blockByteSize, blockBitSize, i,\n        retVal, lastArrayIndex, keyBinLen, hashBitSize,\n        keyWithIPad = [],\n        keyWithOPad = [];\n\n      /* Validate the output format selection */\n      switch (outputFormat) {\n        case \"HEX\":\n          formatFunc = binb2hex;\n          break;\n        case \"B64\":\n          formatFunc = binb2b64;\n          break;\n        case \"ASCII\":\n          formatFunc = binb2str;\n          break;\n        default:\n          return \"FORMAT NOT RECOGNIZED\";\n      }\n\n      /* Validate the hash variant selection and set needed variables */\n      switch (variant) {\n        case \"SHA-1\":\n          blockByteSize = 64;\n          hashBitSize = 160;\n          break;\n        case \"SHA-224\":\n          blockByteSize = 64;\n          hashBitSize = 224;\n          break;\n        case \"SHA-256\":\n          blockByteSize = 64;\n          hashBitSize = 256;\n          break;\n        case \"SHA-384\":\n          blockByteSize = 128;\n          hashBitSize = 384;\n          break;\n        case \"SHA-512\":\n          blockByteSize = 128;\n          hashBitSize = 512;\n          break;\n        default:\n          return \"HASH NOT RECOGNIZED\";\n      }\n\n      /* Validate input format selection */\n      if (\"HEX\" === inputFormat) {\n        /* Nibbles must come in pairs */\n        if (0 !== (key.length % 2)) {\n          return \"KEY MUST BE IN BYTE INCREMENTS\";\n        }\n        keyToUse = hex2binb(key);\n        keyBinLen = key.length * 4;\n      } else if (\"ASCII\" === inputFormat) {\n        keyToUse = str2binb(key);\n        keyBinLen = key.length * charSize;\n      } else {\n        return \"UNKNOWN KEY INPUT TYPE\";\n      }\n\n      /* These are used multiple times, calculate and store them */\n      blockBitSize = blockByteSize * 8;\n      lastArrayIndex = (blockByteSize / 4) - 1;\n\n      /* Figure out what to do with the key based on its size relative to\n       * the hash's block size */\n      if (blockByteSize < (keyBinLen / 8)) {\n        if (\"SHA-1\" === variant) {\n          keyToUse = coreSHA1(keyToUse, keyBinLen);\n        } else {\n          keyToUse = coreSHA2(keyToUse, keyBinLen, variant);\n        }\n        /* For all variants, the block size is bigger than the output\n         * size so there will never be a useful byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      } else if (blockByteSize > (keyBinLen / 8)) {\n        /* If the blockByteSize is greater than the key length, there\n         * will always be at LEAST one \"useless\" byte at the end of the\n         * string */\n        keyToUse[lastArrayIndex] &= 0xFFFFFF00;\n      }\n\n      /* Create ipad and opad */\n      for (i = 0; i <= lastArrayIndex; i += 1) {\n        keyWithIPad[i] = keyToUse[i] ^ 0x36363636;\n        keyWithOPad[i] = keyToUse[i] ^ 0x5C5C5C5C;\n      }\n\n      /* Calculate the HMAC */\n      if (\"SHA-1\" === variant) {\n        retVal = coreSHA1(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen);\n        retVal = coreSHA1(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize);\n      } else {\n        retVal = coreSHA2(\n          keyWithIPad.concat(this.strToHash),\n          blockBitSize + this.strBinLen, variant);\n        retVal = coreSHA2(\n          keyWithOPad.concat(retVal),\n          blockBitSize + hashBitSize, variant);\n      }\n\n      return (formatFunc(retVal));\n    }\n  };\n\n  return jsSHA;\n}());\n\nmodule.exports = {\n  sha1: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-1\", \"ASCII\");\n  },\n  sha224: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-224\", \"ASCII\");\n  },\n  sha256: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-256\", \"ASCII\");\n  },\n  sha384: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-384\", \"ASCII\");\n\n  },\n  sha512: function(str) {\n    var shaObj = new jsSHA(str, \"ASCII\");\n    return shaObj.getHash(\"SHA-512\", \"ASCII\");\n  }\n}\n","module.exports = {\n  cipher: require('./cipher'),\n  hash: require('./hash'),\n  cfb: require('./cfb.js'),\n  publicKey: require('./public_key'),\n  signature: require('./signature.js'),\n  random: require('./random.js'),\n  pkcs1: require('./pkcs1.js')\n\n}\n\nvar crypto = require('./crypto.js');\n\nfor (var i in crypto)\n  module.exports[i] = crypto[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * ASN1 object identifiers for hashes (See RFC4880 5.2.2)\n */\nhash_headers = new Array();\nhash_headers[1] = [0x30, 0x20, 0x30, 0x0c, 0x06, 0x08, 0x2a, 0x86, 0x48, 0x86, 0xf7, 0x0d, 0x02, 0x05, 0x05, 0x00, 0x04,\n    0x10\n];\nhash_headers[3] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2B, 0x24, 0x03, 0x02, 0x01, 0x05, 0x00, 0x04, 0x14];\nhash_headers[2] = [0x30, 0x21, 0x30, 0x09, 0x06, 0x05, 0x2b, 0x0e, 0x03, 0x02, 0x1a, 0x05, 0x00, 0x04, 0x14];\nhash_headers[8] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x01, 0x05, 0x00,\n    0x04, 0x20\n];\nhash_headers[9] = [0x30, 0x41, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x02, 0x05, 0x00,\n    0x04, 0x30\n];\nhash_headers[10] = [0x30, 0x51, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x03, 0x05,\n    0x00, 0x04, 0x40\n];\nhash_headers[11] = [0x30, 0x31, 0x30, 0x0d, 0x06, 0x09, 0x60, 0x86, 0x48, 0x01, 0x65, 0x03, 0x04, 0x02, 0x04, 0x05,\n    0x00, 0x04, 0x1C\n];\n\n\nvar crypto = require('./crypto.js'),\n  random = require('./random.js'),\n  util = require('../util'),\n  BigInteger = require('./public_key/jsbn.js'),\n  hash = require('./hash');\n\nmodule.exports = {\n  eme: {\n    /**\n     * create a EME-PKCS1-v1_5 padding (See RFC4880 13.1.1)\n     * @param {String} message message to be padded\n     * @param {Integer} length Length to the resulting message\n     * @return {String} EME-PKCS1 padded message\n     */\n    encode: function(message, length) {\n      if (message.length > length - 11)\n        return -1;\n      var result = \"\";\n      result += String.fromCharCode(0);\n      result += String.fromCharCode(2);\n      for (var i = 0; i < length - message.length - 3; i++) {\n        result += String.fromCharCode(random.getPseudoRandom(1, 255));\n      }\n      result += String.fromCharCode(0);\n      result += message;\n      return result;\n    },\n\n    /**\n     * decodes a EME-PKCS1-v1_5 padding (See RFC4880 13.1.2)\n     * @param {String} message EME-PKCS1 padded message\n     * @return {String} decoded message \n     */\n    decode: function(message, len) {\n      if (message.length < len)\n        message = String.fromCharCode(0) + message;\n      if (message.length < 12 || message.charCodeAt(0) != 0 || message.charCodeAt(1) != 2)\n        return -1;\n      var i = 2;\n      while (message.charCodeAt(i) != 0 && message.length > i)\n        i++;\n      return message.substring(i + 1, message.length);\n    },\n  },\n\n  emsa: {\n\n    /**\n     * create a EMSA-PKCS1-v1_5 padding (See RFC4880 13.1.3)\n     * @param {Integer} algo Hash algorithm type used\n     * @param {String} data Data to be hashed\n     * @param {Integer} keylength Key size of the public mpi in bytes\n     * @returns {String} Hashcode with pkcs1padding as string\n     */\n    encode: function(algo, data, keylength) {\n      var data2 = \"\";\n      data2 += String.fromCharCode(0x00);\n      data2 += String.fromCharCode(0x01);\n      for (var i = 0; i < (keylength - hash_headers[algo].length - 3 -\n        hash.getHashByteLength(algo)); i++)\n\n        data2 += String.fromCharCode(0xff);\n\n      data2 += String.fromCharCode(0x00);\n\n      for (var i = 0; i < hash_headers[algo].length; i++)\n        data2 += String.fromCharCode(hash_headers[algo][i]);\n\n      data2 += hash.digest(algo, data);\n      return new BigInteger(util.hexstrdump(data2), 16);\n    },\n\n    /**\n     * extract the hash out of an EMSA-PKCS1-v1.5 padding (See RFC4880 13.1.3) \n     * @param {String} data Hash in pkcs1 encoding\n     * @returns {String} The hash as string\n     */\n    decode: function(algo, data) {\n      var i = 0;\n      if (data.charCodeAt(0) == 0) i++;\n      else if (data.charCodeAt(0) != 1) return -1;\n      else i++;\n\n      while (data.charCodeAt(i) == 0xFF) i++;\n      if (data.charCodeAt(i++) != 0) return -1;\n      var j = 0;\n      for (j = 0; j < hash_headers[algo].length && j + i < data.length; j++) {\n        if (data.charCodeAt(j + i) != hash_headers[algo][j]) return -1;\n      }\n      i += j;\n      if (data.substring(i).length < hash.getHashByteLength(algo)) return -1;\n      return data.substring(i);\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// A Digital signature algorithm implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  hashModule = require('../hash'),\n  util = require('../../util');\n\nfunction DSA() {\n  // s1 = ((g**s) mod p) mod q\n  // s1 = ((s**-1)*(sha-1(m)+(s1*x) mod q)\n  function sign(hashalgo, m, g, p, q, x) {\n    // If the output size of the chosen hash is larger than the number of\n    // bits of q, the hash result is truncated to fit by taking the number\n    // of leftmost bits equal to the number of bits of q.  This (possibly\n    // truncated) hash function result is treated as a number and used\n    // directly in the DSA signature algorithm.\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    var k = random.getRandomBigIntegerInRange(BigInteger.ONE.add(BigInteger.ONE), q.subtract(BigInteger.ONE));\n    var s1 = (g.modPow(k, p)).mod(q);\n    var s2 = (k.modInverse(q).multiply(hash.add(x.multiply(s1)))).mod(q);\n    var result = new Array();\n    result[0] = s1.toMPI();\n    result[1] = s2.toMPI();\n    return result;\n  }\n\n  function select_hash_algorithm(q) {\n    var usersetting = openpgp.config.config.prefer_hash_algorithm;\n    /*\n     * 1024-bit key, 160-bit q, SHA-1, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 224-bit q, SHA-224, SHA-256, SHA-384, or SHA-512 hash\n     * 2048-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     * 3072-bit key, 256-bit q, SHA-256, SHA-384, or SHA-512 hash\n     */\n    switch (Math.round(q.bitLength() / 8)) {\n      case 20:\n        // 1024 bit\n        if (usersetting != 2 &&\n          usersetting > 11 &&\n          usersetting != 10 &&\n          usersetting < 8)\n          return 2; // prefer sha1\n        return usersetting;\n      case 28:\n        // 2048 bit\n        if (usersetting > 11 &&\n          usersetting < 8)\n          return 11;\n        return usersetting;\n      case 32:\n        // 4096 bit // prefer sha224\n        if (usersetting > 10 &&\n          usersetting < 8)\n          return 8; // prefer sha256\n        return usersetting;\n      default:\n        util.print_debug(\"DSA select hash algorithm: returning null for an unknown length of q\");\n        return null;\n\n    }\n  }\n  this.select_hash_algorithm = select_hash_algorithm;\n\n  function verify(hashalgo, s1, s2, m, p, q, g, y) {\n    var hashed_data = util.getLeftNBits(hashModule.digest(hashalgo, m), q.bitLength());\n    var hash = new BigInteger(util.hexstrdump(hashed_data), 16);\n    if (BigInteger.ZERO.compareTo(s1) > 0 ||\n      s1.compareTo(q) > 0 ||\n      BigInteger.ZERO.compareTo(s2) > 0 ||\n      s2.compareTo(q) > 0) {\n      util.print_error(\"invalid DSA Signature\");\n      return null;\n    }\n    var w = s2.modInverse(q);\n    var u1 = hash.multiply(w).mod(q);\n    var u2 = s1.multiply(w).mod(q);\n    return g.modPow(u1, p).multiply(y.modPow(u2, p)).mod(p).mod(q);\n  }\n\n  /*\n\t * unused code. This can be used as a start to write a key generator\n\t * function.\n\t\n\tfunction generateKey(bitcount) {\n\t    var qi = new BigInteger(bitcount, primeCenterie);\n\t    var pi = generateP(q, 512);\n\t    var gi = generateG(p, q, bitcount);\n\t    var xi;\n\t    do {\n\t        xi = new BigInteger(q.bitCount(), rand);\n\t    } while (x.compareTo(BigInteger.ZERO) != 1 && x.compareTo(q) != -1);\n\t    var yi = g.modPow(x, p);\n\t    return {x: xi, q: qi, p: pi, g: gi, y: yi};\n\t}\n\n\tfunction generateP(q, bitlength, randomfn) {\n\t    if (bitlength % 64 != 0) {\n\t    \treturn false;\n\t    }\n\t    var pTemp;\n\t    var pTemp2;\n\t    do {\n\t        pTemp = randomfn(bitcount, true);\n\t        pTemp2 = pTemp.subtract(BigInteger.ONE);\n\t        pTemp = pTemp.subtract(pTemp2.remainder(q));\n\t    } while (!pTemp.isProbablePrime(primeCenterie) || pTemp.bitLength() != l);\n\t    return pTemp;\n\t}\n\t\n\tfunction generateG(p, q, bitlength, randomfn) {\n\t    var aux = p.subtract(BigInteger.ONE);\n\t    var pow = aux.divide(q);\n\t    var gTemp;\n\t    do {\n\t        gTemp = randomfn(bitlength);\n\t    } while (gTemp.compareTo(aux) != -1 && gTemp.compareTo(BigInteger.ONE) != 1);\n\t    return gTemp.modPow(pow, p);\n\t}\n\n\tfunction generateK(q, bitlength, randomfn) {\n\t    var tempK;\n\t    do {\n\t        tempK = randomfn(bitlength, false);\n\t    } while (tempK.compareTo(q) != -1 && tempK.compareTo(BigInteger.ZERO) != 1);\n\t    return tempK;\n\t}\n\n\tfunction generateR(q,p) {\n\t    k = generateK(q);\n\t    var r = g.modPow(k, p).mod(q);\n\t    return r;\n\t}\n\n\tfunction generateS(hashfn,k,r,m,q,x) {\n        var hash = hashfn(m);\n        s = (k.modInverse(q).multiply(hash.add(x.multiply(r)))).mod(q);\n\t    return s;\n\t} */\n  this.sign = sign;\n  this.verify = verify;\n  // this.generate = generateKey;\n}\n\nmodule.exports = DSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// ElGamal implementation\n\nvar BigInteger = require('./jsbn.js'),\n  random = require('../random.js'),\n  util = require('../../util');\n\nfunction Elgamal() {\n\n  function encrypt(m, g, p, y) {\n    //  choose k in {2,...,p-2}\n    var two = BigInteger.ONE.add(BigInteger.ONE);\n    var pMinus2 = p.subtract(two);\n    var k = random.getRandomBigIntegerInRange(two, pMinus2);\n    k = k.mod(pMinus2).add(BigInteger.ONE);\n    var c = [];\n    c[0] = g.modPow(k, p);\n    c[1] = y.modPow(k, p).multiply(m).mod(p);\n    return c;\n  }\n\n  function decrypt(c1, c2, p, x) {\n    util.print_debug(\"Elgamal Decrypt:\\nc1:\" + util.hexstrdump(c1.toMPI()) + \"\\n\" +\n      \"c2:\" + util.hexstrdump(c2.toMPI()) + \"\\n\" +\n      \"p:\" + util.hexstrdump(p.toMPI()) + \"\\n\" +\n      \"x:\" + util.hexstrdump(x.toMPI()));\n    return (c1.modPow(x, p).modInverse(p)).multiply(c2).mod(p);\n    //var c = c1.pow(x).modInverse(p); // c0^-a mod p\n    //return c.multiply(c2).mod(p);\n  }\n\n  // signing and signature verification using Elgamal is not required by OpenPGP.\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n}\n\nmodule.exports = Elgamal;\n","module.exports = {\n  rsa: require('./rsa.js'),\n  elgamal: require('./elgamal.js'),\n  dsa: require('./dsa.js')\n}\n","/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH \n * \n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\nvar util = require('../../util');\n\n// Basic JavaScript BN library - subset useful for RSA encryption.\n\n// Bits per digit\nvar dbits;\n\n// JavaScript engine analysis\nvar canary = 0xdeadbeefcafe;\nvar j_lm = ((canary & 0xffffff) == 0xefcafe);\n\n// (public) Constructor\n\nfunction BigInteger(a, b, c) {\n  if (a != null)\n    if (\"number\" == typeof a) this.fromNumber(a, b, c);\n    else if (b == null && \"string\" != typeof a) this.fromString(a, 256);\n  else this.fromString(a, b);\n}\n\n// return new, unset BigInteger\n\nfunction nbi() {\n  return new BigInteger(null);\n}\n\n// am: Compute w_j += (x*this_i), propagate carries,\n// c is initial carry, returns final carry.\n// c < 3*dvalue, x < 2*dvalue, this_i < dvalue\n// We need to select the fastest one that works in this environment.\n\n// am1: use a single mult and divide to get the high bits,\n// max digit bits should be 26 because\n// max internal value = 2*dvalue^2-2*dvalue (< 2^53)\n\nfunction am1(i, x, w, j, c, n) {\n  while (--n >= 0) {\n    var v = x * this[i++] + w[j] + c;\n    c = Math.floor(v / 0x4000000);\n    w[j++] = v & 0x3ffffff;\n  }\n  return c;\n}\n// am2 avoids a big mult-and-extract completely.\n// Max digit bits should be <= 30 because we do bitwise ops\n// on values up to 2*hdvalue^2-hdvalue-1 (< 2^31)\n\nfunction am2(i, x, w, j, c, n) {\n  var xl = x & 0x7fff,\n    xh = x >> 15;\n  while (--n >= 0) {\n    var l = this[i] & 0x7fff;\n    var h = this[i++] >> 15;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x7fff) << 15) + w[j] + (c & 0x3fffffff);\n    c = (l >>> 30) + (m >>> 15) + xh * h + (c >>> 30);\n    w[j++] = l & 0x3fffffff;\n  }\n  return c;\n}\n// Alternately, set max digit bits to 28 since some\n// browsers slow down when dealing with 32-bit numbers.\n\nfunction am3(i, x, w, j, c, n) {\n  var xl = x & 0x3fff,\n    xh = x >> 14;\n  while (--n >= 0) {\n    var l = this[i] & 0x3fff;\n    var h = this[i++] >> 14;\n    var m = xh * l + h * xl;\n    l = xl * l + ((m & 0x3fff) << 14) + w[j] + c;\n    c = (l >> 28) + (m >> 14) + xh * h;\n    w[j++] = l & 0xfffffff;\n  }\n  return c;\n}\n/*if(j_lm && (navigator != undefined && \n\tnavigator.appName == \"Microsoft Internet Explorer\")) {\n  BigInteger.prototype.am = am2;\n  dbits = 30;\n}\nelse if(j_lm && (navigator != undefined && navigator.appName != \"Netscape\")) {*/\nBigInteger.prototype.am = am1;\ndbits = 26;\n/*}\nelse { // Mozilla/Netscape seems to prefer am3\n  BigInteger.prototype.am = am3;\n  dbits = 28;\n}*/\n\nBigInteger.prototype.DB = dbits;\nBigInteger.prototype.DM = ((1 << dbits) - 1);\nBigInteger.prototype.DV = (1 << dbits);\n\nvar BI_FP = 52;\nBigInteger.prototype.FV = Math.pow(2, BI_FP);\nBigInteger.prototype.F1 = BI_FP - dbits;\nBigInteger.prototype.F2 = 2 * dbits - BI_FP;\n\n// Digit conversions\nvar BI_RM = \"0123456789abcdefghijklmnopqrstuvwxyz\";\nvar BI_RC = new Array();\nvar rr, vv;\nrr = \"0\".charCodeAt(0);\nfor (vv = 0; vv <= 9; ++vv) BI_RC[rr++] = vv;\nrr = \"a\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\nrr = \"A\".charCodeAt(0);\nfor (vv = 10; vv < 36; ++vv) BI_RC[rr++] = vv;\n\nfunction int2char(n) {\n  return BI_RM.charAt(n);\n}\n\nfunction intAt(s, i) {\n  var c = BI_RC[s.charCodeAt(i)];\n  return (c == null) ? -1 : c;\n}\n\n// (protected) copy this to r\n\nfunction bnpCopyTo(r) {\n  for (var i = this.t - 1; i >= 0; --i) r[i] = this[i];\n  r.t = this.t;\n  r.s = this.s;\n}\n\n// (protected) set from integer value x, -DV <= x < DV\n\nfunction bnpFromInt(x) {\n  this.t = 1;\n  this.s = (x < 0) ? -1 : 0;\n  if (x > 0) this[0] = x;\n  else if (x < -1) this[0] = x + DV;\n  else this.t = 0;\n}\n\n// return bigint initialized to value\n\nfunction nbv(i) {\n  var r = nbi();\n  r.fromInt(i);\n  return r;\n}\n\n// (protected) set from string and radix\n\nfunction bnpFromString(s, b) {\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 256) k = 8; // byte array\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else {\n    this.fromRadix(s, b);\n    return;\n  }\n  this.t = 0;\n  this.s = 0;\n  var i = s.length,\n    mi = false,\n    sh = 0;\n  while (--i >= 0) {\n    var x = (k == 8) ? s[i] & 0xff : intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\") mi = true;\n      continue;\n    }\n    mi = false;\n    if (sh == 0)\n      this[this.t++] = x;\n    else if (sh + k > this.DB) {\n      this[this.t - 1] |= (x & ((1 << (this.DB - sh)) - 1)) << sh;\n      this[this.t++] = (x >> (this.DB - sh));\n    } else\n      this[this.t - 1] |= x << sh;\n    sh += k;\n    if (sh >= this.DB) sh -= this.DB;\n  }\n  if (k == 8 && (s[0] & 0x80) != 0) {\n    this.s = -1;\n    if (sh > 0) this[this.t - 1] |= ((1 << (this.DB - sh)) - 1) << sh;\n  }\n  this.clamp();\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) clamp off excess high words\n\nfunction bnpClamp() {\n  var c = this.s & this.DM;\n  while (this.t > 0 && this[this.t - 1] == c)--this.t;\n}\n\n// (public) return string representation in given radix\n\nfunction bnToString(b) {\n  if (this.s < 0) return \"-\" + this.negate().toString(b);\n  var k;\n  if (b == 16) k = 4;\n  else if (b == 8) k = 3;\n  else if (b == 2) k = 1;\n  else if (b == 32) k = 5;\n  else if (b == 4) k = 2;\n  else return this.toRadix(b);\n  var km = (1 << k) - 1,\n    d, m = false,\n    r = \"\",\n    i = this.t;\n  var p = this.DB - (i * this.DB) % k;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) > 0) {\n      m = true;\n      r = int2char(d);\n    }\n    while (i >= 0) {\n      if (p < k) {\n        d = (this[i] & ((1 << p) - 1)) << (k - p);\n        d |= this[--i] >> (p += this.DB - k);\n      } else {\n        d = (this[i] >> (p -= k)) & km;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      if (d > 0) m = true;\n      if (m) r += int2char(d);\n    }\n  }\n  return m ? r : \"0\";\n}\n\n// (public) -this\n\nfunction bnNegate() {\n  var r = nbi();\n  BigInteger.ZERO.subTo(this, r);\n  return r;\n}\n\n// (public) |this|\n\nfunction bnAbs() {\n  return (this.s < 0) ? this.negate() : this;\n}\n\n// (public) return + if this > a, - if this < a, 0 if equal\n\nfunction bnCompareTo(a) {\n  var r = this.s - a.s;\n  if (r != 0) return r;\n  var i = this.t;\n  r = i - a.t;\n  if (r != 0) return r;\n  while (--i >= 0) if ((r = this[i] - a[i]) != 0) return r;\n  return 0;\n}\n\n// returns bit length of the integer x\n\nfunction nbits(x) {\n  var r = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    r += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    r += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    r += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    r += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    r += 1;\n  }\n  return r;\n}\n\n// (public) return the number of bits in \"this\"\n\nfunction bnBitLength() {\n  if (this.t <= 0) return 0;\n  return this.DB * (this.t - 1) + nbits(this[this.t - 1] ^ (this.s & this.DM));\n}\n\n// (protected) r = this << n*DB\n\nfunction bnpDLShiftTo(n, r) {\n  var i;\n  for (i = this.t - 1; i >= 0; --i) r[i + n] = this[i];\n  for (i = n - 1; i >= 0; --i) r[i] = 0;\n  r.t = this.t + n;\n  r.s = this.s;\n}\n\n// (protected) r = this >> n*DB\n\nfunction bnpDRShiftTo(n, r) {\n  for (var i = n; i < this.t; ++i) r[i - n] = this[i];\n  r.t = Math.max(this.t - n, 0);\n  r.s = this.s;\n}\n\n// (protected) r = this << n\n\nfunction bnpLShiftTo(n, r) {\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << cbs) - 1;\n  var ds = Math.floor(n / this.DB),\n    c = (this.s << bs) & this.DM,\n    i;\n  for (i = this.t - 1; i >= 0; --i) {\n    r[i + ds + 1] = (this[i] >> cbs) | c;\n    c = (this[i] & bm) << bs;\n  }\n  for (i = ds - 1; i >= 0; --i) r[i] = 0;\n  r[ds] = c;\n  r.t = this.t + ds + 1;\n  r.s = this.s;\n  r.clamp();\n}\n\n// (protected) r = this >> n\n\nfunction bnpRShiftTo(n, r) {\n  r.s = this.s;\n  var ds = Math.floor(n / this.DB);\n  if (ds >= this.t) {\n    r.t = 0;\n    return;\n  }\n  var bs = n % this.DB;\n  var cbs = this.DB - bs;\n  var bm = (1 << bs) - 1;\n  r[0] = this[ds] >> bs;\n  for (var i = ds + 1; i < this.t; ++i) {\n    r[i - ds - 1] |= (this[i] & bm) << cbs;\n    r[i - ds] = this[i] >> bs;\n  }\n  if (bs > 0) r[this.t - ds - 1] |= (this.s & bm) << cbs;\n  r.t = this.t - ds;\n  r.clamp();\n}\n\n// (protected) r = this - a\n\nfunction bnpSubTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] - a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c -= a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c -= a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c -= a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c < -1) r[i++] = this.DV + c;\n  else if (c > 0) r[i++] = c;\n  r.t = i;\n  r.clamp();\n}\n\n// (protected) r = this * a, r != this,a (HAC 14.12)\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyTo(a, r) {\n  var x = this.abs(),\n    y = a.abs();\n  var i = x.t;\n  r.t = i + y.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < y.t; ++i) r[i + x.t] = x.am(0, y[i], r, i, 0, x.t);\n  r.s = 0;\n  r.clamp();\n  if (this.s != a.s) BigInteger.ZERO.subTo(r, r);\n}\n\n// (protected) r = this^2, r != this (HAC 14.16)\n\nfunction bnpSquareTo(r) {\n  var x = this.abs();\n  var i = r.t = 2 * x.t;\n  while (--i >= 0) r[i] = 0;\n  for (i = 0; i < x.t - 1; ++i) {\n    var c = x.am(i, x[i], r, 2 * i, 0, 1);\n    if ((r[i + x.t] += x.am(i + 1, 2 * x[i], r, 2 * i + 1, c, x.t - i - 1)) >= x.DV) {\n      r[i + x.t] -= x.DV;\n      r[i + x.t + 1] = 1;\n    }\n  }\n  if (r.t > 0) r[r.t - 1] += x.am(i, x[i], r, 2 * i, 0, 1);\n  r.s = 0;\n  r.clamp();\n}\n\n// (protected) divide this by m, quotient and remainder to q, r (HAC 14.20)\n// r != q, this != m.  q or r may be null.\n\nfunction bnpDivRemTo(m, q, r) {\n  var pm = m.abs();\n  if (pm.t <= 0) return;\n  var pt = this.abs();\n  if (pt.t < pm.t) {\n    if (q != null) q.fromInt(0);\n    if (r != null) this.copyTo(r);\n    return;\n  }\n  if (r == null) r = nbi();\n  var y = nbi(),\n    ts = this.s,\n    ms = m.s;\n  var nsh = this.DB - nbits(pm[pm.t - 1]); // normalize modulus\n  if (nsh > 0) {\n    pm.lShiftTo(nsh, y);\n    pt.lShiftTo(nsh, r);\n  } else {\n    pm.copyTo(y);\n    pt.copyTo(r);\n  }\n  var ys = y.t;\n  var y0 = y[ys - 1];\n  if (y0 == 0) return;\n  var yt = y0 * (1 << this.F1) + ((ys > 1) ? y[ys - 2] >> this.F2 : 0);\n  var d1 = this.FV / yt,\n    d2 = (1 << this.F1) / yt,\n    e = 1 << this.F2;\n  var i = r.t,\n    j = i - ys,\n    t = (q == null) ? nbi() : q;\n  y.dlShiftTo(j, t);\n  if (r.compareTo(t) >= 0) {\n    r[r.t++] = 1;\n    r.subTo(t, r);\n  }\n  BigInteger.ONE.dlShiftTo(ys, t);\n  t.subTo(y, y); // \"negative\" y so we can replace sub with am later\n  while (y.t < ys) y[y.t++] = 0;\n  while (--j >= 0) {\n    // Estimate quotient digit\n    var qd = (r[--i] == y0) ? this.DM : Math.floor(r[i] * d1 + (r[i - 1] + e) * d2);\n    if ((r[i] += y.am(0, qd, r, j, 0, ys)) < qd) { // Try it out\n      y.dlShiftTo(j, t);\n      r.subTo(t, r);\n      while (r[i] < --qd) r.subTo(t, r);\n    }\n  }\n  if (q != null) {\n    r.drShiftTo(ys, q);\n    if (ts != ms) BigInteger.ZERO.subTo(q, q);\n  }\n  r.t = ys;\n  r.clamp();\n  if (nsh > 0) r.rShiftTo(nsh, r); // Denormalize remainder\n  if (ts < 0) BigInteger.ZERO.subTo(r, r);\n}\n\n// (public) this mod a\n\nfunction bnMod(a) {\n  var r = nbi();\n  this.abs().divRemTo(a, null, r);\n  if (this.s < 0 && r.compareTo(BigInteger.ZERO) > 0) a.subTo(r, r);\n  return r;\n}\n\n// Modular reduction using \"classic\" algorithm\n\nfunction Classic(m) {\n  this.m = m;\n}\n\nfunction cConvert(x) {\n  if (x.s < 0 || x.compareTo(this.m) >= 0) return x.mod(this.m);\n  else return x;\n}\n\nfunction cRevert(x) {\n  return x;\n}\n\nfunction cReduce(x) {\n  x.divRemTo(this.m, null, x);\n}\n\nfunction cMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nfunction cSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\nClassic.prototype.convert = cConvert;\nClassic.prototype.revert = cRevert;\nClassic.prototype.reduce = cReduce;\nClassic.prototype.mulTo = cMulTo;\nClassic.prototype.sqrTo = cSqrTo;\n\n// (protected) return \"-1/this % 2^DB\"; useful for Mont. reduction\n// justification:\n//         xy == 1 (mod m)\n//         xy =  1+km\n//   xy(2-xy) = (1+km)(1-km)\n// x[y(2-xy)] = 1-k^2m^2\n// x[y(2-xy)] == 1 (mod m^2)\n// if y is 1/x mod m, then y(2-xy) is 1/x mod m^2\n// should reduce x and y(2-xy) by m^2 at each step to keep size bounded.\n// JS multiply \"overflows\" differently from C/C++, so care is needed here.\n\nfunction bnpInvDigit() {\n  if (this.t < 1) return 0;\n  var x = this[0];\n  if ((x & 1) == 0) return 0;\n  var y = x & 3; // y == 1/x mod 2^2\n  y = (y * (2 - (x & 0xf) * y)) & 0xf; // y == 1/x mod 2^4\n  y = (y * (2 - (x & 0xff) * y)) & 0xff; // y == 1/x mod 2^8\n  y = (y * (2 - (((x & 0xffff) * y) & 0xffff))) & 0xffff; // y == 1/x mod 2^16\n  // last step - calculate inverse mod DV directly;\n  // assumes 16 < DB <= 32 and assumes ability to handle 48-bit ints\n  y = (y * (2 - x * y % this.DV)) % this.DV; // y == 1/x mod 2^dbits\n  // we really want the negative inverse, and -DV < y < DV\n  return (y > 0) ? this.DV - y : -y;\n}\n\n// Montgomery reduction\n\nfunction Montgomery(m) {\n  this.m = m;\n  this.mp = m.invDigit();\n  this.mpl = this.mp & 0x7fff;\n  this.mph = this.mp >> 15;\n  this.um = (1 << (m.DB - 15)) - 1;\n  this.mt2 = 2 * m.t;\n}\n\n// xR mod m\n\nfunction montConvert(x) {\n  var r = nbi();\n  x.abs().dlShiftTo(this.m.t, r);\n  r.divRemTo(this.m, null, r);\n  if (x.s < 0 && r.compareTo(BigInteger.ZERO) > 0) this.m.subTo(r, r);\n  return r;\n}\n\n// x/R mod m\n\nfunction montRevert(x) {\n  var r = nbi();\n  x.copyTo(r);\n  this.reduce(r);\n  return r;\n}\n\n// x = x/R mod m (HAC 14.32)\n\nfunction montReduce(x) {\n  while (x.t <= this.mt2) // pad x so am has enough room later\n    x[x.t++] = 0;\n  for (var i = 0; i < this.m.t; ++i) {\n    // faster way of calculating u0 = x[i]*mp mod DV\n    var j = x[i] & 0x7fff;\n    var u0 = (j * this.mpl + (((j * this.mph + (x[i] >> 15) * this.mpl) & this.um) << 15)) & x.DM;\n    // use am to combine the multiply-shift-add into one call\n    j = i + this.m.t;\n    x[j] += this.m.am(0, u0, x, i, 0, this.m.t);\n    // propagate carry\n    while (x[j] >= x.DV) {\n      x[j] -= x.DV;\n      x[++j]++;\n    }\n  }\n  x.clamp();\n  x.drShiftTo(this.m.t, x);\n  if (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = \"x^2/R mod m\"; x != r\n\nfunction montSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = \"xy/R mod m\"; x,y != r\n\nfunction montMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nMontgomery.prototype.convert = montConvert;\nMontgomery.prototype.revert = montRevert;\nMontgomery.prototype.reduce = montReduce;\nMontgomery.prototype.mulTo = montMulTo;\nMontgomery.prototype.sqrTo = montSqrTo;\n\n// (protected) true iff this is even\n\nfunction bnpIsEven() {\n  return ((this.t > 0) ? (this[0] & 1) : this.s) == 0;\n}\n\n// (protected) this^e, e < 2^32, doing sqr and mul with \"r\" (HAC 14.79)\n\nfunction bnpExp(e, z) {\n  if (e > 0xffffffff || e < 1) return BigInteger.ONE;\n  var r = nbi(),\n    r2 = nbi(),\n    g = z.convert(this),\n    i = nbits(e) - 1;\n  g.copyTo(r);\n  while (--i >= 0) {\n    z.sqrTo(r, r2);\n    if ((e & (1 << i)) > 0) z.mulTo(r2, g, r);\n    else {\n      var t = r;\n      r = r2;\n      r2 = t;\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) this^e % m, 0 <= e < 2^32\n\nfunction bnModPowInt(e, m) {\n  var z;\n  if (e < 256 || m.isEven()) z = new Classic(m);\n  else z = new Montgomery(m);\n  return this.exp(e, z);\n}\n\n// protected\nBigInteger.prototype.copyTo = bnpCopyTo;\nBigInteger.prototype.fromInt = bnpFromInt;\nBigInteger.prototype.fromString = bnpFromString;\nBigInteger.prototype.clamp = bnpClamp;\nBigInteger.prototype.dlShiftTo = bnpDLShiftTo;\nBigInteger.prototype.drShiftTo = bnpDRShiftTo;\nBigInteger.prototype.lShiftTo = bnpLShiftTo;\nBigInteger.prototype.rShiftTo = bnpRShiftTo;\nBigInteger.prototype.subTo = bnpSubTo;\nBigInteger.prototype.multiplyTo = bnpMultiplyTo;\nBigInteger.prototype.squareTo = bnpSquareTo;\nBigInteger.prototype.divRemTo = bnpDivRemTo;\nBigInteger.prototype.invDigit = bnpInvDigit;\nBigInteger.prototype.isEven = bnpIsEven;\nBigInteger.prototype.exp = bnpExp;\n\n// public\nBigInteger.prototype.toString = bnToString;\nBigInteger.prototype.negate = bnNegate;\nBigInteger.prototype.abs = bnAbs;\nBigInteger.prototype.compareTo = bnCompareTo;\nBigInteger.prototype.bitLength = bnBitLength;\nBigInteger.prototype.mod = bnMod;\nBigInteger.prototype.modPowInt = bnModPowInt;\n\n// \"constants\"\nBigInteger.ZERO = nbv(0);\nBigInteger.ONE = nbv(1);\n\nmodule.exports = BigInteger;\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n\n/*\n * Copyright (c) 2003-2005  Tom Wu (tjw@cs.Stanford.EDU) \n * All Rights Reserved.\n *\n * Modified by Recurity Labs GmbH\n *\n * Permission is hereby granted, free of charge, to any person obtaining\n * a copy of this software and associated documentation files (the\n * \"Software\"), to deal in the Software without restriction, including\n * without limitation the rights to use, copy, modify, merge, publish,\n * distribute, sublicense, and/or sell copies of the Software, and to\n * permit persons to whom the Software is furnished to do so, subject to\n * the following conditions:\n *\n * The above copyright notice and this permission notice shall be\n * included in all copies or substantial portions of the Software.\n *\n * THE SOFTWARE IS PROVIDED \"AS-IS\" AND WITHOUT WARRANTY OF ANY KIND, \n * EXPRESS, IMPLIED OR OTHERWISE, INCLUDING WITHOUT LIMITATION, ANY \n * WARRANTY OF MERCHANTABILITY OR FITNESS FOR A PARTICULAR PURPOSE.  \n *\n * IN NO EVENT SHALL TOM WU BE LIABLE FOR ANY SPECIAL, INCIDENTAL,\n * INDIRECT OR CONSEQUENTIAL DAMAGES OF ANY KIND, OR ANY DAMAGES WHATSOEVER\n * RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER OR NOT ADVISED OF\n * THE POSSIBILITY OF DAMAGE, AND ON ANY THEORY OF LIABILITY, ARISING OUT\n * OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.\n *\n * In addition, the following condition applies:\n *\n * All redistributions must retain an intact copy of this copyright notice\n * and disclaimer.\n */\n\n\n// Extended JavaScript BN functions, required for RSA private ops.\n\n// Version 1.1: new BigInteger(\"0\", 10) returns \"proper\" zero\n// Version 1.2: square() API, isProbablePrime fix\n\n// (public)\nfunction bnClone() {\n  var r = nbi();\n  this.copyTo(r);\n  return r;\n}\n\n// (public) return value as integer\n\nfunction bnIntValue() {\n  if (this.s < 0) {\n    if (this.t == 1) return this[0] - this.DV;\n    else if (this.t == 0) return -1;\n  } else if (this.t == 1) return this[0];\n  else if (this.t == 0) return 0;\n  // assumes 16 < DB < 32\n  return ((this[1] & ((1 << (32 - this.DB)) - 1)) << this.DB) | this[0];\n}\n\n// (public) return value as byte\n\nfunction bnByteValue() {\n  return (this.t == 0) ? this.s : (this[0] << 24) >> 24;\n}\n\n// (public) return value as short (assumes DB>=16)\n\nfunction bnShortValue() {\n  return (this.t == 0) ? this.s : (this[0] << 16) >> 16;\n}\n\n// (protected) return x s.t. r^x < DV\n\nfunction bnpChunkSize(r) {\n  return Math.floor(Math.LN2 * this.DB / Math.log(r));\n}\n\n// (public) 0 if this == 0, 1 if this > 0\n\nfunction bnSigNum() {\n  if (this.s < 0) return -1;\n  else if (this.t <= 0 || (this.t == 1 && this[0] <= 0)) return 0;\n  else return 1;\n}\n\n// (protected) convert to radix string\n\nfunction bnpToRadix(b) {\n  if (b == null) b = 10;\n  if (this.signum() == 0 || b < 2 || b > 36) return \"0\";\n  var cs = this.chunkSize(b);\n  var a = Math.pow(b, cs);\n  var d = nbv(a),\n    y = nbi(),\n    z = nbi(),\n    r = \"\";\n  this.divRemTo(d, y, z);\n  while (y.signum() > 0) {\n    r = (a + z.intValue()).toString(b).substr(1) + r;\n    y.divRemTo(d, y, z);\n  }\n  return z.intValue().toString(b) + r;\n}\n\n// (protected) convert from radix string\n\nfunction bnpFromRadix(s, b) {\n  this.fromInt(0);\n  if (b == null) b = 10;\n  var cs = this.chunkSize(b);\n  var d = Math.pow(b, cs),\n    mi = false,\n    j = 0,\n    w = 0;\n  for (var i = 0; i < s.length; ++i) {\n    var x = intAt(s, i);\n    if (x < 0) {\n      if (s.charAt(i) == \"-\" && this.signum() == 0) mi = true;\n      continue;\n    }\n    w = b * w + x;\n    if (++j >= cs) {\n      this.dMultiply(d);\n      this.dAddOffset(w, 0);\n      j = 0;\n      w = 0;\n    }\n  }\n  if (j > 0) {\n    this.dMultiply(Math.pow(b, j));\n    this.dAddOffset(w, 0);\n  }\n  if (mi) BigInteger.ZERO.subTo(this, this);\n}\n\n// (protected) alternate constructor\n\nfunction bnpFromNumber(a, b, c) {\n  if (\"number\" == typeof b) {\n    // new BigInteger(int,int,RNG)\n    if (a < 2) this.fromInt(1);\n    else {\n      this.fromNumber(a, c);\n      if (!this.testBit(a - 1)) // force MSB set\n        this.bitwiseTo(BigInteger.ONE.shiftLeft(a - 1), op_or, this);\n      if (this.isEven()) this.dAddOffset(1, 0); // force odd\n      while (!this.isProbablePrime(b)) {\n        this.dAddOffset(2, 0);\n        if (this.bitLength() > a) this.subTo(BigInteger.ONE.shiftLeft(a - 1), this);\n      }\n    }\n  } else {\n    // new BigInteger(int,RNG)\n    var x = new Array(),\n      t = a & 7;\n    x.length = (a >> 3) + 1;\n    b.nextBytes(x);\n    if (t > 0) x[0] &= ((1 << t) - 1);\n    else x[0] = 0;\n    this.fromString(x, 256);\n  }\n}\n\n// (public) convert to bigendian byte array\n\nfunction bnToByteArray() {\n  var i = this.t,\n    r = new Array();\n  r[0] = this.s;\n  var p = this.DB - (i * this.DB) % 8,\n    d, k = 0;\n  if (i-- > 0) {\n    if (p < this.DB && (d = this[i] >> p) != (this.s & this.DM) >> p)\n      r[k++] = d | (this.s << (this.DB - p));\n    while (i >= 0) {\n      if (p < 8) {\n        d = (this[i] & ((1 << p) - 1)) << (8 - p);\n        d |= this[--i] >> (p += this.DB - 8);\n      } else {\n        d = (this[i] >> (p -= 8)) & 0xff;\n        if (p <= 0) {\n          p += this.DB;\n          --i;\n        }\n      }\n      //if((d&0x80) != 0) d |= -256;\n      //if(k == 0 && (this.s&0x80) != (d&0x80)) ++k;\n      if (k > 0 || d != this.s) r[k++] = d;\n    }\n  }\n  return r;\n}\n\nfunction bnEquals(a) {\n  return (this.compareTo(a) == 0);\n}\n\nfunction bnMin(a) {\n  return (this.compareTo(a) < 0) ? this : a;\n}\n\nfunction bnMax(a) {\n  return (this.compareTo(a) > 0) ? this : a;\n}\n\n// (protected) r = this op a (bitwise)\n\nfunction bnpBitwiseTo(a, op, r) {\n  var i, f, m = Math.min(a.t, this.t);\n  for (i = 0; i < m; ++i) r[i] = op(this[i], a[i]);\n  if (a.t < this.t) {\n    f = a.s & this.DM;\n    for (i = m; i < this.t; ++i) r[i] = op(this[i], f);\n    r.t = this.t;\n  } else {\n    f = this.s & this.DM;\n    for (i = m; i < a.t; ++i) r[i] = op(f, a[i]);\n    r.t = a.t;\n  }\n  r.s = op(this.s, a.s);\n  r.clamp();\n}\n\n// (public) this & a\n\nfunction op_and(x, y) {\n  return x & y;\n}\n\nfunction bnAnd(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_and, r);\n  return r;\n}\n\n// (public) this | a\n\nfunction op_or(x, y) {\n  return x | y;\n}\n\nfunction bnOr(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_or, r);\n  return r;\n}\n\n// (public) this ^ a\n\nfunction op_xor(x, y) {\n  return x ^ y;\n}\n\nfunction bnXor(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_xor, r);\n  return r;\n}\n\n// (public) this & ~a\n\nfunction op_andnot(x, y) {\n  return x & ~y;\n}\n\nfunction bnAndNot(a) {\n  var r = nbi();\n  this.bitwiseTo(a, op_andnot, r);\n  return r;\n}\n\n// (public) ~this\n\nfunction bnNot() {\n  var r = nbi();\n  for (var i = 0; i < this.t; ++i) r[i] = this.DM & ~this[i];\n  r.t = this.t;\n  r.s = ~this.s;\n  return r;\n}\n\n// (public) this << n\n\nfunction bnShiftLeft(n) {\n  var r = nbi();\n  if (n < 0) this.rShiftTo(-n, r);\n  else this.lShiftTo(n, r);\n  return r;\n}\n\n// (public) this >> n\n\nfunction bnShiftRight(n) {\n  var r = nbi();\n  if (n < 0) this.lShiftTo(-n, r);\n  else this.rShiftTo(n, r);\n  return r;\n}\n\n// return index of lowest 1-bit in x, x < 2^31\n\nfunction lbit(x) {\n  if (x == 0) return -1;\n  var r = 0;\n  if ((x & 0xffff) == 0) {\n    x >>= 16;\n    r += 16;\n  }\n  if ((x & 0xff) == 0) {\n    x >>= 8;\n    r += 8;\n  }\n  if ((x & 0xf) == 0) {\n    x >>= 4;\n    r += 4;\n  }\n  if ((x & 3) == 0) {\n    x >>= 2;\n    r += 2;\n  }\n  if ((x & 1) == 0)++r;\n  return r;\n}\n\n// (public) returns index of lowest 1-bit (or -1 if none)\n\nfunction bnGetLowestSetBit() {\n  for (var i = 0; i < this.t; ++i)\n    if (this[i] != 0) return i * this.DB + lbit(this[i]);\n  if (this.s < 0) return this.t * this.DB;\n  return -1;\n}\n\n// return number of 1 bits in x\n\nfunction cbit(x) {\n  var r = 0;\n  while (x != 0) {\n    x &= x - 1;\n    ++r;\n  }\n  return r;\n}\n\n// (public) return number of set bits\n\nfunction bnBitCount() {\n  var r = 0,\n    x = this.s & this.DM;\n  for (var i = 0; i < this.t; ++i) r += cbit(this[i] ^ x);\n  return r;\n}\n\n// (public) true iff nth bit is set\n\nfunction bnTestBit(n) {\n  var j = Math.floor(n / this.DB);\n  if (j >= this.t) return (this.s != 0);\n  return ((this[j] & (1 << (n % this.DB))) != 0);\n}\n\n// (protected) this op (1<<n)\n\nfunction bnpChangeBit(n, op) {\n  var r = BigInteger.ONE.shiftLeft(n);\n  this.bitwiseTo(r, op, r);\n  return r;\n}\n\n// (public) this | (1<<n)\n\nfunction bnSetBit(n) {\n  return this.changeBit(n, op_or);\n}\n\n// (public) this & ~(1<<n)\n\nfunction bnClearBit(n) {\n  return this.changeBit(n, op_andnot);\n}\n\n// (public) this ^ (1<<n)\n\nfunction bnFlipBit(n) {\n  return this.changeBit(n, op_xor);\n}\n\n// (protected) r = this + a\n\nfunction bnpAddTo(a, r) {\n  var i = 0,\n    c = 0,\n    m = Math.min(a.t, this.t);\n  while (i < m) {\n    c += this[i] + a[i];\n    r[i++] = c & this.DM;\n    c >>= this.DB;\n  }\n  if (a.t < this.t) {\n    c += a.s;\n    while (i < this.t) {\n      c += this[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += this.s;\n  } else {\n    c += this.s;\n    while (i < a.t) {\n      c += a[i];\n      r[i++] = c & this.DM;\n      c >>= this.DB;\n    }\n    c += a.s;\n  }\n  r.s = (c < 0) ? -1 : 0;\n  if (c > 0) r[i++] = c;\n  else if (c < -1) r[i++] = this.DV + c;\n  r.t = i;\n  r.clamp();\n}\n\n// (public) this + a\n\nfunction bnAdd(a) {\n  var r = nbi();\n  this.addTo(a, r);\n  return r;\n}\n\n// (public) this - a\n\nfunction bnSubtract(a) {\n  var r = nbi();\n  this.subTo(a, r);\n  return r;\n}\n\n// (public) this * a\n\nfunction bnMultiply(a) {\n  var r = nbi();\n  this.multiplyTo(a, r);\n  return r;\n}\n\n// (public) this^2\n\nfunction bnSquare() {\n  var r = nbi();\n  this.squareTo(r);\n  return r;\n}\n\n// (public) this / a\n\nfunction bnDivide(a) {\n  var r = nbi();\n  this.divRemTo(a, r, null);\n  return r;\n}\n\n// (public) this % a\n\nfunction bnRemainder(a) {\n  var r = nbi();\n  this.divRemTo(a, null, r);\n  return r;\n}\n\n// (public) [this/a,this%a]\n\nfunction bnDivideAndRemainder(a) {\n  var q = nbi(),\n    r = nbi();\n  this.divRemTo(a, q, r);\n  return new Array(q, r);\n}\n\n// (protected) this *= n, this >= 0, 1 < n < DV\n\nfunction bnpDMultiply(n) {\n  this[this.t] = this.am(0, n - 1, this, 0, 0, this.t);\n  ++this.t;\n  this.clamp();\n}\n\n// (protected) this += n << w words, this >= 0\n\nfunction bnpDAddOffset(n, w) {\n  if (n == 0) return;\n  while (this.t <= w) this[this.t++] = 0;\n  this[w] += n;\n  while (this[w] >= this.DV) {\n    this[w] -= this.DV;\n    if (++w >= this.t) this[this.t++] = 0;\n    ++this[w];\n  }\n}\n\n// A \"null\" reducer\n\nfunction NullExp() {}\n\nfunction nNop(x) {\n  return x;\n}\n\nfunction nMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n}\n\nfunction nSqrTo(x, r) {\n  x.squareTo(r);\n}\n\nNullExp.prototype.convert = nNop;\nNullExp.prototype.revert = nNop;\nNullExp.prototype.mulTo = nMulTo;\nNullExp.prototype.sqrTo = nSqrTo;\n\n// (public) this^e\n\nfunction bnPow(e) {\n  return this.exp(e, new NullExp());\n}\n\n// (protected) r = lower n words of \"this * a\", a.t <= n\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyLowerTo(a, n, r) {\n  var i = Math.min(this.t + a.t, n);\n  r.s = 0; // assumes a,this >= 0\n  r.t = i;\n  while (i > 0) r[--i] = 0;\n  var j;\n  for (j = r.t - this.t; i < j; ++i) r[i + this.t] = this.am(0, a[i], r, i, 0, this.t);\n  for (j = Math.min(a.t, n); i < j; ++i) this.am(0, a[i], r, i, 0, n - i);\n  r.clamp();\n}\n\n// (protected) r = \"this * a\" without lower n words, n > 0\n// \"this\" should be the larger one if appropriate.\n\nfunction bnpMultiplyUpperTo(a, n, r) {\n  --n;\n  var i = r.t = this.t + a.t - n;\n  r.s = 0; // assumes a,this >= 0\n  while (--i >= 0) r[i] = 0;\n  for (i = Math.max(n - this.t, 0); i < a.t; ++i)\n    r[this.t + i - n] = this.am(n - i, a[i], r, 0, 0, this.t + i - n);\n  r.clamp();\n  r.drShiftTo(1, r);\n}\n\n// Barrett modular reduction\n\nfunction Barrett(m) {\n  // setup Barrett\n  this.r2 = nbi();\n  this.q3 = nbi();\n  BigInteger.ONE.dlShiftTo(2 * m.t, this.r2);\n  this.mu = this.r2.divide(m);\n  this.m = m;\n}\n\nfunction barrettConvert(x) {\n  if (x.s < 0 || x.t > 2 * this.m.t) return x.mod(this.m);\n  else if (x.compareTo(this.m) < 0) return x;\n  else {\n    var r = nbi();\n    x.copyTo(r);\n    this.reduce(r);\n    return r;\n  }\n}\n\nfunction barrettRevert(x) {\n  return x;\n}\n\n// x = x mod m (HAC 14.42)\n\nfunction barrettReduce(x) {\n  x.drShiftTo(this.m.t - 1, this.r2);\n  if (x.t > this.m.t + 1) {\n    x.t = this.m.t + 1;\n    x.clamp();\n  }\n  this.mu.multiplyUpperTo(this.r2, this.m.t + 1, this.q3);\n  this.m.multiplyLowerTo(this.q3, this.m.t + 1, this.r2);\n  while (x.compareTo(this.r2) < 0) x.dAddOffset(1, this.m.t + 1);\n  x.subTo(this.r2, x);\n  while (x.compareTo(this.m) >= 0) x.subTo(this.m, x);\n}\n\n// r = x^2 mod m; x != r\n\nfunction barrettSqrTo(x, r) {\n  x.squareTo(r);\n  this.reduce(r);\n}\n\n// r = x*y mod m; x,y != r\n\nfunction barrettMulTo(x, y, r) {\n  x.multiplyTo(y, r);\n  this.reduce(r);\n}\n\nBarrett.prototype.convert = barrettConvert;\nBarrett.prototype.revert = barrettRevert;\nBarrett.prototype.reduce = barrettReduce;\nBarrett.prototype.mulTo = barrettMulTo;\nBarrett.prototype.sqrTo = barrettSqrTo;\n\n// (public) this^e % m (HAC 14.85)\n\nfunction bnModPow(e, m) {\n  var i = e.bitLength(),\n    k, r = nbv(1),\n    z;\n  if (i <= 0) return r;\n  else if (i < 18) k = 1;\n  else if (i < 48) k = 3;\n  else if (i < 144) k = 4;\n  else if (i < 768) k = 5;\n  else k = 6;\n  if (i < 8)\n    z = new Classic(m);\n  else if (m.isEven())\n    z = new Barrett(m);\n  else\n    z = new Montgomery(m);\n\n  // precomputation\n  var g = new Array(),\n    n = 3,\n    k1 = k - 1,\n    km = (1 << k) - 1;\n  g[1] = z.convert(this);\n  if (k > 1) {\n    var g2 = nbi();\n    z.sqrTo(g[1], g2);\n    while (n <= km) {\n      g[n] = nbi();\n      z.mulTo(g2, g[n - 2], g[n]);\n      n += 2;\n    }\n  }\n\n  var j = e.t - 1,\n    w, is1 = true,\n    r2 = nbi(),\n    t;\n  i = nbits(e[j]) - 1;\n  while (j >= 0) {\n    if (i >= k1) w = (e[j] >> (i - k1)) & km;\n    else {\n      w = (e[j] & ((1 << (i + 1)) - 1)) << (k1 - i);\n      if (j > 0) w |= e[j - 1] >> (this.DB + i - k1);\n    }\n\n    n = k;\n    while ((w & 1) == 0) {\n      w >>= 1;\n      --n;\n    }\n    if ((i -= n) < 0) {\n      i += this.DB;\n      --j;\n    }\n    if (is1) { // ret == 1, don't bother squaring or multiplying it\n      g[w].copyTo(r);\n      is1 = false;\n    } else {\n      while (n > 1) {\n        z.sqrTo(r, r2);\n        z.sqrTo(r2, r);\n        n -= 2;\n      }\n      if (n > 0) z.sqrTo(r, r2);\n      else {\n        t = r;\n        r = r2;\n        r2 = t;\n      }\n      z.mulTo(r2, g[w], r);\n    }\n\n    while (j >= 0 && (e[j] & (1 << i)) == 0) {\n      z.sqrTo(r, r2);\n      t = r;\n      r = r2;\n      r2 = t;\n      if (--i < 0) {\n        i = this.DB - 1;\n        --j;\n      }\n    }\n  }\n  return z.revert(r);\n}\n\n// (public) gcd(this,a) (HAC 14.54)\n\nfunction bnGCD(a) {\n  var x = (this.s < 0) ? this.negate() : this.clone();\n  var y = (a.s < 0) ? a.negate() : a.clone();\n  if (x.compareTo(y) < 0) {\n    var t = x;\n    x = y;\n    y = t;\n  }\n  var i = x.getLowestSetBit(),\n    g = y.getLowestSetBit();\n  if (g < 0) return x;\n  if (i < g) g = i;\n  if (g > 0) {\n    x.rShiftTo(g, x);\n    y.rShiftTo(g, y);\n  }\n  while (x.signum() > 0) {\n    if ((i = x.getLowestSetBit()) > 0) x.rShiftTo(i, x);\n    if ((i = y.getLowestSetBit()) > 0) y.rShiftTo(i, y);\n    if (x.compareTo(y) >= 0) {\n      x.subTo(y, x);\n      x.rShiftTo(1, x);\n    } else {\n      y.subTo(x, y);\n      y.rShiftTo(1, y);\n    }\n  }\n  if (g > 0) y.lShiftTo(g, y);\n  return y;\n}\n\n// (protected) this % n, n < 2^26\n\nfunction bnpModInt(n) {\n  if (n <= 0) return 0;\n  var d = this.DV % n,\n    r = (this.s < 0) ? n - 1 : 0;\n  if (this.t > 0)\n    if (d == 0) r = this[0] % n;\n    else for (var i = this.t - 1; i >= 0; --i) r = (d * r + this[i]) % n;\n  return r;\n}\n\n// (public) 1/this % m (HAC 14.61)\n\nfunction bnModInverse(m) {\n  var ac = m.isEven();\n  if ((this.isEven() && ac) || m.signum() == 0) return BigInteger.ZERO;\n  var u = m.clone(),\n    v = this.clone();\n  var a = nbv(1),\n    b = nbv(0),\n    c = nbv(0),\n    d = nbv(1);\n  while (u.signum() != 0) {\n    while (u.isEven()) {\n      u.rShiftTo(1, u);\n      if (ac) {\n        if (!a.isEven() || !b.isEven()) {\n          a.addTo(this, a);\n          b.subTo(m, b);\n        }\n        a.rShiftTo(1, a);\n      } else if (!b.isEven()) b.subTo(m, b);\n      b.rShiftTo(1, b);\n    }\n    while (v.isEven()) {\n      v.rShiftTo(1, v);\n      if (ac) {\n        if (!c.isEven() || !d.isEven()) {\n          c.addTo(this, c);\n          d.subTo(m, d);\n        }\n        c.rShiftTo(1, c);\n      } else if (!d.isEven()) d.subTo(m, d);\n      d.rShiftTo(1, d);\n    }\n    if (u.compareTo(v) >= 0) {\n      u.subTo(v, u);\n      if (ac) a.subTo(c, a);\n      b.subTo(d, b);\n    } else {\n      v.subTo(u, v);\n      if (ac) c.subTo(a, c);\n      d.subTo(b, d);\n    }\n  }\n  if (v.compareTo(BigInteger.ONE) != 0) return BigInteger.ZERO;\n  if (d.compareTo(m) >= 0) return d.subtract(m);\n  if (d.signum() < 0) d.addTo(m, d);\n  else return d;\n  if (d.signum() < 0) return d.add(m);\n  else return d;\n}\n\nvar lowprimes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97, 101,\n    103, 107, 109, 113, 127, 131, 137, 139, 149, 151, 157, 163, 167, 173, 179, 181, 191, 193, 197, 199, 211, 223, 227,\n    229, 233, 239, 241, 251, 257, 263, 269, 271, 277, 281, 283, 293, 307, 311, 313, 317, 331, 337, 347, 349, 353, 359,\n    367, 373, 379, 383, 389, 397, 401, 409, 419, 421, 431, 433, 439, 443, 449, 457, 461, 463, 467, 479, 487, 491, 499,\n    503, 509, 521, 523, 541, 547, 557, 563, 569, 571, 577, 587, 593, 599, 601, 607, 613, 617, 619, 631, 641, 643, 647,\n    653, 659, 661, 673, 677, 683, 691, 701, 709, 719, 727, 733, 739, 743, 751, 757, 761, 769, 773, 787, 797, 809, 811,\n    821, 823, 827, 829, 839, 853, 857, 859, 863, 877, 881, 883, 887, 907, 911, 919, 929, 937, 941, 947, 953, 967, 971,\n    977, 983, 991, 997\n];\nvar lplim = (1 << 26) / lowprimes[lowprimes.length - 1];\n\n// (public) test primality with certainty >= 1-.5^t\n\nfunction bnIsProbablePrime(t) {\n  var i, x = this.abs();\n  if (x.t == 1 && x[0] <= lowprimes[lowprimes.length - 1]) {\n    for (i = 0; i < lowprimes.length; ++i)\n      if (x[0] == lowprimes[i]) return true;\n    return false;\n  }\n  if (x.isEven()) return false;\n  i = 1;\n  while (i < lowprimes.length) {\n    var m = lowprimes[i],\n      j = i + 1;\n    while (j < lowprimes.length && m < lplim) m *= lowprimes[j++];\n    m = x.modInt(m);\n    while (i < j) if (m % lowprimes[i++] == 0) return false;\n  }\n  return x.millerRabin(t);\n}\n\n/* added by Recurity Labs */\n\nfunction nbits(x) {\n  var n = 1,\n    t;\n  if ((t = x >>> 16) != 0) {\n    x = t;\n    n += 16;\n  }\n  if ((t = x >> 8) != 0) {\n    x = t;\n    n += 8;\n  }\n  if ((t = x >> 4) != 0) {\n    x = t;\n    n += 4;\n  }\n  if ((t = x >> 2) != 0) {\n    x = t;\n    n += 2;\n  }\n  if ((t = x >> 1) != 0) {\n    x = t;\n    n += 1;\n  }\n  return n;\n}\n\nfunction bnToMPI() {\n  var ba = this.toByteArray();\n  var size = (ba.length - 1) * 8 + nbits(ba[0]);\n  var result = \"\";\n  result += String.fromCharCode((size & 0xFF00) >> 8);\n  result += String.fromCharCode(size & 0xFF);\n  result += util.bin2str(ba);\n  return result;\n}\n/* END of addition */\n\n// (protected) true if probably prime (HAC 4.24, Miller-Rabin)\nfunction bnpMillerRabin(t) {\n  var n1 = this.subtract(BigInteger.ONE);\n  var k = n1.getLowestSetBit();\n  if (k <= 0) return false;\n  var r = n1.shiftRight(k);\n  t = (t + 1) >> 1;\n  if (t > lowprimes.length) t = lowprimes.length;\n  var a = nbi();\n  var j, bases = [];\n  for (var i = 0; i < t; ++i) {\n    //Pick bases at random, instead of starting at 2\n    for (;;) {\n      j = lowprimes[Math.floor(Math.random() * lowprimes.length)];\n      if (bases.indexOf(j) == -1) break;\n    }\n    bases.push(j);\n    a.fromInt(j);\n    var y = a.modPow(r, this);\n    if (y.compareTo(BigInteger.ONE) != 0 && y.compareTo(n1) != 0) {\n      var j = 1;\n      while (j++ < k && y.compareTo(n1) != 0) {\n        y = y.modPowInt(2, this);\n        if (y.compareTo(BigInteger.ONE) == 0) return false;\n      }\n      if (y.compareTo(n1) != 0) return false;\n    }\n  }\n  return true;\n}\n\nvar BigInteger = require('./jsbn.js');\n\n// protected\nBigInteger.prototype.chunkSize = bnpChunkSize;\nBigInteger.prototype.toRadix = bnpToRadix;\nBigInteger.prototype.fromRadix = bnpFromRadix;\nBigInteger.prototype.fromNumber = bnpFromNumber;\nBigInteger.prototype.bitwiseTo = bnpBitwiseTo;\nBigInteger.prototype.changeBit = bnpChangeBit;\nBigInteger.prototype.addTo = bnpAddTo;\nBigInteger.prototype.dMultiply = bnpDMultiply;\nBigInteger.prototype.dAddOffset = bnpDAddOffset;\nBigInteger.prototype.multiplyLowerTo = bnpMultiplyLowerTo;\nBigInteger.prototype.multiplyUpperTo = bnpMultiplyUpperTo;\nBigInteger.prototype.modInt = bnpModInt;\nBigInteger.prototype.millerRabin = bnpMillerRabin;\n\n// public\nBigInteger.prototype.clone = bnClone;\nBigInteger.prototype.intValue = bnIntValue;\nBigInteger.prototype.byteValue = bnByteValue;\nBigInteger.prototype.shortValue = bnShortValue;\nBigInteger.prototype.signum = bnSigNum;\nBigInteger.prototype.toByteArray = bnToByteArray;\nBigInteger.prototype.equals = bnEquals;\nBigInteger.prototype.min = bnMin;\nBigInteger.prototype.max = bnMax;\nBigInteger.prototype.and = bnAnd;\nBigInteger.prototype.or = bnOr;\nBigInteger.prototype.xor = bnXor;\nBigInteger.prototype.andNot = bnAndNot;\nBigInteger.prototype.not = bnNot;\nBigInteger.prototype.shiftLeft = bnShiftLeft;\nBigInteger.prototype.shiftRight = bnShiftRight;\nBigInteger.prototype.getLowestSetBit = bnGetLowestSetBit;\nBigInteger.prototype.bitCount = bnBitCount;\nBigInteger.prototype.testBit = bnTestBit;\nBigInteger.prototype.setBit = bnSetBit;\nBigInteger.prototype.clearBit = bnClearBit;\nBigInteger.prototype.flipBit = bnFlipBit;\nBigInteger.prototype.add = bnAdd;\nBigInteger.prototype.subtract = bnSubtract;\nBigInteger.prototype.multiply = bnMultiply;\nBigInteger.prototype.divide = bnDivide;\nBigInteger.prototype.remainder = bnRemainder;\nBigInteger.prototype.divideAndRemainder = bnDivideAndRemainder;\nBigInteger.prototype.modPow = bnModPow;\nBigInteger.prototype.modInverse = bnModInverse;\nBigInteger.prototype.pow = bnPow;\nBigInteger.prototype.gcd = bnGCD;\nBigInteger.prototype.isProbablePrime = bnIsProbablePrime;\nBigInteger.prototype.toMPI = bnToMPI;\n\n// JSBN-specific extension\nBigInteger.prototype.square = bnSquare;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n//\n// RSA implementation\n\nvar BigInteger = require('./jsbn.js'),\n  util = require('../../util'),\n  random = require('../random.js');\n\nfunction SecureRandom() {\n  function nextBytes(byteArray) {\n    for (var n = 0; n < byteArray.length; n++) {\n      byteArray[n] = random.getSecureRandomOctet();\n    }\n  }\n  this.nextBytes = nextBytes;\n}\n\nfunction RSA() {\n  /**\n   * This function uses jsbn Big Num library to decrypt RSA\n   * @param m\n   *            message\n   * @param d\n   *            RSA d as BigInteger\n   * @param p\n   *            RSA p as BigInteger\n   * @param q\n   *            RSA q as BigInteger\n   * @param u\n   *            RSA u as BigInteger\n   * @return {BigInteger} The decrypted value of the message\n   */\n  function decrypt(m, d, p, q, u) {\n    var xp = m.mod(p).modPow(d.mod(p.subtract(BigInteger.ONE)), p);\n    var xq = m.mod(q).modPow(d.mod(q.subtract(BigInteger.ONE)), q);\n    util.print_debug(\"rsa.js decrypt\\nxpn:\" + util.hexstrdump(xp.toMPI()) + \"\\nxqn:\" + util.hexstrdump(xq.toMPI()));\n\n    var t = xq.subtract(xp);\n    if (t[0] == 0) {\n      t = xp.subtract(xq);\n      t = t.multiply(u).mod(q);\n      t = q.subtract(t);\n    } else {\n      t = t.multiply(u).mod(q);\n    }\n    return t.multiply(p).add(xp);\n  }\n\n  /**\n   * encrypt message\n   * @param m message as BigInteger\n   * @param e public MPI part as BigInteger\n   * @param n public MPI part as BigInteger\n   * @return BigInteger\n   */\n  function encrypt(m, e, n) {\n    return m.modPowInt(e, n);\n  }\n\n  /* Sign and Verify */\n  function sign(m, d, n) {\n    return m.modPow(d, n);\n  }\n\n  function verify(x, e, n) {\n    return x.modPowInt(e, n);\n  }\n\n  // \"empty\" RSA key constructor\n\n  function keyObject() {\n    this.n = null;\n    this.e = 0;\n    this.ee = null;\n    this.d = null;\n    this.p = null;\n    this.q = null;\n    this.dmp1 = null;\n    this.dmq1 = null;\n    this.u = null;\n  }\n\n  // Generate a new random private key B bits long, using public expt E\n\n  function generate(B, E) {\n    var key = new keyObject();\n    var rng = new SecureRandom();\n    var qs = B >> 1;\n    key.e = parseInt(E, 16);\n    key.ee = new BigInteger(E, 16);\n    for (;;) {\n      for (;;) {\n        key.p = new BigInteger(B - qs, 1, rng);\n        if (key.p.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.p.isProbablePrime(10))\n          break;\n      }\n      for (;;) {\n        key.q = new BigInteger(qs, 1, rng);\n        if (key.q.subtract(BigInteger.ONE).gcd(key.ee).compareTo(BigInteger.ONE) == 0 && key.q.isProbablePrime(10))\n          break;\n      }\n      if (key.p.compareTo(key.q) <= 0) {\n        var t = key.p;\n        key.p = key.q;\n        key.q = t;\n      }\n      var p1 = key.p.subtract(BigInteger.ONE);\n      var q1 = key.q.subtract(BigInteger.ONE);\n      var phi = p1.multiply(q1);\n      if (phi.gcd(key.ee).compareTo(BigInteger.ONE) == 0) {\n        key.n = key.p.multiply(key.q);\n        key.d = key.ee.modInverse(phi);\n        key.dmp1 = key.d.mod(p1);\n        key.dmq1 = key.d.mod(q1);\n        key.u = key.p.modInverse(key.q);\n        break;\n      }\n    }\n    return key;\n  }\n\n  this.encrypt = encrypt;\n  this.decrypt = decrypt;\n  this.verify = verify;\n  this.sign = sign;\n  this.generate = generate;\n  this.keyObject = keyObject;\n}\n\nmodule.exports = RSA;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA \n\n// The GPG4Browsers crypto interface\n\nvar type_mpi = require('../type/mpi.js');\n\nmodule.exports = {\n  /**\n   * Retrieve secure random byte string of the specified length\n   * @param {Integer} length Length in bytes to generate\n   * @return {String} Random byte string\n   */\n  getRandomBytes: function(length) {\n    var result = '';\n    for (var i = 0; i < length; i++) {\n      result += String.fromCharCode(this.getSecureRandomOctet());\n    }\n    return result;\n  },\n\n  /**\n   * Return a pseudo-random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A pseudo random number\n   */\n  getPseudoRandom: function(from, to) {\n    return Math.round(Math.random() * (to - from)) + from;\n  },\n\n  /**\n   * Return a secure random number in the specified range\n   * @param {Integer} from Min of the random number\n   * @param {Integer} to Max of the random number (max 32bit)\n   * @return {Integer} A secure random number\n   */\n  getSecureRandom: function(from, to) {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    var bits = ((to - from)).toString(2).length;\n    while ((buf[0] & (Math.pow(2, bits) - 1)) > (to - from))\n      window.crypto.getRandomValues(buf);\n    return from + (Math.abs(buf[0] & (Math.pow(2, bits) - 1)));\n  },\n\n  getSecureRandomOctet: function() {\n    var buf = new Uint32Array(1);\n    window.crypto.getRandomValues(buf);\n    return buf[0] & 0xFF;\n  },\n\n  /**\n   * Create a secure random big integer of bits length\n   * @param {Integer} bits Bit length of the MPI to create\n   * @return {BigInteger} Resulting big integer\n   */\n  getRandomBigInteger: function(bits) {\n    if (bits < 0) {\n      return null;\n    }\n    var numBytes = Math.floor((bits + 7) / 8);\n\n    var randomBits = this.getRandomBytes(numBytes);\n    if (bits % 8 > 0) {\n\n      randomBits = String.fromCharCode(\n      (Math.pow(2, bits % 8) - 1) &\n        randomBits.charCodeAt(0)) +\n        randomBits.substring(1);\n    }\n    var mpi = new type_mpi();\n    mpi.fromBytes(randomBits);\n    return mpi.toBigInteger();\n  },\n\n  getRandomBigIntegerInRange: function(min, max) {\n    if (max.compareTo(min) <= 0) {\n      return;\n    }\n\n    var range = max.subtract(min);\n    var r = this.getRandomBigInteger(range.bitLength());\n    while (r > range) {\n      r = this.getRandomBigInteger(range.bitLength());\n    }\n    return min.add(r);\n  }\n\n};\n","var publicKey = require('./public_key'),\n  pkcs1 = require('./pkcs1.js'),\n  hashModule = require('./hash');\n\nmodule.exports = {\n  /**\n   * \n   * @param {Integer} algo public Key algorithm\n   * @param {Integer} hash_algo Hash algorithm\n   * @param {openpgp_type_mpi[]} msg_MPIs Signature multiprecision integers\n   * @param {openpgp_type_mpi[]} publickey_MPIs Public key multiprecision integers \n   * @param {String} data Data on where the signature was computed on.\n   * @return {Boolean} true if signature (sig_data was equal to data over hash)\n   */\n  verify: function(algo, hash_algo, msg_MPIs, publickey_MPIs, data) {\n    var calc_hash = hashModule.digest(hash_algo, data);\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var n = publickey_MPIs[0].toBigInteger();\n        var e = publickey_MPIs[1].toBigInteger();\n        var x = msg_MPIs[0].toBigInteger();\n        var dopublic = rsa.verify(x, e, n);\n        var hash = pkcs1.emsa.decode(hash_algo, dopublic.toMPI().substring(2));\n        if (hash == -1) {\n          throw new Error('PKCS1 padding in message or key incorrect. Aborting...');\n        }\n        return hash == calc_hash;\n\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error(\"signing with Elgamal is not defined in the OpenPGP standard.\");\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n        var s1 = msg_MPIs[0].toBigInteger();\n        var s2 = msg_MPIs[1].toBigInteger();\n        var p = publickey_MPIs[0].toBigInteger();\n        var q = publickey_MPIs[1].toBigInteger();\n        var g = publickey_MPIs[2].toBigInteger();\n        var y = publickey_MPIs[3].toBigInteger();\n        var m = data;\n        var dopublic = dsa.verify(hash_algo, s1, s2, m, p, q, g, y);\n        return dopublic.compareTo(s1) == 0;\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n\n  },\n\n  /**\n   * Create a signature on data using the specified algorithm\n   * @param {Integer} hash_algo hash Algorithm to use (See RFC4880 9.4)\n   * @param {Integer} algo Asymmetric cipher algorithm to use (See RFC4880 9.1)\n   * @param {openpgp_type_mpi[]} publicMPIs Public key multiprecision integers \n   * of the private key \n   * @param {openpgp_type_mpi[]} secretMPIs Private key multiprecision \n   * integers which is used to sign the data\n   * @param {String} data Data to be signed\n   * @return {openpgp_type_mpi[]}\n   */\n  sign: function(hash_algo, algo, keyIntegers, data) {\n\n    switch (algo) {\n      case 1:\n        // RSA (Encrypt or Sign) [HAC]  \n      case 2:\n        // RSA Encrypt-Only [HAC]\n      case 3:\n        // RSA Sign-Only [HAC]\n        var rsa = new publicKey.rsa();\n        var d = keyIntegers[2].toBigInteger();\n        var n = keyIntegers[0].toBigInteger();\n        var m = pkcs1.emsa.encode(hash_algo,\n          data, keyIntegers[0].byteLength());\n\n        return rsa.sign(m, d, n).toMPI();\n\n      case 17:\n        // DSA (Digital Signature Algorithm) [FIPS186] [HAC]\n        var dsa = new publicKey.dsa();\n\n        var p = keyIntegers[0].toBigInteger();\n        var q = keyIntegers[1].toBigInteger();\n        var g = keyIntegers[2].toBigInteger();\n        var y = keyIntegers[3].toBigInteger();\n        var x = keyIntegers[4].toBigInteger();\n        var m = data;\n        var result = dsa.sign(hash_algo, m, g, p, q, x);\n\n        return result[0].toString() + result[1].toString();\n      case 16:\n        // Elgamal (Encrypt-Only) [ELGAMAL] [HAC]\n        throw new Error('Signing with Elgamal is not defined in the OpenPGP standard.');\n      default:\n        throw new Error('Invalid signature algorithm.');\n    }\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n//\n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n//\n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n//\n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar base64 = require('./base64.js');\nvar enums = require('../enums.js');\n\n/**\n * Finds out which Ascii Armoring type is used. This is an internal function\n * @param {String} text [String] ascii armored text\n * @returns {Integer} 0 = MESSAGE PART n of m\n *         1 = MESSAGE PART n\n *         2 = SIGNED MESSAGE\n *         3 = PGP MESSAGE\n *         4 = PUBLIC KEY BLOCK\n *         5 = PRIVATE KEY BLOCK\n *         null = unknown\n */\nfunction get_type(text) {\n  var splittedtext = text.split('-----');\n  // BEGIN PGP MESSAGE, PART X/Y\n  // Used for multi-part messages, where the armor is split amongst Y\n  // parts, and this is the Xth part out of Y.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+\\/\\d+/)) {\n    return enums.armor.multipart_section;\n  } else\n  // BEGIN PGP MESSAGE, PART X\n  // Used for multi-part messages, where this is the Xth part of an\n  // unspecified number of parts. Requires the MESSAGE-ID Armor\n  // Header to be used.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE, PART \\d+/)) {\n    return enums.armor.multipart_last;\n\n  } else\n  // BEGIN PGP SIGNATURE\n  // Used for detached signatures, OpenPGP/MIME signatures, and\n  // cleartext signatures. Note that PGP 2.x uses BEGIN PGP MESSAGE\n  // for detached signatures.\n  if (splittedtext[1].match(/BEGIN PGP SIGNED MESSAGE/)) {\n    return enums.armor.signed;\n\n  } else\n  // BEGIN PGP MESSAGE\n  // Used for signed, encrypted, or compressed files.\n  if (splittedtext[1].match(/BEGIN PGP MESSAGE/)) {\n    return enums.armor.message;\n\n  } else\n  // BEGIN PGP PUBLIC KEY BLOCK\n  // Used for armoring public keys.\n  if (splittedtext[1].match(/BEGIN PGP PUBLIC KEY BLOCK/)) {\n    return enums.armor.public_key;\n\n  } else\n  // BEGIN PGP PRIVATE KEY BLOCK\n  // Used for armoring private keys.\n  if (splittedtext[1].match(/BEGIN PGP PRIVATE KEY BLOCK/)) {\n    return enums.armor.private_key;\n  }\n}\n\n/**\n * Add additional information to the armor version of an OpenPGP binary\n * packet block.\n * @author  Alex\n * @version 2011-12-16\n * @returns {String} The header information\n */\nfunction armor_addheader(options) {\n  var result = \"\";\n  if (options.show_version) {\n    result += \"Version: \" + options.versionstring + '\\r\\n';\n  }\n  if (options.show_comment) {\n    result += \"Comment: \" + options.commentstring + '\\r\\n';\n  }\n  result += '\\r\\n';\n  return result;\n}\n\n\n\n/**\n * Calculates a checksum over the given data and returns it base64 encoded\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {String} Base64 encoded checksum\n */\nfunction getCheckSum(data) {\n  var c = createcrc24(data);\n  var str = \"\" + String.fromCharCode(c >> 16) +\n    String.fromCharCode((c >> 8) & 0xFF) +\n    String.fromCharCode(c & 0xFF);\n  return base64.encode(str);\n}\n\n/**\n * Calculates the checksum over the given data and compares it with the \n * given base64 encoded checksum\n * @param {String} data Data to create a CRC-24 checksum for\n * @param {String} checksum Base64 encoded checksum\n * @return {Boolean} True if the given checksum is correct; otherwise false\n */\nfunction verifyCheckSum(data, checksum) {\n  var c = getCheckSum(data);\n  var d = checksum;\n  return c[0] == d[0] && c[1] == d[1] && c[2] == d[2];\n}\n/**\n * Internal function to calculate a CRC-24 checksum over a given string (data)\n * @param {String} data Data to create a CRC-24 checksum for\n * @return {Integer} The CRC-24 checksum as number\n */\nvar crc_table = [\n    0x00000000, 0x00864cfb, 0x018ad50d, 0x010c99f6, 0x0393e6e1, 0x0315aa1a, 0x021933ec, 0x029f7f17, 0x07a18139,\n    0x0727cdc2, 0x062b5434, 0x06ad18cf, 0x043267d8, 0x04b42b23, 0x05b8b2d5, 0x053efe2e, 0x0fc54e89, 0x0f430272,\n    0x0e4f9b84, 0x0ec9d77f, 0x0c56a868, 0x0cd0e493, 0x0ddc7d65, 0x0d5a319e, 0x0864cfb0, 0x08e2834b, 0x09ee1abd,\n    0x09685646, 0x0bf72951, 0x0b7165aa, 0x0a7dfc5c, 0x0afbb0a7, 0x1f0cd1e9, 0x1f8a9d12, 0x1e8604e4, 0x1e00481f,\n    0x1c9f3708, 0x1c197bf3, 0x1d15e205, 0x1d93aefe, 0x18ad50d0, 0x182b1c2b, 0x192785dd, 0x19a1c926, 0x1b3eb631,\n    0x1bb8faca, 0x1ab4633c, 0x1a322fc7, 0x10c99f60, 0x104fd39b, 0x11434a6d, 0x11c50696, 0x135a7981, 0x13dc357a,\n    0x12d0ac8c, 0x1256e077, 0x17681e59, 0x17ee52a2, 0x16e2cb54, 0x166487af, 0x14fbf8b8, 0x147db443, 0x15712db5,\n    0x15f7614e, 0x3e19a3d2, 0x3e9fef29, 0x3f9376df, 0x3f153a24, 0x3d8a4533, 0x3d0c09c8, 0x3c00903e, 0x3c86dcc5,\n    0x39b822eb, 0x393e6e10, 0x3832f7e6, 0x38b4bb1d, 0x3a2bc40a, 0x3aad88f1, 0x3ba11107, 0x3b275dfc, 0x31dced5b,\n    0x315aa1a0,\n    0x30563856, 0x30d074ad, 0x324f0bba, 0x32c94741, 0x33c5deb7, 0x3343924c, 0x367d6c62, 0x36fb2099, 0x37f7b96f,\n    0x3771f594, 0x35ee8a83, 0x3568c678, 0x34645f8e, 0x34e21375, 0x2115723b, 0x21933ec0, 0x209fa736, 0x2019ebcd,\n    0x228694da, 0x2200d821, 0x230c41d7, 0x238a0d2c, 0x26b4f302, 0x2632bff9, 0x273e260f, 0x27b86af4, 0x252715e3,\n    0x25a15918, 0x24adc0ee, 0x242b8c15, 0x2ed03cb2, 0x2e567049, 0x2f5ae9bf, 0x2fdca544, 0x2d43da53, 0x2dc596a8,\n    0x2cc90f5e, 0x2c4f43a5, 0x2971bd8b, 0x29f7f170, 0x28fb6886, 0x287d247d, 0x2ae25b6a, 0x2a641791, 0x2b688e67,\n    0x2beec29c, 0x7c3347a4, 0x7cb50b5f, 0x7db992a9, 0x7d3fde52, 0x7fa0a145, 0x7f26edbe, 0x7e2a7448, 0x7eac38b3,\n    0x7b92c69d, 0x7b148a66, 0x7a181390, 0x7a9e5f6b, 0x7801207c, 0x78876c87, 0x798bf571, 0x790db98a, 0x73f6092d,\n    0x737045d6, 0x727cdc20, 0x72fa90db, 0x7065efcc, 0x70e3a337, 0x71ef3ac1, 0x7169763a, 0x74578814, 0x74d1c4ef,\n    0x75dd5d19, 0x755b11e2, 0x77c46ef5, 0x7742220e, 0x764ebbf8, 0x76c8f703, 0x633f964d, 0x63b9dab6, 0x62b54340,\n    0x62330fbb,\n    0x60ac70ac, 0x602a3c57, 0x6126a5a1, 0x61a0e95a, 0x649e1774, 0x64185b8f, 0x6514c279, 0x65928e82, 0x670df195,\n    0x678bbd6e, 0x66872498, 0x66016863, 0x6cfad8c4, 0x6c7c943f, 0x6d700dc9, 0x6df64132, 0x6f693e25, 0x6fef72de,\n    0x6ee3eb28, 0x6e65a7d3, 0x6b5b59fd, 0x6bdd1506, 0x6ad18cf0, 0x6a57c00b, 0x68c8bf1c, 0x684ef3e7, 0x69426a11,\n    0x69c426ea, 0x422ae476, 0x42aca88d, 0x43a0317b, 0x43267d80, 0x41b90297, 0x413f4e6c, 0x4033d79a, 0x40b59b61,\n    0x458b654f, 0x450d29b4, 0x4401b042, 0x4487fcb9, 0x461883ae, 0x469ecf55, 0x479256a3, 0x47141a58, 0x4defaaff,\n    0x4d69e604, 0x4c657ff2, 0x4ce33309, 0x4e7c4c1e, 0x4efa00e5, 0x4ff69913, 0x4f70d5e8, 0x4a4e2bc6, 0x4ac8673d,\n    0x4bc4fecb, 0x4b42b230, 0x49ddcd27, 0x495b81dc, 0x4857182a, 0x48d154d1, 0x5d26359f, 0x5da07964, 0x5cace092,\n    0x5c2aac69, 0x5eb5d37e, 0x5e339f85, 0x5f3f0673, 0x5fb94a88, 0x5a87b4a6, 0x5a01f85d, 0x5b0d61ab, 0x5b8b2d50,\n    0x59145247, 0x59921ebc, 0x589e874a, 0x5818cbb1, 0x52e37b16, 0x526537ed, 0x5369ae1b, 0x53efe2e0, 0x51709df7,\n    0x51f6d10c,\n    0x50fa48fa, 0x507c0401, 0x5542fa2f, 0x55c4b6d4, 0x54c82f22, 0x544e63d9, 0x56d11cce, 0x56575035, 0x575bc9c3,\n    0x57dd8538\n];\n\nfunction createcrc24(input) {\n  var crc = 0xB704CE;\n  var index = 0;\n\n  while ((input.length - index) > 16) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 1)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 2)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 3)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 4)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 5)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 6)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 7)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 8)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 9)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 10)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 11)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 12)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 13)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 14)) & 0xff];\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index + 15)) & 0xff];\n    index += 16;\n  }\n\n  for (var j = index; j < input.length; j++) {\n    crc = (crc << 8) ^ crc_table[((crc >> 16) ^ input.charCodeAt(index++)) & 0xff];\n  }\n  return crc & 0xffffff;\n}\n\n/**\n * DeArmor an OpenPGP armored message; verify the checksum and return \n * the encoded bytes\n * @param {String} text OpenPGP armored message\n * @returns {(Boolean|Object)} Either false in case of an error \n * or an object with attribute \"text\" containing the message text\n * and an attribute \"openpgp\" containing the bytes.\n */\nfunction dearmor(text) {\n  text = text.replace(/\\r/g, '');\n\n  var type = get_type(text);\n\n  if (type != 2) {\n    var splittedtext = text.split('-----');\n\n    var data = {\n      openpgp: base64.decode(\n        splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[0]\n        .replace(/\\n- /g, \"\\n\")),\n      type: type\n    };\n\n    if (verifyCheckSum(data.openpgp,\n      splittedtext[2]\n      .split('\\n\\n')[1]\n      .split(\"\\n=\")[1]\n      .split('\\n')[0]))\n\n      return data;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed: '\" + splittedtext[2]\n        .split('\\n\\n')[1]\n        .split(\"\\n=\")[1]\n        .split('\\n')[0] + \"' should be '\" + getCheckSum(data)) + \"'\";\n      return false;\n    }\n  } else {\n    var splittedtext = text.split('-----');\n\n    var result = {\n      text: splittedtext[2]\n        .replace(/\\n- /g, \"\\n\")\n        .split(\"\\n\\n\")[1],\n      openpgp: base64_decode(splittedtext[4]\n        .split(\"\\n\\n\")[1]\n        .split(\"\\n=\")[0]),\n      type: type\n    };\n\n    if (verifyCheckSum(result.openpgp, splittedtext[4]\n      .split(\"\\n\\n\")[1]\n      .split(\"\\n=\")[1]))\n\n      return result;\n    else {\n      util.print_error(\"Ascii armor integrity check on message failed\");\n      return false;\n    }\n  }\n}\n\n\n/**\n * Armor an OpenPGP binary packet block\n * @param {Integer} messagetype type of the message\n * @param data\n * @param {Integer} partindex\n * @param {Integer} parttotal\n * @returns {String} Armored text\n */\nfunction armor(messagetype, data, options, partindex, parttotal) {\n  var result = \"\";\n  switch (messagetype) {\n    case enums.armor.multipart_section:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"/\" + parttotal + \"-----\\r\\n\";\n      break;\n    case enums.armor.mutlipart_last:\n      result += \"-----BEGIN PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE, PART \" + partindex + \"-----\\r\\n\";\n      break;\n    case enums.armor.signed:\n      result += \"\\r\\n-----BEGIN PGP SIGNED MESSAGE-----\\r\\nHash: \" + data.hash + \"\\r\\n\\r\\n\";\n      result += data.text.replace(/\\n-/g, \"\\n- -\");\n      result += \"\\r\\n-----BEGIN PGP SIGNATURE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data.openpgp);\n      result += \"\\r\\n=\" + getCheckSum(data.openpgp) + \"\\r\\n\";\n      result += \"-----END PGP SIGNATURE-----\\r\\n\";\n      break;\n    case enums.armor.message:\n      result += \"-----BEGIN PGP MESSAGE-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP MESSAGE-----\\r\\n\";\n      break;\n    case enums.armor.public_key:\n      result += \"-----BEGIN PGP PUBLIC KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PUBLIC KEY BLOCK-----\\r\\n\\r\\n\";\n      break;\n    case enums.armor.private_key:\n      result += \"-----BEGIN PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      result += armor_addheader(options);\n      result += base64.encode(data);\n      result += \"\\r\\n=\" + getCheckSum(data) + \"\\r\\n\";\n      result += \"-----END PGP PRIVATE KEY BLOCK-----\\r\\n\";\n      break;\n  }\n\n  return result;\n}\n\nmodule.exports = {\n  encode: armor,\n  decode: dearmor\n};\n","/* OpenPGP radix-64/base64 string encoding/decoding\n * Copyright 2005 Herbert Hanewinkel, www.haneWIN.de\n * version 1.0, check www.haneWIN.de for the latest version\n *\n * This software is provided as-is, without express or implied warranty.  \n * Permission to use, copy, modify, distribute or sell this software, with or\n * without fee, for any purpose and by any individual or organization, is hereby\n * granted, provided that the above copyright notice and this paragraph appear \n * in all copies. Distribution as a part of an application or binary must\n * include the above copyright notice in the documentation and/or other materials\n * provided with the application or distribution.\n */\n\nvar b64s = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';\n\nfunction s2r(t) {\n  var a, c, n;\n  var r = '',\n    l = 0,\n    s = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = t.charCodeAt(n);\n    if (s == 0) {\n      r += b64s.charAt((c >> 2) & 63);\n      a = (c & 3) << 4;\n    } else if (s == 1) {\n      r += b64s.charAt((a | (c >> 4) & 15));\n      a = (c & 15) << 2;\n    } else if (s == 2) {\n      r += b64s.charAt(a | ((c >> 6) & 3));\n      l += 1;\n      if ((l % 60) == 0)\n        r += \"\\n\";\n      r += b64s.charAt(c & 63);\n    }\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n\n    s += 1;\n    if (s == 3)\n      s = 0;\n  }\n  if (s > 0) {\n    r += b64s.charAt(a);\n    l += 1;\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n    l += 1;\n  }\n  if (s == 1) {\n    if ((l % 60) == 0)\n      r += \"\\n\";\n    r += '=';\n  }\n\n  return r;\n}\n\nfunction r2s(t) {\n  var c, n;\n  var r = '',\n    s = 0,\n    a = 0;\n  var tl = t.length;\n\n  for (n = 0; n < tl; n++) {\n    c = b64s.indexOf(t.charAt(n));\n    if (c >= 0) {\n      if (s)\n        r += String.fromCharCode(a | (c >> (6 - s)) & 255);\n      s = (s + 2) & 7;\n      a = (c << s) & 255;\n    }\n  }\n  return r;\n}\n\nmodule.exports = {\n  encode: s2r,\n  decode: r2s\n}\n","var enums = {\n\n  /** A string to key specifier type\n   * @enum {Integer}\n   */\n  s2k: {\n    simple: 0,\n    salted: 1,\n    iterated: 3,\n    gnu: 101\n  },\n\n  /** RFC4880, section 9.1 \n   * @enum {String}\n   */\n  publicKey: {\n    rsa_encrypt_sign: 1,\n    rsa_encrypt: 2,\n    rsa_sign: 3,\n    elgamal: 16,\n    dsa: 17\n  },\n\n  /** RFC4880, section 9.2 \n   * @enum {String}\n   */\n  symmetric: {\n    plaintext: 0,\n    /** Not implemented! */\n    idea: 1,\n    tripledes: 2,\n    cast5: 3,\n    blowfish: 4,\n    aes128: 7,\n    aes192: 8,\n    aes256: 9,\n    twofish: 10\n  },\n\n  /** RFC4880, section 9.3\n   * @enum {String}\n   */\n  compression: {\n    uncompressed: 0,\n    /** RFC1951 */\n    zip: 1,\n    /** RFC1950 */\n    zlib: 2,\n    bzip2: 3\n  },\n\n  /** RFC4880, section 9.4\n   * @enum {String}\n   */\n  hash: {\n    md5: 1,\n    sha1: 2,\n    ripemd: 3,\n    sha256: 8,\n    sha384: 9,\n    sha512: 10,\n    sha224: 11\n  },\n\n\n  /**\n   * @enum {String}\n   * A list of packet types and numeric tags associated with them.\n   */\n  packet: {\n    public_key_encrypted_session_key: 1,\n    signature: 2,\n    sym_encrypted_session_key: 3,\n    one_pass_signature: 4,\n    secret_key: 5,\n    public_key: 6,\n    secret_subkey: 7,\n    compressed: 8,\n    symmetrically_encrypted: 9,\n    marker: 10,\n    literal: 11,\n    trust: 12,\n    userid: 13,\n    public_subkey: 14,\n    user_attribute: 17,\n    sym_encrypted_integrity_protected: 18,\n    modification_detection_code: 19\n  },\n\n\n  /**\n   * Data types in the literal packet\n   * @readonly\n   * @enum {String}\n   */\n  literal: {\n    /** Binary data */\n    binary: 'b'.charCodeAt(),\n    /** Text data */\n    text: 't'.charCodeAt(),\n    /** Utf8 data */\n    utf8: 'u'.charCodeAt()\n  },\n\n\n  /** One pass signature packet type\n   * @enum {String} */\n  signature: {\n    /** 0x00: Signature of a binary document. */\n    binary: 0,\n    /** 0x01: Signature of a canonical text document.\n     * Canonicalyzing the document by converting line endings. */\n    text: 1,\n    /** 0x02: Standalone signature.\n     * This signature is a signature of only its own subpacket contents.\n     * It is calculated identically to a signature over a zero-lengh\n     * binary document.  Note that it doesn't make sense to have a V3\n     * standalone signature. */\n    standalone: 2,\n    /** 0x10: Generic certification of a User ID and Public-Key packet.\n     * The issuer of this certification does not make any particular\n     * assertion as to how well the certifier has checked that the owner\n     * of the key is in fact the person described by the User ID. */\n    cert_generic: 16,\n    /** 0x11: Persona certification of a User ID and Public-Key packet.\n     * The issuer of this certification has not done any verification of\n     * the claim that the owner of this key is the User ID specified. */\n    cert_persona: 17,\n    /** 0x12: Casual certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done some casual\n     * verification of the claim of identity. */\n    cert_casual: 18,\n    /** 0x13: Positive certification of a User ID and Public-Key packet.\n     * The issuer of this certification has done substantial\n     * verification of the claim of identity.\n     * \n     * Most OpenPGP implementations make their \"key signatures\" as 0x10\n     * certifications.  Some implementations can issue 0x11-0x13\n     * certifications, but few differentiate between the types. */\n    cert_positive: 19,\n    /** 0x30: Certification revocation signature\n     * This signature revokes an earlier User ID certification signature\n     * (signature class 0x10 through 0x13) or direct-key signature\n     * (0x1F).  It should be issued by the same key that issued the\n     * revoked signature or an authorized revocation key.  The signature\n     * is computed over the same data as the certificate that it\n     * revokes, and should have a later creation date than that\n     * certificate. */\n    cert_revocation: 48,\n    /** 0x18: Subkey Binding Signature\n     * This signature is a statement by the top-level signing key that\n     * indicates that it owns the subkey.  This signature is calculated\n     * directly on the primary key and subkey, and not on any User ID or\n     * other packets.  A signature that binds a signing subkey MUST have\n     * an Embedded Signature subpacket in this binding signature that\n     * contains a 0x19 signature made by the signing subkey on the\n     * primary key and subkey. */\n    subkey_binding: 24,\n    /** 0x19: Primary Key Binding Signature\n\t\t* This signature is a statement by a signing subkey, indicating\n\t\t* that it is owned by the primary key and subkey.  This signature\n\t\t* is calculated the same way as a 0x18 signature: directly on the\n\t\t* primary key and subkey, and not on any User ID or other packets.\n\t\t\n\t\t* When a signature is made over a key, the hash data starts with the\n\t\t* octet 0x99, followed by a two-octet length of the key, and then body\n\t\t* of the key packet.  (Note that this is an old-style packet header for\n\t\t* a key packet with two-octet length.)  A subkey binding signature\n\t\t* (type 0x18) or primary key binding signature (type 0x19) then hashes\n\t\t* the subkey using the same format as the main key (also using 0x99 as\n\t\t* the first octet). */\n    key_binding: 25,\n    /** 0x1F: Signature directly on a key\n     * This signature is calculated directly on a key.  It binds the\n     * information in the Signature subpackets to the key, and is\n     * appropriate to be used for subpackets that provide information\n     * about the key, such as the Revocation Key subpacket.  It is also\n     * appropriate for statements that non-self certifiers want to make\n     * about the key itself, rather than the binding between a key and a\n     * name. */\n    key: 31,\n    /** 0x20: Key revocation signature\n     * The signature is calculated directly on the key being revoked.  A\n     * revoked key is not to be used.  Only revocation signatures by the\n     * key being revoked, or by an authorized revocation key, should be\n     * considered valid revocation signatures.a */\n    key_revocation: 32,\n    /** 0x28: Subkey revocation signature\n     * The signature is calculated directly on the subkey being revoked.\n     * A revoked subkey is not to be used.  Only revocation signatures\n     * by the top-level signature key that is bound to this subkey, or\n     * by an authorized revocation key, should be considered valid\n     * revocation signatures.\n     * Key revocation signatures (types 0x20 and 0x28)\n     * hash only the key being revoked. */\n    subkey_revocation: 40,\n    /** 0x40: Timestamp signature.\n     * This signature is only meaningful for the timestamp contained in\n     * it. */\n    timestamp: 64,\n    /**    0x50: Third-Party Confirmation signature.\n     * This signature is a signature over some other OpenPGP Signature\n     * packet(s).  It is analogous to a notary seal on the signed data.\n     * A third-party signature SHOULD include Signature Target\n     * subpacket(s) to give easy identification.  Note that we really do\n     * mean SHOULD.  There are plausible uses for this (such as a blind\n     * party that only sees the signature, not the key or source\n     * document) that cannot include a target subpacket. */\n    third_party: 80\n  },\n\n  signatureSubpacket: {\n    signature_creation_time: 2,\n    signature_expiration_time: 3,\n    exportable_certification: 4,\n    trust_signature: 5,\n    regular_expression: 6,\n    revocable: 7,\n    reserved: 8,\n    key_expiration_time: 9,\n    placeholder_backwards_compatibility: 10,\n    preferred_symmetric_algorithms: 11,\n    revocation_key: 12,\n    issuer: 16,\n    notification_data: 20,\n    preferred_hash_algorithms: 21,\n    preferred_compression_algorithms: 22,\n    key_server_preferences: 23,\n    preferred_key_server: 24,\n    primary_user_id: 25,\n    policy_uri: 26,\n    key_flags: 27,\n    signers_user_id: 28,\n    reason_for_revocation: 29,\n    features: 30,\n    signature_target: 31,\n    embedded_signature: 32\n  },\n\n  armor: {\n    multipart_section: 0,\n    multipart_last: 1,\n    signed: 2,\n    message: 3,\n    public_key: 4,\n    private_key: 5\n  },\n\n  // Asserts validity and converts from string/integer to integer.\n  write: function(type, e) {\n    if (typeof e == 'number') {\n      e = this.read(type, e);\n    }\n\n    if (type[e] !== undefined) {\n      return type[e];\n    } else throw new Error('Invalid enum value.');\n  },\n  // Converts from an integer to string.\n  read: function(type, e) {\n    for (var i in type)\n      if (type[i] == e) return i;\n\n    throw new Error('Invalid enum value.');\n  }\n}\n\nmodule.exports = enums;\n","\nmodule.exports = require('./openpgp.js');\nmodule.exports.key = require('./key.js');\nmodule.exports.keyring = require('./keyring.js');\nmodule.exports.message = require('./message.js');\nmodule.exports.util = require('./util');\nmodule.exports.packet = require('./packet');\nmodule.exports.mpi = require('./type/mpi.js');\nmodule.exports.s2k = require('./type/s2k.js');\nmodule.exports.keyid = require('./type/keyid.js');\nmodule.exports.armor = require('./encoding/armor.js');\nmodule.exports.enums = require('./enums.js');\nmodule.exports.config = require('./config');\nmodule.exports.crypto = require('./crypto');\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\nvar config = require('./config');\n\n/**\n * @class\n * @classdesc Class that represents an OpenPGP key. Must contain a master key. \n * @param  {packetlist} packetlist [description]\n * Can contain additional subkeys, signatures,\n * user ids, user attributes.\n */\n\n function key(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n\n  /** \n   * Returns the primary key packet (secret or public)\n   * @returns {packet_secret_key|packet_public_key|null} \n   */\n  this.getKeyPacket = function() {\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_key ||\n        this.packets[i].tag == enums.packet.secret_key) {\n        return this.packets[i];\n      }\n    }\n    return null;\n  }\n\n  /** \n   * Returns all the private and public subkey packets\n   * @returns {[public_subkey|secret_subkey]} \n   */\n  this.getSubkeyPackets = function() {\n\n    var subkeys = [];\n\n    for (var i = 0; i < this.packets.length; i++) {\n      if (this.packets[i].tag == enums.packet.public_subkey ||\n        this.packets[i].tag == enums.packet.secret_subkey) {\n        subkeys.push(this.packets[i]);\n      }\n    }\n\n    return subkeys;\n  }\n\n  /** \n   * Returns all the private and public key and subkey packets\n   * @returns {[public_subkey|secret_subkey|packet_secret_key|packet_public_key]} \n   */\n  this.getAllKeyPackets = function() {\n    return [this.getKeyPacket()].concat(this.getSubkeyPackets());\n  }\n\n  /** \n   * Returns key IDs of all key packets\n   * @returns {[keyid]} \n   */\n  this.getKeyIds = function() {\n    var keyIds = [];\n    var keys = this.getAllKeyPackets();\n    for (var i = 0; i < keys.length; i++) {\n      keyIds.push(keys[i].getKeyId());\n    }\n    return keyIds;\n  }\n\n  /** \n   * Returns key IDs of all key packets in hex\n   * @returns {[String]} \n   */\n  this.getKeyIdsHex = function() {\n    return this.getKeyIds().map(function(keyId) {\n      return keyId.toHex();\n    });\n  }\n\n  /**\n   * Returns first key packet which match to an array of key IDs\n   * @param  {[keyid]} keyIds \n   * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null}       \n   */\n  this.getKeyPacketByIds = function(keyIds) {\n    var keys = this.getAllKeyPackets();\n    for (var i = 0; i < keys.length; i++) {\n      var keyId = keys[i].getKeyId(); \n      for (var j = 0; j < keyIds.length; j++) {\n        if (keyId.equals(keyIds[j])) {\n          //TODO return only verified keys\n          return keys[i];\n        }\n      }\n    }\n    return null;\n  }\n\n  /**\n   * Returns true if this is a public key\n   * @return {Boolean}\n   */\n  this.isPublic = function() {\n    var publicKeyPackets = this.packets.filterByTag(enums.packet.public_key);\n    return publicKeyPackets.length !== 0 ? true : false;\n  }\n\n  /**\n   * Returns true if this is a private key\n   * @return {Boolean}\n   */\n  this.isPrivate = function() {\n    var privateKeyPackets = this.packets.filterByTag(enums.packet.private_key);\n    return privateKeyPackets.length !== 0 ? true : false;\n  }\n\n  /**\n   * Returns first key packet that is available for signing\n   * @return {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null}\n   */\n  this.getSigningKeyPacket = function() {\n\n    var signing = [ enums.publicKey.rsa_encrypt_sign, enums.publicKey.rsa_sign, enums.publicKey.dsa];\n\n    signing = signing.map(function(s) {\n      return enums.read(enums.publicKey, s);\n    });\n\n    var keys = this.getAllKeyPackets();\n\n    for (var i = 0; i < keys.length; i++) {\n      if (signing.indexOf(keys[i].algorithm) !== -1) {\n        return keys[i];\n      }\n    }\n\n    return null;\n  }\n\n  /**\n   * Returns preferred signature hash algorithm of this key\n   * @return {String}\n   */\n  function getPreferredSignatureHashAlgorithm() {\n    //TODO implement: https://tools.ietf.org/html/rfc4880#section-5.2.3.8\n    //separate private key preference from digest preferences\n    return config.prefer_hash_algorithm;\n  }\n\n  /**\n   * Returns the first valid encryption key packet for this key\n   * @returns {public_subkey|secret_subkey|packet_secret_key|packet_public_key|null} key packet or null if no encryption key has been found\n   */\n  this.getEncryptionKeyPacket = function() {\n    // V4: by convention subkeys are prefered for encryption service\n    // V3: keys MUST NOT have subkeys\n    var isValidEncryptionKey = function(key) {\n      //TODO evaluate key flags: http://tools.ietf.org/html/rfc4880#section-5.2.3.21\n      return key.algorithm != enums.read(enums.publicKey, enums.publicKey.dsa) && key.algorithm != enums.read(enums.publicKey,\n        enums.publicKey.rsa_sign);\n      //TODO verify key\n      //&& keys.verifyKey()\n    };\n\n    var subkeys = this.getSubkeyPackets();\n\n    for (var j = 0; j < subkeys.length; j++) {\n      if (isValidEncryptionKey(subkeys[j])) {\n        return subkeys[j];\n      }\n    }\n    // if no valid subkey for encryption, use primary key\n    var primaryKey = this.getKeyPacket();\n    if (isValidEncryptionKey(primaryKey)) {\n      return primaryKey;\n    }\n    return null;\n  }\n\n  /**\n   * Decrypts all secret key and subkey packets\n   * @param  {String} passphrase \n   * @return {undefined}\n   */\n  this.decrypt = function(passphrase) {\n    //TODO boolean return value\n    var keys = this.getAllKeyPackets();\n\n    for (var i in keys)\n      if (keys[i].tag == enums.packet.secret_subkey ||\n        keys[i].tag == enums.packet.secret_key) {\n        keys[i].decrypt(passphrase);\n      }\n  }\n\n\n  // TODO\n  this.verify = function() {\n\n  }\n  // TODO\n  this.revoke = function() {\n\n  }\n\n}\n\n/**\n * reads an OpenPGP armored text and returns a key object\n * @param {String} armoredText text to be parsed\n * @return {key} new key object\n */\nkey.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  //TODO don't accept non-key armored texts\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newKey = new key(packetlist);\n  return newKey;\n}\n\nmodule.exports = key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc The class that deals with storage of the keyring. Currently the only option is to use HTML5 local storage.\n */\nvar keyring = function() {\n  this.armoredPacketlists = [];\n  this.parsedPacketlists = [];\n\n  /**\n   * Initialization routine for the keyring. This method reads the \n   * keyring from HTML5 local storage and initializes this instance.\n   * This method is called by openpgp.init().\n   */\n  function init() {\n    var armoredPacketlists = JSON.parse(window.localStorage.getItem(\"armoredPacketlists\"));\n    if (armoredPacketlists === null || armoredPacketlists.length === 0) {\n      armoredPacketlists = [];\n    }\n    this.armoredPacketlists = armoredPacketlists;\n\n    var packetlist;\n    for (var i = 0; i < armoredPacketlists.length; i++) {\n      packetlist = new packet.list();\n      packetlist.read(armoredPacketlists[i]);\n      this.parsedPacketlists.push(packetlist);\n    }\n  }\n  this.init = init;\n\n  /**\n   * Saves the current state of the keyring to HTML5 local storage.\n   * The privateKeys array and publicKeys array gets Stringified using JSON\n   */\n  function store() {\n    window.localStorage.setItem(\"armoredPacketlists\", JSON.stringify(this.armoredPacketlists));\n  }\n  this.store = store;\n\n  function emailPacketCheck(packet, email) {\n    var emailMatch = false;\n    var packetEmail;\n    email = email.toLowerCase();\n    if (packet.tag == enums.packet.userid) {\n      packetEmail = packet.userid;\n      //we need to get just the email from the userid packet\n      packetEmail = packetEmail.split('<')[1].split('<')[0].trim.toLowerCase();\n      if (packetEmail == email) {\n        emailMatch = true;\n      }\n    }\n    return emailMatch;\n  }\n\n  function idPacketCheck(packet, id) {\n    if (packet.getKeyId && packet.getKeyId().write() == id) {\n      return true;\n    }\n    return false;\n  }\n\n  function helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist) {\n    var packet;\n    for (var l = 0; l < packetlist.length; l++) {\n      packet = packetlist[l];\n      identityMatch = identityFunction(packet, identityInput);\n      if (!packetType) {\n        packetMatch = true;\n      } else if (packet.tag == packetType) {\n        packetMatch = true;\n      }\n      if (packetMatch && identityMatch) {\n        return true;\n      }\n    }\n    return false;\n  }\n\n  function checkForIdentityAndPacketMatch(identityFunction, identityInput, packetType) {\n    var results = [];\n    var packetlist;\n    var identityMatch;\n    var packetMatch;\n    for (var p = 0; p < this.parsedPacketlists.length; p++) {\n      identityMatch = false;\n      packetMatch = false;\n      packetlist = this.parsedPacketlists[p];\n      if (helperCheckIdentityAndPacketMatch(identityFunction, identityInput, packetType, packetlist)) {\n        results.push(packetlist);\n      }\n    }\n    return results;\n  }\n  this.checkForIdentityAndPacketMatch = checkForIdentityAndPacketMatch;\n\n  /**\n   * searches all public keys in the keyring matching the address or address part of the user ids\n   * @param {String} email_address\n   * @return {openpgp_msg_publickey[]} The public keys associated with provided email address.\n   */\n  function getPublicKeyForAddress(email) {\n    return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.public_key);\n  }\n  this.getPublicKeyForAddress = getPublicKeyForAddress;\n\n  /**\n   * Searches the keyring for a private key containing the specified email address\n   * @param {String} email_address email address to search for\n   * @return {openpgp_msg_privatekey[]} private keys found\n   */\n  function getPrivateKeyForAddress(email_address) {\n    return checkForIdentityAndPacketMatch(emailPacketCheck, email, enums.packet.secret_key);\n  }\n  this.getPrivateKeyForAddress = getPrivateKeyForAddress;\n\n  /**\n   * Searches the keyring for public keys having the specified key id\n   * @param {String} keyId provided as string of hex number (lowercase)\n   * @return {openpgp_msg_privatekey[]} public keys found\n   */\n  function getPacketlistsForKeyId(keyId) {\n    return this.checkForIdentityAndPacketMatch(idPacketCheck, keyId);\n  }\n  this.getPacketlistsForKeyId = getPacketlistsForKeyId;\n\n  /**\n   * Imports a packet list (public or private key block) from an ascii armored message \n   * @param {String} armored message to read the packets/key from\n   */\n  function importPacketlist(armored) {\n    this.armoredPacketlists.push(armored);\n\n    var dearmored = armor.decode(armored.replace(/\\r/g, '')).openpgp;\n\n    packetlist = new packet.list();\n    packetlist.read(dearmored);\n    this.parsedPacketlists.push(packetlist);\n\n    return true;\n  }\n  this.importPacketlist = importPacketlist;\n\n  /**\n   * TODO\n   * returns the openpgp_msg_privatekey representation of the public key at public key ring index  \n   * @param {Integer} index the index of the public key within the publicKeys array\n   * @return {openpgp_msg_privatekey} the public key object\n   */\n  function exportPublicKey(index) {\n    return this.publicKey[index];\n  }\n  this.exportPublicKey = exportPublicKey;\n\n  /**\n   * TODO\n   * Removes a public key from the public key keyring at the specified index \n   * @param {Integer} index the index of the public key within the publicKeys array\n   * @return {openpgp_msg_privatekey} The public key object which has been removed\n   */\n  function removePublicKey(index) {\n    var removed = this.publicKeys.splice(index, 1);\n    this.store();\n    return removed;\n  }\n  this.removePublicKey = removePublicKey;\n\n};\n\nmodule.exports = new keyring();\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar packet = require('./packet');\nvar enums = require('./enums.js');\nvar armor = require('./encoding/armor.js');\n\n/**\n * @class\n * @classdesc A generic message containing one or more literal packets.\n */\n\nfunction message(packetlist) {\n\n  this.packets = packetlist || new packet.list();\n\n  /**\n   * Returns the key IDs of the public keys to which the session key is encrypted\n   * @return {[keyId]} array of keyid objects\n   */\n  this.getKeyIds = function() {\n    var keyIds = [];\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    pkESKeyPacketlist.forEach(function(packet) {\n      keyIds.push(packet.publicKeyId);\n    });\n    return keyIds;\n  }\n\n  /**\n   * Returns the key IDs in hex of the public keys to which the session key is encrypted\n   * @return {[String]} keyId provided as string of hex numbers (lowercase)\n   */\n  this.getKeyIdsHex = function() {\n    return this.getKeyIds().map(function(keyId) {\n      return keyId.toHex();\n    });\n  }\n\n  /**\n   * Decrypts the message\n   * @param {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with (corresponding to the session key)\n   * @return {[String]} array with plaintext of decrypted messages\n   */\n  this.decrypt = function(privateKeyPacket) {\n    var decryptedMessages = [];\n    var pkESKeyPacketlist = this.packets.filterByType(enums.packet.public_key_encrypted_session_key);\n    for (var i = 0; i < pkESKeyPacketlist.length; i++) {\n      var pkESKeyPacket = pkESKeyPacketlist[i];\n      if (pkESKeyPacket.publicKeyId.equals(privateKeyPacket.getKeyId())) {\n        pkESKeyPacket.decrypt(privateKeyPacket);\n        var symEncryptedPacketlist = this.packets.filter(function(packet) {\n          return packet.tag == enums.packet.symmetrically_encrypted || packet.tag == enums.packet.sym_encrypted_integrity_protected;\n        });\n        for (var k = 0; k < symEncryptedPacketlist.length; k++) {\n          var symEncryptedPacket = symEncryptedPacketlist[k];\n          symEncryptedPacket.decrypt(pkESKeyPacket.sessionKeyAlgorithm, pkESKeyPacket.sessionKey);\n          for (var l = 0; l < symEncryptedPacket.packets.length; l++) {\n            var dataPacket = symEncryptedPacket.packets[l];\n            switch (dataPacket.tag) {\n              case enums.packet.literal:\n                decryptedMessages.push(dataPacket.getBytes());\n                break;\n              case enums.packet.compressed:\n                //TODO\n                break;\n              default:\n                //TODO\n            }\n          }\n        }\n        break;\n      }\n    }\n    return decryptedMessages;\n  }\n\n  /**\n   * Decrypts a message and generates user interface message out of the found.\n   * MDC will be verified as well as message signatures\n   * @param {openpgp_msg_privatekey} private_key the private the message is encrypted with (corresponding to the session key)\n   * @param {openpgp_packet_encryptedsessionkey} sessionkey the session key to be used to decrypt the message\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {String} plaintext of the message or null on error\n   */\n  function decryptAndVerifySignature(private_key, sessionkey, pubkey) {\n    if (private_key == null || sessionkey == null || sessionkey == \"\")\n      return null;\n    var decrypted = sessionkey.decrypt(this, private_key.keymaterial);\n    if (decrypted == null)\n      return null;\n    var packet;\n    var position = 0;\n    var len = decrypted.length;\n    var validSignatures = new Array();\n    util.print_debug_hexstr_dump(\"openpgp.msg.messge decrypt:\\n\", decrypted);\n\n    var messages = openpgp.read_messages_dearmored({\n      text: decrypted,\n      openpgp: decrypted\n    });\n    for (var m in messages) {\n      if (messages[m].data) {\n        this.text = messages[m].data;\n      }\n      if (messages[m].signature) {\n        validSignatures.push(messages[m].verifySignature(pubkey));\n      }\n    }\n    return {\n      text: this.text,\n      validSignatures: validSignatures\n    };\n  }\n\n  /**\n   * Verifies a message signature. This function can be called after read_message if the message was signed only.\n   * @param {openpgp_msg_publickey} pubkey Array of public keys to check signature against. If not provided, checks local keystore.\n   * @return {boolean} true if the signature was correct; otherwise false\n   */\n  function verifySignature(pubkey) {\n    var result = false;\n    if (this.signature.tagType == 2) {\n      if (!pubkey || pubkey.length == 0) {\n        var pubkey;\n        if (this.signature.version == 4) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.issuerKeyId);\n        } else if (this.signature.version == 3) {\n          pubkey = openpgp.keyring.getPublicKeysForKeyId(this.signature.keyId);\n        } else {\n          util.print_error(\"unknown signature type on message!\");\n          return false;\n        }\n      }\n      if (pubkey.length == 0)\n        util.print_warning(\"Unable to verify signature of issuer: \" + util.hexstrdump(this.signature.issuerKeyId) +\n          \". Public key not found in keyring.\");\n      else {\n        for (var i = 0; i < pubkey.length; i++) {\n          var tohash = this.text.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g, \"\\r\\n\");\n          if (this.signature.verify(tohash, pubkey[i])) {\n            util.print_info(\"Found Good Signature from \" + pubkey[i].obj.userIds[0].text + \" (0x\" + util.hexstrdump(\n              pubkey[i].obj.getKeyId()).substring(8) + \")\");\n            result = true;\n          } else {\n            util.print_error(\"Signature verification failed: Bad Signature from \" + pubkey[i].obj.userIds[0].text +\n              \" (0x\" + util.hexstrdump(pubkey[0].obj.getKeyId()).substring(8) + \")\");\n          }\n        }\n      }\n    }\n    return result;\n  }\n}\n\n/**\n * reads an OpenPGP armored message and returns a message object\n * @param {String} armoredText text to be parsed\n * @return {key} new message object\n */\nmessage.readArmored = function(armoredText) {\n  //TODO how do we want to handle bad text? Exception throwing\n  //TODO don't accept non-message armored texts\n  var input = armor.decode(armoredText).openpgp;\n  var packetlist = new packet.list();\n  packetlist.read(input);\n  var newMessage = new message(packetlist);\n  return newMessage;\n}\n\nmodule.exports = message;","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @fileoverview The openpgp base class should provide all of the functionality \n * to consume the openpgp.js library. All additional classes are documented \n * for extending and developing on top of the base library.\n */\n\nvar armor = require('./encoding/armor.js');\nvar packet = require('./packet');\nvar util = require('./util');\nvar enums = require('./enums.js');\nvar crypto = require('./crypto');\nvar config = require('./config');\n\n/**\n * GPG4Browsers Core interface. A single instance is hold\n * from the beginning. To use this library call \"openpgp.init()\"\n * @alias openpgp\n * @class\n * @classdesc Main Openpgp.js class. Use this to initiate and make all calls to this library.\n */\nfunction _openpgp() {\n  this.tostring = \"\";\n\n  /**\n   * encrypts message with keys\n   * @param  {[key]} keys    array of keys, used to encrypt the message\n   * @param  {String} message text message as native JavaScript string\n   * @return {String}         encrypted ASCII armored message \n   */\n  function encryptMessage(keys, message) {\n\n    var messagePacketlist = new packet.list();\n\n    //TODO get preferred algo from signature\n    var sessionKey = crypto.generateSessionKey(enums.read(enums.symmetric, config.encryption_cipher));\n\n    keys.forEach(function(key) {\n      var encryptionKeyPacket = key.getEncryptionKeyPacket();\n      if (encryptionKeyPacket) {\n        var pkESKeyPacket = new packet.public_key_encrypted_session_key();\n        pkESKeyPacket.publicKeyId = encryptionKeyPacket.getKeyId();\n        pkESKeyPacket.publicKeyAlgorithm = encryptionKeyPacket.algorithm;\n        pkESKeyPacket.sessionKey = sessionKey;\n        //TODO get preferred algo from signature\n        pkESKeyPacket.sessionKeyAlgorithm = enums.read(enums.symmetric, config.encryption_cipher);\n        pkESKeyPacket.encrypt(encryptionKeyPacket);\n        messagePacketlist.push(pkESKeyPacket);\n      }\n    });\n\n    var literalDataPacket = new packet.literal();\n    literalDataPacket.set(message, 'utf8');\n    var literalDataPacketlist = new packet.list();\n    literalDataPacketlist.push(literalDataPacket);\n\n    var symEncryptedPacket;\n    if (config.integrity_protect) {\n      symEncryptedPacket = new packet.sym_encrypted_integrity_protected();\n    } else {\n      symEncryptedPacket = new packet.symmetrically_encrypted();\n    }\n    symEncryptedPacket.packets = literalDataPacketlist;\n    //TODO get preferred algo from signature\n    symEncryptedPacket.encrypt(enums.read(enums.symmetric, config.encryption_cipher), sessionKey);\n    messagePacketlist.push(symEncryptedPacket);\n\n    var armored = armor.encode(enums.armor.message, messagePacketlist.write(), config);\n    return armored;\n  }\n\n  function encryptAndSignMessage(publicKeys, privateKey, message) {\n\n  }\n\n  /**\n   * decrypts message\n   * @param  {secret_subkey|packet_secret_key} privateKeyPacket the private key packet (with decrypted secret part) the message is encrypted with\n   * @param  {message} message the message object with the encrypted data\n   * @return {String}         decrypted message as as native JavaScript string\n   */\n  function decryptMessage(privateKeyPacket, message) {\n    return message.decrypt(privateKeyPacket);\n  }\n\n  function decryptAndVerifyMessage(privateKey, publicKeys, messagePacketlist) {\n\n  }\n\n  function verifyMessage(publicKeys, messagePacketlist) {\n\n  }\n\n  function signMessage(privateKey, messagePacketlist) {\n\n  }\n\n  /**\n   * TODO: update this doc\n   * generates a new key pair for openpgp. Beta stage. Currently only \n   * supports RSA keys, and no subkeys.\n   * @param {Integer} keyType to indicate what type of key to make. \n   * RSA is 1. Follows algorithms outlined in OpenPGP.\n   * @param {Integer} numBits number of bits for the key creation. (should \n   * be 1024+, generally)\n   * @param {String} userId assumes already in form of \"User Name \n   * <username@email.com>\"\n   * @param {String} passphrase The passphrase used to encrypt the resulting private key\n   * @return {Object} {privateKey: [openpgp_msg_privatekey], \n   * privateKeyArmored: [string], publicKeyArmored: [string]}\n   */\n  function generateKeyPair(keyType, numBits, userId, passphrase) {\n    var packetlist = new packet.list();\n\n    var secretKeyPacket = new packet.secret_key();\n    secretKeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretKeyPacket.generate(numBits);\n    secretKeyPacket.encrypt(passphrase);\n\n    var userIdPacket = new packet.userid();\n    userIdPacket.read(userId);\n\n    var dataToSign = {};\n    dataToSign.userid = userIdPacket;\n    dataToSign.key = secretKeyPacket;\n    var signaturePacket = new packet.signature();\n    signaturePacket.issuerKeyId = secretKeyPacket.getKeyId().write();\n    signaturePacket.signatureType = enums.signature.cert_generic;\n    signaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    signaturePacket.hashAlgorithm = enums.hash.sha256;\n    signaturePacket.sign(secretKeyPacket, dataToSign);\n\n    var secretSubkeyPacket = new packet.secret_subkey();\n    secretSubkeyPacket.algorithm = enums.read(enums.publicKey, keyType);\n    secretSubkeyPacket.generate(numBits);\n    secretSubkeyPacket.encrypt(passphrase);\n\n    dataToSign = {};\n    dataToSign.key = secretKeyPacket;\n    dataToSign.bind = secretSubkeyPacket;\n    var subkeySignaturePacket = new packet.signature();\n    subkeySignaturePacket.issuerKeyId = secretSubkeyPacket.getKeyId().write();\n    subkeySignaturePacket.signatureType = enums.signature.subkey_binding;\n    subkeySignaturePacket.publicKeyAlgorithm = keyType;\n    //TODO we should load preferred hash from config, or as input to this function\n    subkeySignaturePacket.hashAlgorithm = enums.hash.sha256;\n    subkeySignaturePacket.sign(secretSubkeyPacket, dataToSign);\n\n    packetlist.push(secretKeyPacket);\n    packetlist.push(userIdPacket);\n    packetlist.push(signaturePacket);\n    packetlist.push(secretSubkeyPacket);\n    packetlist.push(subkeySignaturePacket);\n\n    var armored = armor.encode(enums.armor.private_key, packetlist.write(), this.config);\n    return armored;\n  }\n\n  /**\n   * creates a binary string representation of an encrypted and signed message.\n   * The message will be encrypted with the public keys specified and signed\n   * with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]} Private key \n   * to be used to sign the message\n   * @param {Object[]} publickeys An arraf of {obj: [openpgp_msg_publickey]}\n   * - public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt and sign\n   * @return {String} a binary string representation of the message which \n   * can be OpenPGP armored\n   */\n  function write_signed_and_encrypted_message(privatekey, publickeys, messagetext) {\n    var result = \"\";\n    var i;\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    for (i = 0; i < publickeys.length; i++) {\n      var onepasssignature = new openpgp_packet_onepasssignature();\n      var onepasssigstr = \"\";\n      if (i === 0)\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      else\n        onepasssigstr = onepasssignature.write_packet(1, openpgp.config.config.prefer_hash_algorithm, privatekey, false);\n      util.print_debug_hexstr_dump(\"onepasssigstr: |\" + onepasssigstr + \"|\\n\", onepasssigstr);\n      var datasignature = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(\n        /\\n/g, \"\\r\\n\"), privatekey);\n      util.print_debug_hexstr_dump(\"datasignature: |\" + datasignature.openpgp + \"|\\n\", datasignature.openpgp);\n      if (i === 0) {\n        result = onepasssigstr + literal + datasignature.openpgp;\n      } else {\n        result = onepasssigstr + result + datasignature.openpgp;\n      }\n    }\n\n    util.print_debug_hexstr_dump(\"signed packet: |\" + result + \"|\\n\", result);\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n  /**\n   * creates a binary string representation of an encrypted message.\n   * The message will be encrypted with the public keys specified \n   * @param {Object[]} publickeys An array of {obj: [openpgp_msg_publickey]}\n   * -public keys to be used to encrypt the message \n   * @param {String} messagetext message text to encrypt\n   * @return {String} a binary string representation of the message\n   * which can be OpenPGP armored\n   */\n  function write_encrypted_message(publickeys, messagetext) {\n    var result = \"\";\n    var literal = new openpgp_packet_literaldata().write_packet(messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/g,\n      \"\\r\\n\"));\n    util.print_debug_hexstr_dump(\"literal_packet: |\" + literal + \"|\\n\", literal);\n    result = literal;\n\n    // signatures done.. now encryption\n    var sessionkey = openpgp_crypto_generateSessionKey(openpgp.config.config.encryption_cipher);\n    var result2 = \"\";\n\n    // creating session keys for each recipient\n    for (var i = 0; i < publickeys.length; i++) {\n      var pkey = publickeys[i].getEncryptionKey();\n      if (pkey === null) {\n        util.print_error(\"no encryption key found! Key is for signing only.\");\n        return null;\n      }\n      result2 += new openpgp_packet_encryptedsessionkey().\n      write_pub_key_packet(\n        pkey.getKeyId(),\n        pkey.MPIs,\n        pkey.publicKeyAlgorithm,\n        openpgp.config.config.encryption_cipher,\n        sessionkey);\n    }\n    if (openpgp.config.config.integrity_protect) {\n      result2 += new openpgp_packet_encryptedintegrityprotecteddata().write_packet(openpgp.config.config.encryption_cipher,\n        sessionkey, result);\n    } else {\n      result2 += new openpgp_packet_encrypteddata().write_packet(openpgp.config.config.encryption_cipher, sessionkey,\n        result);\n    }\n    return armor.encode(3, result2, null, null);\n  }\n\n  /**\n   * creates a binary string representation a signed message.\n   * The message will be signed with the specified private key.\n   * @param {Object} privatekey {obj: [openpgp_msg_privatekey]}\n   * - the private key to be used to sign the message \n   * @param {String} messagetext message text to sign\n   * @return {Object} {Object: text [String]}, openpgp: {String} a binary\n   *  string representation of the message which can be OpenPGP\n   *   armored(openpgp) and a text representation of the message (text). \n   * This can be directly used to OpenPGP armor the message\n   */\n  function write_signed_message(privatekey, messagetext) {\n    var sig = new openpgp_packet_signature().write_message_signature(1, messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/,\n      \"\\r\\n\"), privatekey);\n    var result = {\n      text: messagetext.replace(/\\r\\n/g, \"\\n\").replace(/\\n/, \"\\r\\n\"),\n      openpgp: sig.openpgp,\n      hash: sig.hash\n    };\n    return armor.encode(2, result, null, null);\n  }\n\n  this.generateKeyPair = generateKeyPair;\n  this.write_signed_message = write_signed_message;\n  this.write_signed_and_encrypted_message = write_signed_and_encrypted_message;\n  this.encryptMessage = encryptMessage;\n  this.decryptMessage = decryptMessage;\n\n}\n\nmodule.exports = new _openpgp();\n","var enums = require('../enums.js');\n\n// This is pretty ugly, but browserify needs to have the requires explicitly written.\nmodule.exports = {\n  compressed: require('./compressed.js'),\n  sym_encrypted_integrity_protected: require('./sym_encrypted_integrity_protected.js'),\n  public_key_encrypted_session_key: require('./public_key_encrypted_session_key.js'),\n  sym_encrypted_session_key: require('./sym_encrypted_session_key.js'),\n  literal: require('./literal.js'),\n  public_key: require('./public_key.js'),\n  symmetrically_encrypted: require('./symmetrically_encrypted.js'),\n  marker: require('./marker.js'),\n  public_subkey: require('./public_subkey.js'),\n  user_attribute: require('./user_attribute.js'),\n  one_pass_signature: require('./one_pass_signature.js'),\n  secret_key: require('./secret_key.js'),\n  userid: require('./userid.js'),\n  secret_subkey: require('./secret_subkey.js'),\n  signature: require('./signature.js'),\n  trust: require('./trust.js')\n}\n\nfor (var i in enums.packet) {\n  var packetClass = module.exports[i];\n\n  if (packetClass != undefined)\n    packetClass.prototype.tag = enums.packet[i];\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  JXG = require('../compression/jxg.js'),\n  base64 = require('../encoding/base64.js');\n\n/**\n * @class\n * @classdesc Implementation of the Compressed Data Packet (Tag 8)\n * \n * RFC4880 5.6:\n * The Compressed Data packet contains compressed data.  Typically, this\n * packet is found as the contents of an encrypted packet, or following\n * a Signature or One-Pass Signature packet, and contains a literal data\n * packet.\n */\nmodule.exports = function packet_compressed() {\n  /** @type {packetlist} */\n  this.packets;\n  /** @type {compression} */\n  this.algorithm = 'uncompressed';\n\n  this.compressed = null;\n\n\n  /**\n   * Parsing function for the packet.\n   * @param {String} input Payload of a tag 8 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @parAM {iNTEGER} LEN lENGTH OF the packet or the remaining length of \n   * input at position\n   * @return {openpgp_packet_compressed} Object representation\n   */\n  this.read = function(bytes) {\n    // One octet that gives the algorithm used to compress the packet.\n    this.algorithm = enums.read(enums.compression, bytes.charCodeAt(0));\n\n    // Compressed data, which makes up the remainder of the packet.\n    this.compressed = bytes.substr(1);\n\n    this.decompress();\n  }\n\n\n\n  this.write = function() {\n    if (this.compressed == null)\n      this.compress();\n\n    return String.fromCharCode(enums.write(enums.compression, this.algorithm)) + this.compressed;\n  }\n\n\n  /**\n   * Decompression method for decompressing the compressed data\n   * read by read_packet\n   * @return {String} The decompressed data\n   */\n  this.decompress = function() {\n    var decompressed;\n\n    switch (this.algorithm) {\n      case 'uncompressed':\n        decompressed = this.compressed;\n        break;\n\n      case 'zip':\n        var compData = this.compressed;\n\n        var radix = base64.encode(compData).replace(/\\n/g, \"\");\n        // no header in this case, directly call deflate\n        var jxg_obj = new JXG.Util.Unzip(JXG.Util.Base64.decodeAsArray(radix));\n\n        decompressed = unescape(jxg_obj.deflate()[0][0]);\n        break;\n\n      case 'zlib':\n        //RFC 1950. Bits 0-3 Compression Method\n        var compressionMethod = this.compressed.charCodeAt(0) % 0x10;\n\n        //Bits 4-7 RFC 1950 are LZ77 Window. Generally this value is 7 == 32k window size.\n        // 2nd Byte in RFC 1950 is for \"FLAGs\" Allows for a Dictionary \n        // (how is this defined). Basic checksum, and compression level.\n\n        if (compressionMethod == 8) { //CM 8 is for DEFLATE, RFC 1951\n          // remove 4 bytes ADLER32 checksum from the end\n          var compData = this.compressed.substring(0, this.compressed.length - 4);\n          var radix = base64.encode(compData).replace(/\\n/g, \"\");\n          //TODO check ADLER32 checksum\n          decompressed = JXG.decompress(radix);\n          break;\n\n        } else {\n          util.print_error(\"Compression algorithm ZLIB only supports \" +\n            \"DEFLATE compression method.\");\n        }\n        break;\n\n      case 'bzip2':\n        // TODO: need to implement this\n        throw new Error('Compression algorithm BZip2 [BZ2] is not implemented.');\n        break;\n\n      default:\n        throw new Error(\"Compression algorithm unknown :\" + this.alogrithm);\n        break;\n    }\n\n    this.packets.read(decompressed);\n  }\n\n  /**\n   * Compress the packet data (member decompressedData)\n   * @param {Integer} type Algorithm to be used // See RFC 4880 9.3\n   * @param {String} data Data to be compressed\n   * @return {String} The compressed data stored in attribute compressedData\n   */\n  this.compress = function() {\n    switch (this.algorithm) {\n\n      case 'uncompressed':\n        // - Uncompressed\n        this.compressed = this.packets.write();\n        break;\n\n      case 'zip':\n        // - ZIP [RFC1951]\n        util.print_error(\"Compression algorithm ZIP [RFC1951] is not implemented.\");\n        break;\n\n      case 'zlib':\n        // - ZLIB [RFC1950]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm ZLIB [RFC1950] is not implemented.\");\n        break;\n\n      case 'bzip2':\n        //  - BZip2 [BZ2]\n        // TODO: need to implement this\n        util.print_error(\"Compression algorithm BZip2 [BZ2] is not implemented.\");\n        break;\n\n      default:\n        util.print_error(\"Compression algorithm unknown :\" + this.type);\n        break;\n    }\n  }\n};\n","var enums = require('../enums.js');\n\nmodule.exports = {\n  list: require('./packetlist.js'),\n};\n\nvar packets = require('./all_packets.js');\n\nfor (var i in packets)\n  module.exports[i] = packets[i];\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc Implementation of the Literal Data Packet (Tag 11)\n * \n * RFC4880 5.9: A Literal Data packet contains the body of a message; data that\n * is not to be further interpreted.\n */\nmodule.exports = function packet_literal() {\n  this.format = 'utf8';\n  this.data = '';\n  this.date = new Date();\n\n\n  /**\n   * Set the packet data to a javascript native string or a squence of \n   * bytes. Conversion to a proper utf8 encoding takes place when the \n   * packet is written.\n   * @param {String} str Any native javascript string\n   * @param {openpgp_packet_literaldata.format} format \n   */\n  this.set = function(str, format) {\n    this.format = format;\n    this.data = str;\n  }\n\n  /**\n   * Set the packet data to value represented by the provided string\n   * of bytes together with the appropriate conversion format.\n   * @param {String} bytes The string of bytes\n   * @param {openpgp_packet_literaldata.format} format\n   */\n  this.setBytes = function(bytes, format) {\n    this.format = format;\n\n    if (format == 'utf8')\n      bytes = util.decode_utf8(bytes);\n\n    this.data = bytes;\n  }\n\n  /**\n   * Get the byte sequence representing the literal packet data\n   * @returns {String} A sequence of bytes\n   */\n  this.getBytes = function() {\n    if (this.format == 'utf8')\n      return util.encode_utf8(this.data);\n    else\n      return this.data;\n  }\n\n\n\n  /**\n   * Parsing function for a literal data packet (tag 11).\n   * \n   * @param {String} input Payload of a tag 11 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    // - A one-octet field that describes how the data is formatted.\n\n    var format = enums.read(enums.literal, bytes[0].charCodeAt());\n\n    var filename_len = bytes.charCodeAt(1);\n    this.filename = util.decode_utf8(bytes.substr(2, filename_len));\n\n    this.date = util.readDate(bytes.substr(2 + filename_len, 4));\n\n    var data = bytes.substring(6 + filename_len);\n\n    this.setBytes(data, format);\n  }\n\n  /**\n   * Creates a string representation of the packet\n   * \n   * @param {String} data The data to be inserted as body\n   * @return {String} string-representation of the packet\n   */\n  this.write = function() {\n    var filename = util.encode_utf8(\"msg.txt\");\n\n    var data = this.getBytes();\n\n    var result = '';\n    result += String.fromCharCode(enums.write(enums.literal, this.format));\n    result += String.fromCharCode(filename.length);\n    result += filename;\n    result += util.writeDate(this.date);\n    result += data;\n    return result;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the strange \"Marker packet\" (Tag 10)\n * \n * RFC4880 5.8: An experimental version of PGP used this packet as the Literal\n * packet, but no released version of PGP generated Literal packets with this\n * tag. With PGP 5.x, this packet has been reassigned and is reserved for use as\n * the Marker packet.\n * \n * Such a packet MUST be ignored when received.\n */\nfunction packet_marker() {\n  /**\n   * Parsing function for a literal data packet (tag 10).\n   * \n   * @param {String} input Payload of a tag 10 packet\n   * @param {Integer} position\n   *            Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    if (bytes[0].charCodeAt() == 0x50 && // P\n    bytes[1].charCodeAt() == 0x47 && // G\n    bytes[2].charCodeAt() == 0x50) // P\n      return true;\n    // marker packet does not contain \"PGP\"\n    return false;\n  }\n}\n\nmodule.exports = packet_marker;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/**\n * @class\n * @classdesc Implementation of the One-Pass Signature Packets (Tag 4)\n * \n * RFC4880 5.4:\n * The One-Pass Signature packet precedes the signed data and contains\n * enough information to allow the receiver to begin calculating any\n * hashes needed to verify the signature.  It allows the Signature\n * packet to be placed at the end of the message, so that the signer\n * can compute the entire signed message in one pass.\n */\n\nvar enums = require('../enums.js'),\n  type_keyid = require('../type/keyid.js');\n\nmodule.exports = function packet_one_pass_signature() {\n  this.version = null; // A one-octet version number.  The current version is 3.\n  this.type = null; // A one-octet signature type.  Signature types are described in RFC4880 Section 5.2.1.\n  this.hashAlgorithm = null; // A one-octet number describing the hash algorithm used. (See RFC4880 9.4)\n  this.publicKeyAlgorithm = null; // A one-octet number describing the public-key algorithm used. (See RFC4880 9.1)\n  this.signingKeyId = null; // An eight-octet number holding the Key ID of the signing key.\n  this.flags = null; //  A one-octet number holding a flag showing whether the signature is nested.  A zero value indicates that the next packet is another One-Pass Signature packet that describes another signature to be applied to the same message data.\n\n  /**\n   * parsing function for a one-pass signature packet (tag 4).\n   * @param {String} bytes payload of a tag 4 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var mypos = 0;\n    // A one-octet version number.  The current version is 3.\n    this.version = bytes.charCodeAt(mypos++);\n\n    // A one-octet signature type.  Signature types are described in\n    //   Section 5.2.1.\n    this.type = enums.read(enums.signature, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the hash algorithm used.\n    this.hashAlgorithm = enums.read(enums.hash, bytes.charCodeAt(mypos++));\n\n    // A one-octet number describing the public-key algorithm used.\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes.charCodeAt(mypos++));\n\n    // An eight-octet number holding the Key ID of the signing key.\n    this.signingKeyId = new type_keyid();\n    this.signingKeyId.read(bytes.substr(mypos));\n    mypos += 8;\n\n    // A one-octet number holding a flag showing whether the signature\n    //   is nested.  A zero value indicates that the next packet is\n    //   another One-Pass Signature packet that describes another\n    //   signature to be applied to the same message data.\n    this.flags = bytes.charCodeAt(mypos++);\n    return this;\n  }\n\n  /**\n   * creates a string representation of a one-pass signature packet\n   * @param {Integer} type Signature types as described in RFC4880 Section 5.2.1.\n   * @param {Integer} hashalgorithm the hash algorithm used within the signature\n   * @param {openpgp_msg_privatekey} privatekey the private key used to generate the signature\n   * @param {Integer} length length of data to be signed\n   * @param {boolean} nested boolean showing whether the signature is nested. \n   *  \"true\" indicates that the next packet is another One-Pass Signature packet\n   *   that describes another signature to be applied to the same message data. \n   * @return {String} a string representation of a one-pass signature packet\n   */\n  this.write = function(type, hashalgorithm, privatekey, length, nested) {\n    var result = \"\";\n\n    result += String.fromCharCode(3);\n    result += String.fromCharCode(enums.write(enums.signature, type));\n    result += String.fromCharCode(enums.write(enums.hash, this.hashAlgorithm));\n    result += String.fromCharCode(enums.write(enums.publicKey, privatekey.algorithm));\n    result += privatekey.getKeyId().write();\n    if (nested)\n      result += String.fromCharCode(0);\n    else\n      result += String.fromCharCode(1);\n\n    return result;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util');\n\n\nmodule.exports = {\n  readSimpleLength: function(bytes) {\n    var len = 0,\n      offset,\n      type = bytes[0].charCodeAt();\n\n\n    if (type < 192) {\n      len = bytes[0].charCodeAt();\n      offset = 1;\n    } else if (type < 255) {\n      len = ((bytes[0].charCodeAt() - 192) << 8) + (bytes[1].charCodeAt()) + 192;\n      offset = 2;\n    } else if (type == 255) {\n      len = util.readNumber(bytes.substr(1, 4));\n      offset = 5;\n    }\n\n    return {\n      len: len,\n      offset: offset\n    };\n  },\n\n  /**\n   * Encodes a given integer of length to the openpgp length specifier to a\n   * string\n   * \n   * @param {Integer} length The length to encode\n   * @return {String} String with openpgp length representation\n   */\n  writeSimpleLength: function(length) {\n    var result = \"\";\n    if (length < 192) {\n      result += String.fromCharCode(length);\n    } else if (length > 191 && length < 8384) {\n      /*\n       * let a = (total data packet length) - 192 let bc = two octet\n       * representation of a let d = b + 192\n       */\n      result += String.fromCharCode(((length - 192) >> 8) + 192);\n      result += String.fromCharCode((length - 192) & 0xFF);\n    } else {\n      result += String.fromCharCode(255);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Writes a packet header version 4 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeHeader: function(tag_type, length) {\n    /* we're only generating v4 packet headers here */\n    var result = \"\";\n    result += String.fromCharCode(0xC0 | tag_type);\n    result += this.writeSimpleLength(length);\n    return result;\n  },\n\n  /**\n   * Writes a packet header Version 3 with the given tag_type and length to a\n   * string\n   * \n   * @param {Integer} tag_type Tag type\n   * @param {Integer} length Length of the payload\n   * @return {String} String of the header\n   */\n  writeOldHeader: function(tag_type, length) {\n    var result = \"\";\n    if (length < 256) {\n      result += String.fromCharCode(0x80 | (tag_type << 2));\n      result += String.fromCharCode(length);\n    } else if (length < 65536) {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 1);\n      result += util.writeNumber(length, 2);\n    } else {\n      result += String.fromCharCode(0x80 | (tag_type << 2) | 2);\n      result += util.writeNumber(length, 4);\n    }\n    return result;\n  },\n\n  /**\n   * Generic static Packet Parser function\n   * \n   * @param {String} input Input stream as string\n   * @param {integer} position Position to start parsing\n   * @param {integer} len Length of the input from position on\n   * @return {Object} Returns a parsed openpgp_packet\n   */\n  read: function(input, position, len) {\n    // some sanity checks\n    if (input == null || input.length <= position || input.substring(position).length < 2 || (input[position].charCodeAt() &\n      0x80) == 0) {\n      util\n        .print_error(\"Error during parsing. This message / key is probably not containing a valid OpenPGP format.\");\n      return null;\n    }\n    var mypos = position;\n    var tag = -1;\n    var format = -1;\n    var packet_length;\n\n    format = 0; // 0 = old format; 1 = new format\n    if ((input[mypos].charCodeAt() & 0x40) != 0) {\n      format = 1;\n    }\n\n    var packet_length_type;\n    if (format) {\n      // new format header\n      tag = input[mypos].charCodeAt() & 0x3F; // bit 5-0\n    } else {\n      // old format header\n      tag = (input[mypos].charCodeAt() & 0x3F) >> 2; // bit 5-2\n      packet_length_type = input[mypos].charCodeAt() & 0x03; // bit 1-0\n    }\n\n    // header octet parsing done\n    mypos++;\n\n    // parsed length from length field\n    var bodydata = null;\n\n    // used for partial body lengths\n    var real_packet_length = -1;\n    if (!format) {\n      // 4.2.1. Old Format Packet Lengths\n      switch (packet_length_type) {\n        case 0:\n          // The packet has a one-octet length. The header is 2 octets\n          // long.\n          packet_length = input[mypos++].charCodeAt();\n          break;\n        case 1:\n          // The packet has a two-octet length. The header is 3 octets\n          // long.\n          packet_length = (input[mypos++].charCodeAt() << 8) | input[mypos++].charCodeAt();\n          break;\n        case 2:\n          // The packet has a four-octet length. The header is 5\n          // octets long.\n          packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n            8) | input[mypos++].charCodeAt();\n          break;\n        default:\n          // 3 - The packet is of indeterminate length. The header is 1\n          // octet long, and the implementation must determine how long\n          // the packet is. If the packet is in a file, this means that\n          // the packet extends until the end of the file. In general, \n          // an implementation SHOULD NOT use indeterminate-length \n          // packets except where the end of the data will be clear \n          // from the context, and even then it is better to use a \n          // definite length, or a new format header. The new format \n          // headers described below have a mechanism for precisely\n          // encoding data of indeterminate length.\n          packet_length = len;\n          break;\n      }\n\n    } else // 4.2.2. New Format Packet Lengths\n    {\n\n      // 4.2.2.1. One-Octet Lengths\n      if (input[mypos].charCodeAt() < 192) {\n        packet_length = input[mypos++].charCodeAt();\n        util.print_debug(\"1 byte length:\" + packet_length);\n        // 4.2.2.2. Two-Octet Lengths\n      } else if (input[mypos].charCodeAt() >= 192 && input[mypos].charCodeAt() < 224) {\n        packet_length = ((input[mypos++].charCodeAt() - 192) << 8) + (input[mypos++].charCodeAt()) + 192;\n        util.print_debug(\"2 byte length:\" + packet_length);\n        // 4.2.2.4. Partial Body Lengths\n      } else if (input[mypos].charCodeAt() > 223 && input[mypos].charCodeAt() < 255) {\n        packet_length = 1 << (input[mypos++].charCodeAt() & 0x1F);\n        util.print_debug(\"4 byte length:\" + packet_length);\n        // EEEK, we're reading the full data here...\n        var mypos2 = mypos + packet_length;\n        bodydata = input.substring(mypos, mypos + packet_length);\n        while (true) {\n          if (input[mypos2].charCodeAt() < 192) {\n            var tmplen = input[mypos2++].charCodeAt();\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() >= 192 && input[mypos2].charCodeAt() < 224) {\n            var tmplen = ((input[mypos2++].charCodeAt() - 192) << 8) + (input[mypos2++].charCodeAt()) + 192;\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n            break;\n          } else if (input[mypos2].charCodeAt() > 223 && input[mypos2].charCodeAt() < 255) {\n            var tmplen = 1 << (input[mypos2++].charCodeAt() & 0x1F);\n            packet_length += tmplen;\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            mypos2 += tmplen;\n          } else {\n            mypos2++;\n            var tmplen = (input[mypos2++].charCodeAt() << 24) | (input[mypos2++].charCodeAt() << 16) | (input[mypos2++]\n              .charCodeAt() << 8) | input[mypos2++].charCodeAt();\n            bodydata += input.substring(mypos2, mypos2 + tmplen);\n            packet_length += tmplen;\n            mypos2 += tmplen;\n            break;\n          }\n        }\n        real_packet_length = mypos2;\n        // 4.2.2.3. Five-Octet Lengths\n      } else {\n        mypos++;\n        packet_length = (input[mypos++].charCodeAt() << 24) | (input[mypos++].charCodeAt() << 16) | (input[mypos++].charCodeAt() <<\n          8) | input[mypos++].charCodeAt();\n      }\n    }\n\n    // if there was'nt a partial body length: use the specified\n    // packet_length\n    if (real_packet_length == -1) {\n      real_packet_length = packet_length;\n    }\n\n    if (bodydata == null) {\n      bodydata = input.substring(mypos, mypos + real_packet_length);\n    }\n\n    return {\n      tag: tag,\n      packet: bodydata,\n      offset: mypos + real_packet_length\n    };\n  }\n}\n","var packetParser = require('./packet.js'),\n  packets = require('./all_packets.js'),\n  enums = require('../enums.js');\n\n/**\n * @class\n * @classdesc This class represents a list of openpgp packets.\n * Take care when iterating over it - the packets themselves\n * are stored as numerical indices.\n */\nmodule.exports = function packetlist() {\n  /** The number of packets contained within the list.\n   * @readonly\n   * @type {Integer} */\n  this.length = 0;\n\n  /**\n   * Reads a stream of binary data and interprents it as a list of packets.\n   * @param {openpgp_bytearray} An array of bytes.\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    while (i < bytes.length) {\n      var parsed = packetParser.read(bytes, i, bytes.length - i);\n      i = parsed.offset;\n\n      var tag = enums.read(enums.packet, parsed.tag);\n      var packet = new packets[tag]();\n\n      this.push(packet);\n\n      packet.read(parsed.packet);\n    }\n  }\n\n  /**\n   * Creates a binary representation of openpgp objects contained within the\n   * class instance.\n   * @returns {openpgp_bytearray} An array of bytes containing valid openpgp packets.\n   */\n  this.write = function() {\n    var bytes = '';\n\n    for (var i = 0; i < this.length; i++) {\n      var packetbytes = this[i].write();\n      bytes += packetParser.writeHeader(this[i].tag, packetbytes.length);\n      bytes += packetbytes;\n    }\n\n    return bytes;\n  }\n\n  /**\n   * Adds a packet to the list. This is the only supported method of doing so;\n   * writing to packetlist[i] directly will result in an error.\n   */\n  this.push = function(packet) {\n    packet.packets = new packetlist();\n\n    this[this.length] = packet;\n    this.length++;\n  }\n\n    /**\n   * Creates a new packetList with all packets that pass the test implemented by the provided function.\n   */\n   this.filter = function(callback) {\n\n    var filtered = new packetlist();\n\n    for (var i = 0; i < this.length; i++) {\n      if (callback(this[i], i, this)) {\n        filtered.push(this[i]);\n      }\n    }\n\n    return filtered; \n   }\n\n   /**\n   * Creates a new packetList with all packets from the given type\n   */\n   this.filterByType = function(packetType) {\n     return this.filter(function(packet) {\n       return packet.tag == packetType;\n     });\n   } \n\n   /**\n    * Executes the provided callback once for each element\n    */\n   this.forEach = function(callback) {\n     for (var i = 0; i < this.length; i++) {\n      callback(this[i]);\n    }\n   }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  type_keyid = require('../type/keyid.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nmodule.exports = function packet_public_key() {\n  /** Key creation date.\n   * @type {Date} */\n  this.created = new Date();\n  /** A list of multiprecision integers\n   * @type {openpgp_type_mpi} */\n  this.mpi = [];\n  /** Public key algorithm\n   * @type {openpgp.publickey} */\n  this.algorithm = 'rsa_sign';\n\n\n  /**\n   * Internal Parser for public keys as specified in RFC 4880 section \n   * 5.5.2 Public-Key Packet Formats\n   * called by read_tag&lt;num&gt;\n   * @param {String} input Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of input\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.readPublicKey = this.read = function(bytes) {\n    // A one-octet version number (3 or 4).\n    var version = bytes[0].charCodeAt();\n\n    if (version == 4) {\n      // - A four-octet number denoting the time that the key was created.\n      this.created = util.readDate(bytes.substr(1, 4));\n\n      // - A one-octet number denoting the public-key algorithm of this key.\n      this.algorithm = enums.read(enums.publicKey, bytes[5].charCodeAt());\n\n      var mpicount = crypto.getPublicMpiCount(this.algorithm);\n      this.mpi = [];\n\n      var bmpi = bytes.substr(6);\n      var p = 0;\n\n      for (var i = 0; i < mpicount && p < bmpi.length; i++) {\n\n        this.mpi[i] = new type_mpi();\n\n        p += this.mpi[i].read(bmpi.substr(p))\n\n        if (p > bmpi.length)\n          util.print_error(\"openpgp.packet.keymaterial.js\\n\" + 'error reading MPI @:' + p);\n      }\n\n      return p + 6;\n    } else {\n      throw new Error('Version ' + version + ' of the key packet is unsupported.');\n    }\n  }\n\n  /*\n   * Same as write_private_key, but has less information because of \n   * public key.\n   * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n   * IE 1 corresponds to RSA.\n   * @param {RSA.keyObject} key\n   * @param timePacket\n   * @return {Object} {body: [string]OpenPGP packet body contents, \n   * header: [string] OpenPGP packet header, string: [string] header+body}\n   */\n  this.writePublicKey = this.write = function() {\n    // Version\n    var result = String.fromCharCode(4);\n    result += util.writeDate(this.created);\n    result += String.fromCharCode(enums.write(enums.publicKey, this.algorithm));\n\n    var mpicount = crypto.getPublicMpiCount(this.algorithm);\n\n    for (var i = 0; i < mpicount; i++) {\n      result += this.mpi[i].write();\n    }\n\n    return result;\n  }\n\n  // Write an old version packet - it's used by some of the internal routines.\n  this.writeOld = function() {\n    var bytes = this.writePublicKey();\n\n    return String.fromCharCode(0x99) +\n      util.writeNumber(bytes.length, 2) +\n      bytes;\n  }\n\n  /**\n   * Calculates the key id of the key \n   * @return {String} A 8 byte key id\n   */\n  this.getKeyId = function() {\n    var keyid = new type_keyid();\n    keyid.read(this.getFingerprint().substr(12, 8));\n    return keyid;\n  }\n\n  /**\n   * Calculates the fingerprint of the key\n   * @return {String} A string containing the fingerprint\n   */\n  this.getFingerprint = function() {\n    var toHash = this.writeOld();\n    return crypto.hash.sha1(toHash, toHash.length);\n  }\n\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_keyid = require('../type/keyid.js'),\n  util = require('../util'),\n  type_mpi = require('../type/mpi.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_public_key_encrypted_session_key() {\n  this.version = 3;\n\n  this.publicKeyId = new type_keyid();\n  this.publicKeyAlgorithm = 'rsa_encrypt';\n\n  this.sessionKey = null;\n  this.sessionKeyAlgorithm = 'aes256';\n\n  /** @type {openpgp_type_mpi[]} */\n  this.encrypted = [];\n\n  /**\n   * Parsing function for a publickey encrypted session key packet (tag 1).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n\n    this.version = bytes[0].charCodeAt();\n    this.publicKeyId.read(bytes.substr(1));\n    this.publicKeyAlgorithm = enums.read(enums.publicKey, bytes[9].charCodeAt());\n\n    var i = 10;\n\n    var integerCount = (function(algo) {\n      switch (algo) {\n        case 'rsa_encrypt':\n        case 'rsa_encrypt_sign':\n          return 1;\n\n        case 'elgamal':\n          return 2;\n\n        default:\n          throw new Error(\"Invalid algorithm.\");\n      }\n    })(this.publicKeyAlgorithm);\n\n    this.encrypted = [];\n\n    for (var j = 0; j < integerCount; j++) {\n      var mpi = new type_mpi();\n      i += mpi.read(bytes.substr(i));\n      this.encrypted.push(mpi);\n    }\n  }\n\n  /**\n   * Create a string representation of a tag 1 packet\n   * \n   * @param {String} publicKeyId\n   *             The public key id corresponding to publicMPIs key as string\n   * @param {openpgp_type_mpi[]} publicMPIs\n   *            Multiprecision integer objects describing the public key\n   * @param {Integer} pubalgo\n   *            The corresponding public key algorithm // See RFC4880 9.1\n   * @param {Integer} symmalgo\n   *            The symmetric cipher algorithm used to encrypt the data \n   *            within an encrypteddatapacket or encryptedintegrity-\n   *            protecteddatapacket \n   *            following this packet //See RFC4880 9.2\n   * @param {String} sessionkey\n   *            A string of randombytes representing the session key\n   * @return {String} The string representation\n   */\n  this.write = function() {\n\n    var result = String.fromCharCode(this.version);\n    result += this.publicKeyId.write();\n    result += String.fromCharCode(\n      enums.write(enums.publicKey, this.publicKeyAlgorithm));\n\n    for (var i = 0; i < this.encrypted.length; i++) {\n      result += this.encrypted[i].write()\n    }\n\n    return result;\n  }\n\n  this.encrypt = function(key) {\n    var data = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm));\n\n    data += this.sessionKey;\n    var checksum = util.calc_checksum(this.sessionKey);\n    data += util.writeNumber(checksum, 2);\n\n    var mpi = new type_mpi();\n    mpi.fromBytes(crypto.pkcs1.eme.encode(\n      data,\n      key.mpi[0].byteLength()));\n\n    this.encrypted = crypto.publicKeyEncrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      mpi);\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(key) {\n    var result = crypto.publicKeyDecrypt(\n      this.publicKeyAlgorithm,\n      key.mpi,\n      this.encrypted).toBytes();\n\n    var checksum = util.readNumber(result.substr(result.length - 2));\n\n    var decoded = crypto.pkcs1.eme.decode(\n      result,\n      key.mpi[0].byteLength());\n\n    var key = decoded.substring(1, decoded.length - 2);\n\n    if (checksum != util.calc_checksum(key)) {\n      throw new Error('Checksum mismatch');\n    } else {\n      this.sessionKey = key;\n      this.sessionKeyAlgorithm =\n        enums.read(enums.symmetric, decoded.charCodeAt(0));\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar public_key = require('./public_key.js');\n\nmodule.exports = function public_subkey() {\n  public_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar publicKey = require('./public_key.js'),\n  enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js'),\n  type_s2k = require('../type/s2k.js');\n\n/**\n * @class\n * @classdesc Implementation of the Key Material Packet (Tag 5,6,7,14)\n *   \n * RFC4480 5.5:\n * A key material packet contains all the information about a public or\n * private key.  There are four variants of this packet type, and two\n * major versions.  Consequently, this section is complex.\n */\nfunction packet_secret_key() {\n  publicKey.call(this);\n\n  this.encrypted = null;\n\n\n  function get_hash_len(hash) {\n    if (hash == 'sha1')\n      return 20;\n    else\n      return 2;\n  }\n\n  function get_hash_fn(hash) {\n    if (hash == 'sha1')\n      return crypto.hash.sha1;\n    else\n      return function(c) {\n        return util.writeNumber(util.calc_checksum(c), 2);\n    };\n  }\n\n  // Helper function\n\n  function parse_cleartext_mpi(hash_algorithm, cleartext, algorithm) {\n    var hashlen = get_hash_len(hash_algorithm),\n      hashfn = get_hash_fn(hash_algorithm);\n\n    var hashtext = cleartext.substr(cleartext.length - hashlen);\n    cleartext = cleartext.substr(0, cleartext.length - hashlen);\n\n    var hash = hashfn(cleartext);\n\n    if (hash != hashtext)\n      throw new Error(\"Hash mismatch.\");\n\n    var mpis = crypto.getPrivateMpiCount(algorithm);\n\n    var j = 0;\n    var mpi = [];\n\n    for (var i = 0; i < mpis && j < cleartext.length; i++) {\n      mpi[i] = new type_mpi();\n      j += mpi[i].read(cleartext.substr(j));\n    }\n\n    return mpi;\n  }\n\n  function write_cleartext_mpi(hash_algorithm, algorithm, mpi) {\n    var bytes = '';\n    var discard = crypto.getPublicMpiCount(algorithm);\n\n    for (var i = discard; i < mpi.length; i++) {\n      bytes += mpi[i].write();\n    }\n\n\n    bytes += get_hash_fn(hash_algorithm)(bytes);\n\n    return bytes;\n  }\n\n\n  // 5.5.3.  Secret-Key Packet Formats\n\n  /**\n   * Internal parser for private keys as specified in RFC 4880 section 5.5.3\n   * @param {String} bytes Input string to read the packet from\n   * @param {Integer} position Start position for the parser\n   * @param {Integer} len Length of the packet or remaining length of bytes\n   * @return {Object} This object with attributes set by the parser\n   */\n  this.read = function(bytes) {\n    // - A Public-Key or Public-Subkey packet, as described above.\n    var len = this.readPublicKey(bytes);\n\n    bytes = bytes.substr(len);\n\n\n    // - One octet indicating string-to-key usage conventions.  Zero\n    //   indicates that the secret-key data is not encrypted.  255 or 254\n    //   indicates that a string-to-key specifier is being given.  Any\n    //   other value is a symmetric-key encryption algorithm identifier.\n    var isEncrypted = bytes[0].charCodeAt();\n\n    if (isEncrypted) {\n      this.encrypted = bytes;\n    } else {\n\n      // - Plain or encrypted multiprecision integers comprising the secret\n      //   key data.  These algorithm-specific fields are as described\n      //   below.\n\n      this.mpi = this.mpi.concat(parse_cleartext_mpi('mod', bytes.substr(1),\n        this.algorithm));\n    }\n\n  };\n\n  /*\n     * Creates an OpenPGP key packet for the given key. much \n\t * TODO in regards to s2k, subkeys.\n     * @param {Integer} keyType Follows the OpenPGP algorithm standard, \n\t * IE 1 corresponds to RSA.\n     * @param {RSA.keyObject} key\n     * @param passphrase\n     * @param s2kHash\n     * @param symmetricEncryptionAlgorithm\n     * @param timePacket\n     * @return {Object} {body: [string]OpenPGP packet body contents, \n\t\theader: [string] OpenPGP packet header, string: [string] header+body}\n     */\n  this.write = function() {\n    var bytes = this.writePublicKey();\n\n    if (!this.encrypted) {\n      bytes += String.fromCharCode(0);\n\n      bytes += write_cleartext_mpi('mod', this.algorithm, this.mpi);\n    } else {\n      bytes += this.encrypted;\n    }\n\n    return bytes;\n  };\n\n\n\n\n  /** Encrypt the payload. By default, we use aes256 and iterated, salted string\n   * to key specifier\n   * @param {String} passphrase\n   */\n  this.encrypt = function(passphrase) {\n\n    var s2k = new type_s2k(),\n      symmetric = 'aes256',\n      cleartext = write_cleartext_mpi('sha1', this.algorithm, this.mpi),\n      key = produceEncryptionKey(s2k, passphrase, symmetric),\n      blockLen = crypto.cipher[symmetric].blockSize,\n      iv = crypto.random.getRandomBytes(blockLen);\n\n\n    this.encrypted = '';\n    this.encrypted += String.fromCharCode(254);\n    this.encrypted += String.fromCharCode(enums.write(enums.symmetric, symmetric));\n    this.encrypted += s2k.write();\n    this.encrypted += iv;\n\n    this.encrypted += crypto.cfb.normalEncrypt(symmetric, key, cleartext, iv);\n  };\n\n  function produceEncryptionKey(s2k, passphrase, algorithm) {\n    return s2k.produce_key(passphrase,\n      crypto.cipher[algorithm].keySize);\n  }\n\n  /**\n   * Decrypts the private key MPIs which are needed to use the key.\n   * openpgp_packet_keymaterial.hasUnencryptedSecretKeyData should be \n   * false otherwise\n   * a call to this function is not needed\n   * \n   * @param {String} str_passphrase The passphrase for this private key \n   * as string\n   * @return {Boolean} True if the passphrase was correct; false if not\n   */\n  this.decrypt = function(passphrase) {\n    if (!this.encrypted)\n      return;\n\n    var i = 0,\n      symmetric,\n      key;\n\n    var s2k_usage = this.encrypted[i++].charCodeAt();\n\n    // - [Optional] If string-to-key usage octet was 255 or 254, a one-\n    //   octet symmetric encryption algorithm.\n    if (s2k_usage == 255 || s2k_usage == 254) {\n      symmetric = this.encrypted[i++].charCodeAt();\n      symmetric = enums.read(enums.symmetric, symmetric);\n\n      // - [Optional] If string-to-key usage octet was 255 or 254, a\n      //   string-to-key specifier.  The length of the string-to-key\n      //   specifier is implied by its type, as described above.\n      var s2k = new type_s2k();\n      i += s2k.read(this.encrypted.substr(i));\n\n      key = produceEncryptionKey(s2k, passphrase, symmetric);\n    } else {\n      symmetric = s2k_usage;\n      symmetric = enums.read(enums.symmetric, symmetric);\n      key = crypto.hash.md5(passphrase);\n    }\n\n\n    // - [Optional] If secret data is encrypted (string-to-key usage octet\n    //   not zero), an Initial Vector (IV) of the same length as the\n    //   cipher's block size.\n    var iv = this.encrypted.substr(i,\n      crypto.cipher[symmetric].blockSize);\n\n    i += iv.length;\n\n    var cleartext,\n      ciphertext = this.encrypted.substr(i);\n\n    cleartext = crypto.cfb.normalDecrypt(symmetric, key, ciphertext, iv);\n\n    var hash = s2k_usage == 254 ?\n      'sha1' :\n      'mod';\n\n\n    this.mpi = this.mpi.concat(parse_cleartext_mpi(hash, cleartext,\n      this.algorithm));\n  };\n\n  this.generate = function(bits, passphrase) {\n    this.mpi = crypto.generateMpi(this.algorithm, bits);\n  };\n\n}\n\npacket_secret_key.prototype = new publicKey;\n\nmodule.exports = packet_secret_key;\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar secret_key = require('./secret_key.js');\n\nmodule.exports = function secret_subkey() {\n  secret_key.call(this);\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  packet = require('./packet.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto'),\n  type_mpi = require('../type/mpi.js');\n\n/**\n * @class\n * @classdesc Implementation of the Signature Packet (Tag 2)\n * \n * RFC4480 5.2:\n * A Signature packet describes a binding between some public key and\n * some data.  The most common signatures are a signature of a file or a\n * block of text, and a signature that is a certification of a User ID.\n */\nmodule.exports = function packet_signature() {\n\n  this.signatureType = null;\n  this.hashAlgorithm = null;\n  this.publicKeyAlgorithm = null;\n\n  this.signatureData = null;\n  this.signedHashValue = null;\n  this.mpi = null;\n\n  this.created = null;\n  this.signatureExpirationTime = null;\n  this.signatureNeverExpires = null;\n  this.exportable = null;\n  this.trustLevel = null;\n  this.trustAmount = null;\n  this.regularExpression = null;\n  this.revocable = null;\n  this.keyExpirationTime = null;\n  this.keyNeverExpires = null;\n  this.preferredSymmetricAlgorithms = null;\n  this.revocationKeyClass = null;\n  this.revocationKeyAlgorithm = null;\n  this.revocationKeyFingerprint = null;\n  this.issuerKeyId = null;\n  this.notation = {};\n  this.preferredHashAlgorithms = null;\n  this.preferredCompressionAlgorithms = null;\n  this.keyServerPreferences = null;\n  this.preferredKeyServer = null;\n  this.isPrimaryUserID = null;\n  this.policyURI = null;\n  this.keyFlags = null;\n  this.signersUserId = null;\n  this.reasonForRevocationFlag = null;\n  this.reasonForRevocationString = null;\n  this.signatureTargetPublicKeyAlgorithm = null;\n  this.signatureTargetHashAlgorithm = null;\n  this.signatureTargetHash = null;\n  this.embeddedSignature = null;\n\n  this.verified = false;\n\n  /**\n   * parsing function for a signature packet (tag 2).\n   * @param {String} bytes payload of a tag 2 packet\n   * @param {Integer} position position to start reading from the bytes string\n   * @param {Integer} len length of the packet or the remaining length of bytes at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n\n    var version = bytes[i++].charCodeAt();\n    // switch on version (3 and 4)\n    switch (version) {\n      case 3:\n        // One-octet length of following hashed material. MUST be 5.\n        if (bytes[i++].charCodeAt() != 5)\n          util.print_debug(\"openpgp.packet.signature.js\\n\" +\n            'invalid One-octet length of following hashed material.' +\n            'MUST be 5. @:' + (i - 1));\n\n        var sigpos = i;\n        // One-octet signature type.\n        this.signatureType = bytes[i++].charCodeAt();\n\n        // Four-octet creation time.\n        this.created = util.readDate(bytes.substr(i, 4));\n        i += 4;\n\n        // storing data appended to data which gets verified\n        this.signatureData = bytes.substring(position, i);\n\n        // Eight-octet Key ID of signer.\n        this.issuerKeyId = bytes.substring(i, i + 8);\n        i += 8;\n\n        // One-octet public-key algorithm.\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n\n        // One-octet hash algorithm.\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n        break;\n      case 4:\n        this.signatureType = bytes[i++].charCodeAt();\n        this.publicKeyAlgorithm = bytes[i++].charCodeAt();\n        this.hashAlgorithm = bytes[i++].charCodeAt();\n\n\n        function subpackets(bytes, signed) {\n          // Two-octet scalar octet count for following hashed subpacket\n          // data.\n          var subpacket_length = util.readNumber(\n            bytes.substr(0, 2));\n\n          var i = 2;\n\n          // Hashed subpacket data set (zero or more subpackets)\n          var subpacked_read = 0;\n          while (i < 2 + subpacket_length) {\n\n            var len = packet.readSimpleLength(bytes.substr(i));\n            i += len.offset;\n\n            // Since it is trivial to add data to the unhashed portion of \n            // the packet we simply ignore all unauthenticated data.\n            if (signed)\n              this.read_sub_packet(bytes.substr(i, len.len));\n\n            i += len.len;\n          }\n\n          return i;\n        }\n\n        i += subpackets.call(this, bytes.substr(i), true);\n\n        // A V4 signature hashes the packet body\n        // starting from its first field, the version number, through the end\n        // of the hashed subpacket data.  Thus, the fields hashed are the\n        // signature version, the signature type, the public-key algorithm, the\n        // hash algorithm, the hashed subpacket length, and the hashed\n        // subpacket body.\n        this.signatureData = bytes.substr(0, i);\n\n        i += subpackets.call(this, bytes.substr(i), false);\n\n        break;\n      default:\n        throw new Error('Version ' + version + ' of the signature is unsupported.');\n        break;\n    }\n\n    // Two-octet field holding left 16 bits of signed hash value.\n    this.signedHashValue = bytes.substr(i, 2);\n    i += 2;\n\n    this.signature = bytes.substr(i);\n  };\n\n  this.write = function() {\n    return this.signatureData +\n      util.writeNumber(0, 2) + // Number of unsigned subpackets.\n    this.signedHashValue +\n      this.signature;\n  };\n\n  /**\n   * Signs provided data. This needs to be done prior to serialization.\n   * @param {Object} data Contains packets to be signed.\n   * @param {openpgp_msg_privatekey} privatekey private key used to sign the message. \n   */\n  this.sign = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var result = String.fromCharCode(4);\n    result += String.fromCharCode(signatureType);\n    result += String.fromCharCode(publicKeyAlgorithm);\n    result += String.fromCharCode(hashAlgorithm);\n\n    //Calculate subpackets\n    var creationTimeSubpacket = write_sub_packet(enums.signatureSubpacket.signature_creation_time,\n      util.writeDate(new Date()));\n\n    var issuerSubpacket = write_sub_packet(enums.signatureSubpacket.issuer, key.getKeyId().write());\n\n    // Add subpackets here\n    result += util.writeNumber(creationTimeSubpacket.length + issuerSubpacket.length, 2);\n    result += creationTimeSubpacket + issuerSubpacket;\n\n    this.signatureData = result;\n\n    var trailer = this.calculateTrailer();\n\n    var toHash = this.toSign(signatureType, data) +\n      this.signatureData + trailer;\n\n    var hash = crypto.hash.digest(hashAlgorithm, toHash);\n\n    this.signedHashValue = hash.substr(0, 2);\n\n    this.signature = crypto.signature.sign(hashAlgorithm,\n      publicKeyAlgorithm, key.mpi, toHash);\n  };\n\n  /**\n   * creates a string representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   * @param {Integer} type subpacket signature type. Signature types as described \n   * in RFC4880 Section 5.2.3.2\n   * @param {String} data data to be included\n   * @return {String} a string-representation of a sub signature packet (See RFC 4880 5.2.3.1)\n   */\n  function write_sub_packet(type, data) {\n    var result = \"\";\n    result += packet.writeSimpleLength(data.length + 1);\n    result += String.fromCharCode(type);\n    result += data;\n    return result;\n  }\n\n  // V4 signature sub packets\n\n  this.read_sub_packet = function(bytes) {\n    var mypos = 0;\n\n    function read_array(prop, bytes) {\n      this[prop] = [];\n\n      for (var i = 0; i < bytes.length; i++) {\n        this[prop].push(bytes[i].charCodeAt());\n      }\n    }\n\n    // The leftwost bit denotes a \"critical\" packet, but we ignore it.\n    var type = bytes[mypos++].charCodeAt() & 0x7F;\n\n    // subpacket type\n    switch (type) {\n      case 2:\n        // Signature Creation Time\n        this.created = util.readDate(bytes.substr(mypos));\n        break;\n      case 3:\n        // Signature Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.signatureNeverExpires = time.getTime() == 0;\n        this.signatureExpirationTime = time;\n\n        break;\n      case 4:\n        // Exportable Certification\n        this.exportable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 5:\n        // Trust Signature\n        this.trustLevel = bytes[mypos++].charCodeAt();\n        this.trustAmount = bytes[mypos++].charCodeAt();\n        break;\n      case 6:\n        // Regular Expression\n        this.regularExpression = bytes.substr(mypos);\n        break;\n      case 7:\n        // Revocable\n        this.revocable = bytes[mypos++].charCodeAt() == 1;\n        break;\n      case 9:\n        // Key Expiration Time\n        var time = util.readDate(bytes.substr(mypos));\n\n        this.keyExpirationTime = time;\n        this.keyNeverExpires = time.getTime() == 0;\n\n        break;\n      case 11:\n        // Preferred Symmetric Algorithms\n        this.preferredSymmetricAlgorithms = [];\n\n        while (mypos != bytes.length) {\n          this.preferredSymmetricAlgorithms.push(bytes[mypos++].charCodeAt());\n        }\n\n        break;\n      case 12:\n        // Revocation Key\n        // (1 octet of class, 1 octet of public-key algorithm ID, 20\n        // octets of\n        // fingerprint)\n        this.revocationKeyClass = bytes[mypos++].charCodeAt();\n        this.revocationKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.revocationKeyFingerprint = bytes.substr(mypos, 20);\n        break;\n\n      case 16:\n        // Issuer\n        this.issuerKeyId = bytes.substr(mypos, 8);\n        break;\n\n      case 20:\n        // Notation Data\n        // We don't know how to handle anything but a text flagged data.\n        if (bytes[mypos].charCodeAt() == 0x80) {\n\n          // We extract key/value tuple from the byte stream.\n          mypos += 4;\n          var m = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n          var n = util.writeNumber(bytes.substr(mypos, 2));\n          mypos += 2\n\n          var name = bytes.substr(mypos, m),\n            value = bytes.substr(mypos + m, n);\n\n          this.notation[name] = value;\n        } else throw new Error(\"Unsupported notation flag.\");\n        break;\n      case 21:\n        // Preferred Hash Algorithms\n        read_array.call(this, 'preferredHashAlgorithms', bytes.substr(mypos));\n        break;\n      case 22:\n        // Preferred Compression Algorithms\n        read_array.call(this, 'preferredCompressionAlgorithms ', bytes.substr(mypos));\n        break;\n      case 23:\n        // Key Server Preferences\n        read_array.call(this, 'keyServerPreferencess', bytes.substr(mypos));\n        break;\n      case 24:\n        // Preferred Key Server\n        this.preferredKeyServer = bytes.substr(mypos);\n        break;\n      case 25:\n        // Primary User ID\n        this.isPrimaryUserID = bytes[mypos++] != 0;\n        break;\n      case 26:\n        // Policy URI\n        this.policyURI = bytes.substr(mypos);\n        break;\n      case 27:\n        // Key Flags\n        read_array.call(this, 'keyFlags', bytes.substr(mypos));\n        break;\n      case 28:\n        // Signer's User ID\n        this.signersUserId += bytes.substr(mypos);\n        break;\n      case 29:\n        // Reason for Revocation\n        this.reasonForRevocationFlag = bytes[mypos++].charCodeAt();\n        this.reasonForRevocationString = bytes.substr(mypos);\n        break;\n      case 30:\n        // Features\n        read_array.call(this, 'features', bytes.substr(mypos));\n        break;\n      case 31:\n        // Signature Target\n        // (1 octet public-key algorithm, 1 octet hash algorithm, N octets hash)\n        this.signatureTargetPublicKeyAlgorithm = bytes[mypos++].charCodeAt();\n        this.signatureTargetHashAlgorithm = bytes[mypos++].charCodeAt();\n\n        var len = crypto.getHashByteLength(this.signatureTargetHashAlgorithm);\n\n        this.signatureTargetHash = bytes.substr(mypos, len);\n        break;\n      case 32:\n        // Embedded Signature\n        this.embeddedSignature = new packet_signature();\n        this.embeddedSignature.read(bytes.substr(mypos));\n        break;\n      default:\n        util.print_error(\"openpgp.packet.signature.js\\n\" +\n          'unknown signature subpacket type ' + type + \" @:\" + mypos +\n          \" subplen:\" + subplen + \" len:\" + len);\n        break;\n    }\n  };\n\n  // Produces data to produce signature on\n  this.toSign = function(type, data) {\n    var t = enums.signature;\n\n    switch (type) {\n      case t.binary:\n        return data.getBytes();\n\n      case t.text:\n        return this.toSign(t.binary, data)\n          .replace(/\\r\\n/g, '\\n')\n          .replace(/\\n/g, '\\r\\n');\n\n      case t.standalone:\n        return '';\n\n      case t.cert_generic:\n      case t.cert_persona:\n      case t.cert_casual:\n      case t.cert_positive:\n      case t.cert_revocation:\n        {\n          var packet, tag;\n\n          if (data.userid !== undefined) {\n            tag = 0xB4;\n            packet = data.userid;\n          } else if (data.userattribute !== undefined) {\n            tag = 0xD1;\n            packet = data.userattribute;\n          } else throw new Error('Either a userid or userattribute packet needs to be ' +\n              'supplied for certification.');\n\n          var bytes = packet.write();\n\n          return this.toSign(t.key, data) +\n            String.fromCharCode(tag) +\n            util.writeNumber(bytes.length, 4) +\n            bytes;\n        }\n      case t.subkey_binding:\n      case t.key_binding:\n        {\n          return this.toSign(t.key, data) + this.toSign(t.key, {\n            key: data.bind\n          });\n        }\n      case t.key:\n        {\n          if (data.key == undefined)\n            throw new Error('Key packet is required for this sigtature.');\n\n          return data.key.writeOld();\n        }\n      case t.key_revocation:\n      case t.subkey_revocation:\n        return this.toSign(t.key, data);\n      case t.timestamp:\n        return '';\n      case t.thrid_party:\n        throw new Error('Not implemented');\n        break;\n      default:\n        throw new Error('Unknown signature type.')\n    }\n  }\n\n\n  this.calculateTrailer = function() {\n    // calculating the trailer\n    var trailer = '';\n    trailer += String.fromCharCode(4); // Version\n    trailer += String.fromCharCode(0xFF);\n    trailer += util.writeNumber(this.signatureData.length, 4);\n    return trailer\n  }\n\n\n  /**\n   * verifys the signature packet. Note: not signature types are implemented\n   * @param {String} data data which on the signature applies\n   * @param {openpgp_msg_privatekey} key the public key to verify the signature\n   * @return {boolean} True if message is verified, else false.\n   */\n  this.verify = function(key, data) {\n    var signatureType = enums.write(enums.signature, this.signatureType),\n      publicKeyAlgorithm = enums.write(enums.publicKey, this.publicKeyAlgorithm),\n      hashAlgorithm = enums.write(enums.hash, this.hashAlgorithm);\n\n    var bytes = this.toSign(signatureType, data),\n      trailer = this.calculateTrailer();\n\n\n    var mpicount = 0;\n    // Algorithm-Specific Fields for RSA signatures:\n    // \t    - multiprecision number (MPI) of RSA signature value m**d mod n.\n    if (publicKeyAlgorithm > 0 && publicKeyAlgorithm < 4)\n      mpicount = 1;\n    //    Algorithm-Specific Fields for DSA signatures:\n    //      - MPI of DSA value r.\n    //      - MPI of DSA value s.\n    else if (publicKeyAlgorithm == 17)\n      mpicount = 2;\n\n    var mpi = [],\n      i = 0;\n    for (var j = 0; j < mpicount; j++) {\n      mpi[j] = new type_mpi();\n      i += mpi[j].read(this.signature.substr(i));\n    }\n\n    this.verified = crypto.signature.verify(publicKeyAlgorithm,\n      hashAlgorithm, mpi, key.mpi,\n      bytes + this.signatureData + trailer);\n\n    return this.verified;\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Sym. Encrypted Integrity Protected Data \n * Packet (Tag 18)\n * \n * RFC4880 5.13: The Symmetrically Encrypted Integrity Protected Data packet is\n * a variant of the Symmetrically Encrypted Data packet. It is a new feature\n * created for OpenPGP that addresses the problem of detecting a modification to\n * encrypted data. It is used in combination with a Modification Detection Code\n * packet.\n */\n\nmodule.exports = function packet_sym_encrypted_integrity_protected() {\n  /** The encrypted payload. */\n  this.encrypted = null; // string\n  /** @type {Boolean}\n   * If after decrypting the packet this is set to true,\n   * a modification has been detected and thus the contents\n   * should be discarded.\n   */\n  this.modification = false;\n  this.packets;\n\n\n  this.read = function(bytes) {\n    // - A one-octet version number. The only currently defined value is\n    // 1.\n    var version = bytes[0].charCodeAt();\n\n    if (version != 1) {\n      throw new Error('Invalid packet version.');\n    }\n\n    // - Encrypted data, the output of the selected symmetric-key cipher\n    //   operating in Cipher Feedback mode with shift amount equal to the\n    //   block size of the cipher (CFB-n where n is the block size).\n    this.encrypted = bytes.substr(1);\n  }\n\n  this.write = function() {\n\n    return String.fromCharCode(1) // Version\n    + this.encrypted;\n  }\n\n  this.encrypt = function(sessionKeyAlgorithm, key) {\n    var bytes = this.packets.write()\n\n    var prefixrandom = crypto.getPrefixRandom(sessionKeyAlgorithm);\n    var prefix = prefixrandom + prefixrandom.charAt(prefixrandom.length - 2) + prefixrandom.charAt(prefixrandom.length -\n      1)\n\n    var tohash = bytes;\n\n\n    // Modification detection code packet.\n    tohash += String.fromCharCode(0xD3);\n    tohash += String.fromCharCode(0x14);\n\n\n    tohash += crypto.hash.sha1(prefix + tohash);\n\n\n    this.encrypted = crypto.cfb.encrypt(prefixrandom,\n      sessionKeyAlgorithm, tohash, key, false).substring(0,\n      prefix.length + tohash.length);\n  }\n\n  /**\n   * Decrypts the encrypted data contained in this object read_packet must\n   * have been called before\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *            The selected symmetric encryption algorithm to be used\n   * @param {String} key The key of cipher blocksize length to be used\n   * @return {String} The decrypted data of this packet\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, false);\n\n\n    // there must be a modification detection code packet as the\n    // last packet and everything gets hashed except the hash itself\n    this.hash = crypto.hash.sha1(\n      crypto.cfb.mdc(sessionKeyAlgorithm, key, this.encrypted) + decrypted.substring(0, decrypted.length - 20));\n\n\n    var mdc = decrypted.substr(decrypted.length - 20, 20);\n\n    if (this.hash != mdc) {\n      throw new Error('Modification detected.');\n    } else\n      this.packets.read(decrypted.substr(0, decrypted.length - 22));\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar type_s2k = require('../type/s2k.js'),\n  enums = require('../enums.js'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Public-Key Encrypted Session Key Packets (Tag 1)\n * \n * RFC4880 5.1: A Public-Key Encrypted Session Key packet holds the session key\n * used to encrypt a message. Zero or more Public-Key Encrypted Session Key\n * packets and/or Symmetric-Key Encrypted Session Key packets may precede a\n * Symmetrically Encrypted Data Packet, which holds an encrypted message. The\n * message is encrypted with the session key, and the session key is itself\n * encrypted and stored in the Encrypted Session Key packet(s). The\n * Symmetrically Encrypted Data Packet is preceded by one Public-Key Encrypted\n * Session Key packet for each OpenPGP key to which the message is encrypted.\n * The recipient of the message finds a session key that is encrypted to their\n * public key, decrypts the session key, and then uses the session key to\n * decrypt the message.\n */\nmodule.exports = function packet_sym_encrypted_session_key() {\n  this.tag = 3;\n  this.sessionKeyEncryptionAlgorithm = null;\n  this.sessionKeyAlgorithm = 'aes256';\n  this.encrypted = null;\n  this.s2k = new type_s2k();\n\n  /**\n   * Parsing function for a symmetric encrypted session key packet (tag 3).\n   * \n   * @param {String} input Payload of a tag 1 packet\n   * @param {Integer} position Position to start reading from the input string\n   * @param {Integer} len\n   *            Length of the packet or the remaining length of\n   *            input at position\n   * @return {openpgp_packet_encrypteddata} Object representation\n   */\n  this.read = function(bytes) {\n    // A one-octet version number. The only currently defined version is 4.\n    this.version = bytes[0].charCodeAt();\n\n    // A one-octet number describing the symmetric algorithm used.\n    var algo = enums.read(enums.symmetric, bytes[1].charCodeAt());\n\n    // A string-to-key (S2K) specifier, length as defined above.\n    var s2klength = this.s2k.read(bytes.substr(2));\n\n    // Optionally, the encrypted session key itself, which is decrypted\n    // with the string-to-key object.\n    var done = s2klength + 2;\n\n    if (done < bytes.length) {\n      this.encrypted = bytes.substr(done);\n      this.sessionKeyEncryptionAlgorithm = algo\n    } else\n      this.sessionKeyAlgorithm = algo;\n  }\n\n  this.write = function() {\n    var algo = this.encrypted == null ?\n      this.sessionKeyAlgorithm :\n      this.sessionKeyEncryptionAlgorithm;\n\n    var bytes = String.fromCharCode(this.version) +\n      String.fromCharCode(enums.write(enums.symmetric, algo)) +\n      this.s2k.write();\n\n    if (this.encrypted != null)\n      bytes += this.encrypted;\n    return bytes;\n  }\n\n  /**\n   * Decrypts the session key (only for public key encrypted session key\n   * packets (tag 1)\n   * \n   * @param {openpgp_msg_message} msg\n   *            The message object (with member encryptedData\n   * @param {openpgp_msg_privatekey} key\n   *            Private key with secMPIs unlocked\n   * @return {String} The unencrypted session key\n   */\n  this.decrypt = function(passphrase) {\n    var algo = this.sessionKeyEncryptionAlgorithm != null ?\n      this.sessionKeyEncryptionAlgorithm :\n      this.sessionKeyAlgorithm;\n\n\n    var length = crypto.cipher[algo].keySize;\n    var key = this.s2k.produce_key(passphrase, length);\n\n    if (this.encrypted == null) {\n      this.sessionKey = key;\n\n    } else {\n      var decrypted = crypto.cfb.decrypt(\n        this.sessionKeyEncryptionAlgorithm, key, this.encrypted, true);\n\n      this.sessionKeyAlgorithm = enums.read(enums.symmetric,\n        decrypted[0].keyCodeAt());\n\n      this.sessionKey = decrypted.substr(1);\n    }\n  }\n\n  this.encrypt = function(passphrase) {\n    var length = crypto.getKeyLength(this.sessionKeyEncryptionAlgorithm);\n    var key = this.s2k.produce_key(passphrase, length);\n\n    var private_key = String.fromCharCode(\n      enums.write(enums.symmetric, this.sessionKeyAlgorithm)) +\n\n    crypto.getRandomBytes(\n      crypto.getKeyLength(this.sessionKeyAlgorithm));\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(this.sessionKeyEncryptionAlgorithm),\n      this.sessionKeyEncryptionAlgorithm, key, private_key, true);\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the Symmetrically Encrypted Data Packet (Tag 9)\n * \n * RFC4880 5.7: The Symmetrically Encrypted Data packet contains data encrypted\n * with a symmetric-key algorithm. When it has been decrypted, it contains other\n * packets (usually a literal data packet or compressed data packet, but in\n * theory other Symmetrically Encrypted Data packets or sequences of packets\n * that form whole OpenPGP messages).\n */\n\nmodule.exports = function packet_symmetrically_encrypted() {\n  this.encrypted = null;\n  /** Decrypted packets contained within. \n   * @type {openpgp_packetlist} */\n  this.packets;\n\n  this.read = function(bytes) {\n    this.encrypted = bytes;\n  }\n\n  this.write = function() {\n    return this.encrypted;\n  }\n\n  /**\n   * Symmetrically decrypt the packet data\n   * \n   * @param {Integer} sessionKeyAlgorithm\n   *             Symmetric key algorithm to use // See RFC4880 9.2\n   * @param {String} key\n   *             Key as string with the corresponding length to the\n   *            algorithm\n   * @return The decrypted data;\n   */\n  this.decrypt = function(sessionKeyAlgorithm, key) {\n    var decrypted = crypto.cfb.decrypt(\n      sessionKeyAlgorithm, key, this.encrypted, true);\n\n    this.packets.read(decrypted);\n  }\n\n  this.encrypt = function(algo, key) {\n    var data = this.packets.write();\n\n    this.encrypted = crypto.cfb.encrypt(\n      crypto.getPrefixRandom(algo), algo, data, key, true);\n  }\n};\n","module.exports = function packet_trust() {\n\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n/** \n * @class\n * @classdesc Implementation of the User Attribute Packet (Tag 17)\n *  The User Attribute packet is a variation of the User ID packet.  It\n *  is capable of storing more types of data than the User ID packet,\n *  which is limited to text.  Like the User ID packet, a User Attribute\n *  packet may be certified by the key owner (\"self-signed\") or any other\n *  key owner who cares to certify it.  Except as noted, a User Attribute\n *  packet may be used anywhere that a User ID packet may be used.\n *\n *  While User Attribute packets are not a required part of the OpenPGP\n *  standard, implementations SHOULD provide at least enough\n *  compatibility to properly handle a certification signature on the\n *  User Attribute packet.  A simple way to do this is by treating the\n *  User Attribute packet as a User ID packet with opaque contents, but\n *  an implementation may use any method desired.\n */\nmodule.exports = function packet_user_attribute() {\n  this.tag = 17;\n  this.attributes = [];\n\n  /**\n   * parsing function for a user attribute packet (tag 17).\n   * @param {String} input payload of a tag 17 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    while (i < bytes.length) {\n      var len = openpgp_packet.read_simple_length(bytes);\n\n      i += len.offset;\n      this.attributes.push(bytes.substr(i, len.len));\n      i += len.len;\n    }\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of the User ID Packet (Tag 13)\n * A User ID packet consists of UTF-8 text that is intended to represent\n * the name and email address of the key holder.  By convention, it\n * includes an RFC 2822 [RFC2822] mail name-addr, but there are no\n * restrictions on its content.  The packet length in the header\n * specifies the length of the User ID. \n */\nmodule.exports = function packet_userid() {\n  /** @type {String} A string containing the user id. Usually in the form\n   * John Doe <john@example.com> \n   */\n  this.userid = '';\n\n\n  /**\n   * Parsing function for a user id packet (tag 13).\n   * @param {String} input payload of a tag 13 packet\n   * @param {Integer} position position to start reading from the input string\n   * @param {Integer} len length of the packet or the remaining length of input \n   * at position\n   * @return {openpgp_packet_encrypteddata} object representation\n   */\n  this.read = function(bytes) {\n    this.userid = util.decode_utf8(bytes);\n  }\n\n  /**\n   * Creates a string representation of the user id packet\n   * @param {String} user_id the user id as string (\"John Doe <john.doe@mail.us\")\n   * @return {String} string representation\n   */\n  this.write = function() {\n    return util.encode_utf8(this.userid);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar util = require('../util');\n\n/**\n * @class\n * @classdesc Implementation of type key id (RFC4880 3.3)\n *  A Key ID is an eight-octet scalar that identifies a key.\n   Implementations SHOULD NOT assume that Key IDs are unique.  The\n   section \"Enhanced Key Formats\" below describes how Key IDs are\n   formed.\n */\nmodule.exports = function keyid() {\n\n  this.bytes = '';\n\n  for (var i = 0; i < 8; i++) {\n    this.bytes += String.fromCharCode(0);\n  }\n  /**\n   * Parsing method for a key id\n   * @param {String} input Input to read the key id from \n   * @param {integer} position Position where to start reading the key \n   * id from input\n   * @return {openpgp_type_keyid} This object\n   */\n  this.read = function(bytes) {\n    this.bytes = bytes.substr(0, 8);\n  }\n\n  this.write = function() {\n    return this.bytes;\n  }\n\n  this.toHex = function() {\n    return util.hexstrdump(this.bytes);\n  }\n\n  this.equals = function(keyid) {\n    return this.bytes == keyid.bytes;\n  }\n};\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\n// Hint: We hold our MPIs as an array of octets in big endian format preceeding a two\n// octet scalar: MPI: [a,b,c,d,e,f]\n// - MPI size: (a << 8) | b \n// - MPI = c | d << 8 | e << ((MPI.length -2)*8) | f ((MPI.length -2)*8)\n\nvar BigInteger = require('../crypto/public_key/jsbn.js'),\n  util = require('../util');\n\n/**\n * @class\n * @classdescImplementation of type MPI (RFC4880 3.2)\n * Multiprecision integers (also called MPIs) are unsigned integers used\n * to hold large integers such as the ones used in cryptographic\n * calculations.\n * An MPI consists of two pieces: a two-octet scalar that is the length\n * of the MPI in bits followed by a string of octets that contain the\n * actual integer.\n */\nmodule.exports = function mpi() {\n  /** An implementation dependent integer */\n  this.data = null;\n\n  /**\n   * Parsing function for a mpi (RFC 4880 3.2).\n   * @param {String} input Payload of mpi data\n   * @param {Integer} position Position to start reading from the input \n   * string\n   * @param {Integer} len Length of the packet or the remaining length of \n   * input at position\n   * @return {openpgp_type_mpi} Object representation\n   */\n  this.read = function(bytes) {\n    var bits = (bytes[0].charCodeAt() << 8) | bytes[1].charCodeAt();\n\n    // Additional rules:\n    //\n    //    The size of an MPI is ((MPI.length + 7) / 8) + 2 octets.\n    //\n    //    The length field of an MPI describes the length starting from its\n    //\t  most significant non-zero bit.  Thus, the MPI [00 02 01] is not\n    //    formed correctly.  It should be [00 01 01].\n\n    // TODO: Verification of this size method! This size calculation as\n    // \t\t specified above is not applicable in JavaScript\n    var bytelen = Math.ceil(bits / 8);\n\n    var raw = bytes.substr(2, bytelen);\n    this.fromBytes(raw);\n\n    return 2 + bytelen;\n  }\n\n  this.fromBytes = function(bytes) {\n    this.data = new BigInteger(util.hexstrdump(bytes), 16);\n  }\n\n  this.toBytes = function() {\n    return this.write().substr(2);\n  }\n\n  this.byteLength = function() {\n    return this.toBytes().length;\n  }\n\n  /**\n   * Converts the mpi object to a string as specified in RFC4880 3.2\n   * @return {String} mpi Byte representation\n   */\n  this.write = function() {\n    return this.data.toMPI();\n  }\n\n  this.toBigInteger = function() {\n    return this.data.clone();\n  }\n\n  this.fromBigInteger = function(bn) {\n    this.data = bn.clone();\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar enums = require('../enums.js'),\n  util = require('../util'),\n  crypto = require('../crypto');\n\n/**\n * @class\n * @classdesc Implementation of the String-to-key specifier (RFC4880 3.7)\n * String-to-key (S2K) specifiers are used to convert passphrase strings\n   into symmetric-key encryption/decryption keys.  They are used in two\n   places, currently: to encrypt the secret part of private keys in the\n   private keyring, and to convert passphrases to encryption keys for\n   symmetrically encrypted messages.\n */\nmodule.exports = function s2k() {\n  /** @type {openpgp.hash} */\n  this.algorithm = 'sha256';\n  /** @type {openpgp_type_s2k.type} */\n  this.type = 'iterated';\n  this.c = 96;\n  /** @type {openpgp_bytearray} \n   * Eight bytes of salt. */\n  this.salt = crypto.random.getRandomBytes(8);\n\n\n  // Exponen bias, defined in RFC4880\n  var expbias = 6;\n\n  this.get_count = function() {\n    return (16 + (this.c & 15)) << ((this.c >> 4) + expbias);\n  }\n\n  /**\n   * Parsing function for a string-to-key specifier (RFC 4880 3.7).\n   * @param {String} input Payload of string-to-key specifier\n   * @return {Integer} Actual length of the object\n   */\n  this.read = function(bytes) {\n    var i = 0;\n    this.type = enums.read(enums.s2k, bytes[i++].charCodeAt());\n    this.algorithm = enums.read(enums.hash, bytes[i++].charCodeAt());\n\n    switch (this.type) {\n      case 'simple':\n        break;\n\n      case 'salted':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n        break;\n\n      case 'iterated':\n        this.salt = bytes.substr(i, 8);\n        i += 8;\n\n        // Octet 10: count, a one-octet, coded value\n        this.c = bytes[i++].charCodeAt();\n        break;\n\n      case 'gnu':\n        if (bytes.substr(i, 3) == \"GNU\") {\n          i += 3; // GNU\n          var gnuExtType = 1000 + bytes[i++].charCodeAt();\n          if (gnuExtType == 1001) {\n            this.type = gnuExtType;\n            // GnuPG extension mode 1001 -- don't write secret key at all\n          } else {\n            throw new Error(\"Unknown s2k gnu protection mode.\");\n          }\n        } else {\n          throw new Error(\"Unknown s2k type.\");\n        }\n        break;\n\n      default:\n        throw new Error(\"Unknown s2k type.\");\n        break;\n    }\n\n    return i;\n  }\n\n\n  /**\n   * writes an s2k hash based on the inputs.\n   * @return {String} Produced key of hashAlgorithm hash length\n   */\n  this.write = function() {\n    var bytes = String.fromCharCode(enums.write(enums.s2k, this.type));\n    bytes += String.fromCharCode(enums.write(enums.hash, this.algorithm));\n\n    switch (this.type) {\n      case 'simple':\n        break;\n      case 'salted':\n        bytes += this.salt;\n        break;\n      case 'iterated':\n        bytes += this.salt;\n        bytes += String.fromCharCode(this.c);\n        break;\n    };\n\n    return bytes;\n  }\n\n  /**\n   * Produces a key using the specified passphrase and the defined \n   * hashAlgorithm \n   * @param {String} passphrase Passphrase containing user input\n   * @return {String} Produced key with a length corresponding to \n   * hashAlgorithm hash length\n   */\n  this.produce_key = function(passphrase, numBytes) {\n    passphrase = util.encode_utf8(passphrase);\n\n    function round(prefix, s2k) {\n      var algorithm = enums.write(enums.hash, s2k.algorithm);\n\n      switch (s2k.type) {\n        case 'simple':\n          return crypto.hash.digest(algorithm, prefix + passphrase);\n\n        case 'salted':\n          return crypto.hash.digest(algorithm,\n            prefix + s2k.salt + passphrase);\n\n        case 'iterated':\n          var isp = [],\n            count = s2k.get_count();\n          data = s2k.salt + passphrase;\n\n          while (isp.length * data.length < count)\n            isp.push(data);\n\n          isp = isp.join('');\n\n          if (isp.length > count)\n            isp = isp.substr(0, count);\n\n          return crypto.hash.digest(algorithm, prefix + isp);\n      };\n    }\n\n    var result = '',\n      prefix = '';\n\n    while (result.length <= numBytes) {\n      result += round(prefix, this);\n      prefix += String.fromCharCode(0);\n    }\n\n    return result.substr(0, numBytes);\n  }\n}\n","// GPG4Browsers - An OpenPGP implementation in javascript\n// Copyright (C) 2011 Recurity Labs GmbH\n// \n// This library is free software; you can redistribute it and/or\n// modify it under the terms of the GNU Lesser General Public\n// License as published by the Free Software Foundation; either\n// version 2.1 of the License, or (at your option) any later version.\n// \n// This library is distributed in the hope that it will be useful,\n// but WITHOUT ANY WARRANTY; without even the implied warranty of\n// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU\n// Lesser General Public License for more details.\n// \n// You should have received a copy of the GNU Lesser General Public\n// License along with this library; if not, write to the Free Software\n// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA\n\nvar Util = function() {\n\n  this.readNumber = function(bytes) {\n    var n = 0;\n\n    for (var i = 0; i < bytes.length; i++) {\n      n <<= 8;\n      n += bytes[i].charCodeAt();\n    }\n\n    return n;\n  };\n\n  this.writeNumber = function(n, bytes) {\n    var b = '';\n    for (var i = 0; i < bytes; i++) {\n      b += String.fromCharCode((n >> (8 * (bytes - i - 1))) & 0xFF);\n    }\n\n    return b;\n  };\n\n\n\n  this.readDate = function(bytes) {\n    var n = this.readNumber(bytes);\n    var d = new Date();\n    d.setTime(n * 1000);\n    return d;\n  };\n\n  this.writeDate = function(time) {\n    var numeric = Math.round(time.getTime() / 1000);\n\n    return this.writeNumber(numeric, 4);\n  };\n\n  this.emailRegEx =\n    /[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*@(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?/;\n\n  this.debug = false;\n\n  this.hexdump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    var i = 0;\n    while (c < e) {\n      h = str.charCodeAt(c++).toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\" \" + h);\n      i++;\n      if (i % 32 == 0)\n        r.push(\"\\n           \");\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create hexstring from a binary\n   * @param {String} str String to convert\n   * @return {String} String containing the hexadecimal values\n   */\n  this.hexstrdump = function(str) {\n    if (str == null)\n      return \"\";\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].charCodeAt().toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n  /**\n   * Create binary string from a hex encoded string\n   * @param {String} str Hex string to convert\n   * @return {String} String containing the binary values\n   */\n  this.hex2bin = function(hex) {\n    var str = '';\n    for (var i = 0; i < hex.length; i += 2)\n      str += String.fromCharCode(parseInt(hex.substr(i, 2), 16));\n    return str;\n  };\n\n  /**\n   * Creating a hex string from an binary array of integers (0..255)\n   * @param {String} str Array of bytes to convert\n   * @return {String} Hexadecimal representation of the array\n   */\n  this.hexidump = function(str) {\n    var r = [];\n    var e = str.length;\n    var c = 0;\n    var h;\n    while (c < e) {\n      h = str[c++].toString(16);\n      while (h.length < 2) h = \"0\" + h;\n      r.push(\"\" + h);\n    }\n    return r.join('');\n  };\n\n\n  /**\n   * Convert a native javascript string to a string of utf8 bytes\n   * @param {String} str The string to convert\n   * @return {String} A valid squence of utf8 bytes\n   */\n  this.encode_utf8 = function(str) {\n    return unescape(encodeURIComponent(str));\n  };\n\n  /**\n   * Convert a string of utf8 bytes to a native javascript string\n   * @param {String} utf8 A valid squence of utf8 bytes\n   * @return {String} A native javascript string\n   */\n  this.decode_utf8 = function(utf8) {\n    try {\n      return decodeURIComponent(escape(utf8));\n    } catch (e) {\n      return utf8;\n    }\n  };\n\n  var str2bin = function(str, result) {\n    for (var i = 0; i < str.length; i++) {\n      result[i] = str.charCodeAt(i);\n    }\n\n    return result;\n  };\n\n  var bin2str = function(bin) {\n    var result = [];\n\n    for (var i = 0; i < bin.length; i++) {\n      result.push(String.fromCharCode(bin[i]));\n    }\n\n    return result.join('');\n  };\n\n  /**\n   * Convert a string to an array of integers(0.255)\n   * @param {String} str String to convert\n   * @return {Integer[]} An array of (binary) integers\n   */\n  this.str2bin = function(str) {\n    return str2bin(str, new Array(str.length));\n  };\n\n\n  /**\n   * Convert an array of integers(0.255) to a string \n   * @param {Integer[]} bin An array of (binary) integers to convert\n   * @return {String} The string representation of the array\n   */\n  this.bin2str = bin2str;\n\n  /**\n   * Convert a string to a Uint8Array\n   * @param {String} str String to convert\n   * @return {Uint8Array} The array of (binary) integers\n   */\n  this.str2Uint8Array = function(str) {\n    return str2bin(str, new Uint8Array(new ArrayBuffer(str.length)));\n  };\n\n  /**\n   * Convert a Uint8Array to a string. This currently functions \n   * the same as bin2str. \n   * @param {Uint8Array} bin An array of (binary) integers to convert\n   * @return {String} String representation of the array\n   */\n  this.Uint8Array2str = bin2str;\n\n  /**\n   * Calculates a 16bit sum of a string by adding each character \n   * codes modulus 65535\n   * @param {String} text String to create a sum of\n   * @return {Integer} An integer containing the sum of all character \n   * codes % 65535\n   */\n  this.calc_checksum = function(text) {\n    var checksum = {\n      s: 0,\n      add: function(sadd) {\n        this.s = (this.s + sadd) % 65536;\n      }\n    };\n    for (var i = 0; i < text.length; i++) {\n      checksum.add(text.charCodeAt(i));\n    }\n    return checksum.s;\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug = function(str) {\n    if (this.debug) {\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print a debug message. Debug \n   * messages are only printed if\n   * openpgp.config.debug is set to true. The calling\n   * Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * Different than print_debug because will call hexstrdump iff necessary.\n   * @param {String} str String of the debug message\n   * @return {String} An HTML tt entity containing a paragraph with a \n   * style attribute where the debug message is HTMLencoded in. \n   */\n  this.print_debug_hexstr_dump = function(str, strToHex) {\n    if (this.debug) {\n      str = str + this.hexstrdump(strToHex);\n      console.log(str);\n    }\n  };\n\n  /**\n   * Helper function to print an error message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'\n   * @param {String} str String of the error message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded error message\n   */\n  this.print_error = function(str) {\n    if (this.debug)\n      throw str;\n    console.log(str);\n  };\n\n  /**\n   * Helper function to print an info message. \n   * The calling Javascript context MUST define\n   * a \"showMessages(text)\" function. Line feeds ('\\n')\n   * are automatically converted to HTML line feeds '<br/>'.\n   * @param {String} str String of the info message\n   * @return {String} A HTML paragraph entity with a style attribute \n   * containing the HTML encoded info message\n   */\n  this.print_info = function(str) {\n    if (this.debug)\n      console.log(str);\n  };\n\n  this.print_warning = function(str) {\n    console.log(str);\n  };\n\n  this.getLeftNBits = function(string, bitcount) {\n    var rest = bitcount % 8;\n    if (rest == 0)\n      return string.substring(0, bitcount / 8);\n    var bytes = (bitcount - rest) / 8 + 1;\n    var result = string.substring(0, bytes);\n    return this.shiftRight(result, 8 - rest); // +String.fromCharCode(string.charCodeAt(bytes -1) << (8-rest) & 0xFF);\n  };\n\n  /**\n   * Shifting a string to n bits right\n   * @param {String} value The string to shift\n   * @param {Integer} bitcount Amount of bits to shift (MUST be smaller \n   * than 9)\n   * @return {String} Resulting string. \n   */\n  this.shiftRight = function(value, bitcount) {\n    var temp = util.str2bin(value);\n    if (bitcount % 8 != 0) {\n      for (var i = temp.length - 1; i >= 0; i--) {\n        temp[i] >>= bitcount % 8;\n        if (i > 0)\n          temp[i] |= (temp[i - 1] << (8 - (bitcount % 8))) & 0xFF;\n      }\n    } else {\n      return value;\n    }\n    return util.bin2str(temp);\n  };\n\n  /**\n   * Return the algorithm type as string\n   * @return {String} String representing the message type\n   */\n  this.get_hashAlgorithmString = function(algo) {\n    switch (algo) {\n      case 1:\n        return \"MD5\";\n      case 2:\n        return \"SHA1\";\n      case 3:\n        return \"RIPEMD160\";\n      case 8:\n        return \"SHA256\";\n      case 9:\n        return \"SHA384\";\n      case 10:\n        return \"SHA512\";\n      case 11:\n        return \"SHA224\";\n    }\n    return \"unknown\";\n  };\n\n};\n\n/**\n * an instance that should be used. \n */\nmodule.exports = new Util();\n","var unit = require('../../unit.js');\n\nunit.register(\"AES Rijndael cipher test with test vectors from ecb_tbl.txt\", function() {\n  var openpgp = require('../../../');\n  var util = openpgp.util;\n\n  var result = new Array();\n\n  function test_aes(input, key, output) {\n    var aes = new openpgp.crypto.cipher.aes128(util.bin2str(key));\n\n    var result = util.bin2str(aes.encrypt(input));\n\n    return util.hexstrdump(result) == util.hexstrdump(util.bin2str(output));\n  };\n\n  var testvectors128 = [[[0x00,0x01,0x02,0x03,0x05,0x06,0x07,0x08,0x0A,0x0B,0x0C,0x0D,0x0F,0x10,0x11,0x12],[0x50,0x68,0x12,0xA4,0x5F,0x08,0xC8,0x89,0xB9,0x7F,0x59,0x80,0x03,0x8B,0x83,0x59],[0xD8,0xF5,0x32,0x53,0x82,0x89,0xEF,0x7D,0x06,0xB5,0x06,0xA4,0xFD,0x5B,0xE9,0xC9]],\n                        [[0x14,0x15,0x16,0x17,0x19,0x1A,0x1B,0x1C,0x1E,0x1F,0x20,0x21,0x23,0x24,0x25,0x26],[0x5C,0x6D,0x71,0xCA,0x30,0xDE,0x8B,0x8B,0x00,0x54,0x99,0x84,0xD2,0xEC,0x7D,0x4B],[0x59,0xAB,0x30,0xF4,0xD4,0xEE,0x6E,0x4F,0xF9,0x90,0x7E,0xF6,0x5B,0x1F,0xB6,0x8C]],\n                        [[0x28,0x29,0x2A,0x2B,0x2D,0x2E,0x2F,0x30,0x32,0x33,0x34,0x35,0x37,0x38,0x39,0x3A],[0x53,0xF3,0xF4,0xC6,0x4F,0x86,0x16,0xE4,0xE7,0xC5,0x61,0x99,0xF4,0x8F,0x21,0xF6],[0xBF,0x1E,0xD2,0xFC,0xB2,0xAF,0x3F,0xD4,0x14,0x43,0xB5,0x6D,0x85,0x02,0x5C,0xB1]],\n                        [[0x3C,0x3D,0x3E,0x3F,0x41,0x42,0x43,0x44,0x46,0x47,0x48,0x49,0x4B,0x4C,0x4D,0x4E],[0xA1,0xEB,0x65,0xA3,0x48,0x71,0x65,0xFB,0x0F,0x1C,0x27,0xFF,0x99,0x59,0xF7,0x03],[0x73,0x16,0x63,0x2D,0x5C,0x32,0x23,0x3E,0xDC,0xB0,0x78,0x05,0x60,0xEA,0xE8,0xB2]],\n                        [[0x50,0x51,0x52,0x53,0x55,0x56,0x57,0x58,0x5A,0x5B,0x5C,0x5D,0x5F,0x60,0x61,0x62],[0x35,0x53,0xEC,0xF0,0xB1,0x73,0x95,0x58,0xB0,0x8E,0x35,0x0A,0x98,0xA3,0x9B,0xFA],[0x40,0x8C,0x07,0x3E,0x3E,0x25,0x38,0x07,0x2B,0x72,0x62,0x5E,0x68,0xB8,0x36,0x4B]],\n                        [[0x64,0x65,0x66,0x67,0x69,0x6A,0x6B,0x6C,0x6E,0x6F,0x70,0x71,0x73,0x74,0x75,0x76],[0x67,0x42,0x99,0x69,0x49,0x0B,0x97,0x11,0xAE,0x2B,0x01,0xDC,0x49,0x7A,0xFD,0xE8],[0xE1,0xF9,0x4D,0xFA,0x77,0x65,0x97,0xBE,0xAC,0xA2,0x62,0xF2,0xF6,0x36,0x6F,0xEA]],\n                        [[0x78,0x79,0x7A,0x7B,0x7D,0x7E,0x7F,0x80,0x82,0x83,0x84,0x85,0x87,0x88,0x89,0x8A],[0x93,0x38,0x5C,0x1F,0x2A,0xEC,0x8B,0xED,0x19,0x2F,0x5A,0x8E,0x16,0x1D,0xD5,0x08],[0xF2,0x9E,0x98,0x6C,0x6A,0x1C,0x27,0xD7,0xB2,0x9F,0xFD,0x7E,0xE9,0x2B,0x75,0xF1]],\n                        [[0x8C,0x8D,0x8E,0x8F,0x91,0x92,0x93,0x94,0x96,0x97,0x98,0x99,0x9B,0x9C,0x9D,0x9E],[0xB5,0xBF,0x94,0x6B,0xE1,0x9B,0xEB,0x8D,0xB3,0x98,0x3B,0x5F,0x4C,0x6E,0x8D,0xDB],[0x13,0x1C,0x88,0x6A,0x57,0xF8,0xC2,0xE7,0x13,0xAB,0xA6,0x95,0x5E,0x2B,0x55,0xB5]],\n                        [[0xA0,0xA1,0xA2,0xA3,0xA5,0xA6,0xA7,0xA8,0xAA,0xAB,0xAC,0xAD,0xAF,0xB0,0xB1,0xB2],[0x41,0x32,0x1E,0xE1,0x0E,0x21,0xBD,0x90,0x72,0x27,0xC4,0x45,0x0F,0xF4,0x23,0x24],[0xD2,0xAB,0x76,0x62,0xDF,0x9B,0x8C,0x74,0x02,0x10,0xE5,0xEE,0xB6,0x1C,0x19,0x9D]],\n                        [[0xB4,0xB5,0xB6,0xB7,0xB9,0xBA,0xBB,0xBC,0xBE,0xBF,0xC0,0xC1,0xC3,0xC4,0xC5,0xC6],[0x00,0xA8,0x2F,0x59,0xC9,0x1C,0x84,0x86,0xD1,0x2C,0x0A,0x80,0x12,0x4F,0x60,0x89],[0x14,0xC1,0x05,0x54,0xB2,0x85,0x9C,0x48,0x4C,0xAB,0x58,0x69,0xBB,0xE7,0xC4,0x70]],\n                        [[0xC8,0xC9,0xCA,0xCB,0xCD,0xCE,0xCF,0xD0,0xD2,0xD3,0xD4,0xD5,0xD7,0xD8,0xD9,0xDA],[0x7C,0xE0,0xFD,0x07,0x67,0x54,0x69,0x1B,0x4B,0xBD,0x9F,0xAF,0x8A,0x13,0x72,0xFE],[0xDB,0x4D,0x49,0x8F,0x0A,0x49,0xCF,0x55,0x44,0x5D,0x50,0x2C,0x1F,0x9A,0xB3,0xB5]],\n                        [[0xDC,0xDD,0xDE,0xDF,0xE1,0xE2,0xE3,0xE4,0xE6,0xE7,0xE8,0xE9,0xEB,0xEC,0xED,0xEE],[0x23,0x60,0x5A,0x82,0x43,0xD0,0x77,0x64,0x54,0x1B,0xC5,0xAD,0x35,0x5B,0x31,0x29],[0x6D,0x96,0xFE,0xF7,0xD6,0x65,0x90,0xA7,0x7A,0x77,0xBB,0x20,0x56,0x66,0x7F,0x7F]],\n                        [[0xF0,0xF1,0xF2,0xF3,0xF5,0xF6,0xF7,0xF8,0xFA,0xFB,0xFC,0xFD,0xFE,0x01,0x00,0x02],[0x12,0xA8,0xCF,0xA2,0x3E,0xA7,0x64,0xFD,0x87,0x62,0x32,0xB4,0xE8,0x42,0xBC,0x44],[0x31,0x6F,0xB6,0x8E,0xDB,0xA7,0x36,0xC5,0x3E,0x78,0x47,0x7B,0xF9,0x13,0x72,0x5C]],\n                        [[0x04,0x05,0x06,0x07,0x09,0x0A,0x0B,0x0C,0x0E,0x0F,0x10,0x11,0x13,0x14,0x15,0x16],[0xBC,0xAF,0x32,0x41,0x5E,0x83,0x08,0xB3,0x72,0x3E,0x5F,0xDD,0x85,0x3C,0xCC,0x80],[0x69,0x36,0xF2,0xB9,0x3A,0xF8,0x39,0x7F,0xD3,0xA7,0x71,0xFC,0x01,0x1C,0x8C,0x37]],\n                        [[0x2C,0x2D,0x2E,0x2F,0x31,0x32,0x33,0x34,0x36,0x37,0x38,0x39,0x3B,0x3C,0x3D,0x3E],[0x89,0xAF,0xAE,0x68,0x5D,0x80,0x1A,0xD7,0x47,0xAC,0xE9,0x1F,0xC4,0x9A,0xDD,0xE0],[0xF3,0xF9,0x2F,0x7A,0x9C,0x59,0x17,0x9C,0x1F,0xCC,0x2C,0x2B,0xA0,0xB0,0x82,0xCD]]];\n  \n  var testvectors192 = [[[0x00,0x01,0x02,0x03,0x05,0x06,0x07,0x08,0x0A,0x0B,0x0C,0x0D,0x0F,0x10,0x11,0x12,0x14,0x15,0x16,0x17,0x19,0x1A,0x1B,0x1C],[0x2D,0x33,0xEE,0xF2,0xC0,0x43,0x0A,0x8A,0x9E,0xBF,0x45,0xE8,0x09,0xC4,0x0B,0xB6],[0xDF,0xF4,0x94,0x5E,0x03,0x36,0xDF,0x4C,0x1C,0x56,0xBC,0x70,0x0E,0xFF,0x83,0x7F]],\n                        [[0x1E,0x1F,0x20,0x21,0x23,0x24,0x25,0x26,0x28,0x29,0x2A,0x2B,0x2D,0x2E,0x2F,0x30,0x32,0x33,0x34,0x35,0x37,0x38,0x39,0x3A],[0x6A,0xA3,0x75,0xD1,0xFA,0x15,0x5A,0x61,0xFB,0x72,0x35,0x3E,0x0A,0x5A,0x87,0x56],[0xB6,0xFD,0xDE,0xF4,0x75,0x27,0x65,0xE3,0x47,0xD5,0xD2,0xDC,0x19,0x6D,0x12,0x52]],\n                        [[0x3C,0x3D,0x3E,0x3F,0x41,0x42,0x43,0x44,0x46,0x47,0x48,0x49,0x4B,0x4C,0x4D,0x4E,0x50,0x51,0x52,0x53,0x55,0x56,0x57,0x58],[0xBC,0x37,0x36,0x51,0x8B,0x94,0x90,0xDC,0xB8,0xED,0x60,0xEB,0x26,0x75,0x8E,0xD4],[0xD2,0x36,0x84,0xE3,0xD9,0x63,0xB3,0xAF,0xCF,0x1A,0x11,0x4A,0xCA,0x90,0xCB,0xD6]],\n                        [[0x5A,0x5B,0x5C,0x5D,0x5F,0x60,0x61,0x62,0x64,0x65,0x66,0x67,0x69,0x6A,0x6B,0x6C,0x6E,0x6F,0x70,0x71,0x73,0x74,0x75,0x76],[0xAA,0x21,0x44,0x02,0xB4,0x6C,0xFF,0xB9,0xF7,0x61,0xEC,0x11,0x26,0x3A,0x31,0x1E],[0x3A,0x7A,0xC0,0x27,0x75,0x3E,0x2A,0x18,0xC2,0xCE,0xAB,0x9E,0x17,0xC1,0x1F,0xD0]],\n                        [[0x78,0x79,0x7A,0x7B,0x7D,0x7E,0x7F,0x80,0x82,0x83,0x84,0x85,0x87,0x88,0x89,0x8A,0x8C,0x8D,0x8E,0x8F,0x91,0x92,0x93,0x94],[0x02,0xAE,0xA8,0x6E,0x57,0x2E,0xEA,0xB6,0x6B,0x2C,0x3A,0xF5,0xE9,0xA4,0x6F,0xD6],[0x8F,0x67,0x86,0xBD,0x00,0x75,0x28,0xBA,0x26,0x60,0x3C,0x16,0x01,0xCD,0xD0,0xD8]],\n                        [[0x96,0x97,0x98,0x99,0x9B,0x9C,0x9D,0x9E,0xA0,0xA1,0xA2,0xA3,0xA5,0xA6,0xA7,0xA8,0xAA,0xAB,0xAC,0xAD,0xAF,0xB0,0xB1,0xB2],[0xE2,0xAE,0xF6,0xAC,0xC3,0x3B,0x96,0x5C,0x4F,0xA1,0xF9,0x1C,0x75,0xFF,0x6F,0x36],[0xD1,0x7D,0x07,0x3B,0x01,0xE7,0x15,0x02,0xE2,0x8B,0x47,0xAB,0x55,0x11,0x68,0xB3]],\n                        [[0xB4,0xB5,0xB6,0xB7,0xB9,0xBA,0xBB,0xBC,0xBE,0xBF,0xC0,0xC1,0xC3,0xC4,0xC5,0xC6,0xC8,0xC9,0xCA,0xCB,0xCD,0xCE,0xCF,0xD0],[0x06,0x59,0xDF,0x46,0x42,0x71,0x62,0xB9,0x43,0x48,0x65,0xDD,0x94,0x99,0xF9,0x1D],[0xA4,0x69,0xDA,0x51,0x71,0x19,0xFA,0xB9,0x58,0x76,0xF4,0x1D,0x06,0xD4,0x0F,0xFA]],\n                        [[0xD2,0xD3,0xD4,0xD5,0xD7,0xD8,0xD9,0xDA,0xDC,0xDD,0xDE,0xDF,0xE1,0xE2,0xE3,0xE4,0xE6,0xE7,0xE8,0xE9,0xEB,0xEC,0xED,0xEE],[0x49,0xA4,0x42,0x39,0xC7,0x48,0xFE,0xB4,0x56,0xF5,0x9C,0x27,0x6A,0x56,0x58,0xDF],[0x60,0x91,0xAA,0x3B,0x69,0x5C,0x11,0xF5,0xC0,0xB6,0xAD,0x26,0xD3,0xD8,0x62,0xFF]],\n                        [[0xF0,0xF1,0xF2,0xF3,0xF5,0xF6,0xF7,0xF8,0xFA,0xFB,0xFC,0xFD,0xFE,0x01,0x00,0x02,0x04,0x05,0x06,0x07,0x09,0x0A,0x0B,0x0C],[0x66,0x20,0x8F,0x6E,0x9D,0x04,0x52,0x5B,0xDE,0xDB,0x27,0x33,0xB6,0xA6,0xBE,0x37],[0x70,0xF9,0xE6,0x7F,0x9F,0x8D,0xF1,0x29,0x41,0x31,0x66,0x2D,0xC6,0xE6,0x93,0x64]],\n                        [[0x0E,0x0F,0x10,0x11,0x13,0x14,0x15,0x16,0x18,0x19,0x1A,0x1B,0x1D,0x1E,0x1F,0x20,0x22,0x23,0x24,0x25,0x27,0x28,0x29,0x2A],[0x33,0x93,0xF8,0xDF,0xC7,0x29,0xC9,0x7F,0x54,0x80,0xB9,0x50,0xBC,0x96,0x66,0xB0],[0xD1,0x54,0xDC,0xAF,0xAD,0x8B,0x20,0x7F,0xA5,0xCB,0xC9,0x5E,0x99,0x96,0xB5,0x59]],\n                        [[0x2C,0x2D,0x2E,0x2F,0x31,0x32,0x33,0x34,0x36,0x37,0x38,0x39,0x3B,0x3C,0x3D,0x3E,0x40,0x41,0x42,0x43,0x45,0x46,0x47,0x48],[0x60,0x68,0x34,0xC8,0xCE,0x06,0x3F,0x32,0x34,0xCF,0x11,0x45,0x32,0x5D,0xBD,0x71],[0x49,0x34,0xD5,0x41,0xE8,0xB4,0x6F,0xA3,0x39,0xC8,0x05,0xA7,0xAE,0xB9,0xE5,0xDA]],\n                        [[0x4A,0x4B,0x4C,0x4D,0x4F,0x50,0x51,0x52,0x54,0x55,0x56,0x57,0x59,0x5A,0x5B,0x5C,0x5E,0x5F,0x60,0x61,0x63,0x64,0x65,0x66],[0xFE,0xC1,0xC0,0x4F,0x52,0x9B,0xBD,0x17,0xD8,0xCE,0xCF,0xCC,0x47,0x18,0xB1,0x7F],[0x62,0x56,0x4C,0x73,0x8F,0x3E,0xFE,0x18,0x6E,0x1A,0x12,0x7A,0x0C,0x4D,0x3C,0x61]],\n                        [[0x68,0x69,0x6A,0x6B,0x6D,0x6E,0x6F,0x70,0x72,0x73,0x74,0x75,0x77,0x78,0x79,0x7A,0x7C,0x7D,0x7E,0x7F,0x81,0x82,0x83,0x84],[0x32,0xDF,0x99,0xB4,0x31,0xED,0x5D,0xC5,0xAC,0xF8,0xCA,0xF6,0xDC,0x6C,0xE4,0x75],[0x07,0x80,0x5A,0xA0,0x43,0x98,0x6E,0xB2,0x36,0x93,0xE2,0x3B,0xEF,0x8F,0x34,0x38]],\n                        [[0x86,0x87,0x88,0x89,0x8B,0x8C,0x8D,0x8E,0x90,0x91,0x92,0x93,0x95,0x96,0x97,0x98,0x9A,0x9B,0x9C,0x9D,0x9F,0xA0,0xA1,0xA2],[0x7F,0xDC,0x2B,0x74,0x6F,0x3F,0x66,0x52,0x96,0x94,0x3B,0x83,0x71,0x0D,0x1F,0x82],[0xDF,0x0B,0x49,0x31,0x03,0x8B,0xAD,0xE8,0x48,0xDE,0xE3,0xB4,0xB8,0x5A,0xA4,0x4B]],\n                        [[0xA4,0xA5,0xA6,0xA7,0xA9,0xAA,0xAB,0xAC,0xAE,0xAF,0xB0,0xB1,0xB3,0xB4,0xB5,0xB6,0xB8,0xB9,0xBA,0xBB,0xBD,0xBE,0xBF,0xC0],[0x8F,0xBA,0x15,0x10,0xA3,0xC5,0xB8,0x7E,0x2E,0xAA,0x3F,0x7A,0x91,0x45,0x5C,0xA2],[0x59,0x2D,0x5F,0xDE,0xD7,0x65,0x82,0xE4,0x14,0x3C,0x65,0x09,0x93,0x09,0x47,0x7C]]];\n  \n  var testvectors256 = [[[0x00,0x01,0x02,0x03,0x05,0x06,0x07,0x08,0x0A,0x0B,0x0C,0x0D,0x0F,0x10,0x11,0x12,0x14,0x15,0x16,0x17,0x19,0x1A,0x1B,0x1C,0x1E,0x1F,0x20,0x21,0x23,0x24,0x25,0x26],[0x83,0x4E,0xAD,0xFC,0xCA,0xC7,0xE1,0xB3,0x06,0x64,0xB1,0xAB,0xA4,0x48,0x15,0xAB],[0x19,0x46,0xDA,0xBF,0x6A,0x03,0xA2,0xA2,0xC3,0xD0,0xB0,0x50,0x80,0xAE,0xD6,0xFC]],\n              [[0x28,0x29,0x2A,0x2B,0x2D,0x2E,0x2F,0x30,0x32,0x33,0x34,0x35,0x37,0x38,0x39,0x3A,0x3C,0x3D,0x3E,0x3F,0x41,0x42,0x43,0x44,0x46,0x47,0x48,0x49,0x4B,0x4C,0x4D,0x4E],[0xD9,0xDC,0x4D,0xBA,0x30,0x21,0xB0,0x5D,0x67,0xC0,0x51,0x8F,0x72,0xB6,0x2B,0xF1],[0x5E,0xD3,0x01,0xD7,0x47,0xD3,0xCC,0x71,0x54,0x45,0xEB,0xDE,0xC6,0x2F,0x2F,0xB4]],\n              [[0x50,0x51,0x52,0x53,0x55,0x56,0x57,0x58,0x5A,0x5B,0x5C,0x5D,0x5F,0x60,0x61,0x62,0x64,0x65,0x66,0x67,0x69,0x6A,0x6B,0x6C,0x6E,0x6F,0x70,0x71,0x73,0x74,0x75,0x76],[0xA2,0x91,0xD8,0x63,0x01,0xA4,0xA7,0x39,0xF7,0x39,0x21,0x73,0xAA,0x3C,0x60,0x4C],[0x65,0x85,0xC8,0xF4,0x3D,0x13,0xA6,0xBE,0xAB,0x64,0x19,0xFC,0x59,0x35,0xB9,0xD0]],\n              [[0x78,0x79,0x7A,0x7B,0x7D,0x7E,0x7F,0x80,0x82,0x83,0x84,0x85,0x87,0x88,0x89,0x8A,0x8C,0x8D,0x8E,0x8F,0x91,0x92,0x93,0x94,0x96,0x97,0x98,0x99,0x9B,0x9C,0x9D,0x9E],[0x42,0x64,0xB2,0x69,0x64,0x98,0xDE,0x4D,0xF7,0x97,0x88,0xA9,0xF8,0x3E,0x93,0x90],[0x2A,0x5B,0x56,0xA5,0x96,0x68,0x0F,0xCC,0x0E,0x05,0xF5,0xE0,0xF1,0x51,0xEC,0xAE]],\n              [[0xA0,0xA1,0xA2,0xA3,0xA5,0xA6,0xA7,0xA8,0xAA,0xAB,0xAC,0xAD,0xAF,0xB0,0xB1,0xB2,0xB4,0xB5,0xB6,0xB7,0xB9,0xBA,0xBB,0xBC,0xBE,0xBF,0xC0,0xC1,0xC3,0xC4,0xC5,0xC6],[0xEE,0x99,0x32,0xB3,0x72,0x18,0x04,0xD5,0xA8,0x3E,0xF5,0x94,0x92,0x45,0xB6,0xF6],[0xF5,0xD6,0xFF,0x41,0x4F,0xD2,0xC6,0x18,0x14,0x94,0xD2,0x0C,0x37,0xF2,0xB8,0xC4]],\n              [[0xC8,0xC9,0xCA,0xCB,0xCD,0xCE,0xCF,0xD0,0xD2,0xD3,0xD4,0xD5,0xD7,0xD8,0xD9,0xDA,0xDC,0xDD,0xDE,0xDF,0xE1,0xE2,0xE3,0xE4,0xE6,0xE7,0xE8,0xE9,0xEB,0xEC,0xED,0xEE],[0xE6,0x24,0x8F,0x55,0xC5,0xFD,0xCB,0xCA,0x9C,0xBB,0xB0,0x1C,0x88,0xA2,0xEA,0x77],[0x85,0x39,0x9C,0x01,0xF5,0x9F,0xFF,0xB5,0x20,0x4F,0x19,0xF8,0x48,0x2F,0x00,0xB8]],\n              [[0xF0,0xF1,0xF2,0xF3,0xF5,0xF6,0xF7,0xF8,0xFA,0xFB,0xFC,0xFD,0xFE,0x01,0x00,0x02,0x04,0x05,0x06,0x07,0x09,0x0A,0x0B,0x0C,0x0E,0x0F,0x10,0x11,0x13,0x14,0x15,0x16],[0xB8,0x35,0x8E,0x41,0xB9,0xDF,0xF6,0x5F,0xD4,0x61,0xD5,0x5A,0x99,0x26,0x62,0x47],[0x92,0x09,0x7B,0x4C,0x88,0xA0,0x41,0xDD,0xF9,0x81,0x44,0xBC,0x8D,0x22,0xE8,0xE7]],\n              [[0x18,0x19,0x1A,0x1B,0x1D,0x1E,0x1F,0x20,0x22,0x23,0x24,0x25,0x27,0x28,0x29,0x2A,0x2C,0x2D,0x2E,0x2F,0x31,0x32,0x33,0x34,0x36,0x37,0x38,0x39,0x3B,0x3C,0x3D,0x3E],[0xF0,0xE2,0xD7,0x22,0x60,0xAF,0x58,0xE2,0x1E,0x01,0x5A,0xB3,0xA4,0xC0,0xD9,0x06],[0x89,0xBD,0x5B,0x73,0xB3,0x56,0xAB,0x41,0x2A,0xEF,0x9F,0x76,0xCE,0xA2,0xD6,0x5C]],\n              [[0x40,0x41,0x42,0x43,0x45,0x46,0x47,0x48,0x4A,0x4B,0x4C,0x4D,0x4F,0x50,0x51,0x52,0x54,0x55,0x56,0x57,0x59,0x5A,0x5B,0x5C,0x5E,0x5F,0x60,0x61,0x63,0x64,0x65,0x66],[0x47,0x5B,0x8B,0x82,0x3C,0xE8,0x89,0x3D,0xB3,0xC4,0x4A,0x9F,0x2A,0x37,0x9F,0xF7],[0x25,0x36,0x96,0x90,0x93,0xC5,0x5F,0xF9,0x45,0x46,0x92,0xF2,0xFA,0xC2,0xF5,0x30]],\n              [[0x68,0x69,0x6A,0x6B,0x6D,0x6E,0x6F,0x70,0x72,0x73,0x74,0x75,0x77,0x78,0x79,0x7A,0x7C,0x7D,0x7E,0x7F,0x81,0x82,0x83,0x84,0x86,0x87,0x88,0x89,0x8B,0x8C,0x8D,0x8E],[0x68,0x8F,0x52,0x81,0x94,0x58,0x12,0x86,0x2F,0x5F,0x30,0x76,0xCF,0x80,0x41,0x2F],[0x07,0xFC,0x76,0xA8,0x72,0x84,0x3F,0x3F,0x6E,0x00,0x81,0xEE,0x93,0x96,0xD6,0x37]],\n              [[0x90,0x91,0x92,0x93,0x95,0x96,0x97,0x98,0x9A,0x9B,0x9C,0x9D,0x9F,0xA0,0xA1,0xA2,0xA4,0xA5,0xA6,0xA7,0xA9,0xAA,0xAB,0xAC,0xAE,0xAF,0xB0,0xB1,0xB3,0xB4,0xB5,0xB6],[0x08,0xD1,0xD2,0xBC,0x75,0x0A,0xF5,0x53,0x36,0x5D,0x35,0xE7,0x5A,0xFA,0xCE,0xAA],[0xE3,0x8B,0xA8,0xEC,0x2A,0xA7,0x41,0x35,0x8D,0xCC,0x93,0xE8,0xF1,0x41,0xC4,0x91]],\n              [[0xB8,0xB9,0xBA,0xBB,0xBD,0xBE,0xBF,0xC0,0xC2,0xC3,0xC4,0xC5,0xC7,0xC8,0xC9,0xCA,0xCC,0xCD,0xCE,0xCF,0xD1,0xD2,0xD3,0xD4,0xD6,0xD7,0xD8,0xD9,0xDB,0xDC,0xDD,0xDE],[0x87,0x07,0x12,0x1F,0x47,0xCC,0x3E,0xFC,0xEC,0xA5,0xF9,0xA8,0x47,0x49,0x50,0xA1],[0xD0,0x28,0xEE,0x23,0xE4,0xA8,0x90,0x75,0xD0,0xB0,0x3E,0x86,0x8D,0x7D,0x3A,0x42]],\n              [[0xE0,0xE1,0xE2,0xE3,0xE5,0xE6,0xE7,0xE8,0xEA,0xEB,0xEC,0xED,0xEF,0xF0,0xF1,0xF2,0xF4,0xF5,0xF6,0xF7,0xF9,0xFA,0xFB,0xFC,0xFE,0xFE,0x01,0x01,0x03,0x04,0x05,0x06],[0xE5,0x1A,0xA0,0xB1,0x35,0xDB,0xA5,0x66,0x93,0x9C,0x3B,0x63,0x59,0xA9,0x80,0xC5],[0x8C,0xD9,0x42,0x3D,0xFC,0x45,0x9E,0x54,0x71,0x55,0xC5,0xD1,0xD5,0x22,0xE5,0x40]],\n              [[0x08,0x09,0x0A,0x0B,0x0D,0x0E,0x0F,0x10,0x12,0x13,0x14,0x15,0x17,0x18,0x19,0x1A,0x1C,0x1D,0x1E,0x1F,0x21,0x22,0x23,0x24,0x26,0x27,0x28,0x29,0x2B,0x2C,0x2D,0x2E],[0x06,0x9A,0x00,0x7F,0xC7,0x6A,0x45,0x9F,0x98,0xBA,0xF9,0x17,0xFE,0xDF,0x95,0x21],[0x08,0x0E,0x95,0x17,0xEB,0x16,0x77,0x71,0x9A,0xCF,0x72,0x80,0x86,0x04,0x0A,0xE3]],\n              [[0x30,0x31,0x32,0x33,0x35,0x36,0x37,0x38,0x3A,0x3B,0x3C,0x3D,0x3F,0x40,0x41,0x42,0x44,0x45,0x46,0x47,0x49,0x4A,0x4B,0x4C,0x4E,0x4F,0x50,0x51,0x53,0x54,0x55,0x56],[0x72,0x61,0x65,0xC1,0x72,0x3F,0xBC,0xF6,0xC0,0x26,0xD7,0xD0,0x0B,0x09,0x10,0x27],[0x7C,0x17,0x00,0x21,0x1A,0x39,0x91,0xFC,0x0E,0xCD,0xED,0x0A,0xB3,0xE5,0x76,0xB0]]];\n\n  var res = true;\n  var j = 0;\n  for (var i = 0; i < testvectors128.length; i++) {\n    var res2 = test_aes(testvectors128[i][1],testvectors128[i][0],testvectors128[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing 128 bit key vector with block \"+\n        util.hexidump(testvectors128[i][1])+\n        \" and key \"+util.hexidump(testvectors128[i][0])+\n        \" should be \"+util.hexidump(testvectors128[i][2]),\n        false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"128 bit key test vectors completed.\",true)\n    j++;\n  }\n  \n  res = true;\n  for (var i = 0; i < testvectors192.length; i++) {\n    var res2 = test_aes(testvectors192[i][1],testvectors192[i][0],testvectors192[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing 192 bit key vector with block \"+\n        util.hexidump(testvectors192[i][1])+\n        \" and key \"+util.hexidump(testvectors192[i][0])+\n        \" should be \"+util.hexidump(testvectors192[i][2]),\n        false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"192 bit key test vectors completed.\",true)\n    j++;\n  }\n\n  res = true;\n  for (var i = 0; i < testvectors256.length; i++) {\n    var res2 = test_aes(testvectors256[i][1],testvectors256[i][0],testvectors256[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing 256 bit key vector with block \"+\n        util.hexidump(testvectors256[i][1])+\n        \" and key \"+util.hexidump(testvectors256[i][0])+\n        \" should be \"+util.hexidump(testvectors256[i][2]),\n        false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"256 bit key test vectors completed.\", true)\n    j++;\n  }\n\n  return result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"Blowfish cipher test with test vectors from http://www.schneier.com/code/vectors.txt\", function() {\n  var openpgp = require('../../../'),\n    util = openpgp.util,\n    BFencrypt = openpgp.crypto.cipher.blowfish;\n\n  var result = [];\n  function test_bf(input, key, output) {\n    var blowfish = new openpgp.crypto.cipher.blowfish(util.bin2str(key));\n    var result = util.bin2str(blowfish.encrypt(input));\n\n    return (util.hexstrdump(result) == util.hexstrdump(util.bin2str(output)));\n  }\n  var testvectors = [[[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4E,0xF9,0x97,0x45,0x61,0x98,0xDD,0x78]],\n                     [[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x51,0x86,0x6F,0xD5,0xB8,0x5E,0xCB,0x8A]],\n                     [[0x30,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x01],[0x7D,0x85,0x6F,0x9A,0x61,0x30,0x63,0xF2]],\n                     [[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x24,0x66,0xDD,0x87,0x8B,0x96,0x3C,0x9D]],\n                     [[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x61,0xF9,0xC3,0x80,0x22,0x81,0xB0,0x96]],\n                     [[0x11,0x11,0x11,0x11,0x11,0x11,0x11,0x11],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x7D,0x0C,0xC6,0x30,0xAF,0xDA,0x1E,0xC7]],\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4E,0xF9,0x97,0x45,0x61,0x98,0xDD,0x78]],\n                     [[0xFE,0xDC,0xBA,0x98,0x76,0x54,0x32,0x10],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x0A,0xCE,0xAB,0x0F,0xC6,0xA0,0xA2,0x8D]],\n                     [[0x7C,0xA1,0x10,0x45,0x4A,0x1A,0x6E,0x57],[0x01,0xA1,0xD6,0xD0,0x39,0x77,0x67,0x42],[0x59,0xC6,0x82,0x45,0xEB,0x05,0x28,0x2B]],\n                     [[0x01,0x31,0xD9,0x61,0x9D,0xC1,0x37,0x6E],[0x5C,0xD5,0x4C,0xA8,0x3D,0xEF,0x57,0xDA],[0xB1,0xB8,0xCC,0x0B,0x25,0x0F,0x09,0xA0]],\n                     [[0x07,0xA1,0x13,0x3E,0x4A,0x0B,0x26,0x86],[0x02,0x48,0xD4,0x38,0x06,0xF6,0x71,0x72],[0x17,0x30,0xE5,0x77,0x8B,0xEA,0x1D,0xA4]],\n                     [[0x38,0x49,0x67,0x4C,0x26,0x02,0x31,0x9E],[0x51,0x45,0x4B,0x58,0x2D,0xDF,0x44,0x0A],[0xA2,0x5E,0x78,0x56,0xCF,0x26,0x51,0xEB]],\n                     [[0x04,0xB9,0x15,0xBA,0x43,0xFE,0xB5,0xB6],[0x42,0xFD,0x44,0x30,0x59,0x57,0x7F,0xA2],[0x35,0x38,0x82,0xB1,0x09,0xCE,0x8F,0x1A]],\n                     [[0x01,0x13,0xB9,0x70,0xFD,0x34,0xF2,0xCE],[0x05,0x9B,0x5E,0x08,0x51,0xCF,0x14,0x3A],[0x48,0xF4,0xD0,0x88,0x4C,0x37,0x99,0x18]],\n                     [[0x01,0x70,0xF1,0x75,0x46,0x8F,0xB5,0xE6],[0x07,0x56,0xD8,0xE0,0x77,0x47,0x61,0xD2],[0x43,0x21,0x93,0xB7,0x89,0x51,0xFC,0x98]],\n                     [[0x43,0x29,0x7F,0xAD,0x38,0xE3,0x73,0xFE],[0x76,0x25,0x14,0xB8,0x29,0xBF,0x48,0x6A],[0x13,0xF0,0x41,0x54,0xD6,0x9D,0x1A,0xE5]],\n                     [[0x07,0xA7,0x13,0x70,0x45,0xDA,0x2A,0x16],[0x3B,0xDD,0x11,0x90,0x49,0x37,0x28,0x02],[0x2E,0xED,0xDA,0x93,0xFF,0xD3,0x9C,0x79]],\n                     [[0x04,0x68,0x91,0x04,0xC2,0xFD,0x3B,0x2F],[0x26,0x95,0x5F,0x68,0x35,0xAF,0x60,0x9A],[0xD8,0x87,0xE0,0x39,0x3C,0x2D,0xA6,0xE3]],\n                     [[0x37,0xD0,0x6B,0xB5,0x16,0xCB,0x75,0x46],[0x16,0x4D,0x5E,0x40,0x4F,0x27,0x52,0x32],[0x5F,0x99,0xD0,0x4F,0x5B,0x16,0x39,0x69]],\n                     [[0x1F,0x08,0x26,0x0D,0x1A,0xC2,0x46,0x5E],[0x6B,0x05,0x6E,0x18,0x75,0x9F,0x5C,0xCA],[0x4A,0x05,0x7A,0x3B,0x24,0xD3,0x97,0x7B]],\n                     [[0x58,0x40,0x23,0x64,0x1A,0xBA,0x61,0x76],[0x00,0x4B,0xD6,0xEF,0x09,0x17,0x60,0x62],[0x45,0x20,0x31,0xC1,0xE4,0xFA,0xDA,0x8E]],\n                     [[0x02,0x58,0x16,0x16,0x46,0x29,0xB0,0x07],[0x48,0x0D,0x39,0x00,0x6E,0xE7,0x62,0xF2],[0x75,0x55,0xAE,0x39,0xF5,0x9B,0x87,0xBD]],\n                     [[0x49,0x79,0x3E,0xBC,0x79,0xB3,0x25,0x8F],[0x43,0x75,0x40,0xC8,0x69,0x8F,0x3C,0xFA],[0x53,0xC5,0x5F,0x9C,0xB4,0x9F,0xC0,0x19]],\n                     [[0x4F,0xB0,0x5E,0x15,0x15,0xAB,0x73,0xA7],[0x07,0x2D,0x43,0xA0,0x77,0x07,0x52,0x92],[0x7A,0x8E,0x7B,0xFA,0x93,0x7E,0x89,0xA3]],\n                     [[0x49,0xE9,0x5D,0x6D,0x4C,0xA2,0x29,0xBF],[0x02,0xFE,0x55,0x77,0x81,0x17,0xF1,0x2A],[0xCF,0x9C,0x5D,0x7A,0x49,0x86,0xAD,0xB5]],\n                     [[0x01,0x83,0x10,0xDC,0x40,0x9B,0x26,0xD6],[0x1D,0x9D,0x5C,0x50,0x18,0xF7,0x28,0xC2],[0xD1,0xAB,0xB2,0x90,0x65,0x8B,0xC7,0x78]],\n                     [[0x1C,0x58,0x7F,0x1C,0x13,0x92,0x4F,0xEF],[0x30,0x55,0x32,0x28,0x6D,0x6F,0x29,0x5A],[0x55,0xCB,0x37,0x74,0xD1,0x3E,0xF2,0x01]],\n                     [[0x01,0x01,0x01,0x01,0x01,0x01,0x01,0x01],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0xFA,0x34,0xEC,0x48,0x47,0xB2,0x68,0xB2]],\n                     [[0x1F,0x1F,0x1F,0x1F,0x0E,0x0E,0x0E,0x0E],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0xA7,0x90,0x79,0x51,0x08,0xEA,0x3C,0xAE]],\n                     [[0xE0,0xFE,0xE0,0xFE,0xF1,0xFE,0xF1,0xFE],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0xC3,0x9E,0x07,0x2D,0x9F,0xAC,0x63,0x1D]],\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x01,0x49,0x33,0xE0,0xCD,0xAF,0xF6,0xE4]],\n                     [[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xF2,0x1E,0x9A,0x77,0xB7,0x1C,0x49,0xBC]],\n                     [[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x24,0x59,0x46,0x88,0x57,0x54,0x36,0x9A]],\n                     [[0xFE,0xDC,0xBA,0x98,0x76,0x54,0x32,0x10],[0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF,0xFF],[0x6B,0x5C,0x5A,0x9C,0x5D,0x9E,0x0A,0x5A]]];\n\n  var res = true;\n  var j = 0;\n  for (var i = 0; i < testvectors.length; i++) {\n    var res2 = test_bf(testvectors[i][1],testvectors[i][0],testvectors[i][2]);\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing vector \"+i+\" with block \"+\n        util.hexidump(testvectors[i][0])+\n        \" and key \"+util.hexidump(testvectors[i][1])+\n        \" should be \"+util.hexidump(testvectors[i][2]), false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"34 test vectors completed \", true);\n  }\n  return result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"CAST-128 cipher test with test vectors from RFC2144\", function() {\n\tvar openpgp = require('../../../'),\n\t\tutil = openpgp.util;\n\n\tvar result = [];\n\tfunction test_cast(input, key, output) {\n\t\tvar cast5 = new openpgp.crypto.cipher.cast5(util.bin2str(key));\n\t\tvar result = util.bin2str(cast5.encrypt(input));\n\n\t\treturn util.hexstrdump(result) == util.hexstrdump(util.bin2str(output));\n\t}\n\t\n\tvar testvectors = [[[0x01,0x23,0x45,0x67,0x12,0x34,0x56,0x78,0x23,0x45,0x67,0x89,0x34,0x56,0x78,0x9A],[0x01,0x23,0x45,0x67,0x89,0xAB,0xCD,0xEF],[0x23,0x8B,0x4F,0xE5,0x84,0x7E,0x44,0xB2]]];\n\n\tfor (var i = 0; i < testvectors.length; i++) {\n\t\tresult[i] = new unit.result(\"Testing vector with block \"+\n\t\t\t\tutil.hexidump(testvectors[i][0])+\n\t\t\t\t\" and key \"+util.hexidump(testvectors[i][1])+\n\t\t\t\t\" should be \"+util.hexidump(testvectors[i][2]),\n\t\t\ttest_cast(testvectors[i][1],testvectors[i][0],testvectors[i][2]));\n\t}\n\treturn result;\n});\n","var unit = require('../../unit.js');\r\n\r\nunit.register(\"TripleDES (EDE) cipher test with test vectors from http://csrc.nist.gov/publications/nistpubs/800-20/800-20.pdf\", function() {\r\n  var openpgp = require('../../../'),\r\n    util = openpgp.util;\r\n\r\n  var result = [];\r\n  var key = util.bin2str([1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1]);\r\n  var testvectors = [[[0x80,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x95,0xF8,0xA5,0xE5,0xDD,0x31,0xD9,0x00]],\r\n                     [[0x40,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0xDD,0x7F,0x12,0x1C,0xA5,0x01,0x56,0x19]],\r\n                     [[0x20,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x2E,0x86,0x53,0x10,0x4F,0x38,0x34,0xEA]],\r\n                     [[0x10,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x4B,0xD3,0x88,0xFF,0x6C,0xD8,0x1D,0x4F]],\r\n                     [[0x08,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x20,0xB9,0xE7,0x67,0xB2,0xFB,0x14,0x56]],\r\n                     [[0x04,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x55,0x57,0x93,0x80,0xD7,0x71,0x38,0xEF]],\r\n                     [[0x02,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x6C,0xC5,0xDE,0xFA,0xAF,0x04,0x51,0x2F]],\r\n                     [[0x01,0x00,0x00,0x00,0x00,0x00,0x00,0x00],[0x0D,0x9F,0x27,0x9B,0xA5,0xD8,0x72,0x60]],\r\n                     [[0x00,0x80,0x00,0x00,0x00,0x00,0x00,0x00],[0xD9,0x03,0x1B,0x02,0x71,0xBD,0x5A,0x0A]],\r\n                     [[0x00,0x40,0x00,0x00,0x00,0x00,0x00,0x00],[0x42,0x42,0x50,0xB3,0x7C,0x3D,0xD9,0x51]],\r\n                     [[0x00,0x20,0x00,0x00,0x00,0x00,0x00,0x00],[0xB8,0x06,0x1B,0x7E,0xCD,0x9A,0x21,0xE5]],\r\n                     [[0x00,0x10,0x00,0x00,0x00,0x00,0x00,0x00],[0xF1,0x5D,0x0F,0x28,0x6B,0x65,0xBD,0x28]],\r\n                     [[0x00,0x08,0x00,0x00,0x00,0x00,0x00,0x00],[0xAD,0xD0,0xCC,0x8D,0x6E,0x5D,0xEB,0xA1]],\r\n                     [[0x00,0x04,0x00,0x00,0x00,0x00,0x00,0x00],[0xE6,0xD5,0xF8,0x27,0x52,0xAD,0x63,0xD1]],\r\n                     [[0x00,0x02,0x00,0x00,0x00,0x00,0x00,0x00],[0xEC,0xBF,0xE3,0xBD,0x3F,0x59,0x1A,0x5E]],\r\n                     [[0x00,0x01,0x00,0x00,0x00,0x00,0x00,0x00],[0xF3,0x56,0x83,0x43,0x79,0xD1,0x65,0xCD]],\r\n                     [[0x00,0x00,0x80,0x00,0x00,0x00,0x00,0x00],[0x2B,0x9F,0x98,0x2F,0x20,0x03,0x7F,0xA9]],\r\n                     [[0x00,0x00,0x40,0x00,0x00,0x00,0x00,0x00],[0x88,0x9D,0xE0,0x68,0xA1,0x6F,0x0B,0xE6]],\r\n                     [[0x00,0x00,0x20,0x00,0x00,0x00,0x00,0x00],[0xE1,0x9E,0x27,0x5D,0x84,0x6A,0x12,0x98]],\r\n                     [[0x00,0x00,0x10,0x00,0x00,0x00,0x00,0x00],[0x32,0x9A,0x8E,0xD5,0x23,0xD7,0x1A,0xEC]],\r\n                     [[0x00,0x00,0x08,0x00,0x00,0x00,0x00,0x00],[0xE7,0xFC,0xE2,0x25,0x57,0xD2,0x3C,0x97]],\r\n                     [[0x00,0x00,0x04,0x00,0x00,0x00,0x00,0x00],[0x12,0xA9,0xF5,0x81,0x7F,0xF2,0xD6,0x5D]],\r\n                     [[0x00,0x00,0x02,0x00,0x00,0x00,0x00,0x00],[0xA4,0x84,0xC3,0xAD,0x38,0xDC,0x9C,0x19]],\r\n                     [[0x00,0x00,0x01,0x00,0x00,0x00,0x00,0x00],[0xFB,0xE0,0x0A,0x8A,0x1E,0xF8,0xAD,0x72]],\r\n                     [[0x00,0x00,0x00,0x80,0x00,0x00,0x00,0x00],[0x75,0x0D,0x07,0x94,0x07,0x52,0x13,0x63]],\r\n                     [[0x00,0x00,0x00,0x40,0x00,0x00,0x00,0x00],[0x64,0xFE,0xED,0x9C,0x72,0x4C,0x2F,0xAF]],\r\n                     [[0x00,0x00,0x00,0x20,0x00,0x00,0x00,0x00],[0xF0,0x2B,0x26,0x3B,0x32,0x8E,0x2B,0x60]],\r\n                     [[0x00,0x00,0x00,0x10,0x00,0x00,0x00,0x00],[0x9D,0x64,0x55,0x5A,0x9A,0x10,0xB8,0x52]],\r\n                     [[0x00,0x00,0x00,0x08,0x00,0x00,0x00,0x00],[0xD1,0x06,0xFF,0x0B,0xED,0x52,0x55,0xD7]],\r\n                     [[0x00,0x00,0x00,0x04,0x00,0x00,0x00,0x00],[0xE1,0x65,0x2C,0x6B,0x13,0x8C,0x64,0xA5]],\r\n                     [[0x00,0x00,0x00,0x02,0x00,0x00,0x00,0x00],[0xE4,0x28,0x58,0x11,0x86,0xEC,0x8F,0x46]],\r\n                     [[0x00,0x00,0x00,0x01,0x00,0x00,0x00,0x00],[0xAE,0xB5,0xF5,0xED,0xE2,0x2D,0x1A,0x36]],\r\n                     [[0x00,0x00,0x00,0x00,0x80,0x00,0x00,0x00],[0xE9,0x43,0xD7,0x56,0x8A,0xEC,0x0C,0x5C]],\r\n                     [[0x00,0x00,0x00,0x00,0x40,0x00,0x00,0x00],[0xDF,0x98,0xC8,0x27,0x6F,0x54,0xB0,0x4B]],\r\n                     [[0x00,0x00,0x00,0x00,0x20,0x00,0x00,0x00],[0xB1,0x60,0xE4,0x68,0x0F,0x6C,0x69,0x6F]],\r\n                     [[0x00,0x00,0x00,0x00,0x10,0x00,0x00,0x00],[0xFA,0x07,0x52,0xB0,0x7D,0x9C,0x4A,0xB8]],\r\n                     [[0x00,0x00,0x00,0x00,0x08,0x00,0x00,0x00],[0xCA,0x3A,0x2B,0x03,0x6D,0xBC,0x85,0x02]],\r\n                     [[0x00,0x00,0x00,0x00,0x04,0x00,0x00,0x00],[0x5E,0x09,0x05,0x51,0x7B,0xB5,0x9B,0xCF]],\r\n                     [[0x00,0x00,0x00,0x00,0x02,0x00,0x00,0x00],[0x81,0x4E,0xEB,0x3B,0x91,0xD9,0x07,0x26]],\r\n                     [[0x00,0x00,0x00,0x00,0x01,0x00,0x00,0x00],[0x4D,0x49,0xDB,0x15,0x32,0x91,0x9C,0x9F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x80,0x00,0x00],[0x25,0xEB,0x5F,0xC3,0xF8,0xCF,0x06,0x21]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x40,0x00,0x00],[0xAB,0x6A,0x20,0xC0,0x62,0x0D,0x1C,0x6F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x20,0x00,0x00],[0x79,0xE9,0x0D,0xBC,0x98,0xF9,0x2C,0xCA]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x10,0x00,0x00],[0x86,0x6E,0xCE,0xDD,0x80,0x72,0xBB,0x0E]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x08,0x00,0x00],[0x8B,0x54,0x53,0x6F,0x2F,0x3E,0x64,0xA8]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x04,0x00,0x00],[0xEA,0x51,0xD3,0x97,0x55,0x95,0xB8,0x6B]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x02,0x00,0x00],[0xCA,0xFF,0xC6,0xAC,0x45,0x42,0xDE,0x31]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x01,0x00,0x00],[0x8D,0xD4,0x5A,0x2D,0xDF,0x90,0x79,0x6C]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x80,0x00],[0x10,0x29,0xD5,0x5E,0x88,0x0E,0xC2,0xD0]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x40,0x00],[0x5D,0x86,0xCB,0x23,0x63,0x9D,0xBE,0xA9]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x20,0x00],[0x1D,0x1C,0xA8,0x53,0xAE,0x7C,0x0C,0x5F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x10,0x00],[0xCE,0x33,0x23,0x29,0x24,0x8F,0x32,0x28]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x08,0x00],[0x84,0x05,0xD1,0xAB,0xE2,0x4F,0xB9,0x42]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x04,0x00],[0xE6,0x43,0xD7,0x80,0x90,0xCA,0x42,0x07]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x02,0x00],[0x48,0x22,0x1B,0x99,0x37,0x74,0x8A,0x23]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x01,0x00],[0xDD,0x7C,0x0B,0xBD,0x61,0xFA,0xFD,0x54]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x80],[0x2F,0xBC,0x29,0x1A,0x57,0x0D,0xB5,0xC4]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x40],[0xE0,0x7C,0x30,0xD7,0xE4,0xE2,0x6E,0x12]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x20],[0x09,0x53,0xE2,0x25,0x8E,0x8E,0x90,0xA1]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x10],[0x5B,0x71,0x1B,0xC4,0xCE,0xEB,0xF2,0xEE]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x08],[0xCC,0x08,0x3F,0x1E,0x6D,0x9E,0x85,0xF6]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x04],[0xD2,0xFD,0x88,0x67,0xD5,0x0D,0x2D,0xFE]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x02],[0x06,0xE7,0xEA,0x22,0xCE,0x92,0x70,0x8F]],\r\n                     [[0x00,0x00,0x00,0x00,0x00,0x00,0x00,0x01],[0x16,0x6B,0x40,0xB4,0x4A,0xBA,0x4B,0xD6]]];\r\n\r\n  var res = true;\r\n  var j = 0;\r\n  for (var i = 0; i < testvectors.length; i++) {\r\n    var des = new openpgp.crypto.cipher.des(key);\r\n\r\n    var encr = util.bin2str(des.encrypt(testvectors[i][0], key));\r\n    var res2 = encr == util.bin2str(testvectors[i][1]);\r\n\r\n    res &= res2;\r\n\r\n    if (!res2) {\r\n    result[j] = new unit.result(\"Testing vector with block \" +\r\n        util.hexidump(testvectors[i][0]) +\r\n        \" and key \" + util.hexstrdump(key) +\r\n        \" should be \" + util.hexidump(testvectors[i][1]) + \" != \" +\r\n        util.hexidump(encr),\r\n      false);\r\n    j++;\r\n    }\r\n  }\r\n  if (res) {\r\n    result[j] = new unit.result(\"All 3DES EDE test vectors completed\", true);\r\n  }\r\n  return result;\r\n});\r\n\r\n\r\nunit.register(\"DES encrypt/decrypt padding tests\", function () {\r\n    var openpgp = require('../../../'),\r\n      util = openpgp.util;\r\n\r\n    var result = [];\r\n    var key = util.bin2str([0x01, 0x23, 0x45, 0x67, 0x89, 0xAB, 0xCD, 0xEF]);\r\n    var testvectors = new Array();\r\n    testvectors[0] = [[[0x01], [0x24, 0xC7, 0x4A, 0x9A, 0x79, 0x75, 0x4B, 0xC7]],\r\n\t                  [[0x02, 0x03], [0xA7, 0x7A, 0x9A, 0x59, 0x8A, 0x86, 0x85, 0xC5]],\r\n\t                  [[0x03, 0x04, 0x05], [0x01, 0xCF, 0xEB, 0x6A, 0x74, 0x60, 0xF5, 0x02]],\r\n\t                  [[0x04, 0x05, 0x06, 0x07], [0xA8, 0xF0, 0x3D, 0x59, 0xBA, 0x6B, 0x0E, 0x76]],\r\n\t                  [[0x05, 0x06, 0x07, 0x08, 0x09], [0x86, 0x40, 0x33, 0x61, 0x3F, 0x55, 0x73, 0x49]],\r\n\t                  [[0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B], [0x13, 0x21, 0x3E, 0x0E, 0xCE, 0x2C, 0x94, 0x01]],\r\n\t                  [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0x30, 0x49, 0x97, 0xC1, 0xDA, 0xD5, 0x59, 0xA5]],\r\n\t                  [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9]]];\r\n    testvectors[1] = [[[0x01], [0xF2, 0xAB, 0x1C, 0x9E, 0x70, 0x7D, 0xCC, 0x92]],\r\n\t                  [[0x02, 0x03], [0x6B, 0x4C, 0x67, 0x24, 0x9F, 0xB7, 0x4D, 0xAC]],\r\n\t                  [[0x03, 0x04, 0x05], [0x68, 0x95, 0xAB, 0xA8, 0xEA, 0x53, 0x13, 0x23]],\r\n\t                  [[0x04, 0x05, 0x06, 0x07], [0xC8, 0xDE, 0x60, 0x8F, 0xF6, 0x09, 0x90, 0xB5]],\r\n\t                  [[0x05, 0x06, 0x07, 0x08, 0x09], [0x19, 0x13, 0x50, 0x20, 0x70, 0x40, 0x2E, 0x09]],\r\n\t                  [[0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B], [0xA8, 0x23, 0x40, 0xC6, 0x17, 0xA6, 0x31, 0x4A]],\r\n\t                  [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0x36, 0x62, 0xF2, 0x99, 0x68, 0xD4, 0xBF, 0x7C]],\r\n\t                  [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9, 0x08, 0x6F, 0x9A, 0x1D, 0x74, 0xC9, 0x4D, 0x4E]]];\r\n    testvectors[2] = [[[0x01], [0x83, 0x68, 0xE4, 0x9C, 0x84, 0xCC, 0xCB, 0xF0]],\r\n\t                  [[0x02, 0x03], [0xBB, 0xA8, 0x0B, 0x66, 0x1B, 0x62, 0xC4, 0xC8]],\r\n\t                  [[0x03, 0x04, 0x05], [0x9A, 0xD7, 0x5A, 0x24, 0xFD, 0x3F, 0xBF, 0x22]],\r\n\t                  [[0x04, 0x05, 0x06, 0x07], [0x14, 0x4E, 0x68, 0x6D, 0x2E, 0xC1, 0xB7, 0x52]],\r\n\t                  [[0x05, 0x06, 0x07, 0x08, 0x09], [0x12, 0x0A, 0x51, 0x08, 0xF9, 0xA3, 0x03, 0x74]],\r\n\t                  [[0x06, 0x07, 0x08, 0x09, 0x0A, 0x0B], [0xB2, 0x07, 0xD1, 0x05, 0xF6, 0x67, 0xAF, 0xBA]],\r\n\t                  [[0x07, 0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D], [0xCA, 0x59, 0x61, 0x3A, 0x83, 0x23, 0x26, 0xDD]],\r\n\t                  [[0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F], [0x83, 0x25, 0x79, 0x06, 0x54, 0xA4, 0x44, 0xD9]]];\r\n\r\n    var des = new openpgp.crypto.cipher.originalDes(key);\r\n\r\n    var res = true;\r\n    var j = 0;\r\n\r\n    for (var padding = 0; padding < 3; padding++) {\r\n        var thisVectorSet = testvectors[padding];\r\n\r\n        for (var i = 0; i < thisVectorSet.length; i++) {\r\n            var encrypted = des.encrypt(thisVectorSet[i][0], padding);\r\n            var decrypted = des.decrypt(encrypted, padding);\r\n\r\n            var res2 = (util.bin2str(encrypted) == util.bin2str(thisVectorSet[i][1]));\r\n            var res3 = (util.bin2str(decrypted) == util.bin2str(thisVectorSet[i][0]));\r\n            res &= res2;\r\n            res &= res3;\r\n            if (!res2 || !res3) {\r\n                result[j] = new unit.result(\r\n                    \"Testing vector with block [\" +\r\n                        util.hexidump(thisVectorSet[i][0]) +\r\n                        \"] and key [\" + util.hexstrdump(key) +\r\n                        \"] and padding [\" + padding +\r\n                        \"] should be \" + util.hexidump(thisVectorSet[i][1]) + \" - Actually [ENC:\" + util.hexidump(encrypted) + \", DEC:\" + util.hexidump(decrypted) + \"]\",\r\n                    false);\r\n                j++;\r\n            }\r\n        }\r\n    }\r\n    if (res) {\r\n        result[j] = new unit.result(\"All DES test vectors completed\", true);\r\n    }\r\n    return result;\r\n});\r\n","var unit = require('../../unit.js');\n\nunit.register(\"Twofish test with test vectors from http://www.schneier.com/code/ecb_ival.txt\", function() {\n  var openpgp = require('../../../'),\n    util = openpgp.util;\n\n  function TFencrypt(block, key) {\n    var tf = new openpgp.crypto.cipher.twofish(key);\n\n    return tf.encrypt(block);\n  }\n\n\n  var result = [];\n  var start = [0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];\n  var start_short = [0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];\n  var testvectors = [[0x57,0xFF,0x73,0x9D,0x4D,0xC9,0x2C,0x1B,0xD7,0xFC,0x01,0x70,0x0C,0xC8,0x21,0x6F],\n            [0xD4,0x3B,0xB7,0x55,0x6E,0xA3,0x2E,0x46,0xF2,0xA2,0x82,0xB7,0xD4,0x5B,0x4E,0x0D],\n            [0x90,0xAF,0xE9,0x1B,0xB2,0x88,0x54,0x4F,0x2C,0x32,0xDC,0x23,0x9B,0x26,0x35,0xE6],\n            [0x6C,0xB4,0x56,0x1C,0x40,0xBF,0x0A,0x97,0x05,0x93,0x1C,0xB6,0xD4,0x08,0xE7,0xFA],\n            [0x30,0x59,0xD6,0xD6,0x17,0x53,0xB9,0x58,0xD9,0x2F,0x47,0x81,0xC8,0x64,0x0E,0x58],\n            [0xE6,0x94,0x65,0x77,0x05,0x05,0xD7,0xF8,0x0E,0xF6,0x8C,0xA3,0x8A,0xB3,0xA3,0xD6],\n            [0x5A,0xB6,0x7A,0x5F,0x85,0x39,0xA4,0xA5,0xFD,0x9F,0x03,0x73,0xBA,0x46,0x34,0x66],\n            [0xDC,0x09,0x6B,0xCD,0x99,0xFC,0x72,0xF7,0x99,0x36,0xD4,0xC7,0x48,0xE7,0x5A,0xF7],\n            [0xC5,0xA3,0xE7,0xCE,0xE0,0xF1,0xB7,0x26,0x05,0x28,0xA6,0x8F,0xB4,0xEA,0x05,0xF2],\n            [0x43,0xD5,0xCE,0xC3,0x27,0xB2,0x4A,0xB9,0x0A,0xD3,0x4A,0x79,0xD0,0x46,0x91,0x51]];\n  testvectors[47] =  [0x43,0x10,0x58,0xF4,0xDB,0xC7,0xF7,0x34,0xDA,0x4F,0x02,0xF0,0x4C,0xC4,0xF4,0x59];\n  testvectors[48] =  [0x37,0xFE,0x26,0xFF,0x1C,0xF6,0x61,0x75,0xF5,0xDD,0xF4,0xC3,0x3B,0x97,0xA2,0x05];\n  var res = true;\n  var j = 0;\n  for (var i = 0; i < 49; i++) {\n    var res2 = false;\n    var blk, key, ct;\n    if (i === 0) {\n      blk = start_short;\n      key = util.bin2str(start);\n      ct = testvectors[0];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else if (i === 1) {\n      blk = testvectors[0];\n      key = util.bin2str(start);\n      ct = testvectors[1];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else if (i === 2) {\n      blk = testvectors[i-1];\n      key = util.bin2str(testvectors[i-2].concat(start_short));\n      ct = testvectors[i];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else if (i < 10 || i > 46) {\n      blk = testvectors[i-1];\n      key = util.bin2str(testvectors[i-2].concat(testvectors[i-3]));\n      ct = testvectors[i];\n      res2 = (util.bin2str(TFencrypt(blk,key)) == util.bin2str(ct));\n    } else {\n      testvectors[i] = TFencrypt(testvectors[i-1],util.bin2str(testvectors[i-2].concat(testvectors[i-3])));\n      res2 = true;\n    }\n    res &= res2;\n    if (!res2) {\n      result[j] = new unit.result(\"Testing vector with block \"+util.hexidump(blk)+\" with key \"+ util.hexstrdump(key) +\" should be \"+util.hexidump(ct)+\" but is \"+util.hexidump(TFencrypt(blk,key)), false);\n      j++;\n    }\n  }\n  if (res) {\n    result[j] = new unit.result(\"49 test vectors completed\", true);\n  }\n  return result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"MD5 test with test vectors from RFC 1321\", function() {\n\tvar openpgp = require('../../../'),\n\t\tutil = openpgp.util,\n\t\tMD5 = openpgp.crypto.hash.md5;\n\n\tvar result = new Array();\n\tresult[0] = new unit.result(\"MD5 (\\\"\\\") = d41d8cd98f00b204e9800998ecf8427e\",\n\t\t\tutil.hexstrdump(MD5(\"\")) == \"d41d8cd98f00b204e9800998ecf8427e\");\n\tresult[1] = new unit.result(\"MD5 (\\\"a\\\") = 0cc175b9c0f1b6a831c399e269772661\",\n\t\t\tutil.hexstrdump(MD5 (\"abc\")) == \"900150983cd24fb0d6963f7d28e17f72\");\n\tresult[2] = new unit.result(\"MD5 (\\\"message digest\\\") = f96b697d7cb7938d525a2f31aaf161d0\",\n\t\t\tutil.hexstrdump(MD5 (\"message digest\")) == \"f96b697d7cb7938d525a2f31aaf161d0\");\n\tresult[3] = new unit.result(\"MD5 (\\\"abcdefghijklmnopqrstuvwxyz\\\") = c3fcd3d76192e4007dfb496cca67e13b\",\n\t\t\tutil.hexstrdump(MD5 (\"abcdefghijklmnopqrstuvwxyz\")) == \"c3fcd3d76192e4007dfb496cca67e13b\");\n\tresult[4] = new unit.result(\"MD5 (\\\"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\\\") = d174ab98d277d9f5a5611c2c9f419d9f\",\n\t\t\tutil.hexstrdump(MD5 (\"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789\")) == \"d174ab98d277d9f5a5611c2c9f419d9f\");\n\tresult[5] = new unit.result(\"MD5 (\\\"12345678901234567890123456789012345678901234567890123456789012345678901234567890\\\") = 57edf4a22be3c955ac49da2e2107b67a\",\n\t\t\tutil.hexstrdump(MD5 (\"12345678901234567890123456789012345678901234567890123456789012345678901234567890\")) == \"57edf4a22be3c955ac49da2e2107b67a\");\n\treturn result;\n});\n","var unit = require('../../unit.js');\n\nunit.register(\"RIPE-MD 160 bits test with test vectors from http://homes.esat.kuleuven.be/~bosselae/ripemd160.html\", function() {\n\n\tvar openpgp = require('../../../'),\n\t\tutil = openpgp.util,\n\t\tRMDstring = openpgp.crypto.hash.ripemd;\n\n\tvar result = new Array();\n\tresult[0] = new unit.result(\"RMDstring (\\\"\\\") = 9c1185a5c5e9fc54612808977ee8f548b2258d31\",\n\t\t\tutil.hexstrdump(RMDstring(\"\")) == \"9c1185a5c5e9fc54612808977ee8f548b2258d31\");\n\tresult[1] = new unit.result(\"RMDstring (\\\"a\\\") = 0bdc9d2d256b3ee9daae347be6f4dc835a467ffe\",\n\t\t\tutil.hexstrdump(RMDstring(\"a\")) == \"0bdc9d2d256b3ee9daae347be6f4dc835a467ffe\");\n\tresult[2] = new unit.result(\"RMDstring (\\\"abc\\\") = 8eb208f7e05d987a9b044a8e98c6b087f15a0bfc\",\n\t\t\tutil.hexstrdump(RMDstring(\"abc\")) == \"8eb208f7e05d987a9b044a8e98c6b087f15a0bfc\");\n\tresult[3] = new unit.result(\"RMDstring (\\\"message digest\\\") = 5d0689ef49d2fae572b881b123a85ffa21595f36\",\n\t\t\tutil.hexstrdump(RMDstring(\"message digest\")) == \"5d0689ef49d2fae572b881b123a85ffa21595f36\");\n\treturn result;\n});\n","var unit = require('../../unit.js');\n\n\nunit.register(\"SHA* test with test vectors from NIST FIPS 180-2\", function() {\n\tvar openpgp = require('../../../'),\n\t\tutil = openpgp.util,\n\t\thash = openpgp.crypto.hash;\n\n\tvar result = new Array();\n\t\n\tresult[0] = new unit.result(\"SHA1 - a9993e364706816aba3e25717850c26c9cd0d89d = hash.sha1(\\\"abc\\\") \",\n\t\t\t\"a9993e364706816aba3e25717850c26c9cd0d89d\" == util.hexstrdump(hash.sha1(\"abc\")));\n\tresult[1] = new unit.result(\"SHA1 - 84983e441c3bd26ebaae4aa1f95129e5e54670f1 = hash.sha1(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"84983e441c3bd26ebaae4aa1f95129e5e54670f1\" == util.hexstrdump(hash.sha1(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\n\tresult[2] = new unit.result(\"SHA224 - 23097d223405d8228642a477bda255b32aadbce4bda0b3f7e36c9da7 = hash.sha224(\\\"abc\\\") \",\n\t\t\t\"23097d223405d8228642a477bda255b32aadbce4bda0b3f7e36c9da7\" == util.hexstrdump(hash.sha224(\"abc\")));\n\tresult[3] = new unit.result(\"SHA224 - 75388b16512776cc5dba5da1fd890150b0c6455cb4f58b1952522525 = hash.sha224(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"75388b16512776cc5dba5da1fd890150b0c6455cb4f58b1952522525\" == util.hexstrdump(hash.sha224(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\n\tresult[4] = new unit.result(\"SHA256 - ba7816bf8f01cfea414140de5dae2223b00361a396177a9cb410ff61f20015ad = hash.sha256(\\\"abc\\\") \",\n\t\t\t\"ba7816bf8f01cfea414140de5dae2223b00361a396177a9cb410ff61f20015ad\" == util.hexstrdump(hash.sha256(\"abc\")));\n\tresult[5] = new unit.result(\"SHA256 - 248d6a61d20638b8e5c026930c3e6039a33ce45964ff2167f6ecedd419db06c1 = hash.sha256(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"248d6a61d20638b8e5c026930c3e6039a33ce45964ff2167f6ecedd419db06c1\" == util.hexstrdump(hash.sha256(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\n\tresult[6] = new unit.result(\"SHA384 - cb00753f45a35e8bb5a03d699ac65007272c32ab0eded1631a8b605a43ff5bed8086072ba1e7cc2358baeca134c825a7 = hash.sha384(\\\"abc\\\") \",\n\t\t\t\"cb00753f45a35e8bb5a03d699ac65007272c32ab0eded1631a8b605a43ff5bed8086072ba1e7cc2358baeca134c825a7\" == util.hexstrdump(hash.sha384(\"abc\")));\n\tresult[7] = new unit.result(\"SHA384 - 3391fdddfc8dc7393707a65b1b4709397cf8b1d162af05abfe8f450de5f36bc6b0455a8520bc4e6f5fe95b1fe3c8452b = str384(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"3391fdddfc8dc7393707a65b1b4709397cf8b1d162af05abfe8f450de5f36bc6b0455a8520bc4e6f5fe95b1fe3c8452b\" == util.hexstrdump(hash.sha384(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\t\t\t\t\t\n\tresult[8] = new unit.result(\"SHA512 - ddaf35a193617abacc417349ae20413112e6fa4e89a97ea20a9eeee64b55d39a2192992a274fc1a836ba3c23a3feebbd454d4423643ce80e2a9ac94fa54ca49f = hash.sha512(\\\"abc\\\") \",\n\t\t\t\"ddaf35a193617abacc417349ae20413112e6fa4e89a97ea20a9eeee64b55d39a2192992a274fc1a836ba3c23a3feebbd454d4423643ce80e2a9ac94fa54ca49f\" == util.hexstrdump(hash.sha512(\"abc\")));\n\tresult[9] = new unit.result(\"SHA512 - 204a8fc6dda82f0a0ced7beb8e08a41657c16ef468b228a8279be331a703c33596fd15c13b1b07f9aa1d3bea57789ca031ad85c7a71dd70354ec631238ca3445 = hash.sha512(\\\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\\\") \",\n\t\t\t\"204a8fc6dda82f0a0ced7beb8e08a41657c16ef468b228a8279be331a703c33596fd15c13b1b07f9aa1d3bea57789ca031ad85c7a71dd70354ec631238ca3445\" == util.hexstrdump(hash.sha512(\"abcdbcdecdefdefgefghfghighijhijkijkljklmklmnlmnomnopnopq\")));\t\t\t\t\t\n\treturn result;\n});\n","var unit = require('../unit.js');\n\nunit.register(\"Functional testing of openpgp.crypto.* methods\", function() {\n\tvar openpgp = require('../../');\n  var util = openpgp.util;\n  var result = [];\n  var RSApubMPIstrs = [\n              util.bin2str([0x08,0x00,0xac,0x15,0xb3,0xd6,0xd2,0x0f,0xf0,0x7a,0xdd,0x21,0xb7,\n                            0xbf,0x61,0xfa,0xca,0x93,0x86,0xc8,0x55,0x5a,0x4b,0xa6,0xa4,0x1a,\n                            0x60,0xa2,0x3a,0x37,0x06,0x08,0xd8,0x15,0x8e,0x85,0x45,0xaa,0xb7,\n                            0x1d,0x7b,0x0b,0x73,0x94,0x55,0x0c,0x5c,0xec,0xc0,0x22,0x4b,0xa1,\n                            0x64,0x20,0x7d,0x4d,0xa8,0x96,0x1a,0x64,0x38,0x93,0xcd,0xec,0x73,\n                            0x5d,0xf9,0x89,0x88,0x24,0x3d,0x48,0xff,0x3b,0x87,0x62,0xd0,0x84,\n                            0xea,0xff,0x39,0xb5,0x27,0x70,0xea,0x4a,0xb2,0x2e,0x9d,0xf1,0x7c,\n                            0x23,0xec,0xf4,0x5e,0xea,0x61,0x3d,0xe5,0xd8,0x0d,0xf9,0x59,0x6d,\n                            0x28,0x00,0xeb,0xcb,0xc9,0x55,0x00,0x72,0x30,0x1f,0x65,0x9d,0xd6,\n                            0x17,0x58,0x5f,0xa6,0x4a,0xa0,0xdd,0xe1,0x76,0xf2,0xef,0x21,0x9f,\n                            0x84,0xfc,0xaa,0x5b,0x52,0x6e,0xc1,0xa2,0xb9,0xbd,0xb9,0xf4,0x9e,\n                            0x49,0x92,0xf2,0xaf,0x57,0x86,0xf2,0xef,0x70,0xbf,0x51,0x40,0xfd,\n                            0xbf,0x56,0x51,0xe8,0x2c,0xa2,0x4f,0xf8,0xa4,0xd7,0x36,0x18,0x85,\n                            0xce,0x09,0x0d,0xbc,0x8d,0x65,0x5e,0x8a,0x1d,0x98,0xb0,0x4d,0x9d,\n                            0xc1,0xcf,0x82,0xe1,0xb7,0x43,0x5d,0x5a,0x72,0xcd,0x55,0xd2,0xff,\n                            0xb1,0xb4,0x78,0xbf,0xa1,0x7d,0xac,0xd9,0x1b,0xc4,0xfa,0x39,0x34,\n                            0x92,0x09,0xf9,0x08,0x2a,0x6b,0x9d,0x14,0x56,0x12,0x4c,0xe9,0xa6,\n                            0x29,0xc1,0xf3,0xa9,0x0b,0xfc,0x31,0x75,0x58,0x74,0x2a,0x88,0xaf,\n                            0xee,0xc9,0xa4,0xcd,0x15,0xdc,0x1b,0x8d,0x64,0xc1,0x36,0x17,0xc4,\n                            0x8d,0x5e,0x99,0x7a,0x5b,0x9f,0x39,0xd0,0x00,0x6e,0xf9]),\n              util.bin2str([0x00,0x11,0x01,0x00,0x01])];\n  var RSAsecMPIstrs = [\n              util.bin2str([0x07,0xfe,0x23,0xff,0xce,0x45,0x6c,0x60,0x65,0x40,0x6e,0xae,0x35,\n                            0x10,0x56,0x60,0xee,0xab,0xfa,0x10,0x42,0xba,0xc7,0x04,0xaf,0x63,\n                            0xcd,0x3f,0x62,0xca,0x4b,0xfa,0xe1,0xa9,0x70,0xcd,0x34,0x8b,0xc8,\n                            0x0e,0xe4,0xc4,0xba,0x83,0x17,0x5f,0xa4,0xb8,0xea,0x60,0xc2,0x4d,\n                            0x9a,0xf2,0xa9,0x03,0xeb,0xf6,0xaa,0xc2,0xb8,0x8b,0x43,0x12,0xe9,\n                            0xf7,0x88,0xd2,0x5a,0xa6,0xaa,0x23,0x71,0x31,0x74,0xdb,0x19,0x20,\n                            0x15,0x41,0x1b,0x43,0x68,0x62,0xd8,0xc0,0x93,0x91,0xe8,0xfc,0xb1,\n                            0xa9,0x9a,0x52,0x6c,0xe0,0xbf,0x43,0x01,0xa8,0x37,0x14,0x28,0xbf,\n                            0x0b,0x15,0x56,0x3e,0xa5,0x79,0xa0,0xc4,0x42,0x88,0xee,0xeb,0x1b,\n                            0xf4,0x7a,0x4a,0x58,0x31,0x58,0x81,0xd2,0x3e,0xde,0x07,0x64,0x92,\n                            0xf0,0x60,0xd3,0x9a,0x29,0xca,0xc6,0x67,0x75,0x07,0xca,0x92,0x39,\n                            0x56,0xf6,0x11,0x84,0xba,0x6d,0x4b,0xe6,0x6f,0x66,0xc2,0x17,0xeb,\n                            0x46,0x69,0x1c,0xbb,0xdf,0xc0,0x38,0x00,0xd6,0x01,0xe6,0x70,0x9d,\n                            0x4b,0x9b,0x70,0xed,0x5c,0xb8,0xcf,0xe8,0x68,0x71,0xbe,0x24,0x6d,\n                            0xb1,0xa3,0x13,0xcc,0xf1,0xbc,0x67,0xdc,0xe0,0x69,0x09,0x82,0x3c,\n                            0x3b,0x0f,0x14,0x98,0x48,0x30,0xb2,0x70,0xc6,0x9e,0xfa,0x46,0x8f,\n                            0xf1,0xc0,0x65,0x8e,0xc6,0xae,0xdc,0x47,0x91,0x13,0x1e,0xd6,0x4a,\n                            0xf2,0xad,0xda,0xc2,0xc7,0x39,0x78,0x99,0xde,0x57,0x14,0x45,0x7f,\n                            0x32,0x38,0xa3,0x44,0x0f,0xe7,0x39,0x4c,0x6f,0x0f,0x32,0x7e,0xf1,\n                            0x5c,0x84,0x97,0xdd,0xa0,0x0c,0x87,0x66,0x7d,0x75,0x79]),\n              util.bin2str([0x04,0x00,0xc2,0xbc,0x71,0xf7,0x41,0x4a,0x09,0x66,0x70,0x02,0x68,\n                            0x8b,0xeb,0xe2,0x34,0xd1,0x12,0x83,0x93,0x75,0xe9,0x71,0x32,0xe2,\n                            0xed,0x18,0x6f,0x8e,0x3a,0xff,0x22,0x70,0x28,0x01,0xbf,0x4a,0x39,\n                            0x41,0xbb,0x3c,0x4a,0xbc,0xb8,0x13,0xfc,0x14,0xf6,0x71,0xa1,0x44,\n                            0x1c,0x02,0xa1,0x73,0x81,0xcc,0xa0,0x35,0x02,0x3e,0x97,0xb5,0xc4,\n                            0x94,0x33,0xf1,0xd1,0xdf,0x14,0x3f,0xfb,0x8f,0xb9,0x75,0x70,0xdc,\n                            0x74,0x3f,0x07,0x35,0x8f,0x53,0xaa,0xb2,0xd6,0x88,0x51,0x71,0x4e,\n                            0x01,0x24,0xec,0x7d,0xca,0xf6,0xa2,0xb3,0xbb,0xad,0x2e,0x60,0xfb,\n                            0x1c,0xee,0x49,0xd0,0x4e,0x5c,0xe3,0x1f,0x88,0x48,0xe4,0x68,0x14,\n                            0x3d,0x71,0xba,0xd7,0x4d,0x35,0x10,0x86,0x37,0x62,0xe0,0xa5,0x0b]),\n              util.bin2str([0x04,0x00,0xe2,0x38,0xf9,0xc8,0x3c,0xd1,0xcf,0x62,0x93,0xc3,0x77,\n                            0x76,0x97,0x44,0xe8,0xc8,0xca,0x93,0x9a,0xef,0xf0,0x63,0x76,0x25,\n                            0x3b,0x1c,0x46,0xff,0x90,0x13,0x91,0x15,0x97,0x7e,0x88,0x95,0xd4,\n                            0x7f,0x2f,0x52,0x6e,0x0d,0x55,0x55,0x2e,0xf1,0x58,0x5c,0x7e,0x56,\n                            0xd4,0x48,0xaa,0xdb,0x8c,0x44,0x4d,0x84,0x69,0x33,0x87,0x07,0xb2,\n                            0x7e,0xf5,0xa0,0x60,0xfb,0x73,0x59,0x46,0x29,0xcb,0x1e,0x3f,0x7c,\n                            0x2f,0xa6,0x53,0xe3,0x8c,0xef,0xd5,0xeb,0xbb,0xc8,0x9a,0x8e,0x66,\n                            0x4a,0x47,0x2f,0xe1,0xba,0x5e,0x32,0xd4,0x52,0x04,0x88,0x9d,0x63,\n                            0x3e,0xba,0x71,0x2d,0xf7,0x61,0xd5,0xfc,0x26,0xbf,0xd8,0x60,0x92,\n                            0x7b,0x94,0xf8,0x6f,0x3d,0x97,0x0b,0x0c,0x52,0x8c,0xb3,0xb6,0x8b]),\n              util.bin2str([0x04,0x00,0xb7,0xc5,0x4d,0x6e,0x2f,0xdd,0xef,0xec,0x07,0x70,0xa2,\n                            0x7c,0x1c,0x9d,0x8e,0x66,0x60,0x7c,0x61,0x1e,0x45,0xe9,0xdc,0x82,\n                            0x2f,0xc5,0x7e,0x1a,0xc6,0xd0,0x92,0xc5,0x22,0x9b,0x9a,0xfb,0x73,\n                            0x95,0x99,0xf2,0x7c,0xdb,0x2a,0x93,0x7b,0x5a,0x29,0x73,0x24,0x16,\n                            0x41,0x49,0xb5,0xf2,0x5f,0xbe,0xe7,0x64,0x4d,0xda,0x52,0x9e,0xc1,\n                            0x41,0x40,0x5e,0x03,0x92,0x8d,0x39,0x95,0x1f,0x68,0x9f,0x00,0x2e,\n                            0x0c,0x6f,0xcf,0xd9,0x6d,0x68,0xf7,0x00,0x4f,0x0e,0xc8,0x0b,0xfa,\n                            0x51,0xe0,0x22,0xf0,0xff,0xa7,0x42,0xd4,0xde,0x0b,0x47,0x8f,0x2b,\n                            0xf5,0x4d,0x04,0x32,0x91,0x89,0x4b,0x0e,0x05,0x8d,0x70,0xf9,0xbb,\n                            0xe7,0xd6,0x76,0xea,0x0e,0x1a,0x90,0x30,0xf5,0x98,0x01,0xc5,0x73])];\n  \n  var DSApubMPIstrs = [\n          util.bin2str([0x08,0x00,0xa8,0x85,0x5c,0x28,0x05,0x94,0x03,0xbe,0x07,0x6c,0x13,0x3e,0x65,\n                        0xfb,0xb5,0xe1,0x99,0x7c,0xfa,0x84,0xe3,0xac,0x47,0xa5,0xc4,0x46,0xd8,0x5f,\n                        0x44,0xe9,0xc1,0x6b,0x69,0xf7,0x10,0x76,0x49,0xa7,0x25,0x85,0xf4,0x1b,0xed,\n                        0xc6,0x60,0xc4,0x5b,0xaa,0xd4,0x87,0xd6,0x8f,0x92,0x56,0x7d,0x55,0x3f,0x45,\n                        0xae,0x12,0x73,0xda,0x29,0x8c,0xba,0x32,0xcc,0xd7,0xa4,0xd0,0x24,0xb0,0x7c,\n                        0xd8,0x0c,0x3a,0x91,0x6f,0x98,0x40,0x9c,0x9a,0xa8,0xcc,0x28,0x27,0x95,0x0b,\n                        0xe1,0x5b,0xb9,0x3b,0x1c,0x1c,0xd2,0xec,0xab,0x07,0x25,0x8d,0x7a,0x2a,0x2b,\n                        0x16,0x14,0xe8,0xda,0x71,0xd2,0xab,0xba,0x85,0x14,0x0d,0xc5,0xe0,0x88,0xeb,\n                        0xa5,0xe2,0xd5,0x48,0x3d,0x74,0x0c,0x41,0xeb,0xfd,0xb6,0x4e,0xf9,0x2c,0x82,\n                        0x17,0xdd,0x64,0x1e,0x19,0x39,0xa3,0x7f,0xf9,0x00,0xcd,0x9b,0xda,0x2e,0xbd,\n                        0x71,0x12,0xdf,0x0d,0x7c,0x0a,0x6b,0x2d,0x21,0x3b,0x9c,0x66,0x93,0x4a,0x1e,\n                        0x90,0x79,0xd3,0x5a,0x5b,0xe5,0xb9,0x94,0x1b,0xe6,0x47,0x99,0x06,0x98,0xd8,\n                        0x2a,0xe5,0xe2,0xa6,0x95,0x6a,0x07,0xc8,0xac,0x7c,0xe9,0xfc,0xa2,0x6a,0x16,\n                        0x2c,0x94,0x98,0xbd,0x91,0x0a,0x7c,0x7c,0x2c,0xb9,0x7e,0xa2,0x51,0x8b,0x45,\n                        0x1d,0x46,0x34,0xa8,0x52,0x2b,0xdd,0xd9,0xa8,0xbc,0x46,0x78,0x66,0xe1,0x72,\n                        0x11,0xf1,0xcb,0x1a,0xb6,0x4e,0x05,0x54,0xf7,0xe9,0xbe,0x4c,0x25,0x59,0x08,\n                        0x9f,0xf8,0xea,0x25,0x97,0x33,0xd6,0xc9,0x0f,0x59,0x0e,0xfd,0x9f,0xdc,0xe2,\n                        0xc0,0xcf,0x2f]),\n      util.bin2str([0x01,0x00,0xe1,0x72,0x2c,0xd0,0xbb,0x1a,0x4f,0xb6,0xb6,0x95,0x77,0x71,0x2e,\n                    0x01,0x48,0x3e,0x35,0x54,0x64,0x2b,0xed,0x40,0x5f,0x65,0x0c,0x57,0x28,0x5f,\n                    0xfd,0xfd,0xff,0xd7]),\n      util.bin2str([0x07,0xff,0x5d,0x9f,0xc4,0xb5,0x63,0x25,0x9d,0x72,0x88,0xe5,0x53,0x46,0x98,\n                    0xe3,0xe9,0x62,0xcb,0x0c,0xa1,0xb7,0x75,0x9f,0x18,0x41,0x94,0x32,0x28,0x29,\n                    0x6d,0x69,0xe0,0x3f,0x7d,0x7b,0x2b,0x06,0x5a,0x33,0x5c,0xd4,0x36,0x31,0x09,\n                    0x54,0x85,0x9d,0xb8,0x20,0xfe,0xda,0xfc,0xcd,0x1f,0xb1,0x2c,0x15,0x08,0x9d,\n                    0x32,0x53,0x2f,0xc1,0x42,0x22,0x69,0xff,0x67,0x2e,0x39,0x97,0x50,0x66,0x39,\n                    0xda,0xcf,0xfd,0x64,0x6f,0x91,0x05,0x64,0x37,0xc5,0x07,0x24,0xaa,0x40,0xa0,\n                    0x75,0x82,0x1d,0x97,0x96,0x12,0xf1,0xbd,0x9e,0x09,0x26,0x3c,0x97,0x5d,0x57,\n                    0xb8,0x5c,0x7d,0x89,0x03,0x82,0xcd,0x40,0xe5,0x03,0xe6,0x4a,0xfb,0xbc,0xd2,\n                    0xef,0x7a,0x89,0x02,0x08,0xc8,0x52,0xfa,0x97,0x74,0x66,0x32,0xae,0xa6,0x52,\n                    0x4b,0xef,0x5f,0xce,0x91,0x23,0x3f,0xab,0x9d,0x62,0x21,0xef,0x48,0x6d,0x07,\n                    0x5a,0xba,0xdf,0x00,0x91,0x54,0xea,0x5c,0xfa,0x4b,0x16,0x28,0x1a,0xce,0x48,\n                    0xb7,0x5c,0x50,0xa5,0x59,0xa4,0xb4,0xaf,0x1f,0xeb,0x8d,0x58,0x3f,0x0a,0xa5,\n                    0x97,0x2b,0x51,0x56,0xe8,0x88,0xf6,0x07,0xbc,0xdf,0xfa,0x2b,0x7b,0x88,0xe0,\n                    0x46,0xc8,0x7a,0x3e,0xd8,0x80,0xdb,0x4d,0x87,0x61,0x4f,0x64,0xcd,0xeb,0xe8,\n                    0x0d,0x86,0x16,0xcc,0xdd,0x6c,0x76,0x66,0xc1,0x73,0xb7,0x08,0x98,0x89,0x2f,\n                    0x67,0x69,0xd1,0xfc,0x97,0x4d,0xa2,0xce,0xad,0xbb,0x6f,0xab,0xa5,0xd6,0x18,\n                    0xb3,0x1a,0x96,0x02,0xbc,0x31,0x42,0xa2,0xad,0x77,0xe8,0xe2,0x4c,0x99,0xf9,\n                    0xdd,0xbe,0xcd]),\n      util.bin2str([0x07,0xff,0x5d,0xfe,0x9c,0x98,0xef,0x3a,0xa6,0x49,0xf0,0x10,0x67,0x79,0x2a,\n                    0x9d,0x79,0x43,0x06,0xa4,0xa8,0x6b,0x1a,0x6d,0x1f,0x77,0x6e,0x00,0x31,0xb9,\n                    0xed,0xc9,0x66,0xff,0xf1,0x21,0x32,0xfa,0x62,0x43,0xcd,0x97,0xd3,0x3d,0xaf,\n                    0xb4,0x29,0x29,0x26,0x4e,0x1c,0xa0,0xad,0x1c,0x07,0x28,0x3f,0xe5,0x43,0x10,\n                    0xba,0xb4,0x08,0xe0,0xdc,0xa2,0xc3,0x5b,0x1f,0xbd,0x94,0xc7,0x43,0xe5,0xf2,\n                    0x17,0x30,0x54,0x7f,0x14,0xbe,0xf4,0xbd,0x91,0x3b,0xe4,0x36,0xa4,0x50,0x5b,\n                    0x37,0x89,0x5e,0xcc,0xc7,0x74,0x54,0x32,0x20,0x09,0x63,0x98,0xb7,0xd9,0xaf,\n                    0x7f,0xb0,0x6c,0x27,0x43,0xfe,0x52,0xe6,0x1a,0x88,0x59,0x25,0xfc,0xeb,0x43,\n                    0x50,0xc7,0x65,0x43,0xc1,0x86,0x73,0x58,0x53,0x3a,0xcf,0x7a,0xa3,0x1d,0x56,\n                    0xc8,0x4a,0x80,0x70,0xb7,0xbf,0xf2,0xa3,0xec,0xe8,0x77,0x05,0x33,0x09,0x9d,\n                    0xaa,0xca,0xa0,0xe1,0x64,0x64,0x6f,0x76,0x99,0x41,0x75,0x78,0x90,0xf6,0xe7,\n                    0x23,0xe6,0xec,0x50,0xe5,0x99,0xa8,0x3e,0x1a,0x4b,0xc9,0x88,0x58,0x66,0xae,\n                    0x1a,0x53,0x5e,0xe4,0xb7,0x86,0xcf,0xa6,0xe5,0xad,0xb4,0x80,0xa0,0xf1,0x0d,\n                    0x96,0xb8,0x41,0xd0,0x07,0x9a,0x21,0x8d,0x50,0x7f,0x4f,0x73,0x13,0xa2,0xe2,\n                    0x02,0x07,0xc3,0xa3,0x0f,0x09,0x18,0x7f,0xf7,0x6b,0x90,0x70,0xc0,0xf9,0x0c,\n                    0x67,0x8d,0x9d,0x14,0xb6,0x9d,0x32,0x82,0xd0,0xb5,0xc6,0x57,0xf0,0x91,0xd9,\n                    0xc3,0x26,0xae,0x9f,0xa9,0x67,0x49,0x96,0x5c,0x07,0x3e,0x47,0x5c,0xed,0x60,\n                    0x07,0xac,0x6a])];\n  var DSAsecMPIstrs = [util.bin2str([0x01,0x00,0x9b,0x58,0xa8,0xf4,0x04,0xb1,0xd5,0x14,0x09,0xe1,\n                           0xe1,0xa1,0x8a,0x0b,0xa3,0xc3,0xa3,0x66,0xaa,0x27,0x99,0x50,\n                           0x1c,0x4d,0xba,0x24,0xee,0xdf,0xdf,0xb8,0x8e,0x8e])];\n            \n  var ElgamalpubMPIstrs = \n          [util.bin2str([0x08,0x00,0xea,0xcc,0xbe,0xe2,0xe4,0x5a,0x51,0x18,0x93,0xa1,0x12,0x2f,\n                     0x00,0x99,0x42,0xd8,0x5c,0x1c,0x2f,0xb6,0x3c,0xd9,0x94,0x61,0xb4,0x55,\n                     0x8d,0x4e,0x73,0xe6,0x69,0xbc,0x1d,0x33,0xe3,0x2d,0x91,0x23,0x69,0x95,\n                     0x98,0xd7,0x18,0x5a,0xaf,0xa7,0x93,0xc6,0x05,0x93,0x3a,0xc7,0xea,0xd0,\n                     0xb1,0xa9,0xc7,0xab,0x41,0x89,0xc8,0x38,0x99,0xdc,0x1a,0x57,0x35,0x1a,\n                     0x27,0x62,0x40,0x71,0x9f,0x36,0x1c,0x6d,0x18,0x1c,0x93,0xf7,0xba,0x35,\n                     0x06,0xed,0x30,0xb8,0xd9,0x8a,0x7c,0x03,0xaf,0xba,0x40,0x1f,0x62,0xf1,\n                     0x6d,0x87,0x2c,0xa6,0x2e,0x46,0xb0,0xaa,0xbc,0xbc,0x93,0xfa,0x9b,0x47,\n                     0x3f,0x70,0x1f,0x2a,0xc2,0x66,0x9c,0x7c,0x69,0xe0,0x2b,0x05,0xee,0xb7,\n                     0xa7,0x7f,0xf3,0x21,0x48,0x85,0xc2,0x95,0x5f,0x6f,0x1e,0xb3,0x9b,0x97,\n                     0xf8,0x14,0xc3,0xff,0x4d,0x97,0x25,0x29,0x94,0x41,0x4b,0x90,0xd8,0xba,\n                     0x71,0x45,0x4b,0x1e,0x2f,0xca,0x82,0x5f,0x56,0x77,0xe9,0xd3,0x88,0x5d,\n                     0x8b,0xec,0x92,0x8b,0x8a,0x23,0x88,0x05,0xf8,0x2c,0xa8,0xf1,0x70,0x76,\n                     0xe7,0xbf,0x75,0xa8,0x31,0x14,0x8e,0x76,0xc8,0x01,0xa6,0x25,0x27,0x49,\n                     0xaf,0xdc,0xf4,0xf6,0xf4,0xce,0x90,0x84,0x15,0x2b,0x4d,0xb3,0xcc,0x77,\n                     0xdb,0x65,0x71,0x75,0xd3,0x00,0x1d,0x22,0xc5,0x42,0x2f,0x51,0xfa,0x7b,\n                     0xeb,0x6e,0x03,0xd9,0x41,0xdd,0x2d,0x1a,0xdd,0x07,0x74,0x8b,0xb7,0xa2,\n                     0xfa,0xb2,0x59,0x0e,0x0e,0x94,0x7c,0x00,0xad,0x95,0x23,0x42,0x91,0x18,\n                     0x4c,0x97,0xf1,0x27,0x62,0x77]),\n       util.bin2str([0x00,0x03,0x05]),\n       util.bin2str([0x07,0xff,0x57,0x19,0x76,0xfc,0x09,0x6a,0x7a,0xf7,0xba,0xb2,0x42,0xbf,\n                     0xcd,0x2b,0xc1,0x1a,0x79,0x25,0x8c,0xad,0xf4,0x3a,0x0a,0x7a,0x9b,0x4c,\n                     0x46,0x3c,0xe0,0x4f,0xcc,0x6e,0xe5,0x7a,0x33,0x3a,0x4e,0x80,0xcb,0xd3,\n                     0x62,0xd7,0x8f,0xe2,0xc8,0xb0,0xd0,0xcb,0x49,0xc9,0x9e,0x2d,0x97,0x16,\n                     0x3a,0x7d,0xb1,0xe1,0xd3,0xd9,0xd7,0x3f,0x20,0x60,0xe3,0x3e,0x77,0xea,\n                     0x0c,0xe4,0x7b,0xf0,0x39,0x1a,0x0d,0xd9,0x8f,0x73,0xd2,0x51,0xb8,0x0c,\n                     0x0e,0x15,0x1e,0xad,0x7c,0xd8,0x9d,0x74,0x6e,0xa2,0x17,0x6b,0x58,0x14,\n                     0x2b,0xb7,0xad,0x8a,0xd7,0x66,0xc0,0xdf,0xea,0x2d,0xfc,0xc4,0x6e,0x68,\n                     0xb6,0x4c,0x9a,0x16,0xa4,0x3d,0xc2,0x26,0x0c,0xb7,0xd4,0x13,0x7b,0x22,\n                     0xfd,0x84,0xd7,0x0f,0xdc,0x42,0x75,0x05,0x85,0x29,0x00,0x31,0x1d,0xec,\n                     0x4e,0x22,0x8b,0xf6,0x37,0x83,0x45,0xe5,0xb3,0x31,0x61,0x2c,0x02,0xa1,\n                     0xc6,0x9d,0xea,0xba,0x3d,0x8a,0xab,0x0f,0x61,0x5e,0x14,0x64,0x69,0x1e,\n                     0xa0,0x15,0x48,0x86,0xe5,0x11,0x06,0xe8,0xde,0x34,0xc7,0xa7,0x3d,0x35,\n                     0xd1,0x76,0xc2,0xbe,0x01,0x82,0x61,0x8d,0xe7,0x7e,0x28,0x1d,0x4e,0x8c,\n                     0xb9,0xe8,0x7e,0xa4,0x5f,0xa6,0x3a,0x9e,0x5d,0xac,0xf3,0x60,0x22,0x14,\n                     0xd5,0xd5,0xbe,0x1f,0xf0,0x19,0xe6,0x81,0xfd,0x5d,0xe1,0xf8,0x76,0x5f,\n                     0xe3,0xda,0xba,0x19,0xf3,0xcb,0x10,0xa0,0x6b,0xd0,0x2d,0xbe,0x40,0x42,\n                     0x7b,0x9b,0x15,0xa4,0x2d,0xec,0xcf,0x09,0xd6,0xe3,0x92,0xc3,0x8d,0x65,\n                     0x6b,0x60,0x97,0xda,0x6b,0xca])];\n\n  var ElgamalsecMPIstrs = [\n             util.bin2str([0x01,0x52,0x02,0x80,0x87,0xf6,0xe4,0x49,0xd7,0x2e,0x3e,0xfe,0x60,0xb9,\n                           0xa3,0x2a,0xf0,0x67,0x58,0xe9,0xf6,0x47,0x83,0xde,0x7e,0xfb,0xbb,0xbd,\n                           0xdf,0x48,0x12,0x1b,0x06,0x7d,0x13,0xbc,0x3b,0x49,0xf9,0x86,0xd4,0x53,\n                           0xed,0x2d,0x68])];\n\n  var RSApubMPIs = [];\n  var i;\n  for (i = 0; i < 2; i++) {\n    RSApubMPIs[i] = new openpgp.mpi();\n    RSApubMPIs[i].read(RSApubMPIstrs[i]);\n  }\n\n  var RSAsecMPIs = [];\n  for (i = 0; i < 4; i++) {\n    RSAsecMPIs[i] = new openpgp.mpi();\n    RSAsecMPIs[i].read(RSAsecMPIstrs[i]);\n  }\n    \n  var DSAsecMPIs = [];\n  for (i = 0; i < 1; i++) {\n    DSAsecMPIs[i] = new openpgp.mpi();\n    DSAsecMPIs[i].read(DSAsecMPIstrs[i]);\n  }\n    \n  var DSApubMPIs = [];\n  for (i = 0; i < 4; i++) {\n    DSApubMPIs[i] = new openpgp.mpi();\n    DSApubMPIs[i].read(DSApubMPIstrs[i]);\n  }\n  var ElgamalsecMPIs = [];\n  for (i = 0; i < 1; i++) {\n    ElgamalsecMPIs[i] = new openpgp.mpi();\n    ElgamalsecMPIs[i].read(ElgamalsecMPIstrs[i]);\n  }\n    \n  var ElgamalpubMPIs = [];\n  for (i = 0; i < 3; i++) {\n    ElgamalpubMPIs[i] = new openpgp.mpi();\n    ElgamalpubMPIs[i].read(ElgamalpubMPIstrs[i]);\n  }\n\n  //Originally we passed public and secret MPI separately, now they are joined. Is this what we want to do long term?\n  // RSA\n  var RSAsignedData = openpgp.crypto.signature.sign(2, 1, RSApubMPIs.concat(RSAsecMPIs), \"foobar\");\n  var RSAsignedDataMPI = new openpgp.mpi();\n  RSAsignedDataMPI.read(RSAsignedData);\n  result[0] = new unit.result(\"Testing RSA Sign and Verify\",\n      openpgp.crypto.signature.verify(1, 2, [RSAsignedDataMPI], RSApubMPIs, \"foobar\"));\n\n  // DSA \n  var DSAsignedData = openpgp.crypto.signature.sign(2, 17, DSApubMPIs.concat(DSAsecMPIs), \"foobar\");\n  \n  var DSAmsgMPIs = [];\n    DSAmsgMPIs[0] = new openpgp.mpi();\n    DSAmsgMPIs[1] = new openpgp.mpi();\n    DSAmsgMPIs[0].read(DSAsignedData.substring(0,34));\n    DSAmsgMPIs[1].read(DSAsignedData.substring(34,68));\n  result[1] = new unit.result(\"Testing DSA Sign and Verify\",\n      openpgp.crypto.signature.verify(17, 2, DSAmsgMPIs, DSApubMPIs, \"foobar\"));\n  \n  var symmAlgo = \"aes256\"; // AES256\n  var symmKey = openpgp.crypto.generateSessionKey(symmAlgo);\n  var symmencDataOCFB = openpgp.crypto.cfb.encrypt(openpgp.crypto.getPrefixRandom(symmAlgo), symmAlgo, \"foobarfoobar1234567890\", symmKey, true);\n  var symmencDataCFB  = openpgp.crypto.cfb.encrypt(openpgp.crypto.getPrefixRandom(symmAlgo), symmAlgo, \"foobarfoobar1234567890\", symmKey, false);\n  \n  result[2] = new unit.result(\"Testing symmetric encrypt and decrypt with OpenPGP CFB resync\",\n      openpgp.crypto.cfb.decrypt(symmAlgo,symmKey,symmencDataOCFB,true) == \"foobarfoobar1234567890\");\n  result[3] = new unit.result(\"Testing symmetric encrypt and decrypt without OpenPGP CFB resync (used in modification detection code \\\"MDC\\\" packets)\",\n      openpgp.crypto.cfb.decrypt(symmAlgo,symmKey,symmencDataCFB,false) == \"foobarfoobar1234567890\");\n  \n  var RSAUnencryptedData = new openpgp.mpi();\n  RSAUnencryptedData.fromBytes(openpgp.crypto.pkcs1.eme.encode(symmKey, RSApubMPIs[0].byteLength()));\n  var RSAEncryptedData = openpgp.crypto.publicKeyEncrypt(\"rsa_encrypt_sign\", RSApubMPIs, RSAUnencryptedData);\n\n  result[4] = new unit.result(\"Testing asymmetric encrypt and decrypt using RSA with eme_pkcs1 padding\",\n      openpgp.crypto.pkcs1.eme.decode(openpgp.crypto.publicKeyDecrypt(\"rsa_encrypt_sign\", RSApubMPIs.concat(RSAsecMPIs), RSAEncryptedData).write().substring(2), RSApubMPIs[0].byteLength()) == symmKey);\n\n  var ElgamalUnencryptedData = new openpgp.mpi();\n  ElgamalUnencryptedData.fromBytes(openpgp.crypto.pkcs1.eme.encode(symmKey, ElgamalpubMPIs[0].byteLength()));\n  var ElgamalEncryptedData = openpgp.crypto.publicKeyEncrypt(\"elgamal\", ElgamalpubMPIs, ElgamalUnencryptedData);\n\n  result[5] = new unit.result(\"Testing asymmetric encrypt and decrypt using Elgamal with eme_pkcs1 padding\",\n      openpgp.crypto.pkcs1.eme.decode(openpgp.crypto.publicKeyDecrypt(\"elgamal\", ElgamalpubMPIs.concat(ElgamalsecMPIs), ElgamalEncryptedData).write().substring(2), ElgamalpubMPIs[0].byteLength()) == symmKey);\n\n  return result;\n});\n","var unit = require('../unit.js');\n\nunit.register(\"Testing of binary signature checking\", function() {\n  var openpgp = require('../../');\n  var keyring = openpgp.keyring;\n  var result = [];\n  var priv_key = openpgp.key.readArmored([\n        '-----BEGIN PGP PRIVATE KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'lQHhBFERnrMRBADmM0hIfkI3yosjgbWo9v0Lnr3CCE+8KsMszgVS+hBu0XfGraKm',\n        'ivcA2aaJimHqVYOP7gEnwFAxHBBpeTJcu5wzCFyJwEYqVeS3nnaIhBPplSF14Duf',\n        'i6bB9RV7KxVAg6aunmM2tAutqC+a0y2rDaf7jkJoZ9gWJe2zI+vraD6fiwCgxvHo',\n        '3IgULB9RqIqpLoMgXfcjC+cD/1jeJlKRm+n71ryYwT/ECKsspFz7S36z6q3XyS8Q',\n        'QfrsUz2p1fbFicvJwIOJ8B20J/N2/nit4P0gBUTUxv3QEa7XCM/56/xrGkyBzscW',\n        'AzBoy/AK9K7GN6z13RozuAS60F1xO7MQc6Yi2VU3eASDQEKiyL/Ubf/s/rkZ+sGj',\n        'yJizBACtwCbQzA+z9XBZNUat5NPgcZz5Qeh1nwF9Nxnr6pyBv7tkrLh/3gxRGHqG',\n        '063dMbUk8pmUcJzBUyRsNiIPDoEUsLjY5zmZZmp/waAhpREsnK29WLCbqLdpUors',\n        'c1JJBsObkA1IM8TZY8YUmvsMEvBLCCanuKpclZZXqeRAeOHJ0v4DAwK8WfuTe5B+',\n        'M2BOOeZbN8BpfiA1l//fMMHLRS3UvbLBv4P1+4SyvhyYTR7M76Q0xPc03MFOWHL+',\n        'S9VumbQWVGVzdDIgPHRlc3QyQHRlc3QuY29tPohiBBMRAgAiBQJREZ6zAhsDBgsJ',\n        'CAcDAgYVCAIJCgsEFgIDAQIeAQIXgAAKCRARJ5QDyxae+MXNAKCzWSDR3tMrTrDb',\n        'TAri73N1Xb3j1ACfSl9y+SAah2q7GvmiR1+6+/ekqJGdAVgEURGesxAEANlpMZjW',\n        '33jMxlKHDdyRFXtKOq8RreXhq00plorHbgz9zFEWm4VF53+E/KGnmHGyY5Cy8TKy',\n        'ZjaueZZ9XuG0huZg5If68irFfNZtxdA26jv8//PdZ0Uj+X6J3RVa2peMLDDswTYL',\n        'OL1ZO1fxdtDD40fdAiIZ1QyjwEG0APtz41EfAAMFBAC5/dtgBBPtHe8UjDBaUe4n',\n        'NzHuUBBp6XE+H7eqHNFCuZAJ7yqJLGVHNIaQR419cNy08/OO/+YUQ7rg78LxjFiv',\n        'CH7IzhfU+6yvELSbgRMicY6EnAP2GT+b1+MtFNa3lBGtBHcJla52c2rTAHthYZWk',\n        'fT5R5DnJuQ2cJHBMS9HWyP4DAwK8WfuTe5B+M2C7a/YJSUv6SexdGCaiaTcAm6g/',\n        'PvA6hw/FLzIEP67QcQSSTmhftQIwnddt4S4MyJJH3U4fJaFfYQ1zCniYJohJBBgR',\n        'AgAJBQJREZ6zAhsMAAoJEBEnlAPLFp74QbMAn3V4857xwnO9/+vzIVnL93W3k0/8',\n        'AKC8omYPPomN1E/UJFfXdLDIMi5LoA==',\n        '=LSrW',\n        '-----END PGP PRIVATE KEY BLOCK-----'\n      ].join(\"\\n\")).packets;\n  var pub_key = openpgp.key.readArmored(\n      [ '-----BEGIN PGP PUBLIC KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'mQGiBFERlw4RBAD6Bmcf2w1dtUmtCLkdxeqZLArk3vYoQAjdibxA3gXVyur7fsWb',\n        'ro0jVbBHqOCtC6jDxE2l52NP9+tTlWeVMaqqNvUE47LSaPq2DGI8Wx1Rj6bF3mTs',\n        'obYEwhGbGh/MhJnME9AHODarvk8AZbzo0+k1EwrBWF6dTUBPfqO7rGU2ewCg80WV',\n        'x5pt3evj8rRK3jQ8SMKTNRsD/1PhTdxdZTdXARAFzcW1VaaruWW0Rr1+XHKKwDCz',\n        'i7HE76SO9qjnQfZCZG75CdQxI0h8GFeN3zsDqmhob2iSz2aJ1krtjM+iZ1FBFd57',\n        'OqCV6wmk5IT0RBN12ZzMS19YvzN/ONXHrmTZlKExd9Mh9RKLeVNw+bf6JsKQEzcY',\n        'JzFkBACX9X+hDYchO/2hiTwx4iOO9Fhsuh7eIWumB3gt+aUpm1jrSbas/QLTymmk',\n        'uZuQVXI4NtnlvzlNgWv4L5s5RU5WqNGG7WSaKNdcrvJZRC2dgbUJt04J5CKrWp6R',\n        'aIYal/81Ut1778lU01PEt563TcQnUBlnjU5OR25KhfSeN5CZY7QUVGVzdCA8dGVz',\n        'dEB0ZXN0LmNvbT6IYgQTEQIAIgUCURGXDgIbAwYLCQgHAwIGFQgCCQoLBBYCAwEC',\n        'HgECF4AACgkQikDlZK/UvLSspgCfcNaOpTg1W2ucR1JwBbBGvaERfuMAnRgt3/rs',\n        'EplqEakMckCtikEnpxYe',\n        '=b2Ln',\n        '-----END PGP PUBLIC KEY BLOCK-----'\n      ].join(\"\\n\")).packets;\n  var msg = openpgp.message.readArmored([\n        '-----BEGIN PGP MESSAGE-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'hQEOA1N4OCSSjECBEAP/diDJCQn4e88193PgqhbfAkohk9RQ0v0MPnXpJbCRTHKO',\n        '8r9nxiAr/TQv4ZOingXdAp2JZEoE9pXxZ3r1UWew04czxmgJ8FP1ztZYWVFAWFVi',\n        'Tj930TBD7L1fY/MD4fK6xjEG7z5GT8k4tn4mLm/PpWMbarIglfMopTy1M/py2cID',\n        '/2Sj7Ikh3UFiG+zm4sViYc5roNbMy8ixeoKixxi99Mx8INa2cxNfqbabjblFyc0Z',\n        'BwmbIc+ZiY2meRNI5y/tk0gRD7hT84IXGGl6/mH00bsX/kkWdKGeTvz8s5G8RDHa',\n        'Za4HgLbXItkX/QarvRS9kvkD01ujHfj+1ZvgmOBttNfP0p8BQLIICqvg1eYD9aPB',\n        '+GtOZ2F3+k5VyBL5yIn/s65SBjNO8Fqs3aL0x+p7s1cfUzx8J8a8nWpqq/qIQIqg',\n        'ZJH6MZRKuQwscwH6NWgsSVwcnVCAXnYOpbHxFQ+j7RbF/+uiuqU+DFH/Rd5pik8b',\n        '0Dqnp0yfefrkjQ0nuvubgB6Rv89mHpnvuJfFJRInpg4lrHwLvRwdpN2HDozFHcKK',\n        'aOU=',\n        '=4iGt',\n        '-----END PGP MESSAGE-----'\n      ].join(\"\\n\")).packets;\n  //TODO need both?\n  priv_key[0].decrypt(\"abcd\");\n  priv_key[3].decrypt(\"abcd\");\n  msg[0].decrypt(priv_key[3]);\n  msg[1].decrypt(msg[0].sessionKeyAlgorithm, msg[0].sessionKey);\n  msg[1].packets[2].verify(pub_key[0], msg[1].packets[1]);\n  result[0] = new unit.result(\"Testing signature checking on CAST5-enciphered message\",\n          msg[1].packets[2].verified === true);\n\n  // exercises the GnuPG s2k type 1001 extension:\n  // the secrets on the primary key have been stripped.\n  var priv_key_gnupg_ext = openpgp.key.readArmored([\n        '-----BEGIN PGP PRIVATE KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'lQGqBFERnrMRBADmM0hIfkI3yosjgbWo9v0Lnr3CCE+8KsMszgVS+hBu0XfGraKm',\n        'ivcA2aaJimHqVYOP7gEnwFAxHBBpeTJcu5wzCFyJwEYqVeS3nnaIhBPplSF14Duf',\n        'i6bB9RV7KxVAg6aunmM2tAutqC+a0y2rDaf7jkJoZ9gWJe2zI+vraD6fiwCgxvHo',\n        '3IgULB9RqIqpLoMgXfcjC+cD/1jeJlKRm+n71ryYwT/ECKsspFz7S36z6q3XyS8Q',\n        'QfrsUz2p1fbFicvJwIOJ8B20J/N2/nit4P0gBUTUxv3QEa7XCM/56/xrGkyBzscW',\n        'AzBoy/AK9K7GN6z13RozuAS60F1xO7MQc6Yi2VU3eASDQEKiyL/Ubf/s/rkZ+sGj',\n        'yJizBACtwCbQzA+z9XBZNUat5NPgcZz5Qeh1nwF9Nxnr6pyBv7tkrLh/3gxRGHqG',\n        '063dMbUk8pmUcJzBUyRsNiIPDoEUsLjY5zmZZmp/waAhpREsnK29WLCbqLdpUors',\n        'c1JJBsObkA1IM8TZY8YUmvsMEvBLCCanuKpclZZXqeRAeOHJ0v4DZQJHTlUBtBZU',\n        'ZXN0MiA8dGVzdDJAdGVzdC5jb20+iGIEExECACIFAlERnrMCGwMGCwkIBwMCBhUI',\n        'AgkKCwQWAgMBAh4BAheAAAoJEBEnlAPLFp74xc0AoLNZINHe0ytOsNtMCuLvc3Vd',\n        'vePUAJ9KX3L5IBqHarsa+aJHX7r796SokZ0BWARREZ6zEAQA2WkxmNbfeMzGUocN',\n        '3JEVe0o6rxGt5eGrTSmWisduDP3MURabhUXnf4T8oaeYcbJjkLLxMrJmNq55ln1e',\n        '4bSG5mDkh/ryKsV81m3F0DbqO/z/891nRSP5fondFVral4wsMOzBNgs4vVk7V/F2',\n        '0MPjR90CIhnVDKPAQbQA+3PjUR8AAwUEALn922AEE+0d7xSMMFpR7ic3Me5QEGnp',\n        'cT4ft6oc0UK5kAnvKoksZUc0hpBHjX1w3LTz847/5hRDuuDvwvGMWK8IfsjOF9T7',\n        'rK8QtJuBEyJxjoScA/YZP5vX4y0U1reUEa0EdwmVrnZzatMAe2FhlaR9PlHkOcm5',\n        'DZwkcExL0dbI/gMDArxZ+5N7kH4zYLtr9glJS/pJ7F0YJqJpNwCbqD8+8DqHD8Uv',\n        'MgQ/rtBxBJJOaF+1AjCd123hLgzIkkfdTh8loV9hDXMKeJgmiEkEGBECAAkFAlER',\n        'nrMCGwwACgkQESeUA8sWnvhBswCfdXjznvHCc73/6/MhWcv3dbeTT/wAoLyiZg8+',\n        'iY3UT9QkV9d0sMgyLkug',\n        '=GQsY',\n        '-----END PGP PRIVATE KEY BLOCK-----',\n      ].join(\"\\n\")).packets;\n  priv_key_gnupg_ext[3].decrypt(\"abcd\");\n  msg[0].decrypt(priv_key_gnupg_ext[3]);\n  msg[1].decrypt(msg[0].sessionKeyAlgorithm, msg[0].sessionKey);\n  msg[1].packets[2].verify(pub_key[0], msg[1].packets[1]);\n  result[1] = new unit.result(\"Testing GnuPG stripped-key extensions\",\n          msg[1].packets[2].verified === true);\n\n  // Exercises the ability of openpgp_keyring.getPublicKeysForKeyId to return subkeys\n  keyring.init();\n  keyring.importPacketlist([\n        '-----BEGIN PGP PUBLIC KEY BLOCK-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'mQGiBFERvI4RBAD0M/HGglCtVNXPF72ehT8riAXrl0rSec4RJC61Bh+UAOhxn5+U',\n        'fDgos5p1SpIzYmn+M87JoSSVLAjfakFk0gHgR9I3bu7SIwq3Bikk1Vw3gO+yDSO6',\n        'TKpLUFGYDiBSSE1MGdxBadWLE1hlRf5B2x62gnGmjSpSVbly33PFkoDmrwCg9rAp',\n        'RmncnF9GhWjOLFkEoQw9Yx8EAOsxvq8Ig5Z1gk+ZKfDZeftpHRe3FdrRtbnhxvYY',\n        '7z+w9uz1EpoZUwDR5G4X3hTwJQ7lXmIOskg/+eRMLEAqEY7b/7tW6RaUJ2d6Ehsi',\n        'dOS89fIxElwjAOnVOM5S24f0FDQTTto7QBOoxcNTfkEJCHXSlpoOUmGAP85fXh3l',\n        'yPTGBACJfKc76Un3UWC1sWIRDxYiyh3ZpZyNEskoV6ESW8jEI1RnMnv5TrfGJH5K',\n        'E8jWX7TTnoFyPJtBTjlucAtkQaS4Bb7dg1LLja17zAqKNGOJK2b9fb2Z+lnTjPiY',\n        'i7DPH1XHnfaEexjlh/U7mYa5RrwIphRxNi8gCuxv874ZMmhEn7QWVGVzdDMgPHRl',\n        'c3QzQHRlc3QuY29tPohiBBMRAgAiBQJREbyOAhsDBgsJCAcDAgYVCAIJCgsEFgID',\n        'AQIeAQIXgAAKCRC0u8O0Moa2JYxyAJ9Oi2UlcUT0VJNgwjyl/VF9Xcjf9gCeJPvy',\n        'g/fp4EAU8MJIaN2yMI8pLFS5AaIEURG8nhEEAKVgeNDuYDPufLuJ0GrJV/CbXEjj',\n        'aEPA0iTUqV0nTCPdAfQ/nmE3gh5UlNMr/zSHJ+c4FQhYdLrzRGDOSzV+mfPHH3t+',\n        'YVx+wat0BYwABpHAtsIuLIVo2RQqYZYH85tatwBkm71HHT3jmlEAvr6NFH38+v3s',\n        '3w4Wl0/sdHyaeiSXAKCxJ4X1eOdN7L1rrbJozQ/gDCFuVQP/dcV6Ksss8Aw443jG',\n        'AYBLHWh6o4GhAY6/h1kijF0xD+uc+tNmTQnQi1tEOoTeIZMXnSRwtk8XEuJkkbAP',\n        '+uyvMgyV3wrk9zkaTAin7nrjAERxezFOdBEOtnB1CovJxtMn+RRxaMEGpC4GnETy',\n        'N5+6FkLuLcNXiCQP75ajzOAN1aID/juNjUNpBbNpfqBV7j1K+Kn0n9HYTyQl9ghy',\n        '026+/4c8ag2HV+bg3BD7c2VTVu9xBODHsfu0q8Ql/QB9W8tmYugU6DeXMHaeWPUH',\n        'ph98guM9kF2yHIiRBvAd5i7wOjwn+I/Ir6nBR2yxJ3p31CDUnUlbjTPYg7mtQvHW',\n        'EY2Cp4SWiJEEGBECAAkFAlERvJ4CGwIAUgkQtLvDtDKGtiVHIAQZEQIABgUCURG8',\n        'ngAKCRAMiMeR296Y2SjyAJ9V3wRJJ2Szazqal4khWGfLu5R6/wCfQQIRD24yVdz8',\n        '2a+2eCrwyALT2GAihACfS0nWM3a0gtITqngpJsRws+Ep+eIAn15qD2itutxNb8NI',\n        'bR2gBB5QmVJ3',\n        '=pGA6',\n        '-----END PGP PUBLIC KEY BLOCK-----'\n        ].join(\"\\n\"));\n\n  var msg2 = openpgp.message.readArmored([\n        '-----BEGIN PGP MESSAGE-----',\n        'Version: GnuPG v1.4.11 (GNU/Linux)',\n        '',\n        'kA0DAAIRDIjHkdvemNkBrB1iB2Zvby50eHRREbz3VEVTVCBEQVRBIDEyMzQ1NohG',\n        'BAARAgAGBQJREbz3AAoJEAyIx5Hb3pjZ2TcAn32LpDEuHe9QrSRlyvSuREKNOFwz',\n        'AJ9zh4zsK4GIPuEu81YPNmHsju7DYg==',\n        '=WaSx',\n        '-----END PGP MESSAGE-----'\n        ].join(\"\\n\")).packets;\n  var packetlists = keyring.getPacketlistsForKeyId(msg2[0].signingKeyId.write());\n  var pubKey = packetlists[0];\n  msg2[2].verify(pubKey[3], msg2[1]);\n  result[2] = new unit.result(\"Testing keyring public subkey support\",\n          packetlists !== null && \n          packetlists.length == 1 && \n          msg2[2].verified);\n  return result;\n});\n\n","var unit = require('../unit.js');\n\nunit.register(\"Key generation/encryption/decryption\", function() {\n  var openpgp = require('../../');\n  var keyring = openpgp.keyring;\n  var result = [];\n  var testHelper = function(passphrase, userid, message) {\n    var key = openpgp.generateKeyPair(openpgp.enums.publicKey.rsa_encrypt_sign, 512, \n                                      userid, message, passphrase);\n\n    var info = '\\npassphrase: ' + passphrase + '\\n'\n        + 'userid: ' + userid + '\\n'\n        + 'message: ' + message;\n\n    debugger;\n    var privKey = openpgp.key.readArmored(key);\n\n    var encrypted = openpgp.encryptMessage([privKey], message);\n\n    var msg = openpgp.message.readArmored(encrypted);\n\n    // Find the private (sub)key for the session key of the message\n    var privKeyPacket = privKey.getKeyPacketByIds(msg.getKeyIds());\n\n    if (!privKeyPacket) {\n      return new unit.result(\"No private key found!\" + info, false);\n    }\n\n    try {\n      if (!privKeyPacket.decrypt(passphrase)) {\n        return new unit.result(\"Password for secrect key was incorrect!\" + info, false);\n      }\n    } catch (e) {\n      return new unit.result(\"Exception on decrypt of private key packet!\" + info, false);\n    }\n\n    var decrypted = msg.decrypt(privKeyPacket)[0];\n\n    return new unit.result(message + ' == ' + decrypted + info, message == decrypted);\n  };\n\n  result.push(testHelper('password', 'Test McTestington <test@example.com>', 'hello world'));\n  result.push(testHelper('●●●●', '♔♔♔♔ <test@example.com>', 'łäóć'));\n\n  return result;\n});\n\nunit.register(\"Encryption/decryption\", function() {\n  var openpgp = require('../../');\n\n  var result = [];\n\n  var pub_key = \n     ['-----BEGIN PGP PUBLIC KEY BLOCK-----',\n      'Version: GnuPG v2.0.19 (GNU/Linux)',\n      'Type: RSA/RSA',\n      '',\n      'mI0EUmEvTgEEANyWtQQMOybQ9JltDqmaX0WnNPJeLILIM36sw6zL0nfTQ5zXSS3+',\n      'fIF6P29lJFxpblWk02PSID5zX/DYU9/zjM2xPO8Oa4xo0cVTOTLj++Ri5mtr//f5',\n      'GLsIXxFrBJhD/ghFsL3Op0GXOeLJ9A5bsOn8th7x6JucNKuaRB6bQbSPABEBAAG0',\n      'JFRlc3QgTWNUZXN0aW5ndG9uIDx0ZXN0QGV4YW1wbGUuY29tPoi5BBMBAgAjBQJS',\n      'YS9OAhsvBwsJCAcDAgEGFQgCCQoLBBYCAwECHgECF4AACgkQSmNhOk1uQJQwDAP6',\n      'AgrTyqkRlJVqz2pb46TfbDM2TDF7o9CBnBzIGoxBhlRwpqALz7z2kxBDmwpQa+ki',\n      'Bq3jZN/UosY9y8bhwMAlnrDY9jP1gdCo+H0sD48CdXybblNwaYpwqC8VSpDdTndf',\n      '9j2wE/weihGp/DAdy/2kyBCaiOY1sjhUfJ1GogF49rC4jQRSYS9OAQQA6R/PtBFa',\n      'JaT4jq10yqASk4sqwVMsc6HcifM5lSdxzExFP74naUMMyEsKHP53QxTF0Grqusag',\n      'Qg/ZtgT0CN1HUM152y7ACOdp1giKjpMzOTQClqCoclyvWOFB+L/SwGEIJf7LSCEr',\n      'woBuJifJc8xAVr0XX0JthoW+uP91eTQ3XpsAEQEAAYkBPQQYAQIACQUCUmEvTgIb',\n      'LgCoCRBKY2E6TW5AlJ0gBBkBAgAGBQJSYS9OAAoJEOCE90RsICyXuqIEANmmiRCA',\n      'SF7YK7PvFkieJNwzeK0V3F2lGX+uu6Y3Q/Zxdtwc4xR+me/CSBmsURyXTO29OWhP',\n      'GLszPH9zSJU9BdDi6v0yNprmFPX/1Ng0Abn/sCkwetvjxC1YIvTLFwtUL/7v6NS2',\n      'bZpsUxRTg9+cSrMWWSNjiY9qUKajm1tuzPDZXAUEAMNmAN3xXN/Kjyvj2OK2ck0X',\n      'W748sl/tc3qiKPMJ+0AkMF7Pjhmh9nxqE9+QCEl7qinFqqBLjuzgUhBU4QlwX1GD',\n      'AtNTq6ihLMD5v1d82ZC7tNatdlDMGWnIdvEMCv2GZcuIqDQ9rXWs49e7tq1NncLY',\n      'hz3tYjKhoFTKEIq3y3Pp',\n      '=h/aX',\n      '-----END PGP PUBLIC KEY BLOCK-----'].join('\\n');\n\n  var priv_key =\n      ['-----BEGIN PGP PRIVATE KEY BLOCK-----',\n      'Version: GnuPG v2.0.19 (GNU/Linux)',\n      'Type: RSA/RSA',\n      'Pwd: hello world',\n      '',\n      'lQH+BFJhL04BBADclrUEDDsm0PSZbQ6pml9FpzTyXiyCyDN+rMOsy9J300Oc10kt',\n      '/nyBej9vZSRcaW5VpNNj0iA+c1/w2FPf84zNsTzvDmuMaNHFUzky4/vkYuZra//3',\n      '+Ri7CF8RawSYQ/4IRbC9zqdBlzniyfQOW7Dp/LYe8eibnDSrmkQem0G0jwARAQAB',\n      '/gMDAu7L//czBpE40p1ZqO8K3k7UejemjsQqc7kOqnlDYd1Z6/3NEA/UM30Siipr',\n      'KjdIFY5+hp0hcs6EiiNq0PDfm/W2j+7HfrZ5kpeQVxDek4irezYZrl7JS2xezaLv',\n      'k0Fv/6fxasnFtjOM6Qbstu67s5Gpl9y06ZxbP3VpT62+Xeibn/swWrfiJjuGEEhM',\n      'bgnsMpHtzAz/L8y6KSzViG/05hBaqrvk3/GeEA6nE+o0+0a6r0LYLTemmq6FbaA1',\n      'PHo+x7k7oFcBFUUeSzgx78GckuPwqr2mNfeF+IuSRnrlpZl3kcbHASPAOfEkyMXS',\n      'sWGE7grCAjbyQyM3OEXTSyqnehvGS/1RdB6kDDxGwgE/QFbwNyEh6K4eaaAThW2j',\n      'IEEI0WEnRkPi9fXyxhFsCLSI1XhqTaq7iDNqJTxE+AX2b9ZuZXAxI3Tc/7++vEyL',\n      '3p18N/MB2kt1Wb1azmXWL2EKlT1BZ5yDaJuBQ8BhphM3tCRUZXN0IE1jVGVzdGlu',\n      'Z3RvbiA8dGVzdEBleGFtcGxlLmNvbT6IuQQTAQIAIwUCUmEvTgIbLwcLCQgHAwIB',\n      'BhUIAgkKCwQWAgMBAh4BAheAAAoJEEpjYTpNbkCUMAwD+gIK08qpEZSVas9qW+Ok',\n      '32wzNkwxe6PQgZwcyBqMQYZUcKagC8+89pMQQ5sKUGvpIgat42Tf1KLGPcvG4cDA',\n      'JZ6w2PYz9YHQqPh9LA+PAnV8m25TcGmKcKgvFUqQ3U53X/Y9sBP8HooRqfwwHcv9',\n      'pMgQmojmNbI4VHydRqIBePawnQH+BFJhL04BBADpH8+0EVolpPiOrXTKoBKTiyrB',\n      'UyxzodyJ8zmVJ3HMTEU/vidpQwzISwoc/ndDFMXQauq6xqBCD9m2BPQI3UdQzXnb',\n      'LsAI52nWCIqOkzM5NAKWoKhyXK9Y4UH4v9LAYQgl/stIISvCgG4mJ8lzzEBWvRdf',\n      'Qm2Ghb64/3V5NDdemwARAQAB/gMDAu7L//czBpE40iPcpLzL7GwBbWFhSWgSLy53',\n      'Md99Kxw3cApWCok2E8R9/4VS0490xKZIa5y2I/K8thVhqk96Z8Kbt7MRMC1WLHgC',\n      'qJvkeQCI6PrFM0PUIPLHAQtDJYKtaLXxYuexcAdKzZj3FHdtLNWCooK6n3vJlL1c',\n      'WjZcHJ1PH7USlj1jup4XfxsbziuysRUSyXkjn92GZLm+64vCIiwhqAYoizF2NHHG',\n      'hRTN4gQzxrxgkeVchl+ag7DkQUDANIIVI+A63JeLJgWJiH1fbYlwESByHW+zBFNt',\n      'qStjfIOhjrfNIc3RvsggbDdWQLcbxmLZj4sB0ydPSgRKoaUdRHJY0S4vp9ouKOtl',\n      '2au/P1BP3bhD0fDXl91oeheYth+MSmsJFDg/vZJzCJhFaQ9dp+2EnjN5auNCNbaI',\n      'beFJRHFf9cha8p3hh+AK54NRCT++B2MXYf+TPwqX88jYMBv8kk8vYUgo8128r1zQ',\n      'EzjviQE9BBgBAgAJBQJSYS9OAhsuAKgJEEpjYTpNbkCUnSAEGQECAAYFAlJhL04A',\n      'CgkQ4IT3RGwgLJe6ogQA2aaJEIBIXtgrs+8WSJ4k3DN4rRXcXaUZf667pjdD9nF2',\n      '3BzjFH6Z78JIGaxRHJdM7b05aE8YuzM8f3NIlT0F0OLq/TI2muYU9f/U2DQBuf+w',\n      'KTB62+PELVgi9MsXC1Qv/u/o1LZtmmxTFFOD35xKsxZZI2OJj2pQpqObW27M8Nlc',\n      'BQQAw2YA3fFc38qPK+PY4rZyTRdbvjyyX+1zeqIo8wn7QCQwXs+OGaH2fGoT35AI',\n      'SXuqKcWqoEuO7OBSEFThCXBfUYMC01OrqKEswPm/V3zZkLu01q12UMwZach28QwK',\n      '/YZly4ioND2tdazj17u2rU2dwtiHPe1iMqGgVMoQirfLc+k=',\n      '=lw5e',\n      '-----END PGP PRIVATE KEY BLOCK-----'].join('\\n');\n\n  var armored = \n     ['-----BEGIN PGP MESSAGE-----',\n      'Version: OpenPGP.js VERSION',\n      'Comment: http://openpgpjs.org',\n      '',\n      'wYwD4IT3RGwgLJcBA/wN8H9qUvsJwsarDOcsczk1L9Iif47jy+3vF6LcpSgA',\n      'DfKSARWvatyakvEJmuCNfcpzS8NUPnFA51p0VWmI7FnYG5LkPVUIHb4sN07G',\n      '9PeqhaayZIeNVvS6DoYuxiTG2sbDmyrv6MMSLivs7VcTAN6L7XXJRI2IumOS',\n      'x2WFgYNKANJFAQmen4R4IGf9nZDI7NJ4QHUlK1GENgLix9RzMK+Cri6p9Gx5',\n      '4MDV23jgCBWrUHfFYximgcXiUk0NfpVD3x6chcnKUKJv',\n      '=Eaix',\n      '-----END PGP MESSAGE-----'].join('\\n');\n\n\n  var plaintext = 'short message';\n\n  var key = openpgp.key.readArmored(pub_key);\n\n  var encrypted = openpgp.encryptMessage([key], plaintext);\n\n  var message = openpgp.message.readArmored(encrypted);\n\n  var privKey = openpgp.key.readArmored(priv_key);\n\n  var privKeyPacket = privKey.getKeyPacketByIds(message.getKeyIds());\n\n  privKeyPacket.decrypt('hello world');\n\n  var decrypted = openpgp.decryptMessage(privKeyPacket, message);\n\n  result[0] = new unit.result('Encrypt plain text and afterwards decrypt leads to same result', plaintext == decrypted);\n\n  return result;\n\n});\n","module.exports=require('QjPZ1q');","var process=require(\"__browserify_process\");\nmodule.exports = {\n\ttests: [],\n\tregister: function(str_title, func_runtest) {\n\t\tthis.tests.push({ title: str_title, run: func_runtest });\n\t},\n\t\n\trun: function() {\n\t\tvar test = this.tests.shift();\n\n\t\tvar result = {\n\t\t\ttitle: test.title\n\t\t};\n\n\n\t\tresult.tests = test.run();\n\n\t\treturn result;\n\t},\n\n\trun_all: function() {\n\t\tvar passed = true;\n\n\t\twhile(this.tests.length > 0) {\n\t\t\tvar result = this.run();\n\n\t\t\tconsole.log('Test: ' + result.title);\n\n\t\t\tfor(var i in result.tests) {\n\n\t\t\t\tvar res = result.tests[i].result ?\n\t\t\t\t\t'SUCCESS' : 'FAILED';\n\n\t\t\t\tconsole.log(result.tests[i].description + ' ' + res);\n\n\t\t\t\tpassed = passed && result.tests[i].result;\n\t\t\t}\t\t\t\t\n\t\t}\n\n\t\tif(!passed) process.exit(1);\n\t},\n\t\t\n\tresult: function(str_description, boolean_result) {\n\t\tthis.description = str_description;\n\t\tthis.result = boolean_result;\n\t}\n}\n\n"]} ; \ No newline at end of file